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