mirror of
https://github.com/thomiceli/opengist.git
synced 2025-01-25 15:20:36 +00:00
616 lines
13 KiB
Go
616 lines
13 KiB
Go
package git
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net/url"
|
|
"os"
|
|
"os/exec"
|
|
"path"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
"github.com/rs/zerolog/log"
|
|
"github.com/thomiceli/opengist/internal/config"
|
|
)
|
|
|
|
var (
|
|
ReposDirectory = "repos"
|
|
)
|
|
|
|
const truncateLimit = 2 << 18
|
|
const diffSize = 2 << 12
|
|
const maxFilesPerDiffCommit = 10
|
|
|
|
type RevisionNotFoundError struct{}
|
|
|
|
func (m *RevisionNotFoundError) Error() string {
|
|
return "revision not found"
|
|
}
|
|
|
|
func RepositoryPath(user string, gist string) string {
|
|
return filepath.Join(config.GetHomeDir(), ReposDirectory, strings.ToLower(user), gist)
|
|
}
|
|
|
|
func RepositoryUrl(ctx echo.Context, user string, gist string) string {
|
|
httpProtocol := "http"
|
|
if ctx.Request().TLS != nil || ctx.Request().Header.Get("X-Forwarded-Proto") == "https" {
|
|
httpProtocol = "https"
|
|
}
|
|
|
|
var baseHttpUrl string
|
|
// if a custom external url is set, use it
|
|
if config.C.ExternalUrl != "" {
|
|
baseHttpUrl = config.C.ExternalUrl
|
|
} else {
|
|
baseHttpUrl = httpProtocol + "://" + ctx.Request().Host
|
|
}
|
|
|
|
return fmt.Sprintf("%s/%s/%s", baseHttpUrl, user, gist)
|
|
}
|
|
|
|
func TmpRepositoryPath(gistId string) string {
|
|
dirname := TmpRepositoriesPath()
|
|
return filepath.Join(dirname, gistId)
|
|
}
|
|
|
|
func TmpRepositoriesPath() string {
|
|
return filepath.Join(config.GetHomeDir(), "tmp", "repos")
|
|
}
|
|
|
|
func InitRepository(user string, gist string) error {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
var args []string
|
|
args = append(args, "init")
|
|
if config.C.GitDefaultBranch != "" {
|
|
args = append(args, "--initial-branch", config.C.GitDefaultBranch)
|
|
}
|
|
args = append(args, "--bare", repositoryPath)
|
|
|
|
cmd := exec.Command("git", args...)
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return CreateDotGitFiles(user, gist)
|
|
}
|
|
|
|
func CountCommits(user string, gist string) (string, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command(
|
|
"git",
|
|
"rev-list",
|
|
"--all",
|
|
"--count",
|
|
)
|
|
cmd.Dir = repositoryPath
|
|
|
|
stdout, err := cmd.Output()
|
|
return strings.TrimSuffix(string(stdout), "\n"), err
|
|
}
|
|
|
|
func GetFilesOfRepository(user string, gist string, revision string) ([]string, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command(
|
|
"git",
|
|
"ls-tree",
|
|
"--name-only",
|
|
"--",
|
|
revision,
|
|
)
|
|
cmd.Dir = repositoryPath
|
|
|
|
stdout, err := cmd.Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
slice := strings.Split(string(stdout), "\n")
|
|
for i, s := range slice {
|
|
slice[i] = convertOctalToUTF8(s)
|
|
}
|
|
return slice[:len(slice)-1], nil
|
|
}
|
|
|
|
type catFileBatch struct {
|
|
Name, Hash, Content string
|
|
Size uint64
|
|
Truncated bool
|
|
}
|
|
|
|
func CatFileBatch(user string, gist string, revision string, truncate bool) ([]*catFileBatch, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
lsTreeCmd := exec.Command("git", "ls-tree", "-l", revision)
|
|
lsTreeCmd.Dir = repositoryPath
|
|
lsTreeOutput, err := lsTreeCmd.Output()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
fileMap := make([]*catFileBatch, 0)
|
|
|
|
lines := strings.Split(string(lsTreeOutput), "\n")
|
|
for _, line := range lines {
|
|
fields := strings.Fields(line)
|
|
if len(fields) < 4 {
|
|
continue // Skip lines that don't have enough fields
|
|
}
|
|
|
|
hash := fields[2]
|
|
size, err := strconv.ParseUint(fields[3], 10, 64)
|
|
if err != nil {
|
|
continue // Skip lines with invalid size field
|
|
}
|
|
name := strings.Join(fields[4:], " ") // File name may contain spaces
|
|
|
|
fileMap = append(fileMap, &catFileBatch{
|
|
Hash: hash,
|
|
Size: size,
|
|
Name: convertOctalToUTF8(name),
|
|
})
|
|
}
|
|
|
|
catFileCmd := exec.Command("git", "cat-file", "--batch")
|
|
catFileCmd.Dir = repositoryPath
|
|
stdin, err := catFileCmd.StdinPipe()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
stdout, err := catFileCmd.StdoutPipe()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err = catFileCmd.Start(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
reader := bufio.NewReader(stdout)
|
|
|
|
for _, file := range fileMap {
|
|
_, err = stdin.Write([]byte(file.Hash + "\n"))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
header, err := reader.ReadString('\n')
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
parts := strings.Fields(header)
|
|
if len(parts) > 3 {
|
|
continue // Not a valid header, skip this entry
|
|
}
|
|
|
|
size, err := strconv.ParseUint(parts[2], 10, 64)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sizeToRead := size
|
|
if truncate && sizeToRead > truncateLimit {
|
|
sizeToRead = truncateLimit
|
|
}
|
|
|
|
// Read exactly size bytes from header, or the max allowed if truncated
|
|
content := make([]byte, sizeToRead)
|
|
if _, err = io.ReadFull(reader, content); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
file.Content = string(content)
|
|
|
|
if truncate && size > truncateLimit {
|
|
// skip other bytes if truncated
|
|
if _, err = reader.Discard(int(size - truncateLimit)); err != nil {
|
|
return nil, err
|
|
}
|
|
file.Truncated = true
|
|
}
|
|
|
|
// Read the blank line following the content
|
|
if _, err := reader.ReadByte(); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
if err = stdin.Close(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err = catFileCmd.Wait(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return fileMap, nil
|
|
}
|
|
|
|
func GetFileContent(user string, gist string, revision string, filename string, truncate bool) (string, bool, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
var maxBytes int64 = -1
|
|
if truncate {
|
|
maxBytes = truncateLimit
|
|
}
|
|
|
|
// Set up a context with a timeout
|
|
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Minute)
|
|
defer cancel()
|
|
|
|
cmd := exec.CommandContext(
|
|
ctx,
|
|
"git",
|
|
"--no-pager",
|
|
"show",
|
|
revision+":"+convertURLToOctal(filename),
|
|
)
|
|
cmd.Dir = repositoryPath
|
|
|
|
output, err := cmd.Output()
|
|
if err != nil {
|
|
return "", false, err
|
|
}
|
|
|
|
content, truncated, err := truncateCommandOutput(bytes.NewReader(output), maxBytes)
|
|
if err != nil {
|
|
return "", false, err
|
|
}
|
|
|
|
return content, truncated, nil
|
|
}
|
|
|
|
func GetFileSize(user string, gist string, revision string, filename string) (uint64, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command(
|
|
"git",
|
|
"cat-file",
|
|
"-s",
|
|
revision+":"+convertURLToOctal(filename),
|
|
)
|
|
cmd.Dir = repositoryPath
|
|
|
|
stdout, err := cmd.Output()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return strconv.ParseUint(strings.TrimSuffix(string(stdout), "\n"), 10, 64)
|
|
}
|
|
|
|
func GetLog(user string, gist string, skip int) ([]*Commit, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command(
|
|
"git",
|
|
"--no-pager",
|
|
"log",
|
|
"-n",
|
|
"11",
|
|
"--no-color",
|
|
"-p",
|
|
"--skip",
|
|
strconv.Itoa(skip),
|
|
"--format=format:c %H%na %aN%nm %ae%nt %at",
|
|
"--shortstat",
|
|
"HEAD",
|
|
)
|
|
cmd.Dir = repositoryPath
|
|
stdout, _ := cmd.StdoutPipe()
|
|
err := cmd.Start()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func(cmd *exec.Cmd) {
|
|
waitErr := cmd.Wait()
|
|
if waitErr != nil {
|
|
err = waitErr
|
|
}
|
|
}(cmd)
|
|
|
|
return parseLog(stdout, maxFilesPerDiffCommit, diffSize)
|
|
}
|
|
|
|
func CloneTmp(user string, gist string, gistTmpId string, email string, remove bool) error {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
tmpPath := TmpRepositoriesPath()
|
|
|
|
tmpRepositoryPath := path.Join(tmpPath, gistTmpId)
|
|
|
|
err := os.RemoveAll(tmpRepositoryPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cmd := exec.Command("git", "clone", repositoryPath, gistTmpId)
|
|
cmd.Dir = tmpPath
|
|
if err = cmd.Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
// remove every file (keep the .git directory)
|
|
// useful when user wants to edit multiple files from an existing gist
|
|
if remove {
|
|
if err = removeFilesExceptGit(tmpRepositoryPath); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
cmd = exec.Command("git", "config", "--local", "user.name", user)
|
|
cmd.Dir = tmpRepositoryPath
|
|
if err = cmd.Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
cmd = exec.Command("git", "config", "--local", "user.email", email)
|
|
cmd.Dir = tmpRepositoryPath
|
|
return cmd.Run()
|
|
}
|
|
|
|
func ForkClone(userSrc string, gistSrc string, userDst string, gistDst string) error {
|
|
repositoryPathSrc := RepositoryPath(userSrc, gistSrc)
|
|
repositoryPathDst := RepositoryPath(userDst, gistDst)
|
|
|
|
cmd := exec.Command("git", "clone", "--bare", repositoryPathSrc, repositoryPathDst)
|
|
if err := cmd.Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return CreateDotGitFiles(userDst, gistDst)
|
|
}
|
|
|
|
func SetFileContent(gistTmpId string, filename string, content string) error {
|
|
repositoryPath := TmpRepositoryPath(gistTmpId)
|
|
|
|
return os.WriteFile(filepath.Join(repositoryPath, filename), []byte(content), 0644)
|
|
}
|
|
|
|
func AddAll(gistTmpId string) error {
|
|
tmpPath := TmpRepositoryPath(gistTmpId)
|
|
|
|
// in case of a change where only a file name has its case changed
|
|
cmd := exec.Command("git", "rm", "-r", "--cached", "--ignore-unmatch", ".")
|
|
cmd.Dir = tmpPath
|
|
err := cmd.Run()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cmd = exec.Command("git", "add", "-A")
|
|
cmd.Dir = tmpPath
|
|
|
|
return cmd.Run()
|
|
}
|
|
|
|
func CommitRepository(gistTmpId string, authorName string, authorEmail string) error {
|
|
cmd := exec.Command("git",
|
|
"commit",
|
|
"--allow-empty",
|
|
"-m",
|
|
"Opengist commit",
|
|
"--author",
|
|
fmt.Sprintf("%s <%s>", authorName, authorEmail),
|
|
)
|
|
tmpPath := TmpRepositoryPath(gistTmpId)
|
|
cmd.Dir = tmpPath
|
|
|
|
return cmd.Run()
|
|
}
|
|
|
|
func Push(gistTmpId string) error {
|
|
tmpRepositoryPath := TmpRepositoryPath(gistTmpId)
|
|
cmd := exec.Command(
|
|
"git",
|
|
"push",
|
|
)
|
|
cmd.Dir = tmpRepositoryPath
|
|
|
|
err := cmd.Run()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return os.RemoveAll(tmpRepositoryPath)
|
|
}
|
|
|
|
func DeleteRepository(user string, gist string) error {
|
|
return os.RemoveAll(RepositoryPath(user, gist))
|
|
}
|
|
|
|
func UpdateServerInfo(user string, gist string) error {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command("git", "update-server-info")
|
|
cmd.Dir = repositoryPath
|
|
return cmd.Run()
|
|
}
|
|
|
|
func RPC(user string, gist string, service string) ([]byte, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command("git", service, "--stateless-rpc", "--advertise-refs", ".")
|
|
cmd.Dir = repositoryPath
|
|
stdout, err := cmd.Output()
|
|
return stdout, err
|
|
}
|
|
|
|
func GcRepos() error {
|
|
subdirs, err := os.ReadDir(filepath.Join(config.GetHomeDir(), ReposDirectory))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, subdir := range subdirs {
|
|
if !subdir.IsDir() {
|
|
continue
|
|
}
|
|
|
|
subRoot := filepath.Join(config.GetHomeDir(), ReposDirectory, subdir.Name())
|
|
|
|
gitRepos, err := os.ReadDir(subRoot)
|
|
if err != nil {
|
|
log.Warn().Err(err).Msg("Cannot read directory")
|
|
continue
|
|
}
|
|
|
|
for _, repo := range gitRepos {
|
|
if !repo.IsDir() {
|
|
continue
|
|
}
|
|
|
|
repoPath := filepath.Join(subRoot, repo.Name())
|
|
|
|
log.Info().Msg("Running git gc for repository " + repoPath)
|
|
|
|
cmd := exec.Command("git", "gc")
|
|
cmd.Dir = repoPath
|
|
err = cmd.Run()
|
|
if err != nil {
|
|
log.Warn().Err(err).Msg("Cannot run git gc for repository " + repoPath)
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func HasNoCommits(user string, gist string) (bool, error) {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
cmd := exec.Command("git", "rev-parse", "--all")
|
|
cmd.Dir = repositoryPath
|
|
|
|
var out bytes.Buffer
|
|
cmd.Stdout = &out
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if out.String() == "" {
|
|
return true, nil // No commits exist
|
|
}
|
|
|
|
return false, nil // Commits exist
|
|
}
|
|
|
|
func GetGitVersion() (string, error) {
|
|
cmd := exec.Command("git", "--version")
|
|
stdout, err := cmd.Output()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
versionFields := strings.Fields(string(stdout))
|
|
if len(versionFields) < 3 {
|
|
return string(stdout), nil
|
|
}
|
|
|
|
return versionFields[2], nil
|
|
}
|
|
|
|
func CreateDotGitFiles(user string, gist string) error {
|
|
repositoryPath := RepositoryPath(user, gist)
|
|
|
|
f1, err := os.OpenFile(filepath.Join(repositoryPath, "git-daemon-export-ok"), os.O_RDONLY|os.O_CREATE, 0644)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f1.Close()
|
|
|
|
if os.Getenv("OPENGIST_SKIP_GIT_HOOKS") != "1" {
|
|
for _, hook := range []string{"pre-receive", "post-receive"} {
|
|
if err = createDotGitHookFile(repositoryPath, hook, fmt.Sprintf(hookTemplate, hook)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func createDotGitHookFile(repositoryPath string, hook string, content string) error {
|
|
preReceiveDst, err := os.OpenFile(filepath.Join(repositoryPath, "hooks", hook), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0744)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err = preReceiveDst.WriteString(content); err != nil {
|
|
return err
|
|
}
|
|
defer preReceiveDst.Close()
|
|
|
|
return nil
|
|
}
|
|
|
|
func removeFilesExceptGit(dir string) error {
|
|
return filepath.WalkDir(dir, func(path string, d os.DirEntry, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if d.IsDir() && filepath.Base(path) == ".git" {
|
|
return filepath.SkipDir
|
|
}
|
|
if !d.IsDir() {
|
|
return os.Remove(path)
|
|
}
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func convertOctalToUTF8(name string) string {
|
|
name = strings.Trim(name, `"`)
|
|
utf8Name, err := strconv.Unquote(name)
|
|
if err != nil {
|
|
utf8Name, err = strconv.Unquote(`"` + name + `"`)
|
|
if err != nil {
|
|
return name
|
|
}
|
|
}
|
|
return utf8Name
|
|
}
|
|
|
|
func convertUTF8ToOctal(name string) string {
|
|
if strings.Contains(name, "\\") {
|
|
return name
|
|
}
|
|
|
|
needsQuoting := false
|
|
for _, r := range name {
|
|
if r > 127 {
|
|
needsQuoting = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !needsQuoting {
|
|
return name
|
|
}
|
|
|
|
quoted := fmt.Sprintf("%q", name)
|
|
return strings.Trim(quoted, `"`)
|
|
}
|
|
|
|
func convertURLToOctal(name string) string {
|
|
decoded, err := url.QueryUnescape(name)
|
|
if err != nil {
|
|
return name
|
|
}
|
|
|
|
return convertUTF8ToOctal(decoded)
|
|
}
|
|
|
|
const hookTemplate = `#!/bin/sh
|
|
"$OG_OPENGIST_HOME_INTERNAL/symlinks/opengist" --config=$OG_OPENGIST_HOME_INTERNAL/symlinks/config.yml hook %s
|
|
`
|