Monorepo for Tangled
0
fork

Configure Feed

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

at default-knot 1150 lines 32 kB view raw
1package state 2 3import ( 4 "context" 5 "fmt" 6 "net/http" 7 "slices" 8 "strings" 9 "time" 10 11 comatproto "github.com/bluesky-social/indigo/api/atproto" 12 "github.com/bluesky-social/indigo/atproto/identity" 13 "github.com/bluesky-social/indigo/atproto/syntax" 14 lexutil "github.com/bluesky-social/indigo/lex/util" 15 "github.com/go-chi/chi/v5" 16 "github.com/gorilla/feeds" 17 "tangled.org/core/api/tangled" 18 "tangled.org/core/appview/cache" 19 "tangled.org/core/appview/db" 20 "tangled.org/core/appview/middleware" 21 "tangled.org/core/appview/models" 22 "tangled.org/core/appview/pages" 23 "tangled.org/core/appview/pagination" 24 "tangled.org/core/appview/searchquery" 25 "tangled.org/core/orm" 26 "tangled.org/core/xrpc" 27) 28 29func (s *State) Profile(w http.ResponseWriter, r *http.Request) { 30 tabVal := r.URL.Query().Get("tab") 31 switch tabVal { 32 case "repos": 33 middleware. 34 Paginate(http.HandlerFunc(s.reposPage)). 35 ServeHTTP(w, r) 36 case "followers": 37 s.followersPage(w, r) 38 case "following": 39 s.followingPage(w, r) 40 case "starred": 41 middleware. 42 Paginate(http.HandlerFunc(s.starredPage)). 43 ServeHTTP(w, r) 44 case "strings": 45 s.stringsPage(w, r) 46 default: 47 s.profileOverview(w, r) 48 } 49} 50 51func (s *State) profile(r *http.Request) (*pages.ProfileCard, error) { 52 didOrHandle := chi.URLParam(r, "user") 53 if didOrHandle == "" { 54 return nil, fmt.Errorf("empty DID or handle") 55 } 56 57 ident, ok := r.Context().Value("resolvedId").(identity.Identity) 58 if !ok { 59 return nil, fmt.Errorf("failed to resolve ID") 60 } 61 did := ident.DID.String() 62 63 profile, err := db.GetProfile(s.db, did) 64 if err != nil { 65 return nil, fmt.Errorf("failed to get profile: %w", err) 66 } 67 68 hasProfile := profile != nil 69 if !hasProfile { 70 profile = &models.Profile{Did: did} 71 } 72 73 repoCount, err := db.CountRepos(s.db, orm.FilterEq("did", did)) 74 if err != nil { 75 return nil, fmt.Errorf("failed to get repo count: %w", err) 76 } 77 78 stringCount, err := db.CountStrings(s.db, orm.FilterEq("did", did)) 79 if err != nil { 80 return nil, fmt.Errorf("failed to get string count: %w", err) 81 } 82 83 starredCount, err := db.CountStars(s.db, orm.FilterEq("did", did)) 84 if err != nil { 85 return nil, fmt.Errorf("failed to get starred repo count: %w", err) 86 } 87 88 followStats, err := db.GetFollowerFollowingCount(s.db, did) 89 if err != nil { 90 return nil, fmt.Errorf("failed to get follower stats: %w", err) 91 } 92 93 loggedInUser := s.oauth.GetMultiAccountUser(r) 94 followStatus := models.IsNotFollowing 95 var loggedInDid string 96 if loggedInUser != nil { 97 followStatus = db.GetFollowStatus(s.db, loggedInUser.Did, did) 98 loggedInDid = loggedInUser.Did 99 } 100 101 showPunchcard := s.shouldShowPunchcard(did, loggedInDid) 102 103 var punchcard *models.Punchcard 104 if showPunchcard { 105 now := time.Now() 106 startOfYear := time.Date(now.Year(), 1, 1, 0, 0, 0, 0, time.UTC) 107 punchcard, err = db.MakePunchcard( 108 s.db, 109 orm.FilterEq("did", did), 110 orm.FilterGte("date", startOfYear.Format(time.DateOnly)), 111 orm.FilterLte("date", now.Format(time.DateOnly)), 112 ) 113 if err != nil { 114 return nil, fmt.Errorf("failed to get punchcard for %s: %w", did, err) 115 } 116 } 117 118 return &pages.ProfileCard{ 119 UserDid: did, 120 HasProfile: hasProfile, 121 Profile: profile, 122 FollowStatus: followStatus, 123 Stats: pages.ProfileStats{ 124 RepoCount: repoCount, 125 StringCount: stringCount, 126 StarredCount: starredCount, 127 FollowersCount: followStats.Followers, 128 FollowingCount: followStats.Following, 129 }, 130 Punchcard: punchcard, 131 }, nil 132} 133 134func (s *State) profileOverview(w http.ResponseWriter, r *http.Request) { 135 l := s.logger.With("handler", "profileHomePage") 136 137 profile, err := s.profile(r) 138 if err != nil { 139 l.Error("failed to build profile card", "err", err) 140 s.pages.Error500(w) 141 return 142 } 143 l = l.With("profileDid", profile.UserDid) 144 145 repos, err := db.GetRepos( 146 s.db, 147 orm.FilterEq("did", profile.UserDid), 148 ) 149 if err != nil { 150 l.Error("failed to fetch repos", "err", err) 151 } 152 153 // filter out ones that are pinned 154 pinnedRepos := []models.Repo{} 155 for i, r := range repos { 156 if profile.Profile.MatchesPinnedRepo(r) { 157 pinnedRepos = append(pinnedRepos, r) 158 } else if profile.Profile.IsPinnedReposEmpty() && i < 4 { 159 pinnedRepos = append(pinnedRepos, r) 160 } 161 } 162 163 collaboratingRepos, err := db.CollaboratingIn(s.db, profile.UserDid) 164 if err != nil { 165 l.Error("failed to fetch collaborating repos", "err", err) 166 } 167 168 pinnedCollaboratingRepos := []models.Repo{} 169 for _, r := range collaboratingRepos { 170 if profile.Profile.MatchesPinnedRepo(r) { 171 pinnedCollaboratingRepos = append(pinnedCollaboratingRepos, r) 172 } 173 } 174 175 timeline, err := db.MakeProfileTimeline(s.db, profile.UserDid) 176 if err != nil { 177 l.Error("failed to create timeline", "err", err) 178 } 179 180 s.pages.ProfileOverview(w, pages.ProfileOverviewParams{ 181 LoggedInUser: s.oauth.GetMultiAccountUser(r), 182 Card: profile, 183 Repos: pinnedRepos, 184 CollaboratingRepos: pinnedCollaboratingRepos, 185 ProfileTimeline: timeline, 186 }) 187} 188 189func (s *State) shouldShowPunchcard(targetDid, requesterDid string) bool { 190 l := s.logger.With("helper", "shouldShowPunchcard") 191 192 targetPunchcardPreferences, err := db.GetPunchcardPreference(s.db, targetDid) 193 if err != nil { 194 l.Error("failed to get target users punchcard preferences", "err", err) 195 return true 196 } 197 198 requesterPunchcardPreferences, err := db.GetPunchcardPreference(s.db, requesterDid) 199 if err != nil { 200 l.Error("failed to get requester users punchcard preferences", "err", err) 201 return true 202 } 203 204 showPunchcard := true 205 206 // looking at their own profile 207 if targetDid == requesterDid { 208 if targetPunchcardPreferences.HideMine { 209 return false 210 } 211 return true 212 } 213 214 if targetPunchcardPreferences.HideMine || requesterPunchcardPreferences.HideOthers { 215 showPunchcard = false 216 } 217 return showPunchcard 218} 219 220func (s *State) reposPage(w http.ResponseWriter, r *http.Request) { 221 l := s.logger.With("handler", "reposPage") 222 223 profile, err := s.profile(r) 224 if err != nil { 225 l.Error("failed to build profile card", "err", err) 226 s.pages.Error500(w) 227 return 228 } 229 l = l.With("profileDid", profile.UserDid) 230 231 params := r.URL.Query() 232 page := pagination.FromContext(r.Context()) 233 234 query := searchquery.Parse(params.Get("q")) 235 236 var language string 237 if lang := query.Get("language"); lang != nil { 238 language = *lang 239 } 240 241 tf := searchquery.ExtractTextFilters(query) 242 243 searchOpts := models.RepoSearchOptions{ 244 Keywords: tf.Keywords, 245 Phrases: tf.Phrases, 246 NegatedKeywords: tf.NegatedKeywords, 247 NegatedPhrases: tf.NegatedPhrases, 248 Did: profile.UserDid, 249 Language: language, 250 Page: page, 251 } 252 253 var repos []models.Repo 254 var totalRepos int64 255 256 if searchOpts.HasSearchFilters() { 257 res, err := s.indexer.Repos.Search(r.Context(), searchOpts) 258 if err != nil { 259 l.Error("failed to search repos", "err", err) 260 s.pages.Error500(w) 261 return 262 } 263 264 if len(res.Hits) > 0 { 265 repos, err = db.GetRepos(s.db, orm.FilterIn("id", res.Hits)) 266 if err != nil { 267 l.Error("failed to get repos by IDs", "err", err) 268 s.pages.Error500(w) 269 return 270 } 271 272 // sort repos to match search result order (by relevance) 273 repoMap := make(map[int64]models.Repo, len(repos)) 274 for _, repo := range repos { 275 repoMap[repo.Id] = repo 276 } 277 repos = make([]models.Repo, 0, len(res.Hits)) 278 for _, id := range res.Hits { 279 if repo, ok := repoMap[id]; ok { 280 repos = append(repos, repo) 281 } 282 } 283 } 284 totalRepos = int64(res.Total) 285 } else { 286 repos, err = db.GetReposPaginated( 287 s.db, 288 page, 289 orm.FilterEq("did", profile.UserDid), 290 ) 291 if err != nil { 292 l.Error("failed to get repos", "err", err) 293 s.pages.Error500(w) 294 return 295 } 296 297 totalRepos, err = db.CountRepos( 298 s.db, 299 orm.FilterEq("did", profile.UserDid), 300 ) 301 if err != nil { 302 l.Error("failed to count repos", "err", err) 303 s.pages.Error500(w) 304 return 305 } 306 } 307 308 err = s.pages.ProfileRepos(w, pages.ProfileReposParams{ 309 LoggedInUser: s.oauth.GetMultiAccountUser(r), 310 Repos: repos, 311 Card: profile, 312 Page: page, 313 RepoCount: int(totalRepos), 314 FilterQuery: query.String(), 315 }) 316 if err != nil { 317 l.Error("failed to render page", "err", err) 318 } 319} 320 321func (s *State) starredPage(w http.ResponseWriter, r *http.Request) { 322 l := s.logger.With("handler", "starredPage") 323 324 page := pagination.FromContext(r.Context()) 325 l = l.With("page", page) 326 327 profile, err := s.profile(r) 328 if err != nil { 329 l.Error("failed to build profile card", "err", err) 330 s.pages.Error500(w) 331 return 332 } 333 l = l.With("profileDid", profile.UserDid) 334 335 stars, err := db.GetRepoStars(s.db, page, orm.FilterEq("did", profile.UserDid)) 336 if err != nil { 337 l.Error("failed to get stars", "err", err) 338 s.pages.Error500(w) 339 return 340 } 341 var repos []models.Repo 342 for _, s := range stars { 343 repos = append(repos, *s.Repo) 344 } 345 346 err = s.pages.ProfileStarred(w, pages.ProfileStarredParams{ 347 LoggedInUser: s.oauth.GetMultiAccountUser(r), 348 Repos: repos, 349 Total: int(profile.Stats.StarredCount), 350 Card: profile, 351 Page: page, 352 }) 353 if err != nil { 354 l.Error("failed to render", "err", err) 355 } 356} 357 358func (s *State) stringsPage(w http.ResponseWriter, r *http.Request) { 359 l := s.logger.With("handler", "stringsPage") 360 361 profile, err := s.profile(r) 362 if err != nil { 363 l.Error("failed to build profile card", "err", err) 364 s.pages.Error500(w) 365 return 366 } 367 l = l.With("profileDid", profile.UserDid) 368 369 strings, err := db.GetStrings(s.db, 0, orm.FilterEq("did", profile.UserDid)) 370 if err != nil { 371 l.Error("failed to get strings", "err", err) 372 s.pages.Error500(w) 373 return 374 } 375 376 err = s.pages.ProfileStrings(w, pages.ProfileStringsParams{ 377 LoggedInUser: s.oauth.GetMultiAccountUser(r), 378 Strings: strings, 379 Card: profile, 380 }) 381} 382 383type FollowsPageParams struct { 384 Follows []pages.FollowCard 385 Card *pages.ProfileCard 386} 387 388func (s *State) followPage( 389 r *http.Request, 390 fetchFollows func(db.Execer, string) ([]models.Follow, error), 391 extractDid func(models.Follow) string, 392) (*FollowsPageParams, error) { 393 l := s.logger.With("handler", "reposPage") 394 395 profile, err := s.profile(r) 396 if err != nil { 397 return nil, err 398 } 399 l = l.With("profileDid", profile.UserDid) 400 401 loggedInUser := s.oauth.GetMultiAccountUser(r) 402 params := FollowsPageParams{ 403 Card: profile, 404 } 405 406 follows, err := fetchFollows(s.db, profile.UserDid) 407 if err != nil { 408 l.Error("failed to fetch follows", "err", err) 409 return &params, err 410 } 411 412 if len(follows) == 0 { 413 return &params, nil 414 } 415 416 followDids := make([]string, 0, len(follows)) 417 for _, follow := range follows { 418 followDids = append(followDids, extractDid(follow)) 419 } 420 421 profiles, err := db.GetProfiles(s.db, orm.FilterIn("did", followDids)) 422 if err != nil { 423 l.Error("failed to get profiles", "followDids", followDids, "err", err) 424 return &params, err 425 } 426 427 followStatsMap, err := db.GetFollowerFollowingCounts(s.db, followDids) 428 if err != nil { 429 l.Error("getting follow counts", "followDids", followDids, "err", err) 430 } 431 432 loggedInUserFollowing := make(map[string]struct{}) 433 if loggedInUser != nil { 434 following, err := db.GetFollowing(s.db, loggedInUser.Did) 435 if err != nil { 436 l.Error("failed to get follow list", "err", err, "loggedInUser", loggedInUser.Did) 437 return &params, err 438 } 439 loggedInUserFollowing = make(map[string]struct{}, len(following)) 440 for _, follow := range following { 441 loggedInUserFollowing[follow.SubjectDid] = struct{}{} 442 } 443 } 444 445 followCards := make([]pages.FollowCard, len(follows)) 446 for i, did := range followDids { 447 followStats := followStatsMap[did] 448 followStatus := models.IsNotFollowing 449 if _, exists := loggedInUserFollowing[did]; exists { 450 followStatus = models.IsFollowing 451 } else if loggedInUser != nil && loggedInUser.Did == did { 452 followStatus = models.IsSelf 453 } 454 455 var profile *models.Profile 456 if p, exists := profiles[did]; exists { 457 profile = p 458 } else { 459 profile = &models.Profile{} 460 profile.Did = did 461 } 462 followCards[i] = pages.FollowCard{ 463 LoggedInUser: loggedInUser, 464 UserDid: did, 465 FollowStatus: followStatus, 466 FollowersCount: followStats.Followers, 467 FollowingCount: followStats.Following, 468 Profile: profile, 469 } 470 } 471 472 params.Follows = followCards 473 474 return &params, nil 475} 476 477func (s *State) followersPage(w http.ResponseWriter, r *http.Request) { 478 followPage, err := s.followPage(r, db.GetFollowers, func(f models.Follow) string { return f.UserDid }) 479 if err != nil { 480 s.pages.Notice(w, "all-followers", "Failed to load followers") 481 return 482 } 483 484 s.pages.ProfileFollowers(w, pages.ProfileFollowersParams{ 485 LoggedInUser: s.oauth.GetMultiAccountUser(r), 486 Followers: followPage.Follows, 487 Card: followPage.Card, 488 }) 489} 490 491func (s *State) followingPage(w http.ResponseWriter, r *http.Request) { 492 followPage, err := s.followPage(r, db.GetFollowing, func(f models.Follow) string { return f.SubjectDid }) 493 if err != nil { 494 s.pages.Notice(w, "all-following", "Failed to load following") 495 return 496 } 497 498 s.pages.ProfileFollowing(w, pages.ProfileFollowingParams{ 499 LoggedInUser: s.oauth.GetMultiAccountUser(r), 500 Following: followPage.Follows, 501 Card: followPage.Card, 502 }) 503} 504 505func (s *State) AtomFeedPage(w http.ResponseWriter, r *http.Request) { 506 ident, ok := r.Context().Value("resolvedId").(identity.Identity) 507 if !ok { 508 s.pages.Error404(w) 509 return 510 } 511 512 feed, err := s.getProfileFeed(r.Context(), &ident) 513 if err != nil { 514 s.pages.Error500(w) 515 return 516 } 517 518 if feed == nil { 519 return 520 } 521 522 atom, err := feed.ToAtom() 523 if err != nil { 524 s.pages.Error500(w) 525 return 526 } 527 528 w.Header().Set("content-type", "application/atom+xml") 529 w.Write([]byte(atom)) 530} 531 532func (s *State) getProfileFeed(ctx context.Context, id *identity.Identity) (*feeds.Feed, error) { 533 timeline, err := db.MakeProfileTimeline(s.db, id.DID.String()) 534 if err != nil { 535 return nil, err 536 } 537 538 author := &feeds.Author{ 539 Name: fmt.Sprintf("@%s", id.Handle), 540 } 541 542 feed := feeds.Feed{ 543 Title: fmt.Sprintf("%s's timeline", author.Name), 544 Link: &feeds.Link{Href: fmt.Sprintf("%s/@%s", s.config.Core.BaseUrl(), id.Handle), Type: "text/html", Rel: "alternate"}, 545 Items: make([]*feeds.Item, 0), 546 Updated: time.UnixMilli(0), 547 Author: author, 548 } 549 550 for _, byMonth := range timeline.ByMonth { 551 if err := s.addPullRequestItems(ctx, &feed, byMonth.PullEvents.Items, author); err != nil { 552 return nil, err 553 } 554 if err := s.addIssueItems(ctx, &feed, byMonth.IssueEvents.Items, author); err != nil { 555 return nil, err 556 } 557 if err := s.addRepoItems(ctx, &feed, byMonth.RepoEvents, author); err != nil { 558 return nil, err 559 } 560 } 561 562 slices.SortFunc(feed.Items, func(a *feeds.Item, b *feeds.Item) int { 563 return int(b.Created.UnixMilli()) - int(a.Created.UnixMilli()) 564 }) 565 566 if len(feed.Items) > 0 { 567 feed.Updated = feed.Items[0].Created 568 } 569 570 return &feed, nil 571} 572 573func (s *State) addPullRequestItems(ctx context.Context, feed *feeds.Feed, pulls []*models.Pull, author *feeds.Author) error { 574 for _, pull := range pulls { 575 owner, err := s.idResolver.ResolveIdent(ctx, pull.Repo.Did) 576 if err != nil { 577 return err 578 } 579 580 // Add pull request creation item 581 feed.Items = append(feed.Items, s.createPullRequestItem(pull, owner, author)) 582 } 583 return nil 584} 585 586func (s *State) addIssueItems(ctx context.Context, feed *feeds.Feed, issues []*models.Issue, author *feeds.Author) error { 587 for _, issue := range issues { 588 owner, err := s.idResolver.ResolveIdent(ctx, issue.Repo.Did) 589 if err != nil { 590 return err 591 } 592 593 feed.Items = append(feed.Items, s.createIssueItem(issue, owner, author)) 594 } 595 return nil 596} 597 598func (s *State) addRepoItems(ctx context.Context, feed *feeds.Feed, repos []models.RepoEvent, author *feeds.Author) error { 599 for _, repo := range repos { 600 item, err := s.createRepoItem(ctx, repo, author) 601 if err != nil { 602 return err 603 } 604 feed.Items = append(feed.Items, item) 605 } 606 return nil 607} 608 609func (s *State) createPullRequestItem(pull *models.Pull, owner *identity.Identity, author *feeds.Author) *feeds.Item { 610 return &feeds.Item{ 611 Title: fmt.Sprintf("%s created pull request '%s' in @%s/%s", author.Name, pull.Title, owner.Handle, pull.Repo.Name), 612 Link: &feeds.Link{Href: fmt.Sprintf("%s/@%s/%s/pulls/%d", s.config.Core.BaseUrl(), owner.Handle, pull.Repo.Name, pull.PullId), Type: "text/html", Rel: "alternate"}, 613 Created: pull.Created, 614 Author: author, 615 } 616} 617 618func (s *State) createIssueItem(issue *models.Issue, owner *identity.Identity, author *feeds.Author) *feeds.Item { 619 return &feeds.Item{ 620 Title: fmt.Sprintf("%s created issue '%s' in @%s/%s", author.Name, issue.Title, owner.Handle, issue.Repo.Name), 621 Link: &feeds.Link{Href: fmt.Sprintf("%s/@%s/%s/issues/%d", s.config.Core.BaseUrl(), owner.Handle, issue.Repo.Name, issue.IssueId), Type: "text/html", Rel: "alternate"}, 622 Created: issue.Created, 623 Author: author, 624 } 625} 626 627func (s *State) createRepoItem(ctx context.Context, repo models.RepoEvent, author *feeds.Author) (*feeds.Item, error) { 628 var title string 629 if repo.Source != nil { 630 sourceOwner, err := s.idResolver.ResolveIdent(ctx, repo.Source.Did) 631 if err != nil { 632 return nil, err 633 } 634 title = fmt.Sprintf("%s forked repository @%s/%s to '%s'", author.Name, sourceOwner.Handle, repo.Source.Name, repo.Repo.Name) 635 } else { 636 title = fmt.Sprintf("%s created repository '%s'", author.Name, repo.Repo.Name) 637 } 638 639 return &feeds.Item{ 640 Title: title, 641 Link: &feeds.Link{Href: fmt.Sprintf("%s/@%s/%s", s.config.Core.BaseUrl(), author.Name[1:], repo.Repo.Name), Type: "text/html", Rel: "alternate"}, // Remove @ prefix 642 Created: repo.Repo.Created, 643 Author: author, 644 }, nil 645} 646 647func (s *State) UpdateProfileBio(w http.ResponseWriter, r *http.Request) { 648 l := s.logger.With("handler", "UpdateProfileBio") 649 user := s.oauth.GetMultiAccountUser(r) 650 651 err := r.ParseForm() 652 if err != nil { 653 l.Error("invalid profile update form", "err", err) 654 s.pages.Notice(w, "update-profile", "Invalid form.") 655 return 656 } 657 658 profile, err := db.GetProfile(s.db, user.Did) 659 if err != nil { 660 l.Error("getting profile data", "did", user.Did, "err", err) 661 } 662 if profile == nil { 663 profile = &models.Profile{Did: user.Did} 664 } 665 666 profile.Description = r.FormValue("description") 667 profile.IncludeBluesky = r.FormValue("includeBluesky") == "on" 668 profile.Location = r.FormValue("location") 669 profile.Pronouns = r.FormValue("pronouns") 670 rawPreferredHandle := strings.TrimSpace(r.FormValue("preferredHandle")) 671 if rawPreferredHandle != "" { 672 h, err := syntax.ParseHandle(rawPreferredHandle) 673 if err != nil { 674 s.pages.Notice(w, "update-profile", "Invalid handle format.") 675 return 676 } 677 678 ident, err := s.idResolver.ResolveIdent(r.Context(), user.Did) 679 if err != nil || !slices.Contains(ident.AlsoKnownAs, "at://"+rawPreferredHandle) { 680 s.pages.Notice(w, "update-profile", "Handle not found in your DID document.") 681 return 682 } 683 profile.PreferredHandle = h 684 } else { 685 profile.PreferredHandle = "" 686 } 687 688 var links [5]string 689 for i := range 5 { 690 iLink := r.FormValue(fmt.Sprintf("link%d", i)) 691 links[i] = iLink 692 } 693 profile.Links = links 694 695 // Parse stats (exactly 2) 696 stat0 := r.FormValue("stat0") 697 stat1 := r.FormValue("stat1") 698 699 profile.Stats[0].Kind = models.ParseVanityStatKind(stat0) 700 profile.Stats[1].Kind = models.ParseVanityStatKind(stat1) 701 702 if err := db.ValidateProfile(s.db, profile); err != nil { 703 l.Error("invalid profile", "err", err) 704 s.pages.Notice(w, "update-profile", err.Error()) 705 return 706 } 707 708 s.updateProfile(profile, w, r) 709} 710 711func (s *State) UpdateProfilePins(w http.ResponseWriter, r *http.Request) { 712 l := s.logger.With("handler", "UpdateProfilePins") 713 user := s.oauth.GetMultiAccountUser(r) 714 715 err := r.ParseForm() 716 if err != nil { 717 l.Error("invalid profile update form", "err", err) 718 s.pages.Notice(w, "update-profile", "Invalid form.") 719 return 720 } 721 722 profile, err := db.GetProfile(s.db, user.Did) 723 if err != nil { 724 l.Error("getting profile data", "did", user.Did, "err", err) 725 } 726 if profile == nil { 727 profile = &models.Profile{Did: user.Did} 728 } 729 730 i := 0 731 var pinnedRepos [6]string 732 for key, values := range r.Form { 733 if i >= 6 { 734 l.Warn("too many pinned repos") 735 s.pages.Notice(w, "update-profile", "Only 6 repositories can be pinned at a time.") 736 return 737 } 738 if strings.HasPrefix(key, "pinnedRepo") && len(values) > 0 && values[0] != "" && i < 6 { 739 pinnedRepos[i] = values[0] 740 i++ 741 } 742 } 743 profile.PinnedRepos = pinnedRepos 744 745 s.updateProfile(profile, w, r) 746} 747 748func (s *State) UpdateDefaultKnotPreference(w http.ResponseWriter, r *http.Request) { 749 err := r.ParseForm() 750 if err != nil { 751 log.Println("invalid preference update form", err) 752 return 753 } 754 user := s.oauth.GetMultiAccountUser(r) 755 756 defaultKnot := r.Form.Get("default-knot") 757 758 if defaultKnot == "[[none]]" { // see pages/templates/knots/index.html for more info on why we use this value 759 defaultKnot = "" 760 } 761 762 err = db.UpsertKnotPreference(s.db, user.Did(), defaultKnot) 763 if err != nil { 764 log.Println("failed to update default knot preference", err) 765 return 766 } 767 768 s.pages.HxRefresh(w) 769} 770 771func (s *State) updateProfile(profile *models.Profile, w http.ResponseWriter, r *http.Request) { 772 l := s.logger.With("handler", "updateProfile") 773 user := s.oauth.GetMultiAccountUser(r) 774 775 client, err := s.oauth.AuthorizedClient(r) 776 if err != nil { 777 l.Error("failed to get authorized client", "err", err) 778 s.pages.Notice(w, "update-profile", "Failed to update profile, try again later.") 779 return 780 } 781 782 var pinnedRepoStrings []string 783 for _, r := range profile.PinnedRepos { 784 if r != "" { 785 pinnedRepoStrings = append(pinnedRepoStrings, r) 786 } 787 } 788 789 var vanityStats []string 790 for _, v := range profile.Stats { 791 vanityStats = append(vanityStats, string(v.Kind)) 792 } 793 794 ex, _ := comatproto.RepoGetRecord(r.Context(), client, "", tangled.ActorProfileNSID, user.Did, "self") 795 var cid *string 796 var existingAvatar *lexutil.LexBlob 797 if ex != nil { 798 cid = ex.Cid 799 if rec, ok := ex.Value.Val.(*tangled.ActorProfile); ok { 800 existingAvatar = rec.Avatar 801 } 802 } 803 804 _, err = comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ 805 Collection: tangled.ActorProfileNSID, 806 Repo: user.Did, 807 Rkey: "self", 808 Record: &lexutil.LexiconTypeDecoder{ 809 Val: &tangled.ActorProfile{ 810 Avatar: existingAvatar, 811 Bluesky: profile.IncludeBluesky, 812 Description: &profile.Description, 813 Links: profile.Links[:], 814 Location: &profile.Location, 815 PinnedRepositories: pinnedRepoStrings, 816 Stats: vanityStats[:], 817 Pronouns: &profile.Pronouns, 818 PreferredHandle: (*string)(&profile.PreferredHandle), 819 }}, 820 SwapRecord: cid, 821 }) 822 if err != nil { 823 l.Error("failed to update profile on PDS", "err", err) 824 s.pages.Notice(w, "update-profile", "Failed to update PDS, try again later.") 825 return 826 } 827 828 tx, err := s.db.BeginTx(r.Context(), nil) 829 if err != nil { 830 l.Error("failed to start transaction", "err", err) 831 s.pages.Notice(w, "update-profile", "Failed to update profile, try again later.") 832 return 833 } 834 835 if err := db.UpsertProfile(tx, profile); err != nil { 836 l.Error("failed to update profile in DB", "err", err) 837 s.pages.Notice(w, "update-profile", "Failed to update profile, try again later.") 838 return 839 } 840 841 if s.rdb != nil { 842 ctx := r.Context() 843 pipe := s.rdb.Pipeline() 844 didKey := fmt.Sprintf(cache.PreferredHandleByDid, profile.Did) 845 if profile.PreferredHandle != "" { 846 pipe.Set(ctx, didKey, string(profile.PreferredHandle), cache.PreferredHandleTTL) 847 pipe.Set(ctx, fmt.Sprintf(cache.PreferredHandleByHandle, string(profile.PreferredHandle)), profile.Did, cache.PreferredHandleTTL) 848 } else { 849 pipe.Del(ctx, didKey) 850 } 851 if _, execErr := pipe.Exec(ctx); execErr != nil { 852 l.Warn("failed to update preferred handle cache", "err", execErr) 853 } 854 } 855 856 s.notifier.UpdateProfile(r.Context(), profile) 857 858 s.pages.HxRedirect(w, "/"+user.Did) 859} 860 861func (s *State) EditBioFragment(w http.ResponseWriter, r *http.Request) { 862 l := s.logger.With("handler", "EditBioFragment") 863 user := s.oauth.GetMultiAccountUser(r) 864 865 profile, err := db.GetProfile(s.db, user.Did) 866 if err != nil { 867 l.Error("getting profile data", "did", user.Did, "err", err) 868 } 869 if profile == nil { 870 profile = &models.Profile{Did: user.Did} 871 } 872 873 var alsoKnownAs []string 874 ident, err := s.idResolver.ResolveIdent(r.Context(), user.Did) 875 if err == nil { 876 alsoKnownAs = ident.AlsoKnownAs 877 } 878 879 s.pages.EditBioFragment(w, pages.EditBioParams{ 880 LoggedInUser: user, 881 Profile: profile, 882 AlsoKnownAs: alsoKnownAs, 883 }) 884} 885 886func (s *State) EditPinsFragment(w http.ResponseWriter, r *http.Request) { 887 l := s.logger.With("handler", "EditPinsFragment") 888 user := s.oauth.GetMultiAccountUser(r) 889 890 profile, err := db.GetProfile(s.db, user.Did) 891 if err != nil { 892 l.Error("getting profile data", "did", user.Did, "err", err) 893 } 894 if profile == nil { 895 profile = &models.Profile{Did: user.Did} 896 } 897 898 repos, err := db.GetRepos(s.db, orm.FilterEq("did", user.Did)) 899 if err != nil { 900 l.Error("getting repos", "did", user.Did, "err", err) 901 } 902 903 collaboratingRepos, err := db.CollaboratingIn(s.db, user.Did) 904 if err != nil { 905 l.Error("getting collaborating repos", "did", user.Did, "err", err) 906 } 907 908 allRepos := []pages.PinnedRepo{} 909 910 for _, r := range repos { 911 allRepos = append(allRepos, pages.PinnedRepo{ 912 IsPinned: profile.MatchesPinnedRepo(r), 913 Repo: r, 914 }) 915 } 916 for _, r := range collaboratingRepos { 917 allRepos = append(allRepos, pages.PinnedRepo{ 918 IsPinned: profile.MatchesPinnedRepo(r), 919 Repo: r, 920 }) 921 } 922 923 s.pages.EditPinsFragment(w, pages.EditPinsParams{ 924 LoggedInUser: user, 925 Profile: profile, 926 AllRepos: allRepos, 927 }) 928} 929 930func (s *State) UploadProfileAvatar(w http.ResponseWriter, r *http.Request) { 931 l := s.logger.With("handler", "UploadProfileAvatar") 932 user := s.oauth.GetMultiAccountUser(r) 933 l = l.With("did", user.Did) 934 935 // Parse multipart form (10MB max) 936 if err := r.ParseMultipartForm(10 << 20); err != nil { 937 l.Error("failed to parse form", "err", err) 938 s.pages.Notice(w, "avatar-error", "Failed to parse form") 939 return 940 } 941 942 file, header, err := r.FormFile("avatar") 943 if err != nil { 944 l.Error("failed to read avatar file", "err", err) 945 s.pages.Notice(w, "avatar-error", "Failed to read avatar file") 946 return 947 } 948 defer file.Close() 949 950 if header.Size > 5000000 { 951 l.Warn("avatar file too large", "size", header.Size) 952 s.pages.Notice(w, "avatar-error", "Avatar file too large (max 5MB)") 953 return 954 } 955 956 contentType := header.Header.Get("Content-Type") 957 if contentType != "image/png" && contentType != "image/jpeg" { 958 l.Warn("invalid image type", "contentType", contentType) 959 s.pages.Notice(w, "avatar-error", "Invalid image type (only PNG and JPEG allowed)") 960 return 961 } 962 963 client, err := s.oauth.AuthorizedClient(r) 964 if err != nil { 965 l.Error("failed to get PDS client", "err", err) 966 s.pages.Notice(w, "avatar-error", "Failed to connect to your PDS") 967 return 968 } 969 970 uploadBlobResp, err := xrpc.RepoUploadBlob(r.Context(), client, file, header.Header.Get("Content-Type")) 971 if err != nil { 972 l.Error("failed to upload avatar blob", "err", err) 973 s.pages.Notice(w, "avatar-error", "Failed to upload avatar to your PDS") 974 return 975 } 976 977 l.Info("uploaded avatar blob", "cid", uploadBlobResp.Blob.Ref.String()) 978 979 // get current profile record from PDS to get its CID for swap 980 getRecordResp, err := comatproto.RepoGetRecord(r.Context(), client, "", tangled.ActorProfileNSID, user.Did, "self") 981 if err != nil { 982 l.Error("failed to get current profile record", "err", err) 983 s.pages.Notice(w, "avatar-error", "Failed to get current profile from your PDS") 984 return 985 } 986 987 var profileRecord *tangled.ActorProfile 988 if getRecordResp.Value != nil { 989 if val, ok := getRecordResp.Value.Val.(*tangled.ActorProfile); ok { 990 profileRecord = val 991 } else { 992 l.Warn("profile record type assertion failed, creating new record") 993 profileRecord = &tangled.ActorProfile{} 994 } 995 } else { 996 l.Warn("no existing profile record, creating new record") 997 profileRecord = &tangled.ActorProfile{} 998 } 999 1000 profileRecord.Avatar = uploadBlobResp.Blob 1001 1002 _, err = comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ 1003 Collection: tangled.ActorProfileNSID, 1004 Repo: user.Did, 1005 Rkey: "self", 1006 Record: &lexutil.LexiconTypeDecoder{Val: profileRecord}, 1007 SwapRecord: getRecordResp.Cid, 1008 }) 1009 1010 if err != nil { 1011 l.Error("failed to update profile record", "err", err) 1012 s.pages.Notice(w, "avatar-error", "Failed to update profile on your PDS") 1013 return 1014 } 1015 1016 l.Info("successfully updated profile with avatar") 1017 1018 profile, err := db.GetProfile(s.db, user.Did) 1019 if err != nil { 1020 l.Warn("getting profile data from DB", "err", err) 1021 } 1022 if profile == nil { 1023 profile = &models.Profile{Did: user.Did} 1024 } 1025 profile.Avatar = uploadBlobResp.Blob.Ref.String() 1026 1027 tx, err := s.db.BeginTx(r.Context(), nil) 1028 if err != nil { 1029 l.Error("failed to start transaction", "err", err) 1030 s.pages.HxRefresh(w) 1031 w.WriteHeader(http.StatusOK) 1032 return 1033 } 1034 1035 err = db.UpsertProfile(tx, profile) 1036 if err != nil { 1037 l.Error("failed to update profile in DB", "err", err) 1038 s.pages.HxRefresh(w) 1039 w.WriteHeader(http.StatusOK) 1040 return 1041 } 1042 1043 s.pages.HxRedirect(w, r.Header.Get("Referer")) 1044} 1045 1046func (s *State) RemoveProfileAvatar(w http.ResponseWriter, r *http.Request) { 1047 l := s.logger.With("handler", "RemoveProfileAvatar") 1048 user := s.oauth.GetMultiAccountUser(r) 1049 l = l.With("did", user.Did) 1050 1051 client, err := s.oauth.AuthorizedClient(r) 1052 if err != nil { 1053 l.Error("failed to get PDS client", "err", err) 1054 s.pages.Notice(w, "avatar-error", "Failed to connect to your PDS") 1055 return 1056 } 1057 1058 getRecordResp, err := comatproto.RepoGetRecord(r.Context(), client, "", tangled.ActorProfileNSID, user.Did, "self") 1059 if err != nil { 1060 l.Error("failed to get current profile record", "err", err) 1061 s.pages.Notice(w, "avatar-error", "Failed to get current profile from your PDS") 1062 return 1063 } 1064 1065 var profileRecord *tangled.ActorProfile 1066 if getRecordResp.Value != nil { 1067 if val, ok := getRecordResp.Value.Val.(*tangled.ActorProfile); ok { 1068 profileRecord = val 1069 } else { 1070 l.Warn("profile record type assertion failed") 1071 profileRecord = &tangled.ActorProfile{} 1072 } 1073 } else { 1074 l.Warn("no existing profile record") 1075 profileRecord = &tangled.ActorProfile{} 1076 } 1077 1078 profileRecord.Avatar = nil 1079 1080 _, err = comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ 1081 Collection: tangled.ActorProfileNSID, 1082 Repo: user.Did, 1083 Rkey: "self", 1084 Record: &lexutil.LexiconTypeDecoder{Val: profileRecord}, 1085 SwapRecord: getRecordResp.Cid, 1086 }) 1087 1088 if err != nil { 1089 l.Error("failed to update profile record", "err", err) 1090 s.pages.Notice(w, "avatar-error", "Failed to remove avatar from your PDS") 1091 return 1092 } 1093 1094 l.Info("successfully removed avatar from PDS") 1095 1096 profile, err := db.GetProfile(s.db, user.Did) 1097 if err != nil { 1098 l.Warn("getting profile data from DB", "err", err) 1099 } 1100 if profile == nil { 1101 profile = &models.Profile{Did: user.Did} 1102 } 1103 profile.Avatar = "" 1104 1105 tx, err := s.db.BeginTx(r.Context(), nil) 1106 if err != nil { 1107 l.Error("failed to start transaction", "err", err) 1108 s.pages.HxRefresh(w) 1109 w.WriteHeader(http.StatusOK) 1110 return 1111 } 1112 1113 err = db.UpsertProfile(tx, profile) 1114 if err != nil { 1115 l.Error("failed to update profile in DB", "err", err) 1116 s.pages.HxRefresh(w) 1117 w.WriteHeader(http.StatusOK) 1118 return 1119 } 1120 1121 s.pages.HxRedirect(w, r.Header.Get("Referer")) 1122} 1123 1124func (s *State) UpdateProfilePunchcardSetting(w http.ResponseWriter, r *http.Request) { 1125 l := s.logger.With("handler", "UpdateProfilePunchcardSetting") 1126 err := r.ParseForm() 1127 if err != nil { 1128 l.Error("invalid profile update form", "err", err) 1129 return 1130 } 1131 user := s.oauth.GetMultiAccountUser(r) 1132 1133 hideOthers := false 1134 hideMine := false 1135 1136 if r.Form.Get("hideMine") == "on" { 1137 hideMine = true 1138 } 1139 if r.Form.Get("hideOthers") == "on" { 1140 hideOthers = true 1141 } 1142 1143 err = db.UpsertPunchcardPreference(s.db, user.Did, hideMine, hideOthers) 1144 if err != nil { 1145 l.Error("failed to update punchcard preferences", "err", err) 1146 return 1147 } 1148 1149 s.pages.HxRefresh(w) 1150}