forked from tangled.org/core
Monorepo for Tangled — https://tangled.org
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 http.Redirect(w, r, "/"+redirectPath, http.StatusFound) 61 return 62 } 63 64 // if using a handle with @, rewrite to work without @ 65 if normalized := strings.TrimPrefix(firstPart, "@"); userutil.IsHandle(normalized) { 66 redirectPath := strings.Join(append([]string{normalized}, pathParts[1:]...), "/") 67 http.Redirect(w, r, "/"+redirectPath, http.StatusFound) 68 return 69 } 70 } 71 72 standardRouter.ServeHTTP(w, r) 73 }) 74 75 return router 76} 77 78func (s *State) UserRouter(mw *middleware.Middleware) http.Handler { 79 r := chi.NewRouter() 80 81 r.With(mw.ResolveIdent()).Route("/{user}", func(r chi.Router) { 82 r.Get("/", s.Profile) 83 r.Get("/feed.atom", s.AtomFeedPage) 84 85 // redirect /@handle/repo.git -> /@handle/repo 86 r.Get("/{repo}.git", func(w http.ResponseWriter, r *http.Request) { 87 nonDotGitPath := strings.TrimSuffix(r.URL.Path, ".git") 88 http.Redirect(w, r, nonDotGitPath, http.StatusMovedPermanently) 89 }) 90 91 r.With(mw.ResolveRepo()).Route("/{repo}", func(r chi.Router) { 92 r.Use(mw.GoImport()) 93 r.Mount("/", s.RepoRouter(mw)) 94 r.Mount("/issues", s.IssuesRouter(mw)) 95 r.Mount("/pulls", s.PullsRouter(mw)) 96 r.Mount("/pipelines", s.PipelinesRouter()) 97 r.Mount("/labels", s.LabelsRouter()) 98 99 // These routes get proxied to the knot 100 r.Get("/info/refs", s.InfoRefs) 101 r.Post("/git-upload-pack", s.UploadPack) 102 r.Post("/git-receive-pack", s.ReceivePack) 103 // intentionally doesn't use /* as this isn't 104 // a file path 105 r.Get("/archive/{ref}", s.DownloadArchive) 106 }) 107 }) 108 109 r.NotFound(func(w http.ResponseWriter, r *http.Request) { 110 s.pages.Error404(w) 111 }) 112 113 return r 114} 115 116func (s *State) StandardRouter(mw *middleware.Middleware) http.Handler { 117 r := chi.NewRouter() 118 119 r.Handle("/static/*", s.pages.Static()) 120 121 r.Get("/", s.HomeOrTimeline) 122 r.Get("/timeline", s.Timeline) 123 r.Get("/upgradeBanner", s.UpgradeBanner) 124 125 // special-case handler for serving tangled.org/core 126 r.Get("/core", s.Core()) 127 128 r.Get("/login", s.Login) 129 r.Post("/login", s.Login) 130 r.Post("/logout", s.Logout) 131 132 r.Route("/repo", func(r chi.Router) { 133 r.Route("/new", func(r chi.Router) { 134 r.Use(middleware.AuthMiddleware(s.oauth)) 135 r.Get("/", s.NewRepo) 136 r.Post("/", s.NewRepo) 137 }) 138 // r.Post("/import", s.ImportRepo) 139 }) 140 141 r.Get("/goodfirstissues", s.GoodFirstIssues) 142 143 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) { 144 r.Post("/", s.Follow) 145 r.Delete("/", s.Follow) 146 }) 147 148 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) { 149 r.Post("/", s.Star) 150 r.Delete("/", s.Star) 151 }) 152 153 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) { 154 r.Post("/", s.React) 155 r.Delete("/", s.React) 156 }) 157 158 r.Route("/profile", func(r chi.Router) { 159 r.Use(middleware.AuthMiddleware(s.oauth)) 160 r.Get("/edit-bio", s.EditBioFragment) 161 r.Get("/edit-pins", s.EditPinsFragment) 162 r.Post("/bio", s.UpdateProfileBio) 163 r.Post("/pins", s.UpdateProfilePins) 164 }) 165 166 r.Mount("/settings", s.SettingsRouter()) 167 r.Mount("/strings", s.StringsRouter(mw)) 168 r.Mount("/knots", s.KnotsRouter()) 169 r.Mount("/spindles", s.SpindlesRouter()) 170 r.Mount("/notifications", s.NotificationsRouter(mw)) 171 172 r.Mount("/signup", s.SignupRouter()) 173 r.Mount("/", s.oauth.Router()) 174 175 r.Get("/keys/{user}", s.Keys) 176 r.Get("/terms", s.TermsOfService) 177 r.Get("/privacy", s.PrivacyPolicy) 178 r.Get("/brand", s.Brand) 179 180 r.NotFound(func(w http.ResponseWriter, r *http.Request) { 181 s.pages.Error404(w) 182 }) 183 return r 184} 185 186// Core serves tangled.org/core go-import meta tags, and redirects 187// to the core repository if accessed normally. 188func (s *State) Core() http.HandlerFunc { 189 return func(w http.ResponseWriter, r *http.Request) { 190 if r.URL.Query().Get("go-get") == "1" { 191 w.Header().Set("Content-Type", "text/html") 192 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`)) 193 return 194 } 195 196 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound) 197 } 198} 199 200func (s *State) SettingsRouter() http.Handler { 201 settings := &settings.Settings{ 202 Db: s.db, 203 OAuth: s.oauth, 204 Pages: s.pages, 205 Config: s.config, 206 } 207 208 return settings.Router() 209} 210 211func (s *State) SpindlesRouter() http.Handler { 212 logger := log.SubLogger(s.logger, "spindles") 213 214 spindles := &spindles.Spindles{ 215 Db: s.db, 216 OAuth: s.oauth, 217 Pages: s.pages, 218 Config: s.config, 219 Enforcer: s.enforcer, 220 IdResolver: s.idResolver, 221 Logger: logger, 222 } 223 224 return spindles.Router() 225} 226 227func (s *State) KnotsRouter() http.Handler { 228 logger := log.SubLogger(s.logger, "knots") 229 230 knots := &knots.Knots{ 231 Db: s.db, 232 OAuth: s.oauth, 233 Pages: s.pages, 234 Config: s.config, 235 Enforcer: s.enforcer, 236 IdResolver: s.idResolver, 237 Knotstream: s.knotstream, 238 Logger: logger, 239 } 240 241 return knots.Router() 242} 243 244func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler { 245 logger := log.SubLogger(s.logger, "strings") 246 247 strs := &avstrings.Strings{ 248 Db: s.db, 249 OAuth: s.oauth, 250 Pages: s.pages, 251 IdResolver: s.idResolver, 252 Notifier: s.notifier, 253 Logger: logger, 254 } 255 256 return strs.Router(mw) 257} 258 259func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler { 260 issues := issues.New( 261 s.oauth, 262 s.repoResolver, 263 s.pages, 264 s.idResolver, 265 s.db, 266 s.config, 267 s.notifier, 268 s.validator, 269 s.indexer.Issues, 270 log.SubLogger(s.logger, "issues"), 271 ) 272 return issues.Router(mw) 273} 274 275func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler { 276 pulls := pulls.New( 277 s.oauth, 278 s.repoResolver, 279 s.pages, 280 s.idResolver, 281 s.db, 282 s.config, 283 s.notifier, 284 s.enforcer, 285 s.validator, 286 s.indexer.Pulls, 287 log.SubLogger(s.logger, "pulls"), 288 ) 289 return pulls.Router(mw) 290} 291 292func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler { 293 repo := repo.New( 294 s.oauth, 295 s.repoResolver, 296 s.pages, 297 s.spindlestream, 298 s.idResolver, 299 s.db, 300 s.config, 301 s.notifier, 302 s.enforcer, 303 log.SubLogger(s.logger, "repo"), 304 s.validator, 305 ) 306 return repo.Router(mw) 307} 308 309func (s *State) PipelinesRouter() http.Handler { 310 pipes := pipelines.New( 311 s.oauth, 312 s.repoResolver, 313 s.pages, 314 s.spindlestream, 315 s.idResolver, 316 s.db, 317 s.config, 318 s.enforcer, 319 log.SubLogger(s.logger, "pipelines"), 320 ) 321 return pipes.Router() 322} 323 324func (s *State) LabelsRouter() http.Handler { 325 ls := labels.New( 326 s.oauth, 327 s.pages, 328 s.db, 329 s.validator, 330 s.enforcer, 331 log.SubLogger(s.logger, "labels"), 332 ) 333 return ls.Router() 334} 335 336func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler { 337 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications")) 338 return notifs.Router(mw) 339} 340 341func (s *State) SignupRouter() http.Handler { 342 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup")) 343 return sig.Router() 344}