# frozen_string_literal: true require 'spec_helper' RSpec.describe Packages::Pypi::CreatePackageService, :aggregate_failures, feature_category: :package_registry do include PackagesManagerApiSpecHelpers let_it_be(:project) { create(:project) } let_it_be(:user) { create(:user) } let(:sha256) { '1' * 64 } let(:md5) { '567' } let(:requires_python) { '>=2.7' } let(:params) do { name: 'foo', version: '1.0', content: temp_file('foo.tgz'), requires_python: requires_python, sha256_digest: sha256, md5_digest: md5 } end describe '#execute' do subject { described_class.new(project, user, params).execute } let(:created_package) { Packages::Package.pypi.last } context 'without an existing package' do it 'creates the package' do expect { subject }.to change { Packages::Package.pypi.count }.by(1) expect(created_package.name).to eq 'foo' expect(created_package.version).to eq '1.0' expect(created_package.pypi_metadatum.required_python).to eq '>=2.7' expect(created_package.package_files.size).to eq 1 expect(created_package.package_files.first.file_name).to eq 'foo.tgz' expect(created_package.package_files.first.file_sha256).to eq sha256 expect(created_package.package_files.first.file_md5).to eq md5 end end context 'with FIPS mode', :fips_mode do it 'does not generate file_md5' do expect { subject }.to change { Packages::Package.pypi.count }.by(1) expect(created_package.name).to eq 'foo' expect(created_package.version).to eq '1.0' expect(created_package.pypi_metadatum.required_python).to eq '>=2.7' expect(created_package.package_files.size).to eq 1 expect(created_package.package_files.first.file_name).to eq 'foo.tgz' expect(created_package.package_files.first.file_sha256).to eq sha256 expect(created_package.package_files.first.file_md5).to be_nil end end context 'without required_python' do before do params.delete(:requires_python) end it 'creates the package' do expect { subject }.to change { Packages::Package.pypi.count }.by(1) expect(created_package.pypi_metadatum.required_python).to eq '' end end context 'with an invalid metadata' do let(:requires_python) { 'x' * 256 } it 'raises an error' do expect { subject }.to raise_error(ActiveRecord::RecordInvalid) end end it_behaves_like 'assigns the package creator' do let(:package) { created_package } end it_behaves_like 'assigns build to package' it_behaves_like 'assigns status to package' context 'with an existing package' do before do described_class.new(project, user, params).execute end context 'with an existing file' do before do params[:content] = temp_file('foo.tgz') params[:sha256_digest] = sha256 params[:md5_digest] = md5 end it 'throws an error' do expect { subject } .to change { Packages::Package.pypi.count }.by(0) .and change { Packages::PackageFile.count }.by(0) .and raise_error(/File name has already been taken/) end context 'with a pending_destruction package' do before do Packages::Package.pypi.last.pending_destruction! end it 'creates a new package' do expect { subject } .to change { Packages::Package.pypi.count }.by(1) .and change { Packages::PackageFile.count }.by(1) expect(created_package.name).to eq 'foo' expect(created_package.version).to eq '1.0' expect(created_package.pypi_metadatum.required_python).to eq '>=2.7' expect(created_package.package_files.size).to eq 1 expect(created_package.package_files.first.file_name).to eq 'foo.tgz' expect(created_package.package_files.first.file_sha256).to eq sha256 expect(created_package.package_files.first.file_md5).to eq md5 end end end context 'without an existing file' do before do params[:content] = temp_file('another.tgz') end it 'adds the file' do expect { subject } .to change { Packages::Package.pypi.count }.by(0) .and change { Packages::PackageFile.count }.by(1) expect(created_package.package_files.size).to eq 2 expect(created_package.package_files.map(&:file_name).sort).to eq ['another.tgz', 'foo.tgz'] end end end end end