forked from
tangled.org/core
Monorepo for Tangled
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}