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/discussions"
9 "tangled.org/core/appview/issues"
10 "tangled.org/core/appview/knots"
11 "tangled.org/core/appview/labels"
12 "tangled.org/core/appview/middleware"
13 "tangled.org/core/appview/notifications"
14 "tangled.org/core/appview/pipelines"
15 "tangled.org/core/appview/pulls"
16 "tangled.org/core/appview/repo"
17 "tangled.org/core/appview/settings"
18 "tangled.org/core/appview/signup"
19 "tangled.org/core/appview/spindles"
20 "tangled.org/core/appview/state/userutil"
21 avstrings "tangled.org/core/appview/strings"
22 "tangled.org/core/log"
23)
24
25func (s *State) Router() http.Handler {
26 router := chi.NewRouter()
27 middleware := middleware.New(
28 s.oauth,
29 s.db,
30 s.enforcer,
31 s.repoResolver,
32 s.idResolver,
33 s.pages,
34 )
35
36 router.Get("/pwa-manifest.json", s.WebAppManifest)
37 router.Get("/robots.txt", s.RobotsTxt)
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("/discussions", s.DiscussionsRouter(mw))
97 r.Mount("/issues", s.IssuesRouter(mw))
98 r.Mount("/pulls", s.PullsRouter(mw))
99 r.Mount("/pipelines", s.PipelinesRouter(mw))
100 r.Mount("/labels", s.LabelsRouter())
101
102 // These routes get proxied to the knot
103 r.Get("/info/refs", s.InfoRefs)
104 r.Post("/git-upload-archive", s.UploadArchive)
105 r.Post("/git-upload-pack", s.UploadPack)
106 r.Post("/git-receive-pack", s.ReceivePack)
107
108 })
109 })
110
111 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
112 w.WriteHeader(http.StatusNotFound)
113 s.pages.Error404(w)
114 })
115
116 return r
117}
118
119func (s *State) StandardRouter(mw *middleware.Middleware) http.Handler {
120 r := chi.NewRouter()
121
122 r.Handle("/static/*", s.pages.Static())
123
124 r.Get("/", s.HomeOrTimeline)
125 r.Get("/home", s.Home)
126 r.Get("/timeline", s.Timeline)
127 r.Get("/upgradeBanner", s.UpgradeBanner)
128
129 // special-case handler for serving tangled.org/core
130 r.Get("/core", s.Core())
131
132 r.Get("/login", s.Login)
133 r.Post("/login", s.Login)
134 r.Post("/logout", s.Logout)
135
136 r.Post("/account/switch", s.SwitchAccount)
137 r.With(middleware.AuthMiddleware(s.oauth)).Delete("/account/{did}", s.RemoveAccount)
138
139 r.Route("/repo", func(r chi.Router) {
140 r.Route("/new", func(r chi.Router) {
141 r.Use(middleware.AuthMiddleware(s.oauth))
142 r.Get("/", s.NewRepo)
143 r.Post("/", s.NewRepo)
144 })
145 // r.Post("/import", s.ImportRepo)
146 })
147
148 r.With(middleware.Paginate).Get("/goodfirstissues", s.GoodFirstIssues)
149
150 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) {
151 r.Post("/", s.Follow)
152 r.Delete("/", s.Follow)
153 })
154
155 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) {
156 r.Post("/", s.Star)
157 r.Delete("/", s.Star)
158 })
159
160 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) {
161 r.Post("/", s.React)
162 r.Delete("/", s.React)
163 })
164
165 r.Route("/profile", func(r chi.Router) {
166 r.Use(middleware.AuthMiddleware(s.oauth))
167 r.Get("/edit-bio", s.EditBioFragment)
168 r.Get("/edit-pins", s.EditPinsFragment)
169 r.Post("/bio", s.UpdateProfileBio)
170 r.Post("/pins", s.UpdateProfilePins)
171 r.Post("/avatar", s.UploadProfileAvatar)
172 r.Delete("/avatar", s.RemoveProfileAvatar)
173 r.Post("/punchcard", s.UpdateProfilePunchcardSetting)
174 })
175
176 r.Mount("/settings", s.SettingsRouter())
177 r.Mount("/strings", s.StringsRouter(mw))
178
179 r.Mount("/settings/knots", s.KnotsRouter())
180 r.Mount("/settings/spindles", s.SpindlesRouter())
181
182 r.Mount("/notifications", s.NotificationsRouter(mw))
183
184 r.Mount("/signup", s.SignupRouter())
185 r.Mount("/", s.oauth.Router())
186
187 r.Get("/keys/{user}", s.Keys)
188 r.Get("/terms", s.TermsOfService)
189 r.Get("/privacy", s.PrivacyPolicy)
190 r.Get("/brand", s.Brand)
191
192 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
193 w.WriteHeader(http.StatusNotFound)
194 s.pages.Error404(w)
195 })
196 return r
197}
198
199// Core serves tangled.org/core go-import meta tags, and redirects
200// to the core repository if accessed normally.
201func (s *State) Core() http.HandlerFunc {
202 return func(w http.ResponseWriter, r *http.Request) {
203 if r.URL.Query().Get("go-get") == "1" {
204 w.Header().Set("Content-Type", "text/html")
205 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`))
206 return
207 }
208
209 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound)
210 }
211}
212
213func (s *State) SettingsRouter() http.Handler {
214 settings := &settings.Settings{
215 Db: s.db,
216 OAuth: s.oauth,
217 Pages: s.pages,
218 Config: s.config,
219 CfClient: s.cfClient,
220 Logger: log.SubLogger(s.logger, "settings"),
221 }
222
223 return settings.Router()
224}
225
226func (s *State) SpindlesRouter() http.Handler {
227 logger := log.SubLogger(s.logger, "spindles")
228
229 spindles := &spindles.Spindles{
230 Db: s.db,
231 OAuth: s.oauth,
232 Pages: s.pages,
233 Config: s.config,
234 Enforcer: s.enforcer,
235 IdResolver: s.idResolver,
236 Logger: logger,
237 }
238
239 return spindles.Router()
240}
241
242func (s *State) KnotsRouter() http.Handler {
243 logger := log.SubLogger(s.logger, "knots")
244
245 knots := &knots.Knots{
246 Db: s.db,
247 OAuth: s.oauth,
248 Pages: s.pages,
249 Config: s.config,
250 Enforcer: s.enforcer,
251 IdResolver: s.idResolver,
252 Knotstream: s.knotstream,
253 Logger: logger,
254 }
255
256 return knots.Router()
257}
258
259func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler {
260 logger := log.SubLogger(s.logger, "strings")
261
262 strs := &avstrings.Strings{
263 Db: s.db,
264 OAuth: s.oauth,
265 Pages: s.pages,
266 IdResolver: s.idResolver,
267 Notifier: s.notifier,
268 Logger: logger,
269 }
270
271 return strs.Router(mw)
272}
273
274func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler {
275 issues := issues.New(
276 s.oauth,
277 s.repoResolver,
278 s.enforcer,
279 s.pages,
280 s.idResolver,
281 s.mentionsResolver,
282 s.db,
283 s.config,
284 s.notifier,
285 s.validator,
286 s.indexer.Issues,
287 log.SubLogger(s.logger, "issues"),
288 )
289 return issues.Router(mw)
290}
291
292func (s *State) DiscussionsRouter(mw *middleware.Middleware) http.Handler {
293 discussions := discussions.New(
294 s.oauth,
295 s.repoResolver,
296 s.enforcer,
297 s.pages,
298 s.idResolver,
299 s.mentionsResolver,
300 s.db,
301 s.config,
302 s.notifier,
303 s.validator,
304 log.SubLogger(s.logger, "discussions"),
305 )
306 return discussions.Router(mw)
307}
308
309func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler {
310 pulls := pulls.New(
311 s.oauth,
312 s.repoResolver,
313 s.pages,
314 s.idResolver,
315 s.mentionsResolver,
316 s.db,
317 s.config,
318 s.notifier,
319 s.enforcer,
320 s.validator,
321 s.indexer.Pulls,
322 log.SubLogger(s.logger, "pulls"),
323 )
324 return pulls.Router(mw)
325}
326
327func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler {
328 repo := repo.New(
329 s.oauth,
330 s.repoResolver,
331 s.pages,
332 s.spindlestream,
333 s.idResolver,
334 s.db,
335 s.config,
336 s.notifier,
337 s.enforcer,
338 log.SubLogger(s.logger, "repo"),
339 s.validator,
340 s.cfClient,
341 )
342 return repo.Router(mw)
343}
344
345func (s *State) PipelinesRouter(mw *middleware.Middleware) http.Handler {
346 pipes := pipelines.New(
347 s.oauth,
348 s.repoResolver,
349 s.pages,
350 s.spindlestream,
351 s.idResolver,
352 s.db,
353 s.config,
354 s.enforcer,
355 log.SubLogger(s.logger, "pipelines"),
356 )
357 return pipes.Router(mw)
358}
359
360func (s *State) LabelsRouter() http.Handler {
361 ls := labels.New(
362 s.oauth,
363 s.pages,
364 s.db,
365 s.validator,
366 s.enforcer,
367 s.notifier,
368 log.SubLogger(s.logger, "labels"),
369 )
370 return ls.Router()
371}
372
373func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler {
374 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications"))
375 return notifs.Router(mw)
376}
377
378func (s *State) SignupRouter() http.Handler {
379 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup"))
380 return sig.Router()
381}