A vibe coded tangled fork which supports pijul.
1package state
2
3import (
4 "net/http"
5 "strings"
6
7 "github.com/go-chi/chi/v5"
8 "tangled.org/core/appview/issues"
9 "tangled.org/core/appview/knots"
10 "tangled.org/core/appview/labels"
11 "tangled.org/core/appview/middleware"
12 "tangled.org/core/appview/notifications"
13 "tangled.org/core/appview/pipelines"
14 "tangled.org/core/appview/pulls"
15 "tangled.org/core/appview/repo"
16 "tangled.org/core/appview/settings"
17 "tangled.org/core/appview/signup"
18 "tangled.org/core/appview/spindles"
19 "tangled.org/core/appview/state/userutil"
20 avstrings "tangled.org/core/appview/strings"
21 "tangled.org/core/log"
22)
23
24func (s *State) Router() http.Handler {
25 router := chi.NewRouter()
26 middleware := middleware.New(
27 s.oauth,
28 s.db,
29 s.enforcer,
30 s.repoResolver,
31 s.idResolver,
32 s.pages,
33 )
34
35 router.Get("/pwa-manifest.json", s.WebAppManifest)
36 router.Get("/robots.txt", s.RobotsTxt)
37 router.Get("/sitemap.xml", s.Sitemap)
38
39 userRouter := s.UserRouter(&middleware)
40 standardRouter := s.StandardRouter(&middleware)
41
42 router.HandleFunc("/*", func(w http.ResponseWriter, r *http.Request) {
43 pat := chi.URLParam(r, "*")
44 pathParts := strings.SplitN(pat, "/", 2)
45
46 if len(pathParts) > 0 {
47 firstPart := pathParts[0]
48
49 // if using a DID or handle, just continue as per usual
50 if userutil.IsDid(firstPart) || userutil.IsHandle(firstPart) {
51 userRouter.ServeHTTP(w, r)
52 return
53 }
54
55 // if using a flattened DID (like you would in go modules), unflatten
56 if userutil.IsFlattenedDid(firstPart) {
57 unflattenedDid := userutil.UnflattenDid(firstPart)
58 redirectPath := strings.Join(append([]string{unflattenedDid}, pathParts[1:]...), "/")
59
60 redirectURL := *r.URL
61 redirectURL.Path = "/" + redirectPath
62
63 http.Redirect(w, r, redirectURL.String(), http.StatusFound)
64 return
65 }
66
67 // if using a handle with @, rewrite to work without @
68 if normalized := strings.TrimPrefix(firstPart, "@"); userutil.IsHandle(normalized) {
69 redirectPath := strings.Join(append([]string{normalized}, pathParts[1:]...), "/")
70
71 redirectURL := *r.URL
72 redirectURL.Path = "/" + redirectPath
73
74 http.Redirect(w, r, redirectURL.String(), http.StatusFound)
75 return
76 }
77
78 }
79
80 standardRouter.ServeHTTP(w, r)
81 })
82
83 return router
84}
85
86func (s *State) UserRouter(mw *middleware.Middleware) http.Handler {
87 r := chi.NewRouter()
88
89 r.With(mw.ResolveIdent()).Route("/{user}", func(r chi.Router) {
90 r.Get("/", s.Profile)
91 r.Get("/feed.atom", s.AtomFeedPage)
92
93 r.With(mw.ResolveRepo()).Route("/{repo}", func(r chi.Router) {
94 r.Use(mw.GoImport())
95 r.Mount("/", s.RepoRouter(mw))
96 r.Mount("/issues", s.IssuesRouter(mw))
97 r.Mount("/pulls", s.PullsRouter(mw))
98 r.Mount("/pipelines", s.PipelinesRouter(mw))
99 r.Mount("/labels", s.LabelsRouter())
100
101 // These routes get proxied to the knot
102 r.Get("/info/refs", s.InfoRefs)
103 r.Post("/git-upload-archive", s.UploadArchive)
104 r.Post("/git-upload-pack", s.UploadPack)
105 r.Post("/git-receive-pack", s.ReceivePack)
106
107 })
108 })
109
110 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
111 w.WriteHeader(http.StatusNotFound)
112 s.pages.Error404(w)
113 })
114
115 return r
116}
117
118func (s *State) StandardRouter(mw *middleware.Middleware) http.Handler {
119 r := chi.NewRouter()
120
121 r.Handle("/static/*", s.pages.Static())
122
123 r.Get("/", s.HomeOrTimeline)
124 r.Get("/timeline", s.Timeline)
125 r.Get("/upgradeBanner", s.UpgradeBanner)
126
127 // special-case handler for serving tangled.org/core
128 r.Get("/core", s.Core())
129
130 r.Get("/login", s.Login)
131 r.Post("/login", s.Login)
132 r.Post("/logout", s.Logout)
133
134 r.Post("/account/switch", s.SwitchAccount)
135 r.With(middleware.AuthMiddleware(s.oauth)).Delete("/account/{did}", s.RemoveAccount)
136
137 r.Route("/repo", func(r chi.Router) {
138 r.Route("/new", func(r chi.Router) {
139 r.Use(middleware.AuthMiddleware(s.oauth))
140 r.Get("/", s.NewRepo)
141 r.Post("/", s.NewRepo)
142 })
143 // r.Post("/import", s.ImportRepo)
144 })
145
146 r.With(middleware.Paginate).Get("/goodfirstissues", s.GoodFirstIssues)
147
148 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) {
149 r.Post("/", s.Follow)
150 r.Delete("/", s.Follow)
151 })
152
153 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) {
154 r.Post("/", s.Star)
155 r.Delete("/", s.Star)
156 })
157
158 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) {
159 r.Post("/", s.React)
160 r.Delete("/", s.React)
161 })
162
163 r.Route("/profile", func(r chi.Router) {
164 r.Use(middleware.AuthMiddleware(s.oauth))
165 r.Get("/edit-bio", s.EditBioFragment)
166 r.Get("/edit-pins", s.EditPinsFragment)
167 r.Post("/bio", s.UpdateProfileBio)
168 r.Post("/pins", s.UpdateProfilePins)
169 r.Post("/avatar", s.UploadProfileAvatar)
170 r.Delete("/avatar", s.RemoveProfileAvatar)
171 })
172
173 r.Mount("/settings", s.SettingsRouter())
174 r.Mount("/strings", s.StringsRouter(mw))
175
176 r.Mount("/settings/knots", s.KnotsRouter())
177 r.Mount("/settings/spindles", s.SpindlesRouter())
178
179 r.Mount("/notifications", s.NotificationsRouter(mw))
180
181 r.Mount("/signup", s.SignupRouter())
182 r.Mount("/", s.oauth.Router())
183
184 r.Get("/keys/{user}", s.Keys)
185 r.Get("/terms", s.TermsOfService)
186 r.Get("/privacy", s.PrivacyPolicy)
187 r.Get("/brand", s.Brand)
188
189 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
190 w.WriteHeader(http.StatusNotFound)
191 s.pages.Error404(w)
192 })
193 return r
194}
195
196// Core serves tangled.org/core go-import meta tags, and redirects
197// to the core repository if accessed normally.
198func (s *State) Core() http.HandlerFunc {
199 return func(w http.ResponseWriter, r *http.Request) {
200 if r.URL.Query().Get("go-get") == "1" {
201 w.Header().Set("Content-Type", "text/html")
202 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`))
203 return
204 }
205
206 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound)
207 }
208}
209
210func (s *State) SettingsRouter() http.Handler {
211 settings := &settings.Settings{
212 Db: s.db,
213 OAuth: s.oauth,
214 Pages: s.pages,
215 Config: s.config,
216 }
217
218 return settings.Router()
219}
220
221func (s *State) SpindlesRouter() http.Handler {
222 logger := log.SubLogger(s.logger, "spindles")
223
224 spindles := &spindles.Spindles{
225 Db: s.db,
226 OAuth: s.oauth,
227 Pages: s.pages,
228 Config: s.config,
229 Enforcer: s.enforcer,
230 IdResolver: s.idResolver,
231 Logger: logger,
232 }
233
234 return spindles.Router()
235}
236
237func (s *State) KnotsRouter() http.Handler {
238 logger := log.SubLogger(s.logger, "knots")
239
240 knots := &knots.Knots{
241 Db: s.db,
242 OAuth: s.oauth,
243 Pages: s.pages,
244 Config: s.config,
245 Enforcer: s.enforcer,
246 IdResolver: s.idResolver,
247 Knotstream: s.knotstream,
248 Logger: logger,
249 }
250
251 return knots.Router()
252}
253
254func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler {
255 logger := log.SubLogger(s.logger, "strings")
256
257 strs := &avstrings.Strings{
258 Db: s.db,
259 OAuth: s.oauth,
260 Pages: s.pages,
261 IdResolver: s.idResolver,
262 Notifier: s.notifier,
263 Logger: logger,
264 }
265
266 return strs.Router(mw)
267}
268
269func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler {
270 issues := issues.New(
271 s.oauth,
272 s.repoResolver,
273 s.enforcer,
274 s.pages,
275 s.idResolver,
276 s.mentionsResolver,
277 s.db,
278 s.config,
279 s.notifier,
280 s.validator,
281 s.indexer.Issues,
282 log.SubLogger(s.logger, "issues"),
283 )
284 return issues.Router(mw)
285}
286
287func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler {
288 pulls := pulls.New(
289 s.oauth,
290 s.repoResolver,
291 s.pages,
292 s.idResolver,
293 s.mentionsResolver,
294 s.db,
295 s.config,
296 s.notifier,
297 s.enforcer,
298 s.validator,
299 s.indexer.Pulls,
300 log.SubLogger(s.logger, "pulls"),
301 )
302 return pulls.Router(mw)
303}
304
305func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler {
306 repo := repo.New(
307 s.oauth,
308 s.repoResolver,
309 s.pages,
310 s.spindlestream,
311 s.idResolver,
312 s.db,
313 s.config,
314 s.notifier,
315 s.enforcer,
316 log.SubLogger(s.logger, "repo"),
317 s.validator,
318 )
319 return repo.Router(mw)
320}
321
322func (s *State) PipelinesRouter(mw *middleware.Middleware) http.Handler {
323 pipes := pipelines.New(
324 s.oauth,
325 s.repoResolver,
326 s.pages,
327 s.spindlestream,
328 s.idResolver,
329 s.db,
330 s.config,
331 s.enforcer,
332 log.SubLogger(s.logger, "pipelines"),
333 )
334 return pipes.Router(mw)
335}
336
337func (s *State) LabelsRouter() http.Handler {
338 ls := labels.New(
339 s.oauth,
340 s.pages,
341 s.db,
342 s.validator,
343 s.enforcer,
344 log.SubLogger(s.logger, "labels"),
345 )
346 return ls.Router()
347}
348
349func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler {
350 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications"))
351 return notifs.Router(mw)
352}
353
354func (s *State) SignupRouter() http.Handler {
355 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup"))
356 return sig.Router()
357}