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