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