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