internal/gitaly/service/operations/rebase_to_ref.go (85 lines of code) (raw):
package operations
import (
"context"
"errors"
"gitlab.com/gitlab-org/gitaly/v16/internal/git"
"gitlab.com/gitlab-org/gitaly/v16/internal/git/localrepo"
"gitlab.com/gitlab-org/gitaly/v16/internal/structerr"
"gitlab.com/gitlab-org/gitaly/v16/proto/go/gitalypb"
)
// UserRebaseToRef overwrites the given TargetRef with the result of rebasing
// SourceSHA on top of FirstParentRef, and returns the SHA of the HEAD of
// TargetRef.
func (s *Server) UserRebaseToRef(ctx context.Context, request *gitalypb.UserRebaseToRefRequest) (*gitalypb.UserRebaseToRefResponse, error) {
if err := validateUserRebaseToRefRequest(ctx, s.locator, request); err != nil {
return nil, structerr.NewInvalidArgument("%w", err)
}
quarantineDir, quarantineRepo, err := s.quarantinedRepo(ctx, request.GetRepository())
if err != nil {
return nil, structerr.NewInternal("creating repo quarantine: %w", err)
}
oid, err := quarantineRepo.ResolveRevision(ctx, git.Revision(request.GetFirstParentRef()))
if err != nil {
return nil, structerr.NewInvalidArgument("invalid FirstParentRef")
}
sourceOID, err := quarantineRepo.ResolveRevision(ctx, git.Revision(request.GetSourceSha()))
if err != nil {
return nil, structerr.NewInvalidArgument("invalid SourceSha")
}
// Resolve the current state of the target reference. We do not care whether it
// exists or not, but what we do want to assert is that the target reference doesn't
// change while we compute the merge commit as a small protection against races.
objectHash, err := quarantineRepo.ObjectHash(ctx)
if err != nil {
return nil, structerr.NewInternal("detecting object hash: %w", err)
}
var oldTargetOID git.ObjectID
if expectedOldOID := request.GetExpectedOldOid(); expectedOldOID != "" {
oldTargetOID, err = objectHash.FromHex(expectedOldOID)
if err != nil {
return nil, structerr.NewInvalidArgument("invalid expected old object ID: %w", err).WithMetadata("old_object_id", expectedOldOID)
}
oldTargetOID, err = resolveRevision(ctx, quarantineRepo, oldTargetOID)
if err != nil {
return nil, structerr.NewInvalidArgument("cannot resolve expected old object ID: %w", err).
WithMetadata("old_object_id", expectedOldOID)
}
} else if targetRef, err := quarantineRepo.GetReference(ctx, git.ReferenceName(request.GetTargetRef())); err == nil {
if targetRef.IsSymbolic {
return nil, structerr.NewFailedPrecondition("target reference is symbolic: %q", request.GetTargetRef())
}
oid, err := objectHash.FromHex(targetRef.Target)
if err != nil {
return nil, structerr.NewInternal("invalid target revision: %w", err)
}
oldTargetOID = oid
} else if errors.Is(err, git.ErrReferenceAmbiguous) {
return nil, structerr.NewInvalidArgument("target reference is ambiguous: %w", err)
} else if errors.Is(err, git.ErrReferenceNotFound) {
oldTargetOID = objectHash.ZeroOID
} else {
return nil, structerr.NewInternal("could not read target reference: %w", err)
}
committerSignature, err := git.SignatureFromRequest(request)
if err != nil {
return nil, structerr.NewInvalidArgument("%w", err)
}
rebasedOID, err := quarantineRepo.Rebase(
ctx,
oid.String(),
sourceOID.String(),
localrepo.RebaseWithCommitter(committerSignature),
)
if err != nil {
var conflictErr *localrepo.RebaseConflictError
if errors.As(err, &conflictErr) {
return nil, structerr.NewFailedPrecondition("failed to rebase %s on %s while preparing %s due to conflict",
sourceOID, oid, string(request.GetTargetRef()))
}
return nil, structerr.NewInternal("rebasing commits: %w", err)
}
if err := quarantineDir.Migrate(ctx); err != nil {
return nil, structerr.NewInternal("migrating quarantined objects: %w", err)
}
repo := s.localRepoFactory.Build(request.GetRepository())
if err := repo.UpdateRef(ctx, git.ReferenceName(request.GetTargetRef()), rebasedOID, oldTargetOID); err != nil {
return nil, structerr.NewFailedPrecondition("could not update %s. Please refresh and try again", string(request.GetTargetRef()))
}
return &gitalypb.UserRebaseToRefResponse{
CommitId: rebasedOID.String(),
}, nil
}