Monorepo for Tangled
at push-pzoouymnzvnq 943 lines 26 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/models" 21 "tangled.org/core/appview/pages" 22 "tangled.org/core/orm" 23 "tangled.org/core/xrpc" 24) 25 26func (s *State) Profile(w http.ResponseWriter, r *http.Request) { 27 tabVal := r.URL.Query().Get("tab") 28 switch tabVal { 29 case "repos": 30 s.reposPage(w, r) 31 case "followers": 32 s.followersPage(w, r) 33 case "following": 34 s.followingPage(w, r) 35 case "starred": 36 s.starredPage(w, r) 37 case "strings": 38 s.stringsPage(w, r) 39 default: 40 s.profileOverview(w, r) 41 } 42} 43 44func (s *State) profile(r *http.Request) (*pages.ProfileCard, error) { 45 didOrHandle := chi.URLParam(r, "user") 46 if didOrHandle == "" { 47 return nil, fmt.Errorf("empty DID or handle") 48 } 49 50 ident, ok := r.Context().Value("resolvedId").(identity.Identity) 51 if !ok { 52 return nil, fmt.Errorf("failed to resolve ID") 53 } 54 did := ident.DID.String() 55 56 profile, err := db.GetProfile(s.db, did) 57 if err != nil { 58 return nil, fmt.Errorf("failed to get profile: %w", err) 59 } 60 61 hasProfile := profile != nil 62 if !hasProfile { 63 profile = &models.Profile{Did: did} 64 } 65 66 repoCount, err := db.CountRepos(s.db, orm.FilterEq("did", did)) 67 if err != nil { 68 return nil, fmt.Errorf("failed to get repo count: %w", err) 69 } 70 71 stringCount, err := db.CountStrings(s.db, orm.FilterEq("did", did)) 72 if err != nil { 73 return nil, fmt.Errorf("failed to get string count: %w", err) 74 } 75 76 starredCount, err := db.CountStars(s.db, orm.FilterEq("did", did)) 77 if err != nil { 78 return nil, fmt.Errorf("failed to get starred repo count: %w", err) 79 } 80 81 followStats, err := db.GetFollowerFollowingCount(s.db, did) 82 if err != nil { 83 return nil, fmt.Errorf("failed to get follower stats: %w", err) 84 } 85 86 loggedInUser := s.oauth.GetMultiAccountUser(r) 87 followStatus := models.IsNotFollowing 88 if loggedInUser != nil { 89 followStatus = db.GetFollowStatus(s.db, loggedInUser.Active.Did, did) 90 } 91 92 now := time.Now() 93 startOfYear := time.Date(now.Year(), 1, 1, 0, 0, 0, 0, time.UTC) 94 punchcard, err := db.MakePunchcard( 95 s.db, 96 orm.FilterEq("did", did), 97 orm.FilterGte("date", startOfYear.Format(time.DateOnly)), 98 orm.FilterLte("date", now.Format(time.DateOnly)), 99 ) 100 if err != nil { 101 return nil, fmt.Errorf("failed to get punchcard for %s: %w", did, err) 102 } 103 104 return &pages.ProfileCard{ 105 UserDid: did, 106 HasProfile: hasProfile, 107 Profile: profile, 108 FollowStatus: followStatus, 109 Stats: pages.ProfileStats{ 110 RepoCount: repoCount, 111 StringCount: stringCount, 112 StarredCount: starredCount, 113 FollowersCount: followStats.Followers, 114 FollowingCount: followStats.Following, 115 }, 116 Punchcard: punchcard, 117 }, nil 118} 119 120func (s *State) profileOverview(w http.ResponseWriter, r *http.Request) { 121 l := s.logger.With("handler", "profileHomePage") 122 123 profile, err := s.profile(r) 124 if err != nil { 125 l.Error("failed to build profile card", "err", err) 126 s.pages.Error500(w) 127 return 128 } 129 l = l.With("profileDid", profile.UserDid) 130 131 repos, err := db.GetRepos( 132 s.db, 133 0, 134 orm.FilterEq("did", profile.UserDid), 135 ) 136 if err != nil { 137 l.Error("failed to fetch repos", "err", err) 138 } 139 140 // filter out ones that are pinned 141 pinnedRepos := []models.Repo{} 142 for i, r := range repos { 143 // if this is a pinned repo, add it 144 if slices.Contains(profile.Profile.PinnedRepos[:], r.RepoAt()) { 145 pinnedRepos = append(pinnedRepos, r) 146 } 147 148 // if there are no saved pins, add the first 4 repos 149 if profile.Profile.IsPinnedReposEmpty() && i < 4 { 150 pinnedRepos = append(pinnedRepos, r) 151 } 152 } 153 154 collaboratingRepos, err := db.CollaboratingIn(s.db, profile.UserDid) 155 if err != nil { 156 l.Error("failed to fetch collaborating repos", "err", err) 157 } 158 159 pinnedCollaboratingRepos := []models.Repo{} 160 for _, r := range collaboratingRepos { 161 // if this is a pinned repo, add it 162 if slices.Contains(profile.Profile.PinnedRepos[:], r.RepoAt()) { 163 pinnedCollaboratingRepos = append(pinnedCollaboratingRepos, r) 164 } 165 } 166 167 timeline, err := db.MakeProfileTimeline(s.db, profile.UserDid) 168 if err != nil { 169 l.Error("failed to create timeline", "err", err) 170 } 171 172 s.pages.ProfileOverview(w, pages.ProfileOverviewParams{ 173 LoggedInUser: s.oauth.GetMultiAccountUser(r), 174 Card: profile, 175 Repos: pinnedRepos, 176 CollaboratingRepos: pinnedCollaboratingRepos, 177 ProfileTimeline: timeline, 178 }) 179} 180 181func (s *State) reposPage(w http.ResponseWriter, r *http.Request) { 182 l := s.logger.With("handler", "reposPage") 183 184 profile, err := s.profile(r) 185 if err != nil { 186 l.Error("failed to build profile card", "err", err) 187 s.pages.Error500(w) 188 return 189 } 190 l = l.With("profileDid", profile.UserDid) 191 192 repos, err := db.GetRepos( 193 s.db, 194 0, 195 orm.FilterEq("did", profile.UserDid), 196 ) 197 if err != nil { 198 l.Error("failed to get repos", "err", err) 199 s.pages.Error500(w) 200 return 201 } 202 203 err = s.pages.ProfileRepos(w, pages.ProfileReposParams{ 204 LoggedInUser: s.oauth.GetMultiAccountUser(r), 205 Repos: repos, 206 Card: profile, 207 }) 208} 209 210func (s *State) starredPage(w http.ResponseWriter, r *http.Request) { 211 l := s.logger.With("handler", "starredPage") 212 213 profile, err := s.profile(r) 214 if err != nil { 215 l.Error("failed to build profile card", "err", err) 216 s.pages.Error500(w) 217 return 218 } 219 l = l.With("profileDid", profile.UserDid) 220 221 stars, err := db.GetRepoStars(s.db, 0, orm.FilterEq("did", profile.UserDid)) 222 if err != nil { 223 l.Error("failed to get stars", "err", err) 224 s.pages.Error500(w) 225 return 226 } 227 var repos []models.Repo 228 for _, s := range stars { 229 repos = append(repos, *s.Repo) 230 } 231 232 err = s.pages.ProfileStarred(w, pages.ProfileStarredParams{ 233 LoggedInUser: s.oauth.GetMultiAccountUser(r), 234 Repos: repos, 235 Card: profile, 236 }) 237} 238 239func (s *State) stringsPage(w http.ResponseWriter, r *http.Request) { 240 l := s.logger.With("handler", "stringsPage") 241 242 profile, err := s.profile(r) 243 if err != nil { 244 l.Error("failed to build profile card", "err", err) 245 s.pages.Error500(w) 246 return 247 } 248 l = l.With("profileDid", profile.UserDid) 249 250 strings, err := db.GetStrings(s.db, 0, orm.FilterEq("did", profile.UserDid)) 251 if err != nil { 252 l.Error("failed to get strings", "err", err) 253 s.pages.Error500(w) 254 return 255 } 256 257 err = s.pages.ProfileStrings(w, pages.ProfileStringsParams{ 258 LoggedInUser: s.oauth.GetMultiAccountUser(r), 259 Strings: strings, 260 Card: profile, 261 }) 262} 263 264type FollowsPageParams struct { 265 Follows []pages.FollowCard 266 Card *pages.ProfileCard 267} 268 269func (s *State) followPage( 270 r *http.Request, 271 fetchFollows func(db.Execer, string) ([]models.Follow, error), 272 extractDid func(models.Follow) string, 273) (*FollowsPageParams, error) { 274 l := s.logger.With("handler", "reposPage") 275 276 profile, err := s.profile(r) 277 if err != nil { 278 return nil, err 279 } 280 l = l.With("profileDid", profile.UserDid) 281 282 loggedInUser := s.oauth.GetMultiAccountUser(r) 283 params := FollowsPageParams{ 284 Card: profile, 285 } 286 287 follows, err := fetchFollows(s.db, profile.UserDid) 288 if err != nil { 289 l.Error("failed to fetch follows", "err", err) 290 return &params, err 291 } 292 293 if len(follows) == 0 { 294 return &params, nil 295 } 296 297 followDids := make([]string, 0, len(follows)) 298 for _, follow := range follows { 299 followDids = append(followDids, extractDid(follow)) 300 } 301 302 profiles, err := db.GetProfiles(s.db, orm.FilterIn("did", followDids)) 303 if err != nil { 304 l.Error("failed to get profiles", "followDids", followDids, "err", err) 305 return &params, err 306 } 307 308 followStatsMap, err := db.GetFollowerFollowingCounts(s.db, followDids) 309 if err != nil { 310 log.Printf("getting follow counts for %s: %s", followDids, err) 311 } 312 313 loggedInUserFollowing := make(map[string]struct{}) 314 if loggedInUser != nil { 315 following, err := db.GetFollowing(s.db, loggedInUser.Active.Did) 316 if err != nil { 317 l.Error("failed to get follow list", "err", err, "loggedInUser", loggedInUser.Active.Did) 318 return &params, err 319 } 320 loggedInUserFollowing = make(map[string]struct{}, len(following)) 321 for _, follow := range following { 322 loggedInUserFollowing[follow.SubjectDid] = struct{}{} 323 } 324 } 325 326 followCards := make([]pages.FollowCard, len(follows)) 327 for i, did := range followDids { 328 followStats := followStatsMap[did] 329 followStatus := models.IsNotFollowing 330 if _, exists := loggedInUserFollowing[did]; exists { 331 followStatus = models.IsFollowing 332 } else if loggedInUser != nil && loggedInUser.Active.Did == did { 333 followStatus = models.IsSelf 334 } 335 336 var profile *models.Profile 337 if p, exists := profiles[did]; exists { 338 profile = p 339 } else { 340 profile = &models.Profile{} 341 profile.Did = did 342 } 343 followCards[i] = pages.FollowCard{ 344 LoggedInUser: loggedInUser, 345 UserDid: did, 346 FollowStatus: followStatus, 347 FollowersCount: followStats.Followers, 348 FollowingCount: followStats.Following, 349 Profile: profile, 350 } 351 } 352 353 params.Follows = followCards 354 355 return &params, nil 356} 357 358func (s *State) followersPage(w http.ResponseWriter, r *http.Request) { 359 followPage, err := s.followPage(r, db.GetFollowers, func(f models.Follow) string { return f.UserDid }) 360 if err != nil { 361 s.pages.Notice(w, "all-followers", "Failed to load followers") 362 return 363 } 364 365 s.pages.ProfileFollowers(w, pages.ProfileFollowersParams{ 366 LoggedInUser: s.oauth.GetMultiAccountUser(r), 367 Followers: followPage.Follows, 368 Card: followPage.Card, 369 }) 370} 371 372func (s *State) followingPage(w http.ResponseWriter, r *http.Request) { 373 followPage, err := s.followPage(r, db.GetFollowing, func(f models.Follow) string { return f.SubjectDid }) 374 if err != nil { 375 s.pages.Notice(w, "all-following", "Failed to load following") 376 return 377 } 378 379 s.pages.ProfileFollowing(w, pages.ProfileFollowingParams{ 380 LoggedInUser: s.oauth.GetMultiAccountUser(r), 381 Following: followPage.Follows, 382 Card: followPage.Card, 383 }) 384} 385 386func (s *State) AtomFeedPage(w http.ResponseWriter, r *http.Request) { 387 ident, ok := r.Context().Value("resolvedId").(identity.Identity) 388 if !ok { 389 s.pages.Error404(w) 390 return 391 } 392 393 feed, err := s.getProfileFeed(r.Context(), &ident) 394 if err != nil { 395 s.pages.Error500(w) 396 return 397 } 398 399 if feed == nil { 400 return 401 } 402 403 atom, err := feed.ToAtom() 404 if err != nil { 405 s.pages.Error500(w) 406 return 407 } 408 409 w.Header().Set("content-type", "application/atom+xml") 410 w.Write([]byte(atom)) 411} 412 413func (s *State) getProfileFeed(ctx context.Context, id *identity.Identity) (*feeds.Feed, error) { 414 timeline, err := db.MakeProfileTimeline(s.db, id.DID.String()) 415 if err != nil { 416 return nil, err 417 } 418 419 author := &feeds.Author{ 420 Name: fmt.Sprintf("@%s", id.Handle), 421 } 422 423 feed := feeds.Feed{ 424 Title: fmt.Sprintf("%s's timeline", author.Name), 425 Link: &feeds.Link{Href: fmt.Sprintf("%s/@%s", s.config.Core.BaseUrl(), id.Handle), Type: "text/html", Rel: "alternate"}, 426 Items: make([]*feeds.Item, 0), 427 Updated: time.UnixMilli(0), 428 Author: author, 429 } 430 431 for _, byMonth := range timeline.ByMonth { 432 if err := s.addPullRequestItems(ctx, &feed, byMonth.PullEvents.Items, author); err != nil { 433 return nil, err 434 } 435 if err := s.addIssueItems(ctx, &feed, byMonth.IssueEvents.Items, author); err != nil { 436 return nil, err 437 } 438 if err := s.addRepoItems(ctx, &feed, byMonth.RepoEvents, author); err != nil { 439 return nil, err 440 } 441 } 442 443 slices.SortFunc(feed.Items, func(a *feeds.Item, b *feeds.Item) int { 444 return int(b.Created.UnixMilli()) - int(a.Created.UnixMilli()) 445 }) 446 447 if len(feed.Items) > 0 { 448 feed.Updated = feed.Items[0].Created 449 } 450 451 return &feed, nil 452} 453 454func (s *State) addPullRequestItems(ctx context.Context, feed *feeds.Feed, pulls []*models.Pull, author *feeds.Author) error { 455 for _, pull := range pulls { 456 owner, err := s.idResolver.ResolveIdent(ctx, pull.Repo.Did) 457 if err != nil { 458 return err 459 } 460 461 // Add pull request creation item 462 feed.Items = append(feed.Items, s.createPullRequestItem(pull, owner, author)) 463 } 464 return nil 465} 466 467func (s *State) addIssueItems(ctx context.Context, feed *feeds.Feed, issues []*models.Issue, author *feeds.Author) error { 468 for _, issue := range issues { 469 owner, err := s.idResolver.ResolveIdent(ctx, issue.Repo.Did) 470 if err != nil { 471 return err 472 } 473 474 feed.Items = append(feed.Items, s.createIssueItem(issue, owner, author)) 475 } 476 return nil 477} 478 479func (s *State) addRepoItems(ctx context.Context, feed *feeds.Feed, repos []models.RepoEvent, author *feeds.Author) error { 480 for _, repo := range repos { 481 item, err := s.createRepoItem(ctx, repo, author) 482 if err != nil { 483 return err 484 } 485 feed.Items = append(feed.Items, item) 486 } 487 return nil 488} 489 490func (s *State) createPullRequestItem(pull *models.Pull, owner *identity.Identity, author *feeds.Author) *feeds.Item { 491 return &feeds.Item{ 492 Title: fmt.Sprintf("%s created pull request '%s' in @%s/%s", author.Name, pull.Title, owner.Handle, pull.Repo.Name), 493 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"}, 494 Created: pull.Created, 495 Author: author, 496 } 497} 498 499func (s *State) createIssueItem(issue *models.Issue, owner *identity.Identity, author *feeds.Author) *feeds.Item { 500 return &feeds.Item{ 501 Title: fmt.Sprintf("%s created issue '%s' in @%s/%s", author.Name, issue.Title, owner.Handle, issue.Repo.Name), 502 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"}, 503 Created: issue.Created, 504 Author: author, 505 } 506} 507 508func (s *State) createRepoItem(ctx context.Context, repo models.RepoEvent, author *feeds.Author) (*feeds.Item, error) { 509 var title string 510 if repo.Source != nil { 511 sourceOwner, err := s.idResolver.ResolveIdent(ctx, repo.Source.Did) 512 if err != nil { 513 return nil, err 514 } 515 title = fmt.Sprintf("%s forked repository @%s/%s to '%s'", author.Name, sourceOwner.Handle, repo.Source.Name, repo.Repo.Name) 516 } else { 517 title = fmt.Sprintf("%s created repository '%s'", author.Name, repo.Repo.Name) 518 } 519 520 return &feeds.Item{ 521 Title: title, 522 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 523 Created: repo.Repo.Created, 524 Author: author, 525 }, nil 526} 527 528func (s *State) UpdateProfileBio(w http.ResponseWriter, r *http.Request) { 529 user := s.oauth.GetMultiAccountUser(r) 530 531 err := r.ParseForm() 532 if err != nil { 533 log.Println("invalid profile update form", err) 534 s.pages.Notice(w, "update-profile", "Invalid form.") 535 return 536 } 537 538 profile, err := db.GetProfile(s.db, user.Active.Did) 539 if err != nil { 540 log.Printf("getting profile data for %s: %s", user.Active.Did, err) 541 } 542 if profile == nil { 543 profile = &models.Profile{Did: user.Active.Did} 544 } 545 546 profile.Description = r.FormValue("description") 547 profile.IncludeBluesky = r.FormValue("includeBluesky") == "on" 548 profile.Location = r.FormValue("location") 549 profile.Pronouns = r.FormValue("pronouns") 550 551 var links [5]string 552 for i := range 5 { 553 iLink := r.FormValue(fmt.Sprintf("link%d", i)) 554 links[i] = iLink 555 } 556 profile.Links = links 557 558 // Parse stats (exactly 2) 559 stat0 := r.FormValue("stat0") 560 stat1 := r.FormValue("stat1") 561 562 if stat0 != "" { 563 profile.Stats[0].Kind = models.VanityStatKind(stat0) 564 } 565 566 if stat1 != "" { 567 profile.Stats[1].Kind = models.VanityStatKind(stat1) 568 } 569 570 if err := db.ValidateProfile(s.db, profile); err != nil { 571 log.Println("invalid profile", err) 572 s.pages.Notice(w, "update-profile", err.Error()) 573 return 574 } 575 576 s.updateProfile(profile, w, r) 577} 578 579func (s *State) UpdateProfilePins(w http.ResponseWriter, r *http.Request) { 580 user := s.oauth.GetMultiAccountUser(r) 581 582 err := r.ParseForm() 583 if err != nil { 584 log.Println("invalid profile update form", err) 585 s.pages.Notice(w, "update-profile", "Invalid form.") 586 return 587 } 588 589 profile, err := db.GetProfile(s.db, user.Active.Did) 590 if err != nil { 591 log.Printf("getting profile data for %s: %s", user.Active.Did, err) 592 } 593 if profile == nil { 594 profile = &models.Profile{Did: user.Active.Did} 595 } 596 597 i := 0 598 var pinnedRepos [6]syntax.ATURI 599 for key, values := range r.Form { 600 if i >= 6 { 601 log.Println("invalid pin update form", err) 602 s.pages.Notice(w, "update-profile", "Only 6 repositories can be pinned at a time.") 603 return 604 } 605 if strings.HasPrefix(key, "pinnedRepo") && len(values) > 0 && values[0] != "" && i < 6 { 606 aturi, err := syntax.ParseATURI(values[0]) 607 if err != nil { 608 log.Println("invalid profile update form", err) 609 s.pages.Notice(w, "update-profile", "Invalid form.") 610 return 611 } 612 pinnedRepos[i] = aturi 613 i++ 614 } 615 } 616 profile.PinnedRepos = pinnedRepos 617 618 s.updateProfile(profile, w, r) 619} 620 621func (s *State) updateProfile(profile *models.Profile, w http.ResponseWriter, r *http.Request) { 622 user := s.oauth.GetMultiAccountUser(r) 623 tx, err := s.db.BeginTx(r.Context(), nil) 624 if err != nil { 625 log.Println("failed to start transaction", err) 626 s.pages.Notice(w, "update-profile", "Failed to update profile, try again later.") 627 return 628 } 629 630 client, err := s.oauth.AuthorizedClient(r) 631 if err != nil { 632 log.Println("failed to get authorized client", err) 633 s.pages.Notice(w, "update-profile", "Failed to update profile, try again later.") 634 return 635 } 636 637 // yeah... lexgen dose not support syntax.ATURI in the record for some reason, 638 // nor does it support exact size arrays 639 var pinnedRepoStrings []string 640 for _, r := range profile.PinnedRepos { 641 pinnedRepoStrings = append(pinnedRepoStrings, r.String()) 642 } 643 644 var vanityStats []string 645 for _, v := range profile.Stats { 646 vanityStats = append(vanityStats, string(v.Kind)) 647 } 648 649 ex, _ := comatproto.RepoGetRecord(r.Context(), client, "", tangled.ActorProfileNSID, user.Active.Did, "self") 650 var cid *string 651 if ex != nil { 652 cid = ex.Cid 653 } 654 655 _, err = comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ 656 Collection: tangled.ActorProfileNSID, 657 Repo: user.Active.Did, 658 Rkey: "self", 659 Record: &lexutil.LexiconTypeDecoder{ 660 Val: &tangled.ActorProfile{ 661 Bluesky: profile.IncludeBluesky, 662 Description: &profile.Description, 663 Links: profile.Links[:], 664 Location: &profile.Location, 665 PinnedRepositories: pinnedRepoStrings, 666 Stats: vanityStats[:], 667 Pronouns: &profile.Pronouns, 668 }}, 669 SwapRecord: cid, 670 }) 671 if err != nil { 672 log.Println("failed to update profile", err) 673 s.pages.Notice(w, "update-profile", "Failed to update PDS, try again later.") 674 return 675 } 676 677 err = db.UpsertProfile(tx, profile) 678 if err != nil { 679 log.Println("failed to update profile", err) 680 s.pages.Notice(w, "update-profile", "Failed to update profile, try again later.") 681 return 682 } 683 684 s.notifier.UpdateProfile(r.Context(), profile) 685 686 s.pages.HxRedirect(w, "/"+user.Active.Did) 687} 688 689func (s *State) EditBioFragment(w http.ResponseWriter, r *http.Request) { 690 user := s.oauth.GetMultiAccountUser(r) 691 692 profile, err := db.GetProfile(s.db, user.Active.Did) 693 if err != nil { 694 log.Printf("getting profile data for %s: %s", user.Active.Did, err) 695 } 696 if profile == nil { 697 profile = &models.Profile{Did: user.Active.Did} 698 } 699 700 s.pages.EditBioFragment(w, pages.EditBioParams{ 701 LoggedInUser: user, 702 Profile: profile, 703 }) 704} 705 706func (s *State) EditPinsFragment(w http.ResponseWriter, r *http.Request) { 707 user := s.oauth.GetMultiAccountUser(r) 708 709 profile, err := db.GetProfile(s.db, user.Active.Did) 710 if err != nil { 711 log.Printf("getting profile data for %s: %s", user.Active.Did, err) 712 } 713 if profile == nil { 714 profile = &models.Profile{Did: user.Active.Did} 715 } 716 717 repos, err := db.GetRepos(s.db, 0, orm.FilterEq("did", user.Active.Did)) 718 if err != nil { 719 log.Printf("getting repos for %s: %s", user.Active.Did, err) 720 } 721 722 collaboratingRepos, err := db.CollaboratingIn(s.db, user.Active.Did) 723 if err != nil { 724 log.Printf("getting collaborating repos for %s: %s", user.Active.Did, err) 725 } 726 727 allRepos := []pages.PinnedRepo{} 728 729 for _, r := range repos { 730 isPinned := slices.Contains(profile.PinnedRepos[:], r.RepoAt()) 731 allRepos = append(allRepos, pages.PinnedRepo{ 732 IsPinned: isPinned, 733 Repo: r, 734 }) 735 } 736 for _, r := range collaboratingRepos { 737 isPinned := slices.Contains(profile.PinnedRepos[:], r.RepoAt()) 738 allRepos = append(allRepos, pages.PinnedRepo{ 739 IsPinned: isPinned, 740 Repo: r, 741 }) 742 } 743 744 s.pages.EditPinsFragment(w, pages.EditPinsParams{ 745 LoggedInUser: user, 746 Profile: profile, 747 AllRepos: allRepos, 748 }) 749} 750 751func (s *State) UploadProfileAvatar(w http.ResponseWriter, r *http.Request) { 752 l := s.logger.With("handler", "UploadProfileAvatar") 753 user := s.oauth.GetUser(r) 754 l = l.With("did", user.Did) 755 756 // Parse multipart form (10MB max) 757 if err := r.ParseMultipartForm(10 << 20); err != nil { 758 l.Error("failed to parse form", "err", err) 759 s.pages.Notice(w, "avatar-error", "Failed to parse form") 760 return 761 } 762 763 file, header, err := r.FormFile("avatar") 764 if err != nil { 765 l.Error("failed to read avatar file", "err", err) 766 s.pages.Notice(w, "avatar-error", "Failed to read avatar file") 767 return 768 } 769 defer file.Close() 770 771 if header.Size > 1000000 { 772 l.Warn("avatar file too large", "size", header.Size) 773 s.pages.Notice(w, "avatar-error", "Avatar file too large (max 1MB)") 774 return 775 } 776 777 contentType := header.Header.Get("Content-Type") 778 if contentType != "image/png" && contentType != "image/jpeg" { 779 l.Warn("invalid image type", "contentType", contentType) 780 s.pages.Notice(w, "avatar-error", "Invalid image type (only PNG and JPEG allowed)") 781 return 782 } 783 784 client, err := s.oauth.AuthorizedClient(r) 785 if err != nil { 786 l.Error("failed to get PDS client", "err", err) 787 s.pages.Notice(w, "avatar-error", "Failed to connect to your PDS") 788 return 789 } 790 791 uploadBlobResp, err := xrpc.RepoUploadBlob(r.Context(), client, file, header.Header.Get("Content-Type")) 792 if err != nil { 793 l.Error("failed to upload avatar blob", "err", err) 794 s.pages.Notice(w, "avatar-error", "Failed to upload avatar to your PDS") 795 return 796 } 797 798 l.Info("uploaded avatar blob", "cid", uploadBlobResp.Blob.Ref.String()) 799 800 // get current profile record from PDS to get its CID for swap 801 getRecordResp, err := comatproto.RepoGetRecord(r.Context(), client, "", tangled.ActorProfileNSID, user.Did, "self") 802 if err != nil { 803 l.Error("failed to get current profile record", "err", err) 804 s.pages.Notice(w, "avatar-error", "Failed to get current profile from your PDS") 805 return 806 } 807 808 var profileRecord *tangled.ActorProfile 809 if getRecordResp.Value != nil { 810 if val, ok := getRecordResp.Value.Val.(*tangled.ActorProfile); ok { 811 profileRecord = val 812 } else { 813 l.Warn("profile record type assertion failed, creating new record") 814 profileRecord = &tangled.ActorProfile{} 815 } 816 } else { 817 l.Warn("no existing profile record, creating new record") 818 profileRecord = &tangled.ActorProfile{} 819 } 820 821 profileRecord.Avatar = uploadBlobResp.Blob 822 823 _, err = comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ 824 Collection: tangled.ActorProfileNSID, 825 Repo: user.Did, 826 Rkey: "self", 827 Record: &lexutil.LexiconTypeDecoder{Val: profileRecord}, 828 SwapRecord: getRecordResp.Cid, 829 }) 830 831 if err != nil { 832 l.Error("failed to update profile record", "err", err) 833 s.pages.Notice(w, "avatar-error", "Failed to update profile on your PDS") 834 return 835 } 836 837 l.Info("successfully updated profile with avatar") 838 839 profile, err := db.GetProfile(s.db, user.Did) 840 if err != nil { 841 l.Warn("getting profile data from DB", "err", err) 842 } 843 if profile == nil { 844 profile = &models.Profile{Did: user.Did} 845 } 846 profile.Avatar = uploadBlobResp.Blob.Ref.String() 847 848 tx, err := s.db.BeginTx(r.Context(), nil) 849 if err != nil { 850 l.Error("failed to start transaction", "err", err) 851 s.pages.HxRefresh(w) 852 w.WriteHeader(http.StatusOK) 853 return 854 } 855 856 err = db.UpsertProfile(tx, profile) 857 if err != nil { 858 l.Error("failed to update profile in DB", "err", err) 859 s.pages.HxRefresh(w) 860 w.WriteHeader(http.StatusOK) 861 return 862 } 863 864 s.pages.HxRedirect(w, r.Header.Get("Referer")) 865} 866 867func (s *State) RemoveProfileAvatar(w http.ResponseWriter, r *http.Request) { 868 l := s.logger.With("handler", "RemoveProfileAvatar") 869 user := s.oauth.GetUser(r) 870 l = l.With("did", user.Did) 871 872 client, err := s.oauth.AuthorizedClient(r) 873 if err != nil { 874 l.Error("failed to get PDS client", "err", err) 875 s.pages.Notice(w, "avatar-error", "Failed to connect to your PDS") 876 return 877 } 878 879 getRecordResp, err := comatproto.RepoGetRecord(r.Context(), client, "", tangled.ActorProfileNSID, user.Did, "self") 880 if err != nil { 881 l.Error("failed to get current profile record", "err", err) 882 s.pages.Notice(w, "avatar-error", "Failed to get current profile from your PDS") 883 return 884 } 885 886 var profileRecord *tangled.ActorProfile 887 if getRecordResp.Value != nil { 888 if val, ok := getRecordResp.Value.Val.(*tangled.ActorProfile); ok { 889 profileRecord = val 890 } else { 891 l.Warn("profile record type assertion failed") 892 profileRecord = &tangled.ActorProfile{} 893 } 894 } else { 895 l.Warn("no existing profile record") 896 profileRecord = &tangled.ActorProfile{} 897 } 898 899 profileRecord.Avatar = nil 900 901 _, err = comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ 902 Collection: tangled.ActorProfileNSID, 903 Repo: user.Did, 904 Rkey: "self", 905 Record: &lexutil.LexiconTypeDecoder{Val: profileRecord}, 906 SwapRecord: getRecordResp.Cid, 907 }) 908 909 if err != nil { 910 l.Error("failed to update profile record", "err", err) 911 s.pages.Notice(w, "avatar-error", "Failed to remove avatar from your PDS") 912 return 913 } 914 915 l.Info("successfully removed avatar from PDS") 916 917 profile, err := db.GetProfile(s.db, user.Did) 918 if err != nil { 919 l.Warn("getting profile data from DB", "err", err) 920 } 921 if profile == nil { 922 profile = &models.Profile{Did: user.Did} 923 } 924 profile.Avatar = "" 925 926 tx, err := s.db.BeginTx(r.Context(), nil) 927 if err != nil { 928 l.Error("failed to start transaction", "err", err) 929 s.pages.HxRefresh(w) 930 w.WriteHeader(http.StatusOK) 931 return 932 } 933 934 err = db.UpsertProfile(tx, profile) 935 if err != nil { 936 l.Error("failed to update profile in DB", "err", err) 937 s.pages.HxRefresh(w) 938 w.WriteHeader(http.StatusOK) 939 return 940 } 941 942 s.pages.HxRedirect(w, r.Header.Get("Referer")) 943}