# frozen_string_literal: true

RSpec.describe BookmarkManager do
  subject(:manager) { described_class.new(user) }

  fab!(:post)

  let(:user) { Fabricate(:user) }
  let(:reminder_at) { 1.day.from_now }
  let(:name) { "Check this out!" }

  describe ".destroy" do
    let!(:bookmark) { Fabricate(:bookmark, user: user, bookmarkable: post) }
    it "deletes the existing bookmark" do
      manager.destroy(bookmark.id)
      expect(Bookmark.exists?(id: bookmark.id)).to eq(false)
    end

    context "if the bookmark is the last one bookmarked in the topic" do
      it "marks the topic user bookmarked column as false" do
        TopicUser.create(user: user, topic: post.topic, bookmarked: true)
        manager.destroy(bookmark.id)
        tu = TopicUser.find_by(user: user)
        expect(tu.bookmarked).to eq(false)
      end
    end

    context "if the bookmark is belonging to some other user" do
      let!(:bookmark) { Fabricate(:bookmark, user: Fabricate(:admin), bookmarkable: post) }
      it "raises an invalid access error" do
        expect { manager.destroy(bookmark.id) }.to raise_error(Discourse::InvalidAccess)
      end
    end

    context "if the bookmark no longer exists" do
      it "raises a not found error" do
        expect { manager.destroy(9999) }.to raise_error(Discourse::NotFound)
      end
    end
  end

  describe ".update" do
    let!(:bookmark) do
      Fabricate(
        :bookmark_next_business_day_reminder,
        user: user,
        bookmarkable: post,
        name: "Old name",
      )
    end
    let(:new_name) { "Some new name" }
    let(:new_reminder_at) { 10.days.from_now }
    let(:options) { {} }

    def update_bookmark
      manager.update(
        bookmark_id: bookmark.id,
        name: new_name,
        reminder_at: new_reminder_at,
        options: options,
      )
    end

    it "saves the time and new name successfully" do
      update_bookmark
      bookmark.reload
      expect(bookmark.name).to eq(new_name)
      expect(bookmark.reminder_last_sent_at).to eq(nil)
    end

    it "does not reminder_last_sent_at if reminder did not change" do
      bookmark.update(reminder_last_sent_at: 1.day.ago)
      manager.update(bookmark_id: bookmark.id, name: new_name, reminder_at: bookmark.reminder_at)
      bookmark.reload
      expect(bookmark.reminder_last_sent_at).not_to eq(nil)
    end

    context "when options are provided" do
      let(:options) do
        { auto_delete_preference: Bookmark.auto_delete_preferences[:when_reminder_sent] }
      end

      it "saves any additional options successfully" do
        update_bookmark
        bookmark.reload
        expect(bookmark.auto_delete_preference).to eq(1)
      end
    end

    context "if the bookmark is belonging to some other user" do
      let!(:bookmark) { Fabricate(:bookmark, user: Fabricate(:admin), bookmarkable: post) }
      it "raises an invalid access error" do
        expect { update_bookmark }.to raise_error(Discourse::InvalidAccess)
      end
    end

    context "if the bookmark no longer exists" do
      before { bookmark.destroy! }
      it "raises a not found error" do
        expect { update_bookmark }.to raise_error(Discourse::NotFound)
      end
    end
  end

  describe ".destroy_for_topic" do
    let!(:topic) { Fabricate(:topic) }
    let!(:bookmark1) do
      Fabricate(:bookmark, bookmarkable: Fabricate(:post, topic: topic), user: user)
    end
    let!(:bookmark2) do
      Fabricate(:bookmark, bookmarkable: Fabricate(:post, topic: topic), user: user)
    end

    it "destroys all bookmarks for the topic for the specified user" do
      manager.destroy_for_topic(topic)
      expect(Bookmark.for_user_in_topic(user.id, topic.id).length).to eq(0)
    end

    it "does not destroy any other user's topic bookmarks" do
      user2 = Fabricate(:user)
      Fabricate(:bookmark, bookmarkable: Fabricate(:post, topic: topic), user: user2)
      manager.destroy_for_topic(topic)
      expect(Bookmark.for_user_in_topic(user2.id, topic.id).length).to eq(1)
    end

    it "updates the topic user bookmarked column to false" do
      TopicUser.create(user: user, topic: topic, bookmarked: true)
      manager.destroy_for_topic(topic)
      tu = TopicUser.find_by(user: user)
      expect(tu.bookmarked).to eq(false)
    end
  end

  describe ".send_reminder_notification" do
    let(:bookmark) { Fabricate(:bookmark, user: user) }
    it "sets the reminder_last_sent_at" do
      expect(bookmark.reminder_last_sent_at).to eq(nil)
      described_class.send_reminder_notification(bookmark.id)
      bookmark.reload
      expect(bookmark.reminder_last_sent_at).not_to eq(nil)
    end

    it "creates a notification for the reminder" do
      described_class.send_reminder_notification(bookmark.id)
      notif = notifications_for_user.last
      expect(notif.post_number).to eq(bookmark.bookmarkable.post_number)
    end

    context "when the bookmark does no longer exist" do
      before { bookmark.destroy }
      it "does not error, and does not create a notification" do
        described_class.send_reminder_notification(bookmark.id)
        expect(notifications_for_user.any?).to eq(false)
      end
    end

    context "if the post has been deleted" do
      before { bookmark.bookmarkable.trash! }
      it "does not error and does not create a notification" do
        described_class.send_reminder_notification(bookmark.id)
        bookmark.reload
        expect(notifications_for_user.any?).to eq(false)
      end
    end

    def notifications_for_user
      Notification.where(
        notification_type: Notification.types[:bookmark_reminder],
        user_id: bookmark.user.id,
      )
    end
  end

  describe ".toggle_pin" do
    let!(:bookmark) { Fabricate(:bookmark, user: user) }

    it "sets pinned to false if it is true" do
      bookmark.update(pinned: true)
      manager.toggle_pin(bookmark_id: bookmark.id)
      expect(bookmark.reload.pinned).to eq(false)
    end

    it "sets pinned to true if it is false" do
      bookmark.update(pinned: false)
      manager.toggle_pin(bookmark_id: bookmark.id)
      expect(bookmark.reload.pinned).to eq(true)
    end

    context "if the bookmark is belonging to some other user" do
      let!(:bookmark) { Fabricate(:bookmark, user: Fabricate(:admin)) }
      it "raises an invalid access error" do
        expect { manager.toggle_pin(bookmark_id: bookmark.id) }.to raise_error(
          Discourse::InvalidAccess,
        )
      end
    end

    context "if the bookmark no longer exists" do
      before { bookmark.destroy! }
      it "raises a not found error" do
        expect { manager.toggle_pin(bookmark_id: bookmark.id) }.to raise_error(Discourse::NotFound)
      end
    end
  end

  describe "#create_for" do
    it "allows creating a bookmark for the topic and for the first post" do
      manager.create_for(bookmarkable_id: post.topic_id, bookmarkable_type: "Topic", name: name)
      bookmark = Bookmark.find_by(user: user, bookmarkable: post.topic)
      expect(bookmark.present?).to eq(true)

      manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "Post", name: name)
      bookmark = Bookmark.find_by(user: user, bookmarkable: post)
      expect(bookmark).not_to eq(nil)
    end

    it "when topic is deleted it raises invalid access from guardian check" do
      post.topic.trash!
      expect {
        manager.create_for(bookmarkable_id: post.topic_id, bookmarkable_type: "Topic", name: name)
      }.to raise_error(Discourse::InvalidAccess)
    end

    it "when post is deleted it raises invalid access from guardian check" do
      post.trash!
      expect do
        manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "Post", name: name)
      end.to raise_error(Discourse::InvalidAccess)
    end

    it "adds a validation error when the bookmarkable_type is not registered" do
      manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "BlahFactory", name: name)
      expect(manager.errors.full_messages).to include(
        I18n.t("bookmarks.errors.invalid_bookmarkable", type: "BlahFactory"),
      )
    end

    it "updates the topic user bookmarked column to true if any post is bookmarked" do
      manager.create_for(
        bookmarkable_id: post.id,
        bookmarkable_type: "Post",
        name: name,
        reminder_at: reminder_at,
      )
      tu = TopicUser.find_by(user: user)
      expect(tu.bookmarked).to eq(true)
      tu.update(bookmarked: false)
      new_post = Fabricate(:post, topic: post.topic)
      manager.create_for(bookmarkable_id: new_post.id, bookmarkable_type: "Post")
      tu.reload
      expect(tu.bookmarked).to eq(true)
    end

    it "sets auto_delete_preference to clear_reminder by default" do
      bookmark =
        manager.create_for(
          bookmarkable_id: post.id,
          bookmarkable_type: "Post",
          name: name,
          reminder_at: reminder_at,
        )
      expect(bookmark.auto_delete_preference).to eq(
        Bookmark.auto_delete_preferences[:clear_reminder],
      )
    end

    context "when the user has set their bookmark_auto_delete_preference" do
      before do
        user.user_option.update!(
          bookmark_auto_delete_preference: Bookmark.auto_delete_preferences[:on_owner_reply],
        )
      end

      it "sets auto_delete_preferences to the user's user_option.bookmark_auto_delete_preference" do
        bookmark =
          manager.create_for(
            bookmarkable_id: post.id,
            bookmarkable_type: "Post",
            name: name,
            reminder_at: reminder_at,
          )
        expect(bookmark.auto_delete_preference).to eq(
          Bookmark.auto_delete_preferences[:on_owner_reply],
        )
      end

      it "uses the passed in auto_delete_preference option instead of the user's one" do
        bookmark =
          manager.create_for(
            bookmarkable_id: post.id,
            bookmarkable_type: "Post",
            name: name,
            reminder_at: reminder_at,
            options: {
              auto_delete_preference: Bookmark.auto_delete_preferences[:when_reminder_sent],
            },
          )
        expect(bookmark.auto_delete_preference).to eq(
          Bookmark.auto_delete_preferences[:when_reminder_sent],
        )
      end
    end

    context "when a reminder time is provided" do
      it "saves the values correctly" do
        manager.create_for(
          bookmarkable_id: post.id,
          bookmarkable_type: "Post",
          name: name,
          reminder_at: reminder_at,
        )
        bookmark = Bookmark.find_by(user: user, bookmarkable: post)

        expect(bookmark.reminder_at).to eq_time(reminder_at)
        expect(bookmark.reminder_set_at).not_to eq(nil)
      end
    end

    context "when options are provided" do
      let(:options) do
        { auto_delete_preference: Bookmark.auto_delete_preferences[:when_reminder_sent] }
      end

      it "saves any additional options successfully" do
        manager.create_for(
          bookmarkable_id: post.id,
          bookmarkable_type: "Post",
          name: name,
          reminder_at: reminder_at,
          options: options,
        )
        bookmark = Bookmark.find_by(user: user, bookmarkable: post)

        expect(bookmark.auto_delete_preference).to eq(1)
      end
    end

    context "when the bookmark already exists for the user & post" do
      before { Bookmark.create(bookmarkable: post, user: user) }

      it "adds an error to the manager" do
        manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "Post")
        expect(manager.errors.full_messages).to include(
          I18n.t("bookmarks.errors.already_bookmarked", type: "Post"),
        )
      end
    end

    context "when the bookmark name is too long" do
      it "adds an error to the manager" do
        manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "Post", name: "test" * 100)
        expect(manager.errors.full_messages).to include(
          "Name is too long (maximum is 100 characters)",
        )
      end
    end

    context "when the reminder time is in the past" do
      let(:reminder_at) { 10.days.ago }

      it "adds an error to the manager" do
        manager.create_for(
          bookmarkable_id: post.id,
          bookmarkable_type: "Post",
          name: name,
          reminder_at: reminder_at,
        )
        expect(manager.errors.full_messages).to include(
          I18n.t("bookmarks.errors.cannot_set_past_reminder"),
        )
      end
    end

    context "when the reminder time is far-flung (> 10 years from now)" do
      let(:reminder_at) { 11.years.from_now }

      it "adds an error to the manager" do
        manager.create_for(
          bookmarkable_id: post.id,
          bookmarkable_type: "Post",
          name: name,
          reminder_at: reminder_at,
        )
        expect(manager.errors.full_messages).to include(
          I18n.t("bookmarks.errors.cannot_set_reminder_in_distant_future"),
        )
      end
    end

    context "when the post is inaccessible for the user" do
      before { post.trash! }
      it "raises an invalid access error" do
        expect {
          manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "Post", name: name)
        }.to raise_error(Discourse::InvalidAccess)
      end
    end

    context "when the topic is inaccessible for the user" do
      before { post.topic.update(category: Fabricate(:private_category, group: Fabricate(:group))) }
      it "raises an invalid access error" do
        expect {
          manager.create_for(bookmarkable_id: post.id, bookmarkable_type: "Post", name: name)
        }.to raise_error(Discourse::InvalidAccess)
      end
    end
  end
end