A vibe coded tangled fork which supports pijul.
at master 109 lines 3.8 kB view raw
1package repo 2 3import ( 4 "net/http" 5 6 "github.com/go-chi/chi/v5" 7 "tangled.org/core/appview/middleware" 8) 9 10func (rp *Repo) Router(mw *middleware.Middleware) http.Handler { 11 r := chi.NewRouter() 12 r.Get("/", rp.Index) 13 r.Get("/opengraph", rp.Opengraph) 14 r.Get("/feed.atom", rp.AtomFeed) 15 r.Get("/commits/{ref}", rp.Log) 16 r.Get("/changes", rp.Changes) 17 r.Get("/changes/{ref}", rp.Changes) 18 r.Route("/tree/{ref}", func(r chi.Router) { 19 r.Get("/", rp.Index) 20 r.Get("/*", rp.Tree) 21 }) 22 r.Get("/commit/{ref}", rp.Commit) 23 r.Get("/change/{hash}", rp.Change) 24 r.Get("/branches", rp.Branches) 25 r.Delete("/branches", rp.DeleteBranch) 26 r.Route("/tags", func(r chi.Router) { 27 r.Get("/", rp.Tags) 28 r.Route("/{tag}", func(r chi.Router) { 29 r.Get("/", rp.Tag) 30 r.Get("/download/{file}", rp.DownloadArtifact) 31 32 // require repo:push to upload or delete artifacts 33 // 34 // additionally: only the uploader can truly delete an artifact 35 // (record+blob will live on their pds) 36 r.Group(func(r chi.Router) { 37 r.Use(middleware.AuthMiddleware(rp.oauth)) 38 r.Use(mw.RepoPermissionMiddleware("repo:push")) 39 r.Post("/upload", rp.AttachArtifact) 40 r.Delete("/{file}", rp.DeleteArtifact) 41 }) 42 }) 43 }) 44 r.Get("/blob/{ref}/*", rp.Blob) 45 r.Get("/raw/{ref}/*", rp.RepoBlobRaw) 46 47 // intentionally doesn't use /* as this isn't 48 // a file path 49 r.Get("/archive/{ref}", rp.DownloadArchive) 50 51 r.Route("/fork", func(r chi.Router) { 52 r.Use(middleware.AuthMiddleware(rp.oauth)) 53 r.Get("/", rp.ForkRepo) 54 r.Post("/", rp.ForkRepo) 55 r.With(mw.RepoPermissionMiddleware("repo:owner")).Route("/sync", func(r chi.Router) { 56 r.Post("/", rp.SyncRepoFork) 57 }) 58 }) 59 60 r.Route("/compare", func(r chi.Router) { 61 r.Get("/", rp.CompareNew) // start an new comparison 62 63 // we have to wildcard here since we want to support GitHub's compare syntax 64 // /compare/{ref1}...{ref2} 65 // for example: 66 // /compare/master...some/feature 67 // /compare/master...example.com:another/feature <- this is a fork 68 r.Get("/*", rp.Compare) 69 }) 70 71 // label panel in issues/pulls/discussions/tasks 72 r.Route("/label", func(r chi.Router) { 73 r.Get("/", rp.LabelPanel) 74 r.Get("/edit", rp.EditLabelPanel) 75 }) 76 77 // settings routes, needs auth 78 r.Group(func(r chi.Router) { 79 r.Use(middleware.AuthMiddleware(rp.oauth)) 80 r.With(mw.RepoPermissionMiddleware("repo:settings")).Route("/settings", func(r chi.Router) { 81 r.Get("/", rp.Settings) 82 r.With(mw.RepoPermissionMiddleware("repo:owner")).Put("/base", rp.EditBaseSettings) 83 r.With(mw.RepoPermissionMiddleware("repo:owner")).Post("/spindle", rp.EditSpindle) 84 r.With(mw.RepoPermissionMiddleware("repo:owner")).Put("/label", rp.AddLabelDef) 85 r.With(mw.RepoPermissionMiddleware("repo:owner")).Delete("/label", rp.DeleteLabelDef) 86 r.With(mw.RepoPermissionMiddleware("repo:owner")).Post("/label/subscribe", rp.SubscribeLabel) 87 r.With(mw.RepoPermissionMiddleware("repo:owner")).Post("/label/unsubscribe", rp.UnsubscribeLabel) 88 r.With(mw.RepoPermissionMiddleware("repo:invite")).Put("/collaborator", rp.AddCollaborator) 89 r.With(mw.RepoPermissionMiddleware("repo:delete")).Delete("/delete", rp.DeleteRepo) 90 r.Put("/branches/default", rp.SetDefaultBranch) 91 r.Put("/secrets", rp.Secrets) 92 r.Delete("/secrets", rp.Secrets) 93 r.With(mw.RepoPermissionMiddleware("repo:owner")).Route("/sites", func(r chi.Router) { 94 r.Put("/", rp.SaveRepoSiteConfig) 95 r.Delete("/", rp.DeleteRepoSiteConfig) 96 }) 97 r.With(mw.RepoPermissionMiddleware("repo:owner")).Route("/hooks", func(r chi.Router) { 98 r.Get("/", rp.Webhooks) 99 r.Post("/", rp.AddWebhook) 100 r.Put("/{id}", rp.UpdateWebhook) 101 r.Delete("/{id}", rp.DeleteWebhook) 102 r.Post("/{id}/toggle", rp.ToggleWebhook) 103 r.Get("/{id}/deliveries", rp.WebhookDeliveries) 104 }) 105 }) 106 }) 107 108 return r 109}