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