Commit 37389823 authored by Stan Hu's avatar Stan Hu

Refactor handlePostRPC to be more clear

parent 232cd84c
...@@ -127,9 +127,6 @@ func handleGetInfoRefs(rw http.ResponseWriter, r *http.Request, a *api.Response) ...@@ -127,9 +127,6 @@ func handleGetInfoRefs(rw http.ResponseWriter, r *http.Request, a *api.Response)
} }
func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) { func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) {
var err error
var body io.Reader
var isShallowClone bool
var writtenIn int64 var writtenIn int64
w := NewGitHttpResponseWriter(rw) w := NewGitHttpResponseWriter(rw)
...@@ -144,14 +141,73 @@ func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) { ...@@ -144,14 +141,73 @@ func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) {
return return
} }
if action == "git-upload-pack" { if action == "git-receive-pack" {
writtenIn, _ = handleReceivePack(action, w, r, a)
} else {
writtenIn, _ = handleUploadPack(action, w, r, a)
}
}
func handleReceivePack(action string, w *GitHttpResponseWriter, r *http.Request, a *api.Response) (writtenIn int64, err error) {
body := r.Body
cmd, stdin, stdout, err := setupGitCommand(action, a, w, r)
if err != nil {
return
}
defer stdout.Close()
defer stdin.Close()
defer helper.CleanUpProcessGroup(cmd) // Ensure brute force subprocess clean-up
// Write the client request body to Git's standard input
writtenIn, err = io.Copy(stdin, body)
if err != nil {
helper.Fail500(w, r, fmt.Errorf("handleReceivePack: write to %v: %v", cmd.Args, err))
return
}
// Signal to the Git subprocess that no more data is coming
stdin.Close()
// It may take a while before we return and the deferred closes happen
// so let's free up some resources already.
r.Body.Close()
writePostRPCHeader(w, action)
// This io.Copy may take a long time, both for Git push and pull.
_, err = io.Copy(w, stdout)
if err != nil {
helper.LogError(
r,
&copyError{fmt.Errorf("handleReceivePack: copy output of %v: %v", cmd.Args, err)},
)
return
}
err = cmd.Wait()
if err != nil {
helper.LogError(r, fmt.Errorf("handleReceivePack: wait for %v: %v", cmd.Args, err))
return
}
return writtenIn, nil
}
func handleUploadPack(action string, w *GitHttpResponseWriter, r *http.Request, a *api.Response) (writtenIn int64, err error) {
var isShallowClone bool
var body io.Reader
buffer := &bytes.Buffer{} buffer := &bytes.Buffer{}
// Only sniff on the first 4096 bytes: we assume that if we find no // Only sniff on the first 4096 bytes: we assume that if we find no
// 'deepen' message in the first 4096 bytes there won't be one later // 'deepen' message in the first 4096 bytes there won't be one later
// either. // either.
_, err = io.Copy(buffer, io.LimitReader(r.Body, 4096)) _, err = io.Copy(buffer, io.LimitReader(r.Body, 4096))
if err != nil { if err != nil {
helper.Fail500(w, r, &copyError{fmt.Errorf("handlePostRPC: buffer git-upload-pack body: %v", err)}) helper.Fail500(w, r, &copyError{fmt.Errorf("handleUploadPack: buffer git-upload-pack body: %v", err)})
return return
} }
...@@ -162,41 +218,27 @@ func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) { ...@@ -162,41 +218,27 @@ func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) {
buf, err := ioutil.ReadAll(body) buf, err := ioutil.ReadAll(body)
if err != nil { if err != nil {
helper.Fail500(w, r, &copyError{fmt.Errorf("handlePostRPC: full buffer git-upload-pack body: %v", err)}) helper.Fail500(w, r, &copyError{fmt.Errorf("handleUploadPack: full buffer git-upload-pack body: %v", err)})
return return
} }
body = ioutil.NopCloser(bytes.NewBuffer(buf)) body = ioutil.NopCloser(bytes.NewBuffer(buf))
r.Body.Close() r.Body.Close()
} else {
body = r.Body
}
// Prepare our Git subprocess cmd, stdin, stdout, err := setupGitCommand(action, a, w, r)
cmd := gitCommand(a.GL_ID, "git", subCommand(action), "--stateless-rpc", a.RepoPath)
stdout, err := cmd.StdoutPipe()
if err != nil { if err != nil {
helper.Fail500(w, r, fmt.Errorf("handlePostRPC: stdout: %v", err))
return return
} }
defer stdout.Close() defer stdout.Close()
stdin, err := cmd.StdinPipe()
if err != nil {
helper.Fail500(w, r, fmt.Errorf("handlePostRPC: stdin: %v", err))
return
}
defer stdin.Close() defer stdin.Close()
if err := cmd.Start(); err != nil {
helper.Fail500(w, r, fmt.Errorf("handlePostRPC: start %v: %v", cmd.Args, err))
return
}
defer helper.CleanUpProcessGroup(cmd) // Ensure brute force subprocess clean-up defer helper.CleanUpProcessGroup(cmd) // Ensure brute force subprocess clean-up
stdoutError := make(chan error, 1) stdoutError := make(chan error, 1)
var wg sync.WaitGroup var wg sync.WaitGroup
wg.Add(1) wg.Add(1)
if action == "git-upload-pack" {
// Start writing the response // Start writing the response
writePostRPCHeader(w, action) writePostRPCHeader(w, action)
...@@ -206,48 +248,58 @@ func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) { ...@@ -206,48 +248,58 @@ func handlePostRPC(rw http.ResponseWriter, r *http.Request, a *api.Response) {
if _, err := io.Copy(w, stdout); err != nil { if _, err := io.Copy(w, stdout); err != nil {
helper.LogError( helper.LogError(
r, r,
&copyError{fmt.Errorf("handlePostRPC: copy output of %v: %v", cmd.Args, err)}, &copyError{fmt.Errorf("handleUploadPack: copy output of %v: %v", cmd.Args, err)},
) )
stdoutError <- err stdoutError <- err
return return
} }
}() }()
}
// Write the client request body to Git's standard input // Write the client request body to Git's standard input
if writtenIn, err = io.Copy(stdin, body); err != nil { if writtenIn, err = io.Copy(stdin, body); err != nil {
helper.Fail500(w, r, fmt.Errorf("handlePostRPC: write to %v: %v", cmd.Args, err)) helper.Fail500(w, r, fmt.Errorf("handleUploadPack: write to %v: %v", cmd.Args, err))
return return
} }
// Signal to the Git subprocess that no more data is coming // Signal to the Git subprocess that no more data is coming
stdin.Close() stdin.Close()
if action == "git-upload-pack" {
wg.Wait() wg.Wait()
if len(stdoutError) > 0 { if len(stdoutError) > 0 {
return return
} }
} else {
// It may take a while before we return and the deferred closes happen
// so let's free up some resources already.
r.Body.Close()
writePostRPCHeader(w, action) err = cmd.Wait()
// This io.Copy may take a long time, both for Git push and pull. if err != nil && !(isExitError(err) && isShallowClone) {
if _, err := io.Copy(w, stdout); err != nil { helper.LogError(r, fmt.Errorf("handleUploadPack: wait for %v: %v", cmd.Args, err))
helper.LogError(
r,
&copyError{fmt.Errorf("handlePostRPC: copy output of %v: %v", cmd.Args, err)},
)
return return
} }
return writtenIn, nil
}
func setupGitCommand(action string, a *api.Response, w *GitHttpResponseWriter, r *http.Request) (cmd *exec.Cmd, stdin io.WriteCloser, stdout io.ReadCloser, err error) {
// Prepare our Git subprocess
cmd = gitCommand(a.GL_ID, "git", subCommand(action), "--stateless-rpc", a.RepoPath)
stdout, err = cmd.StdoutPipe()
if err != nil {
helper.Fail500(w, r, fmt.Errorf("setupGitCommand: stdout: %v", err))
return nil, nil, nil, err
} }
if err := cmd.Wait(); err != nil && !(isExitError(err) && isShallowClone) {
helper.LogError(r, fmt.Errorf("handlePostRPC: wait for %v: %v", cmd.Args, err)) stdin, err = cmd.StdinPipe()
return if err != nil {
helper.Fail500(w, r, fmt.Errorf("setupGitCommand: stdin: %v", err))
return nil, nil, nil, err
} }
if err = cmd.Start(); err != nil {
helper.Fail500(w, r, fmt.Errorf("setupGitCommand: start %v: %v", cmd.Args, err))
return nil, nil, nil, err
}
return cmd, stdin, stdout, nil
} }
func writePostRPCHeader(w http.ResponseWriter, action string) { func writePostRPCHeader(w http.ResponseWriter, action string) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment