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/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}