this repo has no description
1package state
2
3import (
4 "net/http"
5 "strings"
6
7 "github.com/go-chi/chi/v5"
8 "github.com/gorilla/sessions"
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 oauthhandler "tangled.org/core/appview/oauth/handler"
15 "tangled.org/core/appview/pipelines"
16 "tangled.org/core/appview/pulls"
17 "tangled.org/core/appview/repo"
18 "tangled.org/core/appview/settings"
19 "tangled.org/core/appview/signup"
20 "tangled.org/core/appview/spindles"
21 "tangled.org/core/appview/state/userutil"
22 avstrings "tangled.org/core/appview/strings"
23 "tangled.org/core/log"
24)
25
26func (s *State) Router() http.Handler {
27 router := chi.NewRouter()
28 middleware := middleware.New(
29 s.oauth,
30 s.db,
31 s.enforcer,
32 s.repoResolver,
33 s.idResolver,
34 s.pages,
35 )
36
37 router.Get("/favicon.svg", s.Favicon)
38 router.Get("/favicon.ico", s.Favicon)
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 if strings.HasPrefix(pat, "did:") || strings.HasPrefix(pat, "@") {
46 userRouter.ServeHTTP(w, r)
47 } else {
48 // Check if the first path element is a valid handle without '@' or a flattened DID
49 pathParts := strings.SplitN(pat, "/", 2)
50 if len(pathParts) > 0 {
51 if userutil.IsHandleNoAt(pathParts[0]) {
52 // Redirect to the same path but with '@' prefixed to the handle
53 redirectPath := "@" + pat
54 http.Redirect(w, r, "/"+redirectPath, http.StatusFound)
55 return
56 } else if userutil.IsFlattenedDid(pathParts[0]) {
57 // Redirect to the unflattened DID version
58 unflattenedDid := userutil.UnflattenDid(pathParts[0])
59 var redirectPath string
60 if len(pathParts) > 1 {
61 redirectPath = unflattenedDid + "/" + pathParts[1]
62 } else {
63 redirectPath = unflattenedDid
64 }
65 http.Redirect(w, r, "/"+redirectPath, http.StatusFound)
66 return
67 }
68 }
69 standardRouter.ServeHTTP(w, r)
70 }
71 })
72
73 return router
74}
75
76func (s *State) UserRouter(mw *middleware.Middleware) http.Handler {
77 r := chi.NewRouter()
78
79 r.With(mw.ResolveIdent()).Route("/{user}", func(r chi.Router) {
80 r.Get("/", s.Profile)
81 r.Get("/feed.atom", s.AtomFeedPage)
82
83 // redirect /@handle/repo.git -> /@handle/repo
84 r.Get("/{repo}.git", func(w http.ResponseWriter, r *http.Request) {
85 nonDotGitPath := strings.TrimSuffix(r.URL.Path, ".git")
86 http.Redirect(w, r, nonDotGitPath, http.StatusMovedPermanently)
87 })
88
89 r.With(mw.ResolveRepo()).Route("/{repo}", func(r chi.Router) {
90 r.Use(mw.GoImport())
91 r.Mount("/", s.RepoRouter(mw))
92 r.Mount("/issues", s.IssuesRouter(mw))
93 r.Mount("/pulls", s.PullsRouter(mw))
94 r.Mount("/pipelines", s.PipelinesRouter(mw))
95 r.Mount("/labels", s.LabelsRouter(mw))
96
97 // These routes get proxied to the knot
98 r.Get("/info/refs", s.InfoRefs)
99 r.Post("/git-upload-pack", s.UploadPack)
100 r.Post("/git-receive-pack", s.ReceivePack)
101
102 })
103 })
104
105 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
106 s.pages.Error404(w)
107 })
108
109 return r
110}
111
112func (s *State) StandardRouter(mw *middleware.Middleware) http.Handler {
113 r := chi.NewRouter()
114
115 r.Handle("/static/*", s.pages.Static())
116
117 r.Get("/", s.HomeOrTimeline)
118 r.Get("/timeline", s.Timeline)
119 r.With(middleware.AuthMiddleware(s.oauth)).Get("/upgradeBanner", s.UpgradeBanner)
120
121 // special-case handler for serving tangled.org/core
122 r.Get("/core", s.Core())
123
124 r.Route("/repo", func(r chi.Router) {
125 r.Route("/new", func(r chi.Router) {
126 r.Use(middleware.AuthMiddleware(s.oauth))
127 r.Get("/", s.NewRepo)
128 r.Post("/", s.NewRepo)
129 })
130 // r.Post("/import", s.ImportRepo)
131 })
132
133 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) {
134 r.Post("/", s.Follow)
135 r.Delete("/", s.Follow)
136 })
137
138 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) {
139 r.Post("/", s.Star)
140 r.Delete("/", s.Star)
141 })
142
143 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) {
144 r.Post("/", s.React)
145 r.Delete("/", s.React)
146 })
147
148 r.Route("/profile", func(r chi.Router) {
149 r.Use(middleware.AuthMiddleware(s.oauth))
150 r.Get("/edit-bio", s.EditBioFragment)
151 r.Get("/edit-pins", s.EditPinsFragment)
152 r.Post("/bio", s.UpdateProfileBio)
153 r.Post("/pins", s.UpdateProfilePins)
154 })
155
156 r.Mount("/settings", s.SettingsRouter())
157 r.Mount("/strings", s.StringsRouter(mw))
158 r.Mount("/knots", s.KnotsRouter())
159 r.Mount("/spindles", s.SpindlesRouter())
160 r.Mount("/notifications", s.NotificationsRouter(mw))
161
162 r.Mount("/signup", s.SignupRouter())
163 r.Mount("/", s.OAuthRouter())
164
165 r.Get("/keys/{user}", s.Keys)
166 r.Get("/terms", s.TermsOfService)
167 r.Get("/privacy", s.PrivacyPolicy)
168
169 r.NotFound(func(w http.ResponseWriter, r *http.Request) {
170 s.pages.Error404(w)
171 })
172 return r
173}
174
175// Core serves tangled.org/core go-import meta tags, and redirects
176// to the core repository if accessed normally.
177func (s *State) Core() http.HandlerFunc {
178 return func(w http.ResponseWriter, r *http.Request) {
179 if r.URL.Query().Get("go-get") == "1" {
180 w.Header().Set("Content-Type", "text/html")
181 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`))
182 return
183 }
184
185 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound)
186 }
187}
188
189func (s *State) OAuthRouter() http.Handler {
190 store := sessions.NewCookieStore([]byte(s.config.Core.CookieSecret))
191 oauth := oauthhandler.New(s.config, s.pages, s.idResolver, s.db, s.sess, store, s.oauth, s.enforcer, s.posthog)
192 return oauth.Router()
193}
194
195func (s *State) SettingsRouter() http.Handler {
196 settings := &settings.Settings{
197 Db: s.db,
198 OAuth: s.oauth,
199 Pages: s.pages,
200 Config: s.config,
201 }
202
203 return settings.Router()
204}
205
206func (s *State) SpindlesRouter() http.Handler {
207 logger := log.New("spindles")
208
209 spindles := &spindles.Spindles{
210 Db: s.db,
211 OAuth: s.oauth,
212 Pages: s.pages,
213 Config: s.config,
214 Enforcer: s.enforcer,
215 IdResolver: s.idResolver,
216 Logger: logger,
217 }
218
219 return spindles.Router()
220}
221
222func (s *State) KnotsRouter() http.Handler {
223 logger := log.New("knots")
224
225 knots := &knots.Knots{
226 Db: s.db,
227 OAuth: s.oauth,
228 Pages: s.pages,
229 Config: s.config,
230 Enforcer: s.enforcer,
231 IdResolver: s.idResolver,
232 Knotstream: s.knotstream,
233 Logger: logger,
234 }
235
236 return knots.Router()
237}
238
239func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler {
240 logger := log.New("strings")
241
242 strs := &avstrings.Strings{
243 Db: s.db,
244 OAuth: s.oauth,
245 Pages: s.pages,
246 IdResolver: s.idResolver,
247 Notifier: s.notifier,
248 Logger: logger,
249 }
250
251 return strs.Router(mw)
252}
253
254func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler {
255 issues := issues.New(s.oauth, s.repoResolver, s.pages, s.idResolver, s.db, s.config, s.notifier, s.validator)
256 return issues.Router(mw)
257}
258
259func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler {
260 pulls := pulls.New(s.oauth, s.repoResolver, s.pages, s.idResolver, s.db, s.config, s.notifier)
261 return pulls.Router(mw)
262}
263
264func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler {
265 logger := log.New("repo")
266 repo := repo.New(s.oauth, s.repoResolver, s.pages, s.spindlestream, s.idResolver, s.db, s.config, s.notifier, s.enforcer, logger, s.validator)
267 return repo.Router(mw)
268}
269
270func (s *State) PipelinesRouter(mw *middleware.Middleware) http.Handler {
271 pipes := pipelines.New(s.oauth, s.repoResolver, s.pages, s.spindlestream, s.idResolver, s.db, s.config, s.enforcer)
272 return pipes.Router(mw)
273}
274
275func (s *State) LabelsRouter(mw *middleware.Middleware) http.Handler {
276 ls := labels.New(s.oauth, s.pages, s.db, s.validator)
277 return ls.Router(mw)
278}
279
280func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler {
281 notifs := notifications.New(s.db, s.oauth, s.pages)
282 return notifs.Router(mw)
283}
284
285func (s *State) SignupRouter() http.Handler {
286 logger := log.New("signup")
287
288 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, logger)
289 return sig.Router()
290}