|
|
@ -23,6 +23,7 @@ import (
|
|
|
|
"path"
|
|
|
|
"path"
|
|
|
|
"strconv"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"sync"
|
|
|
|
"syscall"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
|
@ -63,19 +64,22 @@ func (u *User) IsTrusted() bool {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
const (
|
|
|
|
pullsPrefix = "https://api.github.com/repos/matrix-org/matrix-doc/pulls"
|
|
|
|
pullsPrefix = "https://api.github.com/repos/matrix-org/matrix-doc/pulls"
|
|
|
|
matrixDocCloneURL = "https://github.com/matrix-org/matrix-doc.git"
|
|
|
|
matrixDocCloneURL = "https://github.com/matrix-org/matrix-doc.git"
|
|
|
|
|
|
|
|
permissionsOwnerFull = 0700
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
func gitClone(url string, shared bool) (string, error) {
|
|
|
|
func gitClone(url string, shared bool) (string, error) {
|
|
|
|
directory := path.Join("/tmp/matrix-doc", strconv.FormatInt(rand.Int63(), 10))
|
|
|
|
directory := path.Join("/tmp/matrix-doc", strconv.FormatInt(rand.Int63(), 10))
|
|
|
|
cmd := exec.Command("git", "clone", url, directory)
|
|
|
|
if err := os.MkdirAll(directory, permissionsOwnerFull); err != nil {
|
|
|
|
|
|
|
|
return "", fmt.Errorf("error making directory %s: %v", directory, err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
args := []string{"clone", url, directory}
|
|
|
|
if shared {
|
|
|
|
if shared {
|
|
|
|
cmd.Args = append(cmd.Args, "--shared")
|
|
|
|
args = append(args, "--shared")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := runGitCommand(directory, args); err != nil {
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
return "", err
|
|
|
|
return "", fmt.Errorf("error cloning repo: %v", err)
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return directory, nil
|
|
|
|
return directory, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -84,15 +88,13 @@ func gitCheckout(path, sha string) error {
|
|
|
|
return runGitCommand(path, []string{"checkout", sha})
|
|
|
|
return runGitCommand(path, []string{"checkout", sha})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func gitFetch(path string) error {
|
|
|
|
|
|
|
|
return runGitCommand(path, []string{"fetch"})
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func runGitCommand(path string, args []string) error {
|
|
|
|
func runGitCommand(path string, args []string) error {
|
|
|
|
cmd := exec.Command("git", args...)
|
|
|
|
cmd := exec.Command("git", args...)
|
|
|
|
cmd.Dir = path
|
|
|
|
cmd.Dir = path
|
|
|
|
|
|
|
|
var b bytes.Buffer
|
|
|
|
|
|
|
|
cmd.Stderr = &b
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
return fmt.Errorf("error running %q: %v", strings.Join(cmd.Args, " "), err)
|
|
|
|
return fmt.Errorf("error running %q: %v (stderr: %s)", strings.Join(cmd.Args, " "), err, b.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -136,17 +138,35 @@ func writeError(w http.ResponseWriter, code int, err error) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
type server struct {
|
|
|
|
type server struct {
|
|
|
|
|
|
|
|
mu sync.Mutex // Must be locked around any git command on matrixDocCloneURL
|
|
|
|
matrixDocCloneURL string
|
|
|
|
matrixDocCloneURL string
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func (s *server) updateBase() error {
|
|
|
|
|
|
|
|
s.mu.Lock()
|
|
|
|
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
|
|
|
return runGitCommand(s.matrixDocCloneURL, []string{"fetch"})
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// canCheckout returns whether a given sha can currently be checked out from s.matrixDocCloneURL.
|
|
|
|
|
|
|
|
func (s *server) canCheckout(sha string) bool {
|
|
|
|
|
|
|
|
s.mu.Lock()
|
|
|
|
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
|
|
|
return runGitCommand(s.matrixDocCloneURL, []string{"cat-file", "-e", sha + "^{commit}"}) == nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// generateAt generates spec from repo at sha.
|
|
|
|
// generateAt generates spec from repo at sha.
|
|
|
|
// Returns the path where the generation was done.
|
|
|
|
// Returns the path where the generation was done.
|
|
|
|
func (s *server) generateAt(sha string) (dst string, err error) {
|
|
|
|
func (s *server) generateAt(sha string) (dst string, err error) {
|
|
|
|
err = gitFetch(s.matrixDocCloneURL)
|
|
|
|
if !s.canCheckout(sha) {
|
|
|
|
if err != nil {
|
|
|
|
err = s.updateBase()
|
|
|
|
return
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s.mu.Lock()
|
|
|
|
dst, err = gitClone(s.matrixDocCloneURL, true)
|
|
|
|
dst, err = gitClone(s.matrixDocCloneURL, true)
|
|
|
|
|
|
|
|
s.mu.Unlock()
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -164,7 +184,10 @@ func (s *server) getSHAOf(ref string) (string, error) {
|
|
|
|
cmd.Dir = path.Join(s.matrixDocCloneURL)
|
|
|
|
cmd.Dir = path.Join(s.matrixDocCloneURL)
|
|
|
|
var b bytes.Buffer
|
|
|
|
var b bytes.Buffer
|
|
|
|
cmd.Stdout = &b
|
|
|
|
cmd.Stdout = &b
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
s.mu.Lock()
|
|
|
|
|
|
|
|
err := cmd.Run()
|
|
|
|
|
|
|
|
s.mu.Unlock()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("error generating spec: %v\nOutput from gendoc:\n%v", err, b.String())
|
|
|
|
return "", fmt.Errorf("error generating spec: %v\nOutput from gendoc:\n%v", err, b.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return strings.TrimSpace(b.String()), nil
|
|
|
|
return strings.TrimSpace(b.String()), nil
|
|
|
@ -174,16 +197,14 @@ func (s *server) serveSpec(w http.ResponseWriter, req *http.Request) {
|
|
|
|
var sha string
|
|
|
|
var sha string
|
|
|
|
|
|
|
|
|
|
|
|
if strings.ToLower(req.URL.Path) == "/spec/head" {
|
|
|
|
if strings.ToLower(req.URL.Path) == "/spec/head" {
|
|
|
|
if err := gitFetch(s.matrixDocCloneURL); err != nil {
|
|
|
|
// err may be non-nil here but if headSha is non-empty we will serve a possibly-stale result in favour of erroring.
|
|
|
|
writeError(w, 500, err)
|
|
|
|
// This is to deal with cases like where github is down but we still want to serve the spec.
|
|
|
|
return
|
|
|
|
if headSha, err := s.lookupHeadSHA(); headSha == "" {
|
|
|
|
}
|
|
|
|
|
|
|
|
originHead, err := s.getSHAOf("origin/master")
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
writeError(w, 500, err)
|
|
|
|
writeError(w, 500, err)
|
|
|
|
return
|
|
|
|
return
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
sha = headSha
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sha = originHead
|
|
|
|
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
pr, err := lookupPullRequest(*req.URL, "/spec")
|
|
|
|
pr, err := lookupPullRequest(*req.URL, "/spec")
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
@ -220,6 +241,25 @@ func (s *server) serveSpec(w http.ResponseWriter, req *http.Request) {
|
|
|
|
specCache.Add(sha, b)
|
|
|
|
specCache.Add(sha, b)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// lookupHeadSHA looks up what origin/master's HEAD SHA is.
|
|
|
|
|
|
|
|
// It attempts to `git fetch` before doing so.
|
|
|
|
|
|
|
|
// If this fails, it may still return a stale sha, but will also return an error.
|
|
|
|
|
|
|
|
func (s *server) lookupHeadSHA() (sha string, retErr error) {
|
|
|
|
|
|
|
|
retErr = s.updateBase()
|
|
|
|
|
|
|
|
if retErr != nil {
|
|
|
|
|
|
|
|
log.Printf("Error fetching: %v, attempting to fall back to current known value", retErr)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
originHead, err := s.getSHAOf("origin/master")
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
retErr = err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
sha = originHead
|
|
|
|
|
|
|
|
if retErr != nil && originHead != "" {
|
|
|
|
|
|
|
|
log.Printf("Successfully fell back to possibly stale sha: %s", sha)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func checkAuth(pr *PullRequest) error {
|
|
|
|
func checkAuth(pr *PullRequest) error {
|
|
|
|
if !pr.User.IsTrusted() {
|
|
|
|
if !pr.User.IsTrusted() {
|
|
|
|
return fmt.Errorf("%q is not a trusted pull requester", pr.User.Login)
|
|
|
|
return fmt.Errorf("%q is not a trusted pull requester", pr.User.Login)
|
|
|
@ -383,7 +423,7 @@ func main() {
|
|
|
|
if err != nil {
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s := server{masterCloneDir}
|
|
|
|
s := server{matrixDocCloneURL: masterCloneDir}
|
|
|
|
http.HandleFunc("/spec/", forceHTML(s.serveSpec))
|
|
|
|
http.HandleFunc("/spec/", forceHTML(s.serveSpec))
|
|
|
|
http.HandleFunc("/diff/rst/", s.serveRSTDiff)
|
|
|
|
http.HandleFunc("/diff/rst/", s.serveRSTDiff)
|
|
|
|
http.HandleFunc("/diff/html/", forceHTML(s.serveHTMLDiff))
|
|
|
|
http.HandleFunc("/diff/html/", forceHTML(s.serveHTMLDiff))
|
|
|
|