internal/gitaly/service/smarthttp/receive_pack.go (113 lines of code) (raw):
package smarthttp
import (
"context"
"errors"
"fmt"
"gitlab.com/gitlab-org/gitaly/v16/internal/git/gitcmd"
"gitlab.com/gitlab-org/gitaly/v16/internal/gitaly/hook"
"gitlab.com/gitlab-org/gitaly/v16/internal/gitaly/hook/receivepack"
"gitlab.com/gitlab-org/gitaly/v16/internal/gitaly/storage"
"gitlab.com/gitlab-org/gitaly/v16/internal/gitaly/transaction"
"gitlab.com/gitlab-org/gitaly/v16/internal/log"
"gitlab.com/gitlab-org/gitaly/v16/internal/structerr"
"gitlab.com/gitlab-org/gitaly/v16/internal/transaction/voting"
"gitlab.com/gitlab-org/gitaly/v16/proto/go/gitalypb"
"gitlab.com/gitlab-org/gitaly/v16/streamio"
)
func (s *server) PostReceivePack(stream gitalypb.SmartHTTPService_PostReceivePackServer) error {
ctx := stream.Context()
req, err := stream.Recv() // First request contains only Repository and GlId
if err != nil {
return err
}
s.logger.WithFields(log.Fields{
"GlID": req.GetGlId(),
"GlRepository": req.GetGlRepository(),
"GlUsername": req.GetGlUsername(),
"GitConfigOptions": req.GetGitConfigOptions(),
}).DebugContext(ctx, "PostReceivePack")
if err := validateReceivePackRequest(ctx, s.locator, req); err != nil {
return err
}
if err := s.postReceivePack(stream, req); err != nil {
return structerr.NewInternal("%w", err)
}
// In cases where all reference updates are rejected by git-receive-pack(1), we would end up
// with no transactional votes at all. This would lead to scheduling
// replication jobs, which wouldn't accomplish anything since no refs
// were updated.
// To prevent replication jobs from being unnecessarily created, do a
// final vote which concludes this RPC to ensure there's always at least
// one vote. In case there was diverging behaviour in git-receive-pack(1)
// which led to a different outcome across voters, then this final vote
// would fail because the sequence of votes would be different.
if err := transaction.VoteOnContext(ctx, s.txManager, voting.Vote{}, voting.Committed); err != nil {
// When the pre-receive hook failed, git-receive-pack(1) exits with code 0.
// It's arguable whether this is the expected behavior, but anyhow it means
// cmd.Wait() did not error out. On the other hand, the gitaly-hooks command did
// stop the transaction upon failure. So this final vote fails.
// To avoid this error being presented to the end user, ignore it when the
// transaction was stopped.
if !errors.Is(err, transaction.ErrTransactionStopped) {
return structerr.NewAborted("final transactional vote: %w", err)
}
}
return nil
}
func (s *server) postReceivePack(
stream gitalypb.SmartHTTPService_PostReceivePackServer,
req *gitalypb.PostReceivePackRequest,
) (returnedErr error) {
ctx := stream.Context()
stdin := streamio.NewReader(func() ([]byte, error) {
resp, err := stream.Recv()
return resp.GetData(), err
})
stdout := streamio.NewWriter(func(p []byte) error {
return stream.Send(&gitalypb.PostReceivePackResponse{Data: p})
})
repo := s.localRepoFactory.Build(req.GetRepository())
repoPath, err := repo.Path(ctx)
if err != nil {
return err
}
config, err := gitcmd.ConvertConfigOptions(req.GetGitConfigOptions())
if err != nil {
return err
}
transactionID := storage.ExtractTransactionID(ctx)
transactionsEnabled := transactionID > 0
if transactionsEnabled {
procReceiveCleanup, err := receivepack.RegisterProcReceiveHook(
ctx, s.logger, s.cfg, req, repo, s.hookManager, hook.NewTransactionRegistry(s.txRegistry), transactionID,
)
if err != nil {
return err
}
defer func() {
if err := procReceiveCleanup(); err != nil && returnedErr == nil {
returnedErr = err
}
}()
}
objectHash, err := repo.ObjectHash(ctx)
if err != nil {
return fmt.Errorf("detecting object hash: %w", err)
}
cmd, err := repo.Exec(ctx,
gitcmd.Command{
Name: "receive-pack",
Flags: []gitcmd.Option{gitcmd.Flag{Name: "--stateless-rpc"}},
Args: []string{repoPath},
},
gitcmd.WithStdin(stdin),
gitcmd.WithStdout(stdout),
gitcmd.WithReceivePackHooks(objectHash, req, "http", transactionsEnabled),
gitcmd.WithGitProtocol(s.logger, req),
gitcmd.WithConfig(config...),
)
if err != nil {
return structerr.NewFailedPrecondition("spawning receive-pack: %w", err)
}
if err := cmd.Wait(); err != nil {
return structerr.NewFailedPrecondition("waiting for receive-pack: %w", err)
}
return nil
}
func validateReceivePackRequest(ctx context.Context, locator storage.Locator, req *gitalypb.PostReceivePackRequest) error {
if req.GetGlId() == "" {
return structerr.NewInvalidArgument("empty GlId")
}
if req.Data != nil {
return structerr.NewInvalidArgument("non-empty Data")
}
if err := locator.ValidateRepository(ctx, req.GetRepository()); err != nil {
return structerr.NewInvalidArgument("%w", err)
}
return nil
}