Monorepo for Tangled
at master 112 lines 2.9 kB view raw
1package xrpc 2 3import ( 4 "encoding/json" 5 "errors" 6 "fmt" 7 "net/http" 8 9 "github.com/bluesky-social/indigo/atproto/syntax" 10 "tangled.org/core/api/tangled" 11 "tangled.org/core/knotserver/git" 12 "tangled.org/core/patchutil" 13 "tangled.org/core/rbac" 14 "tangled.org/core/types" 15 xrpcerr "tangled.org/core/xrpc/errors" 16) 17 18func (x *Xrpc) Merge(w http.ResponseWriter, r *http.Request) { 19 l := x.Logger.With("handler", "Merge") 20 fail := func(e xrpcerr.XrpcError) { 21 l.Error("failed", "kind", e.Tag, "error", e.Message) 22 writeError(w, e, http.StatusBadRequest) 23 } 24 25 actorDid, ok := r.Context().Value(ActorDid).(syntax.DID) 26 if !ok { 27 fail(xrpcerr.MissingActorDidError) 28 return 29 } 30 31 var data tangled.RepoMerge_Input 32 if err := json.NewDecoder(r.Body).Decode(&data); err != nil { 33 fail(xrpcerr.GenericError(err)) 34 return 35 } 36 37 did := data.Did 38 name := data.Name 39 40 if did == "" || name == "" { 41 fail(xrpcerr.GenericError(fmt.Errorf("did and name are required"))) 42 return 43 } 44 45 repoDid, err := x.Db.GetRepoDid(did, name) 46 if err != nil { 47 fail(xrpcerr.RepoNotFoundError) 48 return 49 } 50 repoPath, _, _, err := x.Db.ResolveRepoDIDOnDisk(x.Config.Repo.ScanPath, repoDid) 51 if err != nil { 52 fail(xrpcerr.RepoNotFoundError) 53 return 54 } 55 56 if ok, err := x.Enforcer.IsPushAllowed(actorDid.String(), rbac.ThisServer, repoDid); !ok || err != nil { 57 l.Error("insufficient permissions", "did", actorDid.String(), "repo", repoDid) 58 writeError(w, xrpcerr.AccessControlError(actorDid.String()), http.StatusUnauthorized) 59 return 60 } 61 62 gr, err := git.Open(repoPath, data.Branch) 63 if err != nil { 64 fail(xrpcerr.GenericError(fmt.Errorf("failed to open repository: %w", err))) 65 return 66 } 67 68 mo := git.MergeOptions{} 69 if data.AuthorName != nil { 70 mo.AuthorName = *data.AuthorName 71 } 72 if data.AuthorEmail != nil { 73 mo.AuthorEmail = *data.AuthorEmail 74 } 75 if data.CommitBody != nil { 76 mo.CommitBody = *data.CommitBody 77 } 78 if data.CommitMessage != nil { 79 mo.CommitMessage = *data.CommitMessage 80 } 81 82 mo.CommitterName = x.Config.Git.UserName 83 mo.CommitterEmail = x.Config.Git.UserEmail 84 mo.FormatPatch = patchutil.IsFormatPatch(data.Patch) 85 86 err = gr.MergeWithOptions(data.Patch, data.Branch, mo) 87 if err != nil { 88 var mergeErr *git.ErrMerge 89 if errors.As(err, &mergeErr) { 90 conflicts := make([]types.ConflictInfo, len(mergeErr.Conflicts)) 91 for i, conflict := range mergeErr.Conflicts { 92 conflicts[i] = types.ConflictInfo{ 93 Filename: conflict.Filename, 94 Reason: conflict.Reason, 95 } 96 } 97 98 conflictErr := xrpcerr.NewXrpcError( 99 xrpcerr.WithTag("MergeConflict"), 100 xrpcerr.WithMessage(fmt.Sprintf("Merge failed due to conflicts: %s", mergeErr.Message)), 101 ) 102 writeError(w, conflictErr, http.StatusConflict) 103 return 104 } else { 105 l.Error("failed to merge", "error", err.Error()) 106 writeError(w, xrpcerr.GitError(err), http.StatusInternalServerError) 107 return 108 } 109 } 110 111 w.WriteHeader(http.StatusOK) 112}