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/git"
13 "tangled.org/core/rbac"
14
15 xrpcerr "tangled.org/core/xrpc/errors"
16)
17
18func (x *Xrpc) DeleteBranch(w http.ResponseWriter, r *http.Request) {
19 l := x.Logger
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.RepoDeleteBranch_Input
32 if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
33 fail(xrpcerr.GenericError(err))
34 return
35 }
36
37 // unfortunately we have to resolve repo-at here
38 repoAt, err := syntax.ParseATURI(data.Repo)
39 if err != nil {
40 fail(xrpcerr.InvalidRepoError(data.Repo))
41 return
42 }
43
44 // resolve this aturi to extract the repo record
45 ident, err := x.Resolver.ResolveIdent(r.Context(), repoAt.Authority().String())
46 if err != nil || ident.Handle.IsInvalidHandle() {
47 fail(xrpcerr.GenericError(fmt.Errorf("failed to resolve handle: %w", err)))
48 return
49 }
50
51 xrpcc := xrpc.Client{Host: ident.PDSEndpoint()}
52 resp, err := comatproto.RepoGetRecord(r.Context(), &xrpcc, "", tangled.RepoNSID, repoAt.Authority().String(), repoAt.RecordKey().String())
53 if err != nil {
54 fail(xrpcerr.GenericError(err))
55 return
56 }
57
58 repo := resp.Value.Val.(*tangled.Repo)
59 repoDid, err := x.Db.GetRepoDid(ident.DID.String(), repo.Name)
60 if err != nil {
61 fail(xrpcerr.RepoNotFoundError)
62 return
63 }
64 repoPath, _, _, err := x.Db.ResolveRepoDIDOnDisk(x.Config.Repo.ScanPath, repoDid)
65 if err != nil {
66 fail(xrpcerr.RepoNotFoundError)
67 return
68 }
69
70 if ok, err := x.Enforcer.IsPushAllowed(actorDid.String(), rbac.ThisServer, repoDid); !ok || err != nil {
71 l.Error("insufficent permissions", "did", actorDid.String(), "repo", repoDid)
72 writeError(w, xrpcerr.AccessControlError(actorDid.String()), http.StatusUnauthorized)
73 return
74 }
75
76 gr, err := git.PlainOpen(repoPath)
77 if err != nil {
78 fail(xrpcerr.GenericError(err))
79 return
80 }
81
82 err = gr.DeleteBranch(data.Branch)
83 if err != nil {
84 l.Error("deleting branch", "error", err.Error(), "branch", data.Branch)
85 writeError(w, xrpcerr.GitError(err), http.StatusInternalServerError)
86 return
87 }
88
89 w.WriteHeader(http.StatusOK)
90}