# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Gitlab::Composer::Cache do
  let_it_be(:package_name) { 'sample-project' }
  let_it_be(:json) { { 'name' => package_name } }
  let_it_be(:group) { create(:group) }
  let_it_be(:project) { create(:project, :custom_repo, files: { 'composer.json' => json.to_json }, group: group) }

  let(:branch) { project.repository.find_branch('master') }
  let(:sha_regex) { /^[A-Fa-f0-9]{64}$/ }

  shared_examples 'Composer create cache page' do
    let(:expected_json) { ::Gitlab::Composer::VersionIndex.new(packages).to_json }

    before do
      stub_composer_cache_object_storage
    end

    it 'creates the cached page' do
      expect { subject }.to change { Packages::Composer::CacheFile.count }.by(1)
      cache_file = Packages::Composer::CacheFile.last
      expect(cache_file.file_sha256).to eq package.reload.composer_metadatum.version_cache_sha
      expect(cache_file.file.read).to eq expected_json
    end
  end

  shared_examples 'Composer marks cache page for deletion' do
    it 'marks the page for deletion' do
      cache_file = Packages::Composer::CacheFile.last

      freeze_time do
        expect { subject }.to change { cache_file.reload.delete_at }.from(nil).to(1.day.from_now)
      end
    end
  end

  describe '#execute' do
    subject { described_class.new(project: project, name: package_name).execute }

    context 'creating packages' do
      context 'with a pre-existing package' do
        let(:package) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '1.0.0', json: json) }
        let(:package2) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '2.0.0', json: json) }
        let(:packages) { [package, package2] }

        before do
          package
          described_class.new(project: project, name: package_name).execute
          package.reload
          package2
        end

        it 'updates the sha and creates the cache page' do
          expect { subject }.to change { package2.reload.composer_metadatum.version_cache_sha }.from(nil).to(sha_regex)
            .and change { package.reload.composer_metadatum.version_cache_sha }.to(sha_regex)
        end

        it_behaves_like 'Composer create cache page'
        it_behaves_like 'Composer marks cache page for deletion'
      end

      context 'first package' do
        let!(:package) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '1.0.0', json: json) }
        let(:packages) { [package] }

        it 'updates the sha and creates the cache page' do
          expect { subject }.to change { package.reload.composer_metadatum.version_cache_sha }.from(nil).to(sha_regex)
        end

        it_behaves_like 'Composer create cache page'
      end
    end

    context 'updating packages' do
      let(:package) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '1.0.0', json: json) }
      let(:package2) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '2.0.0', json: json) }
      let(:packages) { [package, package2] }

      before do
        packages

        described_class.new(project: project, name: package_name).execute

        package.update!(version: '1.2.0')
        package.reload
      end

      it_behaves_like 'Composer create cache page'
      it_behaves_like 'Composer marks cache page for deletion'
    end

    context 'deleting packages' do
      context 'when it is not the last package' do
        let(:package) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '1.0.0', json: json) }
        let(:package2) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '2.0.0', json: json) }
        let(:packages) { [package] }

        before do
          package
          package2

          described_class.new(project: project, name: package_name).execute

          package2.destroy!
        end

        it_behaves_like 'Composer create cache page'
        it_behaves_like 'Composer marks cache page for deletion'
      end

      context 'when it is the last package' do
        let!(:package) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '1.0.0', json: json) }
        let!(:last_sha) do
          described_class.new(project: project, name: package_name).execute
          package.reload.composer_metadatum.version_cache_sha
        end

        before do
          package.destroy!
        end

        subject { described_class.new(project: project, name: package_name, last_page_sha: last_sha).execute }

        it_behaves_like 'Composer marks cache page for deletion'

        it 'does not create a new page' do
          expect { subject }.not_to change { Packages::Composer::CacheFile.count }
        end
      end
    end
  end
end