A vibe coded tangled fork which supports pijul.
at master 381 lines 9.3 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/discussions" 9 "tangled.org/core/appview/issues" 10 "tangled.org/core/appview/knots" 11 "tangled.org/core/appview/labels" 12 "tangled.org/core/appview/middleware" 13 "tangled.org/core/appview/notifications" 14 "tangled.org/core/appview/pipelines" 15 "tangled.org/core/appview/pulls" 16 "tangled.org/core/appview/repo" 17 "tangled.org/core/appview/settings" 18 "tangled.org/core/appview/signup" 19 "tangled.org/core/appview/spindles" 20 "tangled.org/core/appview/state/userutil" 21 avstrings "tangled.org/core/appview/strings" 22 "tangled.org/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("/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("/discussions", s.DiscussionsRouter(mw)) 97 r.Mount("/issues", s.IssuesRouter(mw)) 98 r.Mount("/pulls", s.PullsRouter(mw)) 99 r.Mount("/pipelines", s.PipelinesRouter(mw)) 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("/home", s.Home) 126 r.Get("/timeline", s.Timeline) 127 r.Get("/upgradeBanner", s.UpgradeBanner) 128 129 // special-case handler for serving tangled.org/core 130 r.Get("/core", s.Core()) 131 132 r.Get("/login", s.Login) 133 r.Post("/login", s.Login) 134 r.Post("/logout", s.Logout) 135 136 r.Post("/account/switch", s.SwitchAccount) 137 r.With(middleware.AuthMiddleware(s.oauth)).Delete("/account/{did}", s.RemoveAccount) 138 139 r.Route("/repo", func(r chi.Router) { 140 r.Route("/new", func(r chi.Router) { 141 r.Use(middleware.AuthMiddleware(s.oauth)) 142 r.Get("/", s.NewRepo) 143 r.Post("/", s.NewRepo) 144 }) 145 // r.Post("/import", s.ImportRepo) 146 }) 147 148 r.With(middleware.Paginate).Get("/goodfirstissues", s.GoodFirstIssues) 149 150 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) { 151 r.Post("/", s.Follow) 152 r.Delete("/", s.Follow) 153 }) 154 155 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) { 156 r.Post("/", s.Star) 157 r.Delete("/", s.Star) 158 }) 159 160 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) { 161 r.Post("/", s.React) 162 r.Delete("/", s.React) 163 }) 164 165 r.Route("/profile", func(r chi.Router) { 166 r.Use(middleware.AuthMiddleware(s.oauth)) 167 r.Get("/edit-bio", s.EditBioFragment) 168 r.Get("/edit-pins", s.EditPinsFragment) 169 r.Post("/bio", s.UpdateProfileBio) 170 r.Post("/pins", s.UpdateProfilePins) 171 r.Post("/avatar", s.UploadProfileAvatar) 172 r.Delete("/avatar", s.RemoveProfileAvatar) 173 r.Post("/punchcard", s.UpdateProfilePunchcardSetting) 174 }) 175 176 r.Mount("/settings", s.SettingsRouter()) 177 r.Mount("/strings", s.StringsRouter(mw)) 178 179 r.Mount("/settings/knots", s.KnotsRouter()) 180 r.Mount("/settings/spindles", s.SpindlesRouter()) 181 182 r.Mount("/notifications", s.NotificationsRouter(mw)) 183 184 r.Mount("/signup", s.SignupRouter()) 185 r.Mount("/", s.oauth.Router()) 186 187 r.Get("/keys/{user}", s.Keys) 188 r.Get("/terms", s.TermsOfService) 189 r.Get("/privacy", s.PrivacyPolicy) 190 r.Get("/brand", s.Brand) 191 192 r.NotFound(func(w http.ResponseWriter, r *http.Request) { 193 w.WriteHeader(http.StatusNotFound) 194 s.pages.Error404(w) 195 }) 196 return r 197} 198 199// Core serves tangled.org/core go-import meta tags, and redirects 200// to the core repository if accessed normally. 201func (s *State) Core() http.HandlerFunc { 202 return func(w http.ResponseWriter, r *http.Request) { 203 if r.URL.Query().Get("go-get") == "1" { 204 w.Header().Set("Content-Type", "text/html") 205 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`)) 206 return 207 } 208 209 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound) 210 } 211} 212 213func (s *State) SettingsRouter() http.Handler { 214 settings := &settings.Settings{ 215 Db: s.db, 216 OAuth: s.oauth, 217 Pages: s.pages, 218 Config: s.config, 219 CfClient: s.cfClient, 220 Logger: log.SubLogger(s.logger, "settings"), 221 } 222 223 return settings.Router() 224} 225 226func (s *State) SpindlesRouter() http.Handler { 227 logger := log.SubLogger(s.logger, "spindles") 228 229 spindles := &spindles.Spindles{ 230 Db: s.db, 231 OAuth: s.oauth, 232 Pages: s.pages, 233 Config: s.config, 234 Enforcer: s.enforcer, 235 IdResolver: s.idResolver, 236 Logger: logger, 237 } 238 239 return spindles.Router() 240} 241 242func (s *State) KnotsRouter() http.Handler { 243 logger := log.SubLogger(s.logger, "knots") 244 245 knots := &knots.Knots{ 246 Db: s.db, 247 OAuth: s.oauth, 248 Pages: s.pages, 249 Config: s.config, 250 Enforcer: s.enforcer, 251 IdResolver: s.idResolver, 252 Knotstream: s.knotstream, 253 Logger: logger, 254 } 255 256 return knots.Router() 257} 258 259func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler { 260 logger := log.SubLogger(s.logger, "strings") 261 262 strs := &avstrings.Strings{ 263 Db: s.db, 264 OAuth: s.oauth, 265 Pages: s.pages, 266 IdResolver: s.idResolver, 267 Notifier: s.notifier, 268 Logger: logger, 269 } 270 271 return strs.Router(mw) 272} 273 274func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler { 275 issues := issues.New( 276 s.oauth, 277 s.repoResolver, 278 s.enforcer, 279 s.pages, 280 s.idResolver, 281 s.mentionsResolver, 282 s.db, 283 s.config, 284 s.notifier, 285 s.validator, 286 s.indexer.Issues, 287 log.SubLogger(s.logger, "issues"), 288 ) 289 return issues.Router(mw) 290} 291 292func (s *State) DiscussionsRouter(mw *middleware.Middleware) http.Handler { 293 discussions := discussions.New( 294 s.oauth, 295 s.repoResolver, 296 s.enforcer, 297 s.pages, 298 s.idResolver, 299 s.mentionsResolver, 300 s.db, 301 s.config, 302 s.notifier, 303 s.validator, 304 log.SubLogger(s.logger, "discussions"), 305 ) 306 return discussions.Router(mw) 307} 308 309func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler { 310 pulls := pulls.New( 311 s.oauth, 312 s.repoResolver, 313 s.pages, 314 s.idResolver, 315 s.mentionsResolver, 316 s.db, 317 s.config, 318 s.notifier, 319 s.enforcer, 320 s.validator, 321 s.indexer.Pulls, 322 log.SubLogger(s.logger, "pulls"), 323 ) 324 return pulls.Router(mw) 325} 326 327func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler { 328 repo := repo.New( 329 s.oauth, 330 s.repoResolver, 331 s.pages, 332 s.spindlestream, 333 s.idResolver, 334 s.db, 335 s.config, 336 s.notifier, 337 s.enforcer, 338 log.SubLogger(s.logger, "repo"), 339 s.validator, 340 s.cfClient, 341 ) 342 return repo.Router(mw) 343} 344 345func (s *State) PipelinesRouter(mw *middleware.Middleware) http.Handler { 346 pipes := pipelines.New( 347 s.oauth, 348 s.repoResolver, 349 s.pages, 350 s.spindlestream, 351 s.idResolver, 352 s.db, 353 s.config, 354 s.enforcer, 355 log.SubLogger(s.logger, "pipelines"), 356 ) 357 return pipes.Router(mw) 358} 359 360func (s *State) LabelsRouter() http.Handler { 361 ls := labels.New( 362 s.oauth, 363 s.pages, 364 s.db, 365 s.validator, 366 s.enforcer, 367 s.notifier, 368 log.SubLogger(s.logger, "labels"), 369 ) 370 return ls.Router() 371} 372 373func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler { 374 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications")) 375 return notifs.Router(mw) 376} 377 378func (s *State) SignupRouter() http.Handler { 379 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup")) 380 return sig.Router() 381}