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-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("/timeline", s.Timeline)
126 r.Get("/upgradeBanner", s.UpgradeBanner)
127
128 // special-case handler for serving tangled.org/core
129 r.Get("/core", s.Core())
130
131 r.Get("/login", s.Login)
132 r.Post("/login", s.Login)
133 r.Post("/logout", s.Logout)
134
135 r.Route("/repo", func(r chi.Router) {
136 r.Route("/new", func(r chi.Router) {
137 r.Use(middleware.AuthMiddleware(s.oauth))
138 r.Get("/", s.NewRepo)
139 r.Post("/", s.NewRepo)
140 })
141 // r.Post("/import", s.ImportRepo)
142 })
143
144 r.With(middleware.Paginate).Get("/goodfirstissues", s.GoodFirstIssues)
145
146 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) {
147 r.Post("/", s.Follow)
148 r.Delete("/", s.Follow)
149 })
150
151 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) {
152 r.Post("/", s.Star)
153 r.Delete("/", s.Star)
154 })
155
156 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) {
157 r.Post("/", s.React)
158 r.Delete("/", s.React)
159 })
160
161 r.Route("/profile", func(r chi.Router) {
162 r.Use(middleware.AuthMiddleware(s.oauth))
163 r.Get("/edit-bio", s.EditBioFragment)
164 r.Get("/edit-pins", s.EditPinsFragment)
165 r.Post("/bio", s.UpdateProfileBio)
166 r.Post("/pins", s.UpdateProfilePins)
167 })
168
169 r.Mount("/settings", s.SettingsRouter())
170 r.Mount("/strings", s.StringsRouter(mw))
171
172 r.Mount("/settings/knots", s.KnotsRouter())
173 r.Mount("/settings/spindles", s.SpindlesRouter())
174
175 r.Mount("/notifications", s.NotificationsRouter(mw))
176
177 r.Mount("/signup", s.SignupRouter())
178 r.Mount("/", s.oauth.Router())
179
180 r.Get("/keys/{user}", s.Keys)
181 r.Get("/terms", s.TermsOfService)
182 r.Get("/privacy", s.PrivacyPolicy)
183 r.Get("/brand", s.Brand)
184
185 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
186 w.WriteHeader(http.StatusNotFound)
187 s.pages.Error404(w)
188 })
189 return r
190}
191
192// Core serves tangled.org/core go-import meta tags, and redirects
193// to the core repository if accessed normally.
194func (s *State) Core() http.HandlerFunc {
195 return func(w http.ResponseWriter, r *http.Request) {
196 if r.URL.Query().Get("go-get") == "1" {
197 w.Header().Set("Content-Type", "text/html")
198 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`))
199 return
200 }
201
202 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound)
203 }
204}
205
206func (s *State) SettingsRouter() http.Handler {
207 settings := &settings.Settings{
208 Db: s.db,
209 OAuth: s.oauth,
210 Pages: s.pages,
211 Config: s.config,
212 }
213
214 return settings.Router()
215}
216
217func (s *State) SpindlesRouter() http.Handler {
218 logger := log.SubLogger(s.logger, "spindles")
219
220 spindles := &spindles.Spindles{
221 Db: s.db,
222 OAuth: s.oauth,
223 Pages: s.pages,
224 Config: s.config,
225 Enforcer: s.enforcer,
226 IdResolver: s.idResolver,
227 Logger: logger,
228 }
229
230 return spindles.Router()
231}
232
233func (s *State) KnotsRouter() http.Handler {
234 logger := log.SubLogger(s.logger, "knots")
235
236 knots := &knots.Knots{
237 Db: s.db,
238 OAuth: s.oauth,
239 Pages: s.pages,
240 Config: s.config,
241 Enforcer: s.enforcer,
242 IdResolver: s.idResolver,
243 Knotstream: s.knotstream,
244 Logger: logger,
245 }
246
247 return knots.Router()
248}
249
250func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler {
251 logger := log.SubLogger(s.logger, "strings")
252
253 strs := &avstrings.Strings{
254 Db: s.db,
255 OAuth: s.oauth,
256 Pages: s.pages,
257 IdResolver: s.idResolver,
258 Notifier: s.notifier,
259 Logger: logger,
260 }
261
262 return strs.Router(mw)
263}
264
265func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler {
266 issues := issues.New(
267 s.oauth,
268 s.repoResolver,
269 s.enforcer,
270 s.pages,
271 s.idResolver,
272 s.mentionsResolver,
273 s.db,
274 s.config,
275 s.notifier,
276 s.validator,
277 s.indexer.Issues,
278 log.SubLogger(s.logger, "issues"),
279 )
280 return issues.Router(mw)
281}
282
283func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler {
284 pulls := pulls.New(
285 s.oauth,
286 s.repoResolver,
287 s.pages,
288 s.idResolver,
289 s.mentionsResolver,
290 s.db,
291 s.config,
292 s.notifier,
293 s.enforcer,
294 s.validator,
295 s.indexer.Pulls,
296 log.SubLogger(s.logger, "pulls"),
297 )
298 return pulls.Router(mw)
299}
300
301func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler {
302 repo := repo.New(
303 s.oauth,
304 s.repoResolver,
305 s.pages,
306 s.spindlestream,
307 s.idResolver,
308 s.db,
309 s.config,
310 s.notifier,
311 s.enforcer,
312 log.SubLogger(s.logger, "repo"),
313 s.validator,
314 )
315 return repo.Router(mw)
316}
317
318func (s *State) PipelinesRouter() http.Handler {
319 pipes := pipelines.New(
320 s.oauth,
321 s.repoResolver,
322 s.pages,
323 s.spindlestream,
324 s.idResolver,
325 s.db,
326 s.config,
327 s.enforcer,
328 log.SubLogger(s.logger, "pipelines"),
329 )
330 return pipes.Router()
331}
332
333func (s *State) LabelsRouter() http.Handler {
334 ls := labels.New(
335 s.oauth,
336 s.pages,
337 s.db,
338 s.validator,
339 s.enforcer,
340 log.SubLogger(s.logger, "labels"),
341 )
342 return ls.Router()
343}
344
345func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler {
346 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications"))
347 return notifs.Router(mw)
348}
349
350func (s *State) SignupRouter() http.Handler {
351 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup"))
352 return sig.Router()
353}