Monorepo for Tangled
tangled.org
1package state
2
3import (
4 "context"
5 "database/sql"
6 "errors"
7 "net/http"
8 "strings"
9
10 "github.com/go-chi/chi/v5"
11 "tangled.org/core/appview/db"
12 "tangled.org/core/appview/issues"
13 "tangled.org/core/appview/knots"
14 "tangled.org/core/appview/labels"
15 "tangled.org/core/appview/metrics"
16 "tangled.org/core/appview/middleware"
17 "tangled.org/core/appview/migration"
18 "tangled.org/core/appview/notifications"
19 "tangled.org/core/appview/pipelines"
20 "tangled.org/core/appview/pulls"
21 "tangled.org/core/appview/repo"
22 "tangled.org/core/appview/settings"
23 "tangled.org/core/appview/signup"
24 "tangled.org/core/appview/spindles"
25 "tangled.org/core/appview/state/userutil"
26 avstrings "tangled.org/core/appview/strings"
27 "tangled.org/core/log"
28)
29
30func (s *State) Router() http.Handler {
31 router := chi.NewRouter()
32 middleware := middleware.New(
33 s.oauth,
34 s.db,
35 s.enforcer,
36 s.repoResolver,
37 s.idResolver,
38 s.pages,
39 s.rdb,
40 s.logger,
41 )
42
43 router.Use(metrics.Middleware)
44
45 if err := db.ReapStaleRunningMigrations(context.Background(), s.db); err != nil {
46 s.logger.Warn("failed to reap stale running migrations", "err", err)
47 }
48 m := migration.NewMigration(s.db, s.oauth, s.idResolver.Directory(), s.logger)
49 router.Use(m.BackgroundMigrationMiddleware)
50
51 router.Get("/pwa-manifest.json", s.WebAppManifest)
52 router.Get("/robots.txt", s.RobotsTxt)
53 router.Get("/.well-known/security.txt", s.SecurityTxt)
54
55 userRouter := s.UserRouter(&middleware)
56 standardRouter := s.StandardRouter(&middleware)
57
58 router.HandleFunc("/*", func(w http.ResponseWriter, r *http.Request) {
59 pat := chi.URLParam(r, "*")
60 pathParts := strings.SplitN(pat, "/", 2)
61
62 if len(pathParts) > 0 {
63 firstPart := pathParts[0]
64
65 if userutil.IsDid(firstPart) {
66 repo, err := db.GetRepoByDid(s.db, firstPart)
67 switch {
68 case err == nil:
69 remaining := ""
70 if len(pathParts) > 1 {
71 remaining = "/" + pathParts[1]
72 }
73 rewritten := "/" + repo.Did + "/" + repo.Rkey + remaining
74 r2 := r.Clone(r.Context())
75 r2.URL.Path = rewritten
76 r2.URL.RawPath = rewritten
77 userRouter.ServeHTTP(w, r2)
78 case errors.Is(err, sql.ErrNoRows):
79 userRouter.ServeHTTP(w, r)
80 default:
81 s.logger.Error("db error looking up repo DID", "repoDid", firstPart, "err", err)
82 http.Error(w, "internal server error", http.StatusInternalServerError)
83 }
84 return
85 }
86
87 if userutil.IsHandle(firstPart) {
88 userRouter.ServeHTTP(w, r)
89 return
90 }
91
92 // if using a flattened DID (like you would in go modules), unflatten
93 if userutil.IsFlattenedDid(firstPart) {
94 unflattenedDid := userutil.UnflattenDid(firstPart)
95 redirectPath := strings.Join(append([]string{unflattenedDid}, pathParts[1:]...), "/")
96
97 redirectURL := *r.URL
98 redirectURL.Path = "/" + redirectPath
99
100 http.Redirect(w, r, redirectURL.String(), http.StatusFound)
101 return
102 }
103
104 // if using a handle with @, rewrite to work without @
105 if normalized := strings.TrimPrefix(firstPart, "@"); userutil.IsHandle(normalized) {
106 redirectPath := strings.Join(append([]string{normalized}, pathParts[1:]...), "/")
107
108 redirectURL := *r.URL
109 redirectURL.Path = "/" + redirectPath
110
111 http.Redirect(w, r, redirectURL.String(), http.StatusFound)
112 return
113 }
114
115 }
116
117 standardRouter.ServeHTTP(w, r)
118 })
119
120 return router
121}
122
123func (s *State) UserRouter(mw *middleware.Middleware) http.Handler {
124 r := chi.NewRouter()
125
126 r.With(mw.ResolveIdent()).Route("/{user}", func(r chi.Router) {
127 r.Get("/", s.Profile)
128 r.Get("/feed.atom", s.AtomFeedPage)
129
130 r.With(mw.ResolveRepo()).Route("/{repo}", func(r chi.Router) {
131 r.Use(mw.GoImport())
132
133 // These routes get proxied to the knot
134 r.Get("/info/refs", s.InfoRefs)
135 r.Post("/git-upload-archive", s.UploadArchive)
136 r.Post("/git-upload-pack", s.UploadPack)
137 r.Post("/git-receive-pack", s.ReceivePack)
138
139 r.Group(func(r chi.Router) {
140 r.Use(mw.CanonicalizeRepoURL())
141 r.Mount("/issues", s.IssuesRouter(mw))
142 r.Mount("/pulls", s.PullsRouter(mw))
143 r.Mount("/pipelines", s.PipelinesRouter(mw))
144 r.Mount("/labels", s.LabelsRouter())
145 r.Mount("/", s.RepoRouter(mw))
146 })
147 })
148 })
149
150 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
151 w.WriteHeader(http.StatusNotFound)
152 s.pages.Error404(w)
153 })
154
155 return r
156}
157
158func (s *State) StandardRouter(mw *middleware.Middleware) http.Handler {
159 r := chi.NewRouter()
160
161 r.Handle("/static/*", s.pages.Static())
162
163 r.Get("/", s.HomeOrTimeline)
164 r.Get("/home", s.Home)
165 r.Get("/timeline", s.Timeline)
166 r.Get("/upgradeBanner", s.UpgradeBanner)
167 r.Post("/newsletter/signup", s.NewsletterSignup)
168 r.Post("/newsletter/dismiss", s.NewsletterDismiss)
169
170 // special-case handler for serving tangled.org/core
171 r.Get("/core", s.Core())
172
173 r.Get("/login", s.Login)
174 r.Post("/login", s.Login)
175 r.Post("/logout", s.Logout)
176
177 r.With(middleware.Paginate).Get("/search", s.Search)
178 r.With(middleware.AuthMiddleware(s.oauth)).Get("/search/quick", s.SearchQuick)
179 r.With(middleware.AuthMiddleware(s.oauth)).Get("/search/quick/mobile", s.SearchQuickMobile)
180
181 r.Post("/account/switch", s.SwitchAccount)
182 r.With(middleware.AuthMiddleware(s.oauth)).Delete("/account/{did}", s.RemoveAccount)
183
184 r.Route("/repo", func(r chi.Router) {
185 r.Route("/new", func(r chi.Router) {
186 r.Use(middleware.AuthMiddleware(s.oauth))
187 r.Get("/", s.NewRepo)
188 r.Post("/", s.NewRepo)
189 })
190 // r.Post("/import", s.ImportRepo)
191 })
192
193 r.With(middleware.Paginate).Get("/goodfirstissues", s.GoodFirstIssues)
194
195 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) {
196 r.Post("/", s.Follow)
197 r.Delete("/", s.Follow)
198 })
199
200 r.With(middleware.AuthMiddleware(s.oauth)).Route("/vouch", func(r chi.Router) {
201 r.Post("/", s.Vouch)
202 r.Post("/skip", s.SkipVouchSuggestion)
203 })
204
205 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) {
206 r.Post("/", s.Star)
207 r.Delete("/", s.Star)
208 })
209
210 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) {
211 r.Post("/", s.React)
212 r.Delete("/", s.React)
213 })
214
215 r.With(middleware.AuthMiddleware(s.oauth)).Route("/comment", func(r chi.Router) {
216 r.Get("/", s.CommentBodyFragment)
217 r.Get("/edit", s.EditCommentFragment)
218 r.Get("/reply", s.NewReplyCommentFragment)
219 r.Get("/reply/placeholder", s.ReplyPlaceholderFragment)
220 r.Post("/", s.NewComment)
221 r.Patch("/", s.EditComment)
222 r.Delete("/", s.DeleteComment)
223 })
224
225 r.Get("/profile/popover", s.ProfilePopover)
226
227 r.Route("/profile", func(r chi.Router) {
228 r.Use(middleware.AuthMiddleware(s.oauth))
229 r.Get("/edit-bio", s.EditBioFragment)
230 r.Get("/edit-pins", s.EditPinsFragment)
231 r.Post("/bio", s.UpdateProfileBio)
232 r.Post("/pins", s.UpdateProfilePins)
233 r.Post("/avatar", s.UploadProfileAvatar)
234 r.Delete("/avatar", s.RemoveProfileAvatar)
235 r.Post("/punchcard", s.UpdateProfilePunchcardSetting)
236 })
237
238 r.Mount("/settings", s.SettingsRouter())
239 r.Mount("/strings", s.StringsRouter(mw))
240
241 r.Mount("/settings/knots", s.KnotsRouter())
242 r.Mount("/settings/spindles", s.SpindlesRouter())
243
244 r.Mount("/notifications", s.NotificationsRouter(mw))
245
246 r.Mount("/signup", s.SignupRouter())
247 r.Mount("/", s.oauth.Router())
248
249 r.Get("/keys/{user}", s.Keys)
250 r.Get("/terms", s.TermsOfService)
251 r.Get("/privacy", s.PrivacyPolicy)
252 r.Get("/brand", s.Brand)
253
254 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
255 w.WriteHeader(http.StatusNotFound)
256 s.pages.Error404(w)
257 })
258 return r
259}
260
261// Core serves tangled.org/core go-import meta tags, and redirects
262// to the core repository if accessed normally.
263func (s *State) Core() http.HandlerFunc {
264 return func(w http.ResponseWriter, r *http.Request) {
265 if r.URL.Query().Get("go-get") == "1" {
266 w.Header().Set("Content-Type", "text/html")
267 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`))
268 return
269 }
270
271 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound)
272 }
273}
274
275func (s *State) SettingsRouter() http.Handler {
276 settings := &settings.Settings{
277 Db: s.db,
278 OAuth: s.oauth,
279 Pages: s.pages,
280 Config: s.config,
281 CfClient: s.cfClient,
282 Logger: log.SubLogger(s.logger, "settings"),
283 IdResolver: s.idResolver,
284 }
285
286 return settings.Router()
287}
288
289func (s *State) SpindlesRouter() http.Handler {
290 logger := log.SubLogger(s.logger, "spindles")
291
292 spindles := &spindles.Spindles{
293 Db: s.db,
294 OAuth: s.oauth,
295 Pages: s.pages,
296 Config: s.config,
297 Enforcer: s.enforcer,
298 IdResolver: s.idResolver,
299 Logger: logger,
300 }
301
302 return spindles.Router()
303}
304
305func (s *State) KnotsRouter() http.Handler {
306 logger := log.SubLogger(s.logger, "knots")
307
308 knots := &knots.Knots{
309 Db: s.db,
310 OAuth: s.oauth,
311 Pages: s.pages,
312 Config: s.config,
313 Enforcer: s.enforcer,
314 IdResolver: s.idResolver,
315 Knotstream: s.knotstream,
316 Logger: logger,
317 }
318
319 return knots.Router()
320}
321
322func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler {
323 logger := log.SubLogger(s.logger, "strings")
324
325 strs := &avstrings.Strings{
326 Db: s.db,
327 OAuth: s.oauth,
328 Pages: s.pages,
329 IdResolver: s.idResolver,
330 Notifier: s.notifier,
331 Logger: logger,
332 }
333
334 return strs.Router(mw)
335}
336
337func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler {
338 issues := issues.New(
339 s.oauth,
340 s.repoResolver,
341 s.enforcer,
342 s.pages,
343 s.idResolver,
344 s.mentionsResolver,
345 s.db,
346 s.config,
347 s.notifier,
348 s.validator,
349 s.indexer.Issues,
350 log.SubLogger(s.logger, "issues"),
351 )
352 return issues.Router(mw)
353}
354
355func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler {
356 pulls := pulls.New(
357 s.oauth,
358 s.repoResolver,
359 s.pages,
360 s.idResolver,
361 s.mentionsResolver,
362 s.db,
363 s.config,
364 s.notifier,
365 s.enforcer,
366 s.validator,
367 s.indexer.Pulls,
368 log.SubLogger(s.logger, "pulls"),
369 )
370 return pulls.Router(mw)
371}
372
373func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler {
374 repo := repo.New(
375 s.oauth,
376 s.repoResolver,
377 s.pages,
378 s.spindlestream,
379 s.idResolver,
380 s.db,
381 s.config,
382 s.notifier,
383 s.enforcer,
384 log.SubLogger(s.logger, "repo"),
385 s.validator,
386 s.cfClient,
387 )
388 return repo.Router(mw)
389}
390
391func (s *State) PipelinesRouter(mw *middleware.Middleware) http.Handler {
392 pipes := pipelines.New(
393 s.oauth,
394 s.repoResolver,
395 s.pages,
396 s.spindlestream,
397 s.pipelineNotifier,
398 s.idResolver,
399 s.db,
400 s.config,
401 s.enforcer,
402 log.SubLogger(s.logger, "pipelines"),
403 )
404 return pipes.Router(mw)
405}
406
407func (s *State) LabelsRouter() http.Handler {
408 ls := labels.New(
409 s.oauth,
410 s.pages,
411 s.db,
412 s.validator,
413 s.enforcer,
414 s.notifier,
415 log.SubLogger(s.logger, "labels"),
416 )
417 return ls.Router()
418}
419
420func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler {
421 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications"))
422 return notifs.Router(mw)
423}
424
425func (s *State) SignupRouter() http.Handler {
426 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup"))
427 return sig.Router()
428}