diff options
Diffstat (limited to 'spec/presenters/packages')
10 files changed, 690 insertions, 0 deletions
diff --git a/spec/presenters/packages/composer/packages_presenter_spec.rb b/spec/presenters/packages/composer/packages_presenter_spec.rb new file mode 100644 index 00000000000..0445a346180 --- /dev/null +++ b/spec/presenters/packages/composer/packages_presenter_spec.rb @@ -0,0 +1,78 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe ::Packages::Composer::PackagesPresenter do + using RSpec::Parameterized::TableSyntax + + 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_it_be(:package1) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '1.0.0', json: json) } + let_it_be(:package2) { create(:composer_package, :with_metadatum, project: project, name: package_name, version: '2.0.0', json: json) } + + let(:branch) { project.repository.find_branch('master') } + + let(:packages) { [package1, package2] } + let(:presenter) { described_class.new(group, packages) } + + describe '#package_versions' do + subject { presenter.package_versions } + + def expected_json(package) + { + 'dist' => { + 'reference' => branch.target, + 'shasum' => '', + 'type' => 'zip', + 'url' => "http://localhost/api/v4/projects/#{project.id}/packages/composer/archives/#{package.name}.zip?sha=#{branch.target}" + }, + 'name' => package.name, + 'uid' => package.id, + 'version' => package.version + } + end + + it 'returns the packages json' do + packages = subject['packages'][package_name] + + expect(packages['1.0.0']).to eq(expected_json(package1)) + expect(packages['2.0.0']).to eq(expected_json(package2)) + end + end + + describe '#provider' do + subject { presenter.provider} + + let(:expected_json) do + { + 'providers' => { + package_name => { + 'sha256' => /^\h+$/ + } + } + } + end + + it 'returns the provider json' do + expect(subject).to match(expected_json) + end + end + + describe '#root' do + subject { presenter.root } + + let(:expected_json) do + { + 'packages' => [], + 'provider-includes' => { 'p/%hash%.json' => { 'sha256' => /^\h+$/ } }, + 'providers-url' => "/api/v4/group/#{group.id}/-/packages/composer/%package%.json" + } + end + + it 'returns the provider json' do + expect(subject).to match(expected_json) + end + end +end diff --git a/spec/presenters/packages/conan/package_presenter_spec.rb b/spec/presenters/packages/conan/package_presenter_spec.rb new file mode 100644 index 00000000000..3bc649c5da4 --- /dev/null +++ b/spec/presenters/packages/conan/package_presenter_spec.rb @@ -0,0 +1,181 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe ::Packages::Conan::PackagePresenter do + let_it_be(:user) { create(:user) } + let_it_be(:project) { create(:project) } + let_it_be(:conan_package_reference) { '123456789'} + + RSpec.shared_examples 'not selecting a package with the wrong type' do + context 'with a nuget package with same name and version' do + let_it_be(:wrong_package) { create(:nuget_package, name: 'wrong', version: '1.0.0', project: project) } + + let(:recipe) { "#{wrong_package.name}/#{wrong_package.version}" } + + it { is_expected.to be_empty } + end + end + + describe '#recipe_urls' do + subject { described_class.new(recipe, user, project).recipe_urls } + + context 'no existing package' do + let(:recipe) { "my-pkg/v1.0.0/#{project.full_path}/stable" } + + it { is_expected.to be_empty } + end + + it_behaves_like 'not selecting a package with the wrong type' + + context 'existing package' do + let(:package) { create(:conan_package, project: project) } + let(:recipe) { package.conan_recipe } + + let(:expected_result) do + { + "conanfile.py" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/export/conanfile.py", + "conanmanifest.txt" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/export/conanmanifest.txt" + } + end + + it { is_expected.to eq(expected_result) } + end + end + + describe '#recipe_snapshot' do + subject { described_class.new(recipe, user, project).recipe_snapshot } + + context 'no existing package' do + let(:recipe) { "my-pkg/v1.0.0/#{project.full_path}/stable" } + + it { is_expected.to be_empty } + end + + it_behaves_like 'not selecting a package with the wrong type' + + context 'existing package' do + let(:package) { create(:conan_package, project: project) } + let(:recipe) { package.conan_recipe } + + let(:expected_result) do + { + "conanfile.py" => '12345abcde', + "conanmanifest.txt" => '12345abcde' + } + end + + it { is_expected.to eq(expected_result) } + end + end + + describe '#package_urls' do + let(:reference) { conan_package_reference } + + subject do + described_class.new( + recipe, user, project, conan_package_reference: reference + ).package_urls + end + + context 'no existing package' do + let(:recipe) { "my-pkg/v1.0.0/#{project.full_path}/stable" } + + it { is_expected.to be_empty } + end + + it_behaves_like 'not selecting a package with the wrong type' + + context 'existing package' do + let(:package) { create(:conan_package, project: project) } + let(:recipe) { package.conan_recipe } + + let(:expected_result) do + { + "conaninfo.txt" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/package/#{conan_package_reference}/0/conaninfo.txt", + "conanmanifest.txt" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/package/#{conan_package_reference}/0/conanmanifest.txt", + "conan_package.tgz" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/package/#{conan_package_reference}/0/conan_package.tgz" + } + end + + it { is_expected.to eq(expected_result) } + + context 'multiple packages with different references' do + let(:info_file) { create(:conan_package_file, :conan_package_info, package: package) } + let(:manifest_file) { create(:conan_package_file, :conan_package_manifest, package: package) } + let(:package_file) { create(:conan_package_file, :conan_package, package: package) } + let(:alternative_reference) { 'abcdefghi' } + + before do + [info_file, manifest_file, package_file].each do |file| + file.conan_file_metadatum.conan_package_reference = alternative_reference + file.save + end + end + + it { is_expected.to eq(expected_result) } + + context 'requesting the alternative reference' do + let(:reference) { alternative_reference } + + let(:expected_result) do + { + "conaninfo.txt" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/package/#{alternative_reference}/0/conaninfo.txt", + "conanmanifest.txt" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/package/#{alternative_reference}/0/conanmanifest.txt", + "conan_package.tgz" => "#{Settings.build_base_gitlab_url}/api/v4/packages/conan/v1/files/#{package.conan_recipe_path}/0/package/#{alternative_reference}/0/conan_package.tgz" + } + end + + it { is_expected.to eq(expected_result) } + end + + it 'returns empty if the reference does not exist' do + result = described_class.new( + recipe, user, project, conan_package_reference: 'doesnotexist' + ).package_urls + + expect(result).to eq({}) + end + end + end + end + + describe '#package_snapshot' do + let(:reference) { conan_package_reference } + + subject do + described_class.new( + recipe, user, project, conan_package_reference: reference + ).package_snapshot + end + + context 'no existing package' do + let(:recipe) { "my-pkg/v1.0.0/#{project.full_path}/stable" } + + it { is_expected.to be_empty } + end + + it_behaves_like 'not selecting a package with the wrong type' + + context 'existing package' do + let(:package) { create(:conan_package, project: project) } + let(:recipe) { package.conan_recipe } + + let(:expected_result) do + { + "conaninfo.txt" => '12345abcde', + "conanmanifest.txt" => '12345abcde', + "conan_package.tgz" => '12345abcde' + } + end + + it { is_expected.to eq(expected_result) } + + context 'when requested with invalid reference' do + let(:reference) { 'invalid' } + + it { is_expected.to eq({}) } + end + end + end +end diff --git a/spec/presenters/packages/detail/package_presenter_spec.rb b/spec/presenters/packages/detail/package_presenter_spec.rb new file mode 100644 index 00000000000..34582957364 --- /dev/null +++ b/spec/presenters/packages/detail/package_presenter_spec.rb @@ -0,0 +1,98 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe ::Packages::Detail::PackagePresenter do + let_it_be(:user) { create(:user) } + let_it_be(:project) { create(:project, creator: user) } + let_it_be(:package) { create(:npm_package, :with_build, project: project) } + let(:presenter) { described_class.new(package) } + + let_it_be(:user_info) { { name: user.name, avatar_url: user.avatar_url } } + let!(:expected_package_files) do + npm_file = package.package_files.first + [{ + created_at: npm_file.created_at, + download_path: npm_file.download_path, + file_name: npm_file.file_name, + size: npm_file.size + }] + end + let(:pipeline_info) do + pipeline = package.build_info.pipeline + { + created_at: pipeline.created_at, + id: pipeline.id, + sha: pipeline.sha, + ref: pipeline.ref, + git_commit_message: pipeline.git_commit_message, + user: user_info, + project: { + name: pipeline.project.name, + web_url: pipeline.project.web_url + } + } + end + let!(:dependency_links) { [] } + let!(:expected_package_details) do + { + id: package.id, + created_at: package.created_at, + name: package.name, + package_files: expected_package_files, + package_type: package.package_type, + project_id: package.project_id, + tags: package.tags.as_json, + updated_at: package.updated_at, + version: package.version, + dependency_links: dependency_links + } + end + + context 'detail_view' do + context 'with build_info' do + let_it_be(:package) { create(:npm_package, :with_build, project: project) } + let(:expected_package_details) { super().merge(pipeline: pipeline_info) } + + it 'returns details with pipeline' do + expect(presenter.detail_view).to eq expected_package_details + end + end + + context 'without build info' do + let_it_be(:package) { create(:npm_package, project: project) } + + it 'returns details without pipeline' do + expect(presenter.detail_view).to eq expected_package_details + end + end + + context 'with nuget_metadatum' do + let_it_be(:package) { create(:nuget_package, project: project) } + let_it_be(:nuget_metadatum) { create(:nuget_metadatum, package: package) } + let(:expected_package_details) { super().merge(nuget_metadatum: nuget_metadatum) } + + it 'returns nuget_metadatum' do + expect(presenter.detail_view).to eq expected_package_details + end + end + + context 'with dependency_links' do + let_it_be(:package) { create(:nuget_package, project: project) } + let_it_be(:dependency_link) { create(:packages_dependency_link, package: package) } + let_it_be(:nuget_dependency) { create(:nuget_dependency_link_metadatum, dependency_link: dependency_link) } + let_it_be(:expected_link) do + { + name: dependency_link.dependency.name, + version_pattern: dependency_link.dependency.version_pattern, + target_framework: nuget_dependency.target_framework + } + end + let_it_be(:dependency_links) { [expected_link] } + + it 'returns the correct dependency link' do + expect(presenter.detail_view).to eq expected_package_details + end + end + end +end diff --git a/spec/presenters/packages/npm/package_presenter_spec.rb b/spec/presenters/packages/npm/package_presenter_spec.rb new file mode 100644 index 00000000000..0e8cda5bafd --- /dev/null +++ b/spec/presenters/packages/npm/package_presenter_spec.rb @@ -0,0 +1,65 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe ::Packages::Npm::PackagePresenter do + let_it_be(:project) { create(:project) } + let_it_be(:package_name) { "@#{project.root_namespace.path}/test" } + let!(:package1) { create(:npm_package, version: '1.0.4', project: project, name: package_name) } + let!(:package2) { create(:npm_package, version: '1.0.6', project: project, name: package_name) } + let!(:latest_package) { create(:npm_package, version: '1.0.11', project: project, name: package_name) } + let(:packages) { project.packages.npm.with_name(package_name).last_of_each_version } + let(:presenter) { described_class.new(package_name, packages) } + + describe '#versions' do + subject { presenter.versions } + + context 'for packages without dependencies' do + it { is_expected.to be_a(Hash) } + it { expect(subject[package1.version]).to match_schema('public_api/v4/packages/npm_package_version') } + it { expect(subject[package2.version]).to match_schema('public_api/v4/packages/npm_package_version') } + + described_class::NPM_VALID_DEPENDENCY_TYPES.each do |dependency_type| + it { expect(subject.dig(package1.version, dependency_type)).to be nil } + it { expect(subject.dig(package2.version, dependency_type)).to be nil } + end + end + + context 'for packages with dependencies' do + described_class::NPM_VALID_DEPENDENCY_TYPES.each do |dependency_type| + let!("package_dependency_link_for_#{dependency_type}") { create(:packages_dependency_link, package: package1, dependency_type: dependency_type) } + end + + it { is_expected.to be_a(Hash) } + it { expect(subject[package1.version]).to match_schema('public_api/v4/packages/npm_package_version') } + it { expect(subject[package2.version]).to match_schema('public_api/v4/packages/npm_package_version') } + described_class::NPM_VALID_DEPENDENCY_TYPES.each do |dependency_type| + it { expect(subject.dig(package1.version, dependency_type.to_s)).to be_any } + end + end + end + + describe '#dist_tags' do + subject { presenter.dist_tags } + + context 'for packages without tags' do + it { is_expected.to be_a(Hash) } + it { expect(subject["latest"]).to eq(latest_package.version) } + end + + context 'for packages with tags' do + let!(:package_tag1) { create(:packages_tag, package: package1, name: 'release_a') } + let!(:package_tag2) { create(:packages_tag, package: package1, name: 'test_release') } + let!(:package_tag3) { create(:packages_tag, package: package2, name: 'release_b') } + let!(:package_tag4) { create(:packages_tag, package: latest_package, name: 'release_c') } + let!(:package_tag5) { create(:packages_tag, package: latest_package, name: 'latest') } + + it { is_expected.to be_a(Hash) } + it { expect(subject[package_tag1.name]).to eq(package1.version) } + it { expect(subject[package_tag2.name]).to eq(package1.version) } + it { expect(subject[package_tag3.name]).to eq(package2.version) } + it { expect(subject[package_tag4.name]).to eq(latest_package.version) } + it { expect(subject[package_tag5.name]).to eq(latest_package.version) } + end + end +end diff --git a/spec/presenters/packages/nuget/package_metadata_presenter_spec.rb b/spec/presenters/packages/nuget/package_metadata_presenter_spec.rb new file mode 100644 index 00000000000..d5e7b23d785 --- /dev/null +++ b/spec/presenters/packages/nuget/package_metadata_presenter_spec.rb @@ -0,0 +1,52 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe Packages::Nuget::PackageMetadataPresenter do + include_context 'with expected presenters dependency groups' + + let_it_be(:package) { create(:nuget_package, :with_metadatum) } + let_it_be(:tag1) { create(:packages_tag, name: 'tag1', package: package) } + let_it_be(:tag2) { create(:packages_tag, name: 'tag2', package: package) } + let_it_be(:presenter) { described_class.new(package) } + + describe '#json_url' do + let_it_be(:expected_suffix) { "/api/v4/projects/#{package.project_id}/packages/nuget/metadata/#{package.name}/#{package.version}.json" } + + subject { presenter.json_url } + + it { is_expected.to end_with(expected_suffix) } + end + + describe '#archive_url' do + let_it_be(:expected_suffix) { "/api/v4/projects/#{package.project_id}/packages/nuget/download/#{package.name}/#{package.version}/#{package.package_files.last.file_name}" } + + subject { presenter.archive_url } + + it { is_expected.to end_with(expected_suffix) } + end + + describe '#catalog_entry' do + subject { presenter.catalog_entry } + + before do + create_dependencies_for(package) + end + + it 'returns an entry structure' do + entry = subject + + expect(entry).to be_a Hash + %i[json_url archive_url].each { |field| expect(entry[field]).not_to be_blank } + %i[authors summary].each { |field| expect(entry[field]).to be_blank } + expect(entry[:dependency_groups]).to eq expected_dependency_groups(package.project_id, package.name, package.version) + expect(entry[:package_name]).to eq package.name + expect(entry[:package_version]).to eq package.version + expect(entry[:tags].split(::Packages::Tag::NUGET_TAGS_SEPARATOR)).to contain_exactly('tag1', 'tag2') + + %i[project_url license_url icon_url].each do |field| + expect(entry.dig(:metadatum, field)).to eq(package.nuget_metadatum.send(field)) + end + end + end +end diff --git a/spec/presenters/packages/nuget/packages_metadata_presenter_spec.rb b/spec/presenters/packages/nuget/packages_metadata_presenter_spec.rb new file mode 100644 index 00000000000..b2bcdf8f03d --- /dev/null +++ b/spec/presenters/packages/nuget/packages_metadata_presenter_spec.rb @@ -0,0 +1,66 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe Packages::Nuget::PackagesMetadataPresenter do + include_context 'with expected presenters dependency groups' + + let_it_be(:project) { create(:project) } + let_it_be(:packages) { create_list(:nuget_package, 5, :with_metadatum, name: 'Dummy.Package', project: project) } + let_it_be(:presenter) { described_class.new(packages) } + + describe '#count' do + subject { presenter.count } + + it {is_expected.to eq 1} + end + + describe '#items' do + let(:tag_names) { %w(tag1 tag2) } + + subject { presenter.items } + + before do + packages.each do |pkg| + tag_names.each { |tag| create(:packages_tag, package: pkg, name: tag) } + + create_dependencies_for(pkg) + end + end + + it 'returns an array' do + items = subject + + expect(items).to be_a Array + expect(items.size).to eq 1 + end + + it 'returns a summary structure' do + item = subject.first + + expect(item).to be_a Hash + %i[json_url lower_version upper_version].each { |field| expect(item[field]).not_to be_blank } + expect(item[:packages_count]).to eq packages.count + expect(item[:packages]).to be_a Array + expect(item[:packages].size).to eq packages.count + end + + it 'returns the catalog entries' do + item = subject.first + + item[:packages].each do |pkg| + expect(pkg).to be_a Hash + %i[json_url archive_url catalog_entry].each { |field| expect(pkg[field]).not_to be_blank } + catalog_entry = pkg[:catalog_entry] + %i[json_url archive_url package_name package_version].each { |field| expect(catalog_entry[field]).not_to be_blank } + %i[authors summary].each { |field| expect(catalog_entry[field]).to be_blank } + expect(catalog_entry[:dependency_groups]).to eq(expected_dependency_groups(project.id, catalog_entry[:package_name], catalog_entry[:package_version])) + expect(catalog_entry[:tags].split(::Packages::Tag::NUGET_TAGS_SEPARATOR)).to contain_exactly('tag1', 'tag2') + + %i[project_url license_url icon_url].each do |field| + expect(catalog_entry.dig(:metadatum, field)).not_to be_blank + end + end + end + end +end diff --git a/spec/presenters/packages/nuget/packages_versions_presenter_spec.rb b/spec/presenters/packages/nuget/packages_versions_presenter_spec.rb new file mode 100644 index 00000000000..36aa28243a4 --- /dev/null +++ b/spec/presenters/packages/nuget/packages_versions_presenter_spec.rb @@ -0,0 +1,14 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe Packages::Nuget::PackagesVersionsPresenter do + let_it_be(:packages) { create_list(:nuget_package, 5) } + let_it_be(:presenter) { described_class.new(::Packages::Package.all) } + + describe '#versions' do + subject { presenter.versions } + + it { is_expected.to match_array(packages.map(&:version).sort) } + end +end diff --git a/spec/presenters/packages/nuget/search_results_presenter_spec.rb b/spec/presenters/packages/nuget/search_results_presenter_spec.rb new file mode 100644 index 00000000000..29ec8579dc1 --- /dev/null +++ b/spec/presenters/packages/nuget/search_results_presenter_spec.rb @@ -0,0 +1,59 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe Packages::Nuget::SearchResultsPresenter do + let_it_be(:project) { create(:project) } + let_it_be(:package_a) { create(:nuget_package, :with_metadatum, project: project, name: 'DummyPackageA') } + let_it_be(:tag1) { create(:packages_tag, package: package_a, name: 'tag1') } + let_it_be(:tag2) { create(:packages_tag, package: package_a, name: 'tag2') } + let_it_be(:packages_b) { create_list(:nuget_package, 5, project: project, name: 'DummyPackageB') } + let_it_be(:packages_c) { create_list(:nuget_package, 5, project: project, name: 'DummyPackageC') } + let_it_be(:search_results) { OpenStruct.new(total_count: 3, results: [package_a, packages_b, packages_c].flatten) } + let_it_be(:presenter) { described_class.new(search_results) } + let(:total_count) { presenter.total_count } + let(:data) { presenter.data } + + describe '#total_count' do + it 'expects to have 3 total elements' do + expect(total_count).to eq(3) + end + end + + describe '#data' do + it 'returns the proper data structure' do + expect(data.size).to eq 3 + pkg_a, pkg_b, pkg_c = data + expect_package_result(pkg_a, package_a.name, [package_a.version], %w(tag1 tag2), with_metadatum: true) + expect_package_result(pkg_b, packages_b.first.name, packages_b.map(&:version)) + expect_package_result(pkg_c, packages_c.first.name, packages_c.map(&:version)) + end + + # rubocop:disable Metrics/AbcSize + def expect_package_result(package_json, name, versions, tags = [], with_metadatum: false) + expect(package_json[:type]).to eq 'Package' + expect(package_json[:authors]).to be_blank + expect(package_json[:name]).to eq(name) + expect(package_json[:summary]).to be_blank + expect(package_json[:total_downloads]).to eq 0 + expect(package_json[:verified]).to be + expect(package_json[:version]).to eq VersionSorter.sort(versions).last # rubocop: disable Style/UnneededSort + versions.zip(package_json[:versions]).each do |version, version_json| + expect(version_json[:json_url]).to end_with("#{version}.json") + expect(version_json[:downloads]).to eq 0 + expect(version_json[:version]).to eq version + end + + if tags.any? + expect(package_json[:tags].split(::Packages::Tag::NUGET_TAGS_SEPARATOR)).to contain_exactly(*tags) + else + expect(package_json[:tags]).to be_blank + end + + %i[project_url license_url icon_url].each do |field| + expect(package_json.dig(:metadatum, field)).to with_metadatum ? be_present : be_blank + end + end + # rubocop:enable Metrics/AbcSize + end +end diff --git a/spec/presenters/packages/nuget/service_index_presenter_spec.rb b/spec/presenters/packages/nuget/service_index_presenter_spec.rb new file mode 100644 index 00000000000..19ef890e19f --- /dev/null +++ b/spec/presenters/packages/nuget/service_index_presenter_spec.rb @@ -0,0 +1,28 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe ::Packages::Nuget::ServiceIndexPresenter do + let_it_be(:project) { create(:project) } + let_it_be(:presenter) { described_class.new(project) } + + describe '#version' do + subject { presenter.version } + + it { is_expected.to eq '3.0.0' } + end + + describe '#resources' do + subject { presenter.resources } + + it 'has valid resources' do + expect(subject.size).to eq 8 + subject.each do |resource| + %i[@id @type comment].each do |field| + expect(resource).to have_key(field) + expect(resource[field]).to be_a(String) + end + end + end + end +end diff --git a/spec/presenters/packages/pypi/package_presenter_spec.rb b/spec/presenters/packages/pypi/package_presenter_spec.rb new file mode 100644 index 00000000000..e4d234a4688 --- /dev/null +++ b/spec/presenters/packages/pypi/package_presenter_spec.rb @@ -0,0 +1,49 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe ::Packages::Pypi::PackagePresenter do + using RSpec::Parameterized::TableSyntax + + let_it_be(:project) { create(:project) } + let_it_be(:package_name) { 'sample-project' } + let_it_be(:package1) { create(:pypi_package, project: project, name: package_name, version: '1.0.0') } + let_it_be(:package2) { create(:pypi_package, project: project, name: package_name, version: '2.0.0') } + + let(:packages) { [package1, package2] } + let(:presenter) { described_class.new(packages, project) } + + describe '#body' do + subject { presenter.body} + + shared_examples_for "pypi package presenter" do + let(:file) { package.package_files.first } + let(:filename) { file.file_name } + let(:expected_file) { "<a href=\"http://localhost/api/v4/projects/#{project.id}/packages/pypi/files/#{file.file_sha256}/#{filename}#sha256=#{file.file_sha256}\" data-requires-python=\"#{expected_python_version}\">#{filename}</a><br>" } + + before do + package.pypi_metadatum.required_python = python_version + end + + it { is_expected.to include expected_file } + end + + it_behaves_like "pypi package presenter" do + let(:python_version) { '>=2.7' } + let(:expected_python_version) { '>=2.7' } + let(:package) { package1 } + end + + it_behaves_like "pypi package presenter" do + let(:python_version) { '"><script>alert(1)</script>' } + let(:expected_python_version) { '"><script>alert(1)</script>' } + let(:package) { package1 } + end + + it_behaves_like "pypi package presenter" do + let(:python_version) { '>=2.7, !=3.0' } + let(:expected_python_version) { '>=2.7, !=3.0' } + let(:package) { package2 } + end + end +end |