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