Monorepo for Tangled
0
fork

Configure Feed

Select the types of activity you want to include in your feed.

at master 112 lines 3.0 kB view raw
1package xrpc 2 3import ( 4 "encoding/json" 5 "fmt" 6 "net/http" 7 8 comatproto "github.com/bluesky-social/indigo/api/atproto" 9 "github.com/bluesky-social/indigo/atproto/syntax" 10 "github.com/bluesky-social/indigo/xrpc" 11 "tangled.org/core/api/tangled" 12 "tangled.org/core/knotserver/db" 13 "tangled.org/core/knotserver/git" 14 "tangled.org/core/rbac" 15 "tangled.org/core/tid" 16 17 xrpcerr "tangled.org/core/xrpc/errors" 18) 19 20const ActorDid string = "ActorDid" 21 22func (x *Xrpc) SetDefaultBranch(w http.ResponseWriter, r *http.Request) { 23 l := x.Logger 24 fail := func(e xrpcerr.XrpcError) { 25 l.Error("failed", "kind", e.Tag, "error", e.Message) 26 writeError(w, e, http.StatusBadRequest) 27 } 28 29 actorDid, ok := r.Context().Value(ActorDid).(syntax.DID) 30 if !ok { 31 fail(xrpcerr.MissingActorDidError) 32 return 33 } 34 35 var data tangled.RepoSetDefaultBranch_Input 36 if err := json.NewDecoder(r.Body).Decode(&data); err != nil { 37 fail(xrpcerr.GenericError(err)) 38 return 39 } 40 41 // unfortunately we have to resolve repo-at here 42 repoAt, err := syntax.ParseATURI(data.Repo) 43 if err != nil { 44 fail(xrpcerr.InvalidRepoError(data.Repo)) 45 return 46 } 47 48 // resolve this aturi to extract the repo record 49 ident, err := x.Resolver.ResolveIdent(r.Context(), repoAt.Authority().String()) 50 if err != nil || ident.Handle.IsInvalidHandle() { 51 fail(xrpcerr.GenericError(fmt.Errorf("failed to resolve handle: %w", err))) 52 return 53 } 54 55 xrpcc := xrpc.Client{Host: ident.PDSEndpoint()} 56 resp, err := comatproto.RepoGetRecord(r.Context(), &xrpcc, "", tangled.RepoNSID, repoAt.Authority().String(), repoAt.RecordKey().String()) 57 if err != nil { 58 fail(xrpcerr.GenericError(err)) 59 return 60 } 61 62 repo := resp.Value.Val.(*tangled.Repo) 63 repoDid, err := x.Db.GetRepoDid(actorDid.String(), repo.Name) 64 if err != nil { 65 fail(xrpcerr.RepoNotFoundError) 66 return 67 } 68 repoPath, _, _, err := x.Db.ResolveRepoDIDOnDisk(x.Config.Repo.ScanPath, repoDid) 69 if err != nil { 70 fail(xrpcerr.RepoNotFoundError) 71 return 72 } 73 74 if ok, err := x.Enforcer.IsPushAllowed(actorDid.String(), rbac.ThisServer, repoDid); !ok || err != nil { 75 l.Error("insufficient permissions", "did", actorDid.String()) 76 writeError(w, xrpcerr.AccessControlError(actorDid.String()), http.StatusUnauthorized) 77 return 78 } 79 80 gr, err := git.PlainOpen(repoPath) 81 if err != nil { 82 fail(xrpcerr.GenericError(err)) 83 return 84 } 85 86 err = gr.SetDefaultBranch(data.DefaultBranch) 87 if err != nil { 88 l.Error("setting default branch", "error", err.Error()) 89 writeError(w, xrpcerr.GitError(err), http.StatusInternalServerError) 90 return 91 } 92 93 ownerDid := ident.DID.String() 94 refUpdate := tangled.GitRefUpdate{ 95 RepoDid: repo.RepoDid, 96 OwnerDid: &ownerDid, 97 RepoName: repo.Name, 98 CommitterDid: actorDid.String(), 99 } 100 eventJson, err := json.Marshal(refUpdate) 101 if err != nil { 102 return 103 } 104 105 x.Db.InsertEvent(db.Event{ 106 Rkey: tid.TID(), 107 Nsid: tangled.GitRefUpdateNSID, 108 EventJson: string(eventJson), 109 }, x.Notifier) 110 111 w.WriteHeader(http.StatusOK) 112}