Monorepo for Tangled tangled.org
5

Configure Feed

Select the types of activity you want to include in your feed.

1package state 2 3import ( 4 "context" 5 "database/sql" 6 "errors" 7 "net/http" 8 "strings" 9 10 "github.com/go-chi/chi/v5" 11 "tangled.org/core/appview/db" 12 "tangled.org/core/appview/issues" 13 "tangled.org/core/appview/knots" 14 "tangled.org/core/appview/labels" 15 "tangled.org/core/appview/metrics" 16 "tangled.org/core/appview/middleware" 17 "tangled.org/core/appview/migration" 18 "tangled.org/core/appview/notifications" 19 "tangled.org/core/appview/pipelines" 20 "tangled.org/core/appview/pulls" 21 "tangled.org/core/appview/repo" 22 "tangled.org/core/appview/settings" 23 "tangled.org/core/appview/signup" 24 "tangled.org/core/appview/spindles" 25 "tangled.org/core/appview/state/userutil" 26 avstrings "tangled.org/core/appview/strings" 27 "tangled.org/core/log" 28) 29 30func (s *State) Router() http.Handler { 31 router := chi.NewRouter() 32 middleware := middleware.New( 33 s.oauth, 34 s.db, 35 s.enforcer, 36 s.repoResolver, 37 s.idResolver, 38 s.pages, 39 s.rdb, 40 s.logger, 41 ) 42 43 router.Use(metrics.Middleware) 44 45 if err := db.ReapStaleRunningMigrations(context.Background(), s.db); err != nil { 46 s.logger.Warn("failed to reap stale running migrations", "err", err) 47 } 48 m := migration.NewMigration(s.db, s.oauth, s.idResolver.Directory(), s.logger) 49 router.Use(m.BackgroundMigrationMiddleware) 50 51 router.Get("/pwa-manifest.json", s.WebAppManifest) 52 router.Get("/robots.txt", s.RobotsTxt) 53 router.Get("/.well-known/security.txt", s.SecurityTxt) 54 55 userRouter := s.UserRouter(&middleware) 56 standardRouter := s.StandardRouter(&middleware) 57 58 router.HandleFunc("/*", func(w http.ResponseWriter, r *http.Request) { 59 pat := chi.URLParam(r, "*") 60 pathParts := strings.SplitN(pat, "/", 2) 61 62 if len(pathParts) > 0 { 63 firstPart := pathParts[0] 64 65 if userutil.IsDid(firstPart) { 66 repo, err := db.GetRepoByDid(s.db, firstPart) 67 switch { 68 case err == nil: 69 remaining := "" 70 if len(pathParts) > 1 { 71 remaining = "/" + pathParts[1] 72 } 73 rewritten := "/" + repo.Did + "/" + repo.Rkey + remaining 74 r2 := r.Clone(r.Context()) 75 r2.URL.Path = rewritten 76 r2.URL.RawPath = rewritten 77 userRouter.ServeHTTP(w, r2) 78 case errors.Is(err, sql.ErrNoRows): 79 userRouter.ServeHTTP(w, r) 80 default: 81 s.logger.Error("db error looking up repo DID", "repoDid", firstPart, "err", err) 82 http.Error(w, "internal server error", http.StatusInternalServerError) 83 } 84 return 85 } 86 87 if userutil.IsHandle(firstPart) { 88 userRouter.ServeHTTP(w, r) 89 return 90 } 91 92 // if using a flattened DID (like you would in go modules), unflatten 93 if userutil.IsFlattenedDid(firstPart) { 94 unflattenedDid := userutil.UnflattenDid(firstPart) 95 redirectPath := strings.Join(append([]string{unflattenedDid}, pathParts[1:]...), "/") 96 97 redirectURL := *r.URL 98 redirectURL.Path = "/" + redirectPath 99 100 http.Redirect(w, r, redirectURL.String(), http.StatusFound) 101 return 102 } 103 104 // if using a handle with @, rewrite to work without @ 105 if normalized := strings.TrimPrefix(firstPart, "@"); userutil.IsHandle(normalized) { 106 redirectPath := strings.Join(append([]string{normalized}, pathParts[1:]...), "/") 107 108 redirectURL := *r.URL 109 redirectURL.Path = "/" + redirectPath 110 111 http.Redirect(w, r, redirectURL.String(), http.StatusFound) 112 return 113 } 114 115 } 116 117 standardRouter.ServeHTTP(w, r) 118 }) 119 120 return router 121} 122 123func (s *State) UserRouter(mw *middleware.Middleware) http.Handler { 124 r := chi.NewRouter() 125 126 r.With(mw.ResolveIdent()).Route("/{user}", func(r chi.Router) { 127 r.Get("/", s.Profile) 128 r.Get("/feed.atom", s.AtomFeedPage) 129 130 r.With(mw.ResolveRepo()).Route("/{repo}", func(r chi.Router) { 131 r.Use(mw.GoImport()) 132 133 // These routes get proxied to the knot 134 r.Get("/info/refs", s.InfoRefs) 135 r.Post("/git-upload-archive", s.UploadArchive) 136 r.Post("/git-upload-pack", s.UploadPack) 137 r.Post("/git-receive-pack", s.ReceivePack) 138 139 r.Group(func(r chi.Router) { 140 r.Use(mw.CanonicalizeRepoURL()) 141 r.Mount("/issues", s.IssuesRouter(mw)) 142 r.Mount("/pulls", s.PullsRouter(mw)) 143 r.Mount("/pipelines", s.PipelinesRouter(mw)) 144 r.Mount("/labels", s.LabelsRouter()) 145 r.Mount("/", s.RepoRouter(mw)) 146 }) 147 }) 148 }) 149 150 r.NotFound(func(w http.ResponseWriter, r *http.Request) { 151 w.WriteHeader(http.StatusNotFound) 152 s.pages.Error404(w) 153 }) 154 155 return r 156} 157 158func (s *State) StandardRouter(mw *middleware.Middleware) http.Handler { 159 r := chi.NewRouter() 160 161 r.Handle("/static/*", s.pages.Static()) 162 163 r.Get("/", s.HomeOrTimeline) 164 r.Get("/home", s.Home) 165 r.Get("/timeline", s.Timeline) 166 r.Get("/upgradeBanner", s.UpgradeBanner) 167 r.Post("/newsletter/signup", s.NewsletterSignup) 168 r.Post("/newsletter/dismiss", s.NewsletterDismiss) 169 170 // special-case handler for serving tangled.org/core 171 r.Get("/core", s.Core()) 172 173 r.Get("/login", s.Login) 174 r.Post("/login", s.Login) 175 r.Post("/logout", s.Logout) 176 177 r.With(middleware.Paginate).Get("/search", s.Search) 178 r.With(middleware.AuthMiddleware(s.oauth)).Get("/search/quick", s.SearchQuick) 179 r.With(middleware.AuthMiddleware(s.oauth)).Get("/search/quick/mobile", s.SearchQuickMobile) 180 181 r.Post("/account/switch", s.SwitchAccount) 182 r.With(middleware.AuthMiddleware(s.oauth)).Delete("/account/{did}", s.RemoveAccount) 183 184 r.Route("/repo", func(r chi.Router) { 185 r.Route("/new", func(r chi.Router) { 186 r.Use(middleware.AuthMiddleware(s.oauth)) 187 r.Get("/", s.NewRepo) 188 r.Post("/", s.NewRepo) 189 }) 190 // r.Post("/import", s.ImportRepo) 191 }) 192 193 r.With(middleware.Paginate).Get("/goodfirstissues", s.GoodFirstIssues) 194 195 r.With(middleware.AuthMiddleware(s.oauth)).Route("/follow", func(r chi.Router) { 196 r.Post("/", s.Follow) 197 r.Delete("/", s.Follow) 198 }) 199 200 r.With(middleware.AuthMiddleware(s.oauth)).Route("/vouch", func(r chi.Router) { 201 r.Post("/", s.Vouch) 202 r.Post("/skip", s.SkipVouchSuggestion) 203 }) 204 205 r.With(middleware.AuthMiddleware(s.oauth)).Route("/star", func(r chi.Router) { 206 r.Post("/", s.Star) 207 r.Delete("/", s.Star) 208 }) 209 210 r.With(middleware.AuthMiddleware(s.oauth)).Route("/react", func(r chi.Router) { 211 r.Post("/", s.React) 212 r.Delete("/", s.React) 213 }) 214 215 r.With(middleware.AuthMiddleware(s.oauth)).Route("/comment", func(r chi.Router) { 216 r.Get("/", s.CommentBodyFragment) 217 r.Get("/edit", s.EditCommentFragment) 218 r.Get("/reply", s.NewReplyCommentFragment) 219 r.Get("/reply/placeholder", s.ReplyPlaceholderFragment) 220 r.Post("/", s.NewComment) 221 r.Patch("/", s.EditComment) 222 r.Delete("/", s.DeleteComment) 223 }) 224 225 r.Get("/profile/popover", s.ProfilePopover) 226 227 r.Route("/profile", func(r chi.Router) { 228 r.Use(middleware.AuthMiddleware(s.oauth)) 229 r.Get("/edit-bio", s.EditBioFragment) 230 r.Get("/edit-pins", s.EditPinsFragment) 231 r.Post("/bio", s.UpdateProfileBio) 232 r.Post("/pins", s.UpdateProfilePins) 233 r.Post("/avatar", s.UploadProfileAvatar) 234 r.Delete("/avatar", s.RemoveProfileAvatar) 235 r.Post("/punchcard", s.UpdateProfilePunchcardSetting) 236 }) 237 238 r.Mount("/settings", s.SettingsRouter()) 239 r.Mount("/strings", s.StringsRouter(mw)) 240 241 r.Mount("/settings/knots", s.KnotsRouter()) 242 r.Mount("/settings/spindles", s.SpindlesRouter()) 243 244 r.Mount("/notifications", s.NotificationsRouter(mw)) 245 246 r.Mount("/signup", s.SignupRouter()) 247 r.Mount("/", s.oauth.Router()) 248 249 r.Get("/keys/{user}", s.Keys) 250 r.Get("/terms", s.TermsOfService) 251 r.Get("/privacy", s.PrivacyPolicy) 252 r.Get("/brand", s.Brand) 253 254 r.NotFound(func(w http.ResponseWriter, r *http.Request) { 255 w.WriteHeader(http.StatusNotFound) 256 s.pages.Error404(w) 257 }) 258 return r 259} 260 261// Core serves tangled.org/core go-import meta tags, and redirects 262// to the core repository if accessed normally. 263func (s *State) Core() http.HandlerFunc { 264 return func(w http.ResponseWriter, r *http.Request) { 265 if r.URL.Query().Get("go-get") == "1" { 266 w.Header().Set("Content-Type", "text/html") 267 w.Write([]byte(`<meta name="go-import" content="tangled.org/core git https://tangled.org/@tangled.org/core">`)) 268 return 269 } 270 271 http.Redirect(w, r, "/@tangled.org/core", http.StatusFound) 272 } 273} 274 275func (s *State) SettingsRouter() http.Handler { 276 settings := &settings.Settings{ 277 Db: s.db, 278 OAuth: s.oauth, 279 Pages: s.pages, 280 Config: s.config, 281 CfClient: s.cfClient, 282 Logger: log.SubLogger(s.logger, "settings"), 283 IdResolver: s.idResolver, 284 } 285 286 return settings.Router() 287} 288 289func (s *State) SpindlesRouter() http.Handler { 290 logger := log.SubLogger(s.logger, "spindles") 291 292 spindles := &spindles.Spindles{ 293 Db: s.db, 294 OAuth: s.oauth, 295 Pages: s.pages, 296 Config: s.config, 297 Enforcer: s.enforcer, 298 IdResolver: s.idResolver, 299 Logger: logger, 300 } 301 302 return spindles.Router() 303} 304 305func (s *State) KnotsRouter() http.Handler { 306 logger := log.SubLogger(s.logger, "knots") 307 308 knots := &knots.Knots{ 309 Db: s.db, 310 OAuth: s.oauth, 311 Pages: s.pages, 312 Config: s.config, 313 Enforcer: s.enforcer, 314 IdResolver: s.idResolver, 315 Knotstream: s.knotstream, 316 Logger: logger, 317 } 318 319 return knots.Router() 320} 321 322func (s *State) StringsRouter(mw *middleware.Middleware) http.Handler { 323 logger := log.SubLogger(s.logger, "strings") 324 325 strs := &avstrings.Strings{ 326 Db: s.db, 327 OAuth: s.oauth, 328 Pages: s.pages, 329 IdResolver: s.idResolver, 330 Notifier: s.notifier, 331 Logger: logger, 332 } 333 334 return strs.Router(mw) 335} 336 337func (s *State) IssuesRouter(mw *middleware.Middleware) http.Handler { 338 issues := issues.New( 339 s.oauth, 340 s.repoResolver, 341 s.enforcer, 342 s.pages, 343 s.idResolver, 344 s.mentionsResolver, 345 s.db, 346 s.config, 347 s.notifier, 348 s.validator, 349 s.indexer.Issues, 350 log.SubLogger(s.logger, "issues"), 351 ) 352 return issues.Router(mw) 353} 354 355func (s *State) PullsRouter(mw *middleware.Middleware) http.Handler { 356 pulls := pulls.New( 357 s.oauth, 358 s.repoResolver, 359 s.pages, 360 s.idResolver, 361 s.mentionsResolver, 362 s.db, 363 s.config, 364 s.notifier, 365 s.enforcer, 366 s.validator, 367 s.indexer.Pulls, 368 log.SubLogger(s.logger, "pulls"), 369 ) 370 return pulls.Router(mw) 371} 372 373func (s *State) RepoRouter(mw *middleware.Middleware) http.Handler { 374 repo := repo.New( 375 s.oauth, 376 s.repoResolver, 377 s.pages, 378 s.spindlestream, 379 s.idResolver, 380 s.db, 381 s.config, 382 s.notifier, 383 s.enforcer, 384 log.SubLogger(s.logger, "repo"), 385 s.validator, 386 s.cfClient, 387 ) 388 return repo.Router(mw) 389} 390 391func (s *State) PipelinesRouter(mw *middleware.Middleware) http.Handler { 392 pipes := pipelines.New( 393 s.oauth, 394 s.repoResolver, 395 s.pages, 396 s.spindlestream, 397 s.pipelineNotifier, 398 s.idResolver, 399 s.db, 400 s.config, 401 s.enforcer, 402 log.SubLogger(s.logger, "pipelines"), 403 ) 404 return pipes.Router(mw) 405} 406 407func (s *State) LabelsRouter() http.Handler { 408 ls := labels.New( 409 s.oauth, 410 s.pages, 411 s.db, 412 s.validator, 413 s.enforcer, 414 s.notifier, 415 log.SubLogger(s.logger, "labels"), 416 ) 417 return ls.Router() 418} 419 420func (s *State) NotificationsRouter(mw *middleware.Middleware) http.Handler { 421 notifs := notifications.New(s.db, s.oauth, s.pages, log.SubLogger(s.logger, "notifications")) 422 return notifs.Router(mw) 423} 424 425func (s *State) SignupRouter() http.Handler { 426 sig := signup.New(s.config, s.db, s.posthog, s.idResolver, s.pages, log.SubLogger(s.logger, "signup")) 427 return sig.Router() 428}