forked from tangled.org/core
Monorepo for Tangled — https://tangled.org
at master 8.5 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("/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}