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