1package state
2
3import (
4 "net/http"
5 "strings"
6
7 "github.com/go-chi/chi/v5"
8 "tangled.sh/tangled.sh/core/appview/state/userutil"
9)
10
11func (s *State) Router() http.Handler {
12 router := chi.NewRouter()
13
14 router.HandleFunc("/*", func(w http.ResponseWriter, r *http.Request) {
15 pat := chi.URLParam(r, "*")
16 if strings.HasPrefix(pat, "did:") || strings.HasPrefix(pat, "@") {
17 s.UserRouter().ServeHTTP(w, r)
18 } else {
19 // Check if the first path element is a valid handle without '@' or a flattened DID
20 pathParts := strings.SplitN(pat, "/", 2)
21 if len(pathParts) > 0 {
22 if userutil.IsHandleNoAt(pathParts[0]) {
23 // Redirect to the same path but with '@' prefixed to the handle
24 redirectPath := "@" + pat
25 http.Redirect(w, r, "/"+redirectPath, http.StatusFound)
26 return
27 } else if userutil.IsFlattenedDid(pathParts[0]) {
28 // Redirect to the unflattened DID version
29 unflattenedDid := userutil.UnflattenDid(pathParts[0])
30 var redirectPath string
31 if len(pathParts) > 1 {
32 redirectPath = unflattenedDid + "/" + pathParts[1]
33 } else {
34 redirectPath = unflattenedDid
35 }
36 http.Redirect(w, r, "/"+redirectPath, http.StatusFound)
37 return
38 }
39 }
40 s.StandardRouter().ServeHTTP(w, r)
41 }
42 })
43
44 return router
45}
46
47func (s *State) UserRouter() http.Handler {
48 r := chi.NewRouter()
49
50 // strip @ from user
51 r.Use(StripLeadingAt)
52
53 r.With(ResolveIdent(s)).Route("/{user}", func(r chi.Router) {
54 r.Get("/", s.ProfilePage)
55 r.With(ResolveRepo(s)).Route("/{repo}", func(r chi.Router) {
56 r.Get("/", s.RepoIndex)
57 r.Get("/commits/{ref}", s.RepoLog)
58 r.Route("/tree/{ref}", func(r chi.Router) {
59 r.Get("/", s.RepoIndex)
60 r.Get("/*", s.RepoTree)
61 })
62 r.Get("/commit/{ref}", s.RepoCommit)
63 r.Get("/branches", s.RepoBranches)
64 r.Get("/tags", s.RepoTags)
65 r.Get("/blob/{ref}/*", s.RepoBlob)
66 r.Get("/blob/{ref}/raw/*", s.RepoBlobRaw)
67
68 r.Route("/issues", func(r chi.Router) {
69 r.Get("/", s.RepoIssues)
70 r.Get("/{issue}", s.RepoSingleIssue)
71
72 r.Group(func(r chi.Router) {
73 r.Use(AuthMiddleware(s))
74 r.Get("/new", s.NewIssue)
75 r.Post("/new", s.NewIssue)
76 r.Post("/{issue}/comment", s.NewIssueComment)
77 r.Route("/{issue}/comment/{comment_id}/", func(r chi.Router) {
78 r.Get("/", s.IssueComment)
79 r.Delete("/", s.DeleteIssueComment)
80 r.Get("/edit", s.EditIssueComment)
81 r.Post("/edit", s.EditIssueComment)
82 })
83 r.Post("/{issue}/close", s.CloseIssue)
84 r.Post("/{issue}/reopen", s.ReopenIssue)
85 })
86 })
87
88 r.Route("/pulls", func(r chi.Router) {
89 r.Get("/", s.RepoPulls)
90 r.With(AuthMiddleware(s)).Route("/new", func(r chi.Router) {
91 r.Get("/", s.NewPull)
92 r.Post("/", s.NewPull)
93 })
94
95 r.Route("/{pull}", func(r chi.Router) {
96 r.Use(ResolvePull(s))
97 r.Get("/", s.RepoSinglePull)
98
99 r.Route("/round/{round}", func(r chi.Router) {
100 r.Get("/", s.RepoPullPatch)
101 r.Get("/actions", s.PullActions)
102 r.With(AuthMiddleware(s)).Route("/comment", func(r chi.Router) {
103 r.Get("/", s.PullComment)
104 r.Post("/", s.PullComment)
105 })
106 })
107
108 r.Route("/round/{round}.patch", func(r chi.Router) {
109 r.Get("/", s.RepoPullPatchRaw)
110 })
111
112 r.Group(func(r chi.Router) {
113 r.Use(AuthMiddleware(s))
114 r.Route("/resubmit", func(r chi.Router) {
115 r.Get("/", s.ResubmitPull)
116 r.Post("/", s.ResubmitPull)
117 })
118 r.Post("/close", s.ClosePull)
119 r.Post("/reopen", s.ReopenPull)
120 // collaborators only
121 r.Group(func(r chi.Router) {
122 r.Use(RepoPermissionMiddleware(s, "repo:push"))
123 r.Post("/merge", s.MergePull)
124 // maybe lock, etc.
125 })
126 })
127 })
128 })
129
130 // These routes get proxied to the knot
131 r.Get("/info/refs", s.InfoRefs)
132 r.Post("/git-upload-pack", s.UploadPack)
133
134 // settings routes, needs auth
135 r.Group(func(r chi.Router) {
136 r.Use(AuthMiddleware(s))
137 // repo description can only be edited by owner
138 r.With(RepoPermissionMiddleware(s, "repo:owner")).Route("/description", func(r chi.Router) {
139 r.Put("/", s.RepoDescription)
140 r.Get("/", s.RepoDescription)
141 r.Get("/edit", s.RepoDescriptionEdit)
142 })
143 r.With(RepoPermissionMiddleware(s, "repo:settings")).Route("/settings", func(r chi.Router) {
144 r.Get("/", s.RepoSettings)
145 r.With(RepoPermissionMiddleware(s, "repo:invite")).Put("/collaborator", s.AddCollaborator)
146 })
147 })
148 })
149 })
150
151 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
152 s.pages.Error404(w)
153 })
154
155 return r
156}
157
158func (s *State) StandardRouter() http.Handler {
159 r := chi.NewRouter()
160
161 r.Handle("/static/*", s.pages.Static())
162
163 r.Get("/", s.Timeline)
164
165 r.With(AuthMiddleware(s)).Post("/logout", s.Logout)
166
167 r.Route("/login", func(r chi.Router) {
168 r.Get("/", s.Login)
169 r.Post("/", s.Login)
170 })
171
172 r.Route("/knots", func(r chi.Router) {
173 r.Use(AuthMiddleware(s))
174 r.Get("/", s.Knots)
175 r.Post("/key", s.RegistrationKey)
176
177 r.Route("/{domain}", func(r chi.Router) {
178 r.Post("/init", s.InitKnotServer)
179 r.Get("/", s.KnotServerInfo)
180 r.Route("/member", func(r chi.Router) {
181 r.Use(KnotOwner(s))
182 r.Get("/", s.ListMembers)
183 r.Put("/", s.AddMember)
184 r.Delete("/", s.RemoveMember)
185 })
186 })
187 })
188
189 r.Route("/repo", func(r chi.Router) {
190 r.Route("/new", func(r chi.Router) {
191 r.Use(AuthMiddleware(s))
192 r.Get("/", s.NewRepo)
193 r.Post("/", s.NewRepo)
194 })
195 // r.Post("/import", s.ImportRepo)
196 })
197
198 r.With(AuthMiddleware(s)).Route("/follow", func(r chi.Router) {
199 r.Post("/", s.Follow)
200 r.Delete("/", s.Follow)
201 })
202
203 r.With(AuthMiddleware(s)).Route("/star", func(r chi.Router) {
204 r.Post("/", s.Star)
205 r.Delete("/", s.Star)
206 })
207
208 r.Route("/settings", func(r chi.Router) {
209 r.Use(AuthMiddleware(s))
210 r.Get("/", s.Settings)
211 r.Put("/keys", s.SettingsKeys)
212 r.Delete("/keys", s.SettingsKeys)
213 r.Put("/emails", s.SettingsEmails)
214 r.Delete("/emails", s.SettingsEmails)
215 r.Get("/emails/verify", s.SettingsEmailsVerify)
216 r.Post("/emails/verify/resend", s.SettingsEmailsVerifyResend)
217 r.Post("/emails/primary", s.SettingsEmailsPrimary)
218 })
219
220 r.Get("/keys/{user}", s.Keys)
221
222 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
223 s.pages.Error404(w)
224 })
225 return r
226}