package state import ( "log" "net/http" "time" comatproto "github.com/bluesky-social/indigo/api/atproto" "github.com/bluesky-social/indigo/atproto/syntax" lexutil "github.com/bluesky-social/indigo/lex/util" "tangled.org/core/api/tangled" "tangled.org/core/appview/db" "tangled.org/core/appview/models" "tangled.org/core/appview/pages" "tangled.org/core/tid" ) func (s *State) React(w http.ResponseWriter, r *http.Request) { currentUser := s.oauth.GetMultiAccountUser(r) subject := r.URL.Query().Get("subject") if subject == "" { log.Println("invalid form") return } subjectUri, err := syntax.ParseATURI(subject) if err != nil { log.Println("invalid form") return } reactionKind, ok := models.ParseReactionKind(r.URL.Query().Get("kind")) if !ok { log.Println("invalid reaction kind") return } client, err := s.oauth.AuthorizedClient(r) if err != nil { log.Println("failed to authorize client", err) return } switch r.Method { case http.MethodPost: reaction := models.Reaction{ ReactedByDid: currentUser.Active.Did, Rkey: tid.TID(), Kind: reactionKind, ThreadAt: subjectUri, Created: time.Now(), } tx, err := s.db.BeginTx(r.Context(), nil) if err != nil { s.logger.Error("failed to start transaction", "err", err) return } defer tx.Rollback() if err := db.UpsertReaction(tx, reaction); err != nil { log.Println("failed to react", err) return } record := reaction.AsRecord() resp, err := comatproto.RepoPutRecord(r.Context(), client, &comatproto.RepoPutRecord_Input{ Collection: tangled.FeedReactionNSID, Repo: currentUser.Active.Did, Rkey: reaction.Rkey, Record: &lexutil.LexiconTypeDecoder{ Val: &record, }, }) if err != nil { log.Println("failed to create atproto record", err) return } log.Println("created atproto record: ", resp.Uri) if err := tx.Commit(); err != nil { s.logger.Error("failed to commit transaction", "err", err) // DB op failed but record is created in PDS. Ingester will backfill the missed operation } reactionMap, err := db.GetReactionMap(s.db, 20, subjectUri) if err != nil { log.Println("failed to get reactions for ", subjectUri) } s.pages.ThreadReactionFragment(w, pages.ThreadReactionFragmentParams{ ThreadAt: subjectUri, Kind: reactionKind, Count: reactionMap[reactionKind].Count, Users: reactionMap[reactionKind].Users, IsReacted: true, }) return case http.MethodDelete: tx, err := s.db.BeginTx(r.Context(), nil) if err != nil { s.logger.Error("failed to start transaction", "err", err) } defer tx.Rollback() reactions, err := db.DeleteReaction(tx, syntax.DID(currentUser.Active.Did), subjectUri, reactionKind) if err != nil { s.logger.Error("failed to delete reactions from db", "err", err) return } var writes []*comatproto.RepoApplyWrites_Input_Writes_Elem for _, reactionAt := range reactions { writes = append(writes, &comatproto.RepoApplyWrites_Input_Writes_Elem{ RepoApplyWrites_Delete: &comatproto.RepoApplyWrites_Delete{ Collection: tangled.FeedReactionNSID, Rkey: reactionAt.RecordKey().String(), }, }) } _, err = comatproto.RepoApplyWrites(r.Context(), client, &comatproto.RepoApplyWrites_Input{ Repo: currentUser.Active.Did, Writes: writes, }) if err != nil { s.logger.Error("failed to delete reactions from PDS", "err", err) return } if err := tx.Commit(); err != nil { s.logger.Error("failed to commit transaction", "err", err) // DB op failed but record is created in PDS. Ingester will backfill the missed operation } reactionMap, err := db.GetReactionMap(s.db, 20, subjectUri) if err != nil { log.Println("failed to get reactions for ", subjectUri) return } s.pages.ThreadReactionFragment(w, pages.ThreadReactionFragmentParams{ ThreadAt: subjectUri, Kind: reactionKind, Count: reactionMap[reactionKind].Count, Users: reactionMap[reactionKind].Users, IsReacted: false, }) return } }