summaryrefslogtreecommitdiff
path: root/go/internal/handler/exec.go
diff options
context:
space:
mode:
authorIgor Drozdov <idrozdov@gitlab.com>2019-05-22 15:50:40 +0300
committerIgor Drozdov <idrozdov@gitlab.com>2019-05-31 11:32:27 +0300
commita701af0378cc9a53170f60b95df8cfdcbca04ea7 (patch)
tree9c57bf733e572314e3fbba29b6443805bb9afbca /go/internal/handler/exec.go
parentc86081506c706c6230c247a2391f35d35130ca94 (diff)
downloadgitlab-shell-id-git-receive-pack.tar.gz
Go implementation for git-receive-packid-git-receive-pack
Diffstat (limited to 'go/internal/handler/exec.go')
-rw-r--r--go/internal/handler/exec.go109
1 files changed, 80 insertions, 29 deletions
diff --git a/go/internal/handler/exec.go b/go/internal/handler/exec.go
index ee7b4a8..671263c 100644
--- a/go/internal/handler/exec.go
+++ b/go/internal/handler/exec.go
@@ -14,11 +14,29 @@ import (
"google.golang.org/grpc"
)
-// GitalyHandlerFunc implementations are responsible for deserializing
+// GitalyHandlerFuncWithJSON implementations are responsible for deserializing
// the request JSON into a GRPC request message, making an appropriate Gitaly
// call with the request, using the provided client, and returning the exit code
// or error from the Gitaly call.
-type GitalyHandlerFunc func(ctx context.Context, client *grpc.ClientConn, requestJSON string) (int32, error)
+type GitalyHandlerFuncWithJSON func(ctx context.Context, client *grpc.ClientConn, requestJSON string) (int32, error)
+
+// GitalyHandlerFunc implementations are responsible for making
+// an appropriate Gitaly call using the provided client and context
+// and returning an error from the Gitaly call.
+type GitalyHandlerFunc func(ctx context.Context, client *grpc.ClientConn) (int32, error)
+
+type GitalyConn struct {
+ ctx context.Context
+ conn *grpc.ClientConn
+ close func()
+}
+
+type GitalyCommand struct {
+ Config *config.Config
+ ServiceName string
+ Address string
+ Token string
+}
// RunGitalyCommand provides a bootstrap for Gitaly commands executed
// through GitLab-Shell. It ensures that logging, tracing and other
@@ -26,7 +44,7 @@ type GitalyHandlerFunc func(ctx context.Context, client *grpc.ClientConn, reques
// RunGitalyCommand will handle errors internally and call
// `os.Exit()` on completion. This method will never return to
// the caller.
-func RunGitalyCommand(handler GitalyHandlerFunc) {
+func RunGitalyCommand(handler GitalyHandlerFuncWithJSON) {
exitCode, err := internalRunGitalyCommand(os.Args, handler)
if err != nil {
@@ -36,10 +54,25 @@ func RunGitalyCommand(handler GitalyHandlerFunc) {
os.Exit(exitCode)
}
-// internalRunGitalyCommand is like RunGitalyCommand, except that since it doesn't
-// call os.Exit, we can rely on its deferred handlers executing correctly
-func internalRunGitalyCommand(args []string, handler GitalyHandlerFunc) (int, error) {
+// RunGitalyCommand provides a bootstrap for Gitaly commands executed
+// through GitLab-Shell. It ensures that logging, tracing and other
+// common concerns are configured before executing the `handler`.
+func (gc *GitalyCommand) RunGitalyCommand(handler GitalyHandlerFunc) error {
+ gitalyConn, err := getConn(gc)
+
+ if err != nil {
+ return err
+ }
+ _, err = handler(gitalyConn.ctx, gitalyConn.conn)
+
+ gitalyConn.close()
+
+ return err
+}
+
+// internalRunGitalyCommand runs Gitaly's command by particular Gitaly address and token
+func internalRunGitalyCommand(args []string, handler GitalyHandlerFuncWithJSON) (int, error) {
if len(args) != 3 {
return 1, fmt.Errorf("expected 2 arguments, got %v", args)
}
@@ -53,13 +86,44 @@ func internalRunGitalyCommand(args []string, handler GitalyHandlerFunc) (int, er
return 1, err
}
+ gc := &GitalyCommand{
+ Config: cfg,
+ ServiceName: args[0],
+ Address: args[1],
+ Token: os.Getenv("GITALY_TOKEN"),
+ }
+ requestJSON := string(args[2])
+
+ gitalyConn, err := getConn(gc)
+
+ if err != nil {
+ return 1, err
+ }
+
+ exitCode, err := handler(gitalyConn.ctx, gitalyConn.conn, requestJSON)
+
+ gitalyConn.close()
+
+ return int(exitCode), err
+}
+
+func getConn(gc *GitalyCommand) (*GitalyConn, error) {
+ if gc.Address == "" {
+ return nil, fmt.Errorf("no gitaly_address given")
+ }
+
+ connOpts := client.DefaultDialOpts
+ if gc.Token != "" {
+ connOpts = append(client.DefaultDialOpts, grpc.WithPerRPCCredentials(gitalyauth.RPCCredentialsV2(gc.Token)))
+ }
+
// Use a working directory that won't get removed or unmounted.
if err := os.Chdir("/"); err != nil {
- return 1, err
+ return nil, err
}
// Configure distributed tracing
- serviceName := fmt.Sprintf("gitlab-shell-%v", args[0])
+ serviceName := fmt.Sprintf("gitlab-shell-%v", gc.ServiceName)
closer := tracing.Initialize(
tracing.WithServiceName(serviceName),
@@ -71,34 +135,21 @@ func internalRunGitalyCommand(args []string, handler GitalyHandlerFunc) (int, er
// Processes are spawned as children of the SSH daemon, which tightly
// controls environment variables; doing this means we don't have to
// enable PermitUserEnvironment
- tracing.WithConnectionString(cfg.GitlabTracing),
+ tracing.WithConnectionString(gc.Config.GitlabTracing),
)
- defer closer.Close()
ctx, finished := tracing.ExtractFromEnv(context.Background())
- defer finished()
- gitalyAddress := args[1]
- if gitalyAddress == "" {
- return 1, fmt.Errorf("no gitaly_address given")
- }
-
- conn, err := client.Dial(gitalyAddress, dialOpts())
+ conn, err := client.Dial(gc.Address, connOpts)
if err != nil {
- return 1, err
+ return nil, err
}
- defer conn.Close()
- requestJSON := string(args[2])
- exitCode, err := handler(ctx, conn, requestJSON)
- return int(exitCode), err
-}
-
-func dialOpts() []grpc.DialOption {
- connOpts := client.DefaultDialOpts
- if token := os.Getenv("GITALY_TOKEN"); token != "" {
- connOpts = append(client.DefaultDialOpts, grpc.WithPerRPCCredentials(gitalyauth.RPCCredentialsV2(token)))
+ finish := func() {
+ finished()
+ closer.Close()
+ conn.Close()
}
- return connOpts
+ return &GitalyConn{ctx: ctx, conn: conn, close: finish}, nil
}