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