debian-mirror-gitlab/spec/helpers/blob_helper_spec.rb

601 lines
20 KiB
Ruby
Raw Normal View History

2019-12-21 20:55:43 +05:30
# frozen_string_literal: true
2015-09-11 14:41:01 +05:30
require 'spec_helper'
2020-07-28 23:09:34 +05:30
RSpec.describe BlobHelper do
2016-09-13 17:45:13 +05:30
include TreeHelper
2017-08-17 22:00:37 +05:30
describe "#sanitize_svg_data" do
2016-06-02 11:05:42 +05:30
let(:input_svg_path) { File.join(Rails.root, 'spec', 'fixtures', 'unsanitized.svg') }
2018-11-18 11:00:15 +05:30
let(:data) { File.read(input_svg_path) }
2016-06-02 11:05:42 +05:30
let(:expected_svg_path) { File.join(Rails.root, 'spec', 'fixtures', 'sanitized.svg') }
2018-11-18 11:00:15 +05:30
let(:expected) { File.read(expected_svg_path) }
2016-06-02 11:05:42 +05:30
2016-09-13 17:45:13 +05:30
it 'retains essential elements' do
2017-08-17 22:00:37 +05:30
expect(sanitize_svg_data(data)).to eq(expected)
2016-06-02 11:05:42 +05:30
end
end
2016-09-13 17:45:13 +05:30
describe "#edit_blob_link" do
2020-04-08 14:13:33 +05:30
let(:namespace) { create(:namespace, name: 'gitlab') }
2017-08-17 22:00:37 +05:30
let(:project) { create(:project, :repository, namespace: namespace) }
2016-09-13 17:45:13 +05:30
2020-10-24 23:57:45 +05:30
subject(:link) { helper.edit_blob_button(project, 'master', 'README.md') }
2016-09-13 17:45:13 +05:30
before do
2019-09-30 21:07:59 +05:30
allow(helper).to receive(:current_user).and_return(nil)
allow(helper).to receive(:can?).and_return(true)
allow(helper).to receive(:can_collaborate_with_project?).and_return(true)
2016-09-13 17:45:13 +05:30
end
it 'verifies blob is text' do
expect(helper).not_to receive(:blob_text_viewable?)
2019-09-30 21:07:59 +05:30
button = helper.edit_blob_button(project, 'refs/heads/master', 'README.md')
2016-09-13 17:45:13 +05:30
expect(button).to start_with('<button')
end
it 'uses the passed blob instead retrieve from repository' do
blob = project.repository.blob_at('refs/heads/master', 'README.md')
expect(project.repository).not_to receive(:blob_at)
2019-09-30 21:07:59 +05:30
helper.edit_blob_button(project, 'refs/heads/master', 'README.md', blob: blob)
2016-09-13 17:45:13 +05:30
end
it 'returns a link with the proper route' do
2020-03-13 15:44:24 +05:30
expect(Capybara.string(link).find_link('Edit')[:href]).to eq("/#{project.full_path}/-/edit/master/README.md")
2016-09-13 17:45:13 +05:30
end
it 'returns a link with the passed link_opts on the expected route' do
2020-10-24 23:57:45 +05:30
link_with_mr = helper.edit_blob_button(project, 'master', 'README.md', link_opts: { mr_id: 10 })
2016-09-13 17:45:13 +05:30
2020-10-24 23:57:45 +05:30
expect(Capybara.string(link_with_mr).find_link('Edit')[:href]).to eq("/#{project.full_path}/-/edit/master/README.md?mr_id=10")
end
context 'when edit is the primary button' do
before do
stub_feature_flags(web_ide_primary_edit: false)
end
it 'is rendered as primary' do
expect(link).not_to match(/btn-inverted/)
end
it 'passes on primary tracking attributes' do
parsed_link = Capybara.string(link).find_link('Edit')
expect(parsed_link[:'data-track-event']).to eq("click_edit")
expect(parsed_link[:'data-track-label']).to eq("Edit")
expect(parsed_link[:'data-track-property']).to eq(nil)
end
end
context 'when Web IDE is the primary button' do
before do
stub_feature_flags(web_ide_primary_edit: true)
end
it 'is rendered as inverted' do
expect(link).to match(/btn-inverted/)
end
it 'passes on secondary tracking attributes' do
parsed_link = Capybara.string(link).find_link('Edit')
expect(parsed_link[:'data-track-event']).to eq("click_edit")
expect(parsed_link[:'data-track-label']).to eq("Edit")
expect(parsed_link[:'data-track-property']).to eq("secondary")
end
2016-09-13 17:45:13 +05:30
end
end
2017-08-17 22:00:37 +05:30
context 'viewer related' do
include FakeBlobHelpers
2017-09-10 17:25:29 +05:30
let(:project) { build(:project, lfs_enabled: true) }
2017-08-17 22:00:37 +05:30
before do
allow(Gitlab.config.lfs).to receive(:enabled).and_return(true)
end
let(:viewer_class) do
Class.new(BlobViewer::Base) do
2017-09-10 17:25:29 +05:30
include BlobViewer::ServerSide
self.collapse_limit = 1.megabyte
self.size_limit = 5.megabytes
2017-08-17 22:00:37 +05:30
self.type = :rich
end
end
let(:viewer) { viewer_class.new(blob) }
let(:blob) { fake_blob }
describe '#blob_render_error_reason' do
context 'for error :too_large' do
2017-09-10 17:25:29 +05:30
context 'when the blob size is larger than the absolute size limit' do
2017-08-17 22:00:37 +05:30
let(:blob) { fake_blob(size: 10.megabytes) }
it 'returns an error message' do
expect(helper.blob_render_error_reason(viewer)).to eq('it is larger than 5 MB')
end
end
2017-09-10 17:25:29 +05:30
context 'when the blob size is larger than the size limit' do
2017-08-17 22:00:37 +05:30
let(:blob) { fake_blob(size: 2.megabytes) }
it 'returns an error message' do
expect(helper.blob_render_error_reason(viewer)).to eq('it is larger than 1 MB')
end
end
end
context 'for error :server_side_but_stored_externally' do
let(:blob) { fake_blob(lfs: true) }
it 'returns an error message' do
expect(helper.blob_render_error_reason(viewer)).to eq('it is stored in LFS')
end
end
end
describe '#blob_render_error_options' do
before do
assign(:project, project)
assign(:blob, blob)
assign(:id, File.join('master', blob.path))
controller.params[:controller] = 'projects/blob'
controller.params[:action] = 'show'
controller.params[:namespace_id] = project.namespace.to_param
controller.params[:project_id] = project.to_param
controller.params[:id] = File.join('master', blob.path)
end
2017-09-10 17:25:29 +05:30
context 'for error :collapsed' do
let(:blob) { fake_blob(size: 2.megabytes) }
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
it 'includes a "load it anyway" link' do
expect(helper.blob_render_error_options(viewer)).to include(/load it anyway/)
2017-08-17 22:00:37 +05:30
end
2017-09-10 17:25:29 +05:30
end
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
context 'for error :too_large' do
let(:blob) { fake_blob(size: 10.megabytes) }
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
it 'does not include a "load it anyway" link' do
expect(helper.blob_render_error_options(viewer)).not_to include(/load it anyway/)
2017-08-17 22:00:37 +05:30
end
context 'when the viewer is rich' do
context 'the blob is rendered as text' do
let(:blob) { fake_blob(path: 'file.md', size: 2.megabytes) }
it 'includes a "view the source" link' do
expect(helper.blob_render_error_options(viewer)).to include(/view the source/)
end
end
context 'the blob is not rendered as text' do
let(:blob) { fake_blob(path: 'file.pdf', binary: true, size: 2.megabytes) }
it 'does not include a "view the source" link' do
expect(helper.blob_render_error_options(viewer)).not_to include(/view the source/)
end
end
end
context 'when the viewer is not rich' do
before do
viewer_class.type = :simple
end
let(:blob) { fake_blob(path: 'file.md', size: 2.megabytes) }
it 'does not include a "view the source" link' do
expect(helper.blob_render_error_options(viewer)).not_to include(/view the source/)
end
end
it 'includes a "download it" link' do
expect(helper.blob_render_error_options(viewer)).to include(/download it/)
end
end
context 'for error :server_side_but_stored_externally' do
let(:blob) { fake_blob(path: 'file.md', lfs: true) }
it 'does not include a "load it anyway" link' do
expect(helper.blob_render_error_options(viewer)).not_to include(/load it anyway/)
end
it 'does not include a "view the source" link' do
expect(helper.blob_render_error_options(viewer)).not_to include(/view the source/)
end
it 'includes a "download it" link' do
expect(helper.blob_render_error_options(viewer)).to include(/download it/)
end
end
end
2020-04-08 14:13:33 +05:30
describe '#show_suggest_pipeline_creation_celebration?' do
let(:current_user) { create(:user) }
before do
assign(:project, project)
assign(:blob, blob)
assign(:commit, double('Commit', sha: 'whatever'))
helper.request.cookies["suggest_gitlab_ci_yml_commit_#{project.id}"] = 'true'
allow(helper).to receive(:current_user).and_return(current_user)
end
context 'when file is a pipeline config file' do
let(:data) { File.read(Rails.root.join('spec/support/gitlab_stubs/gitlab_ci.yml')) }
let(:blob) { fake_blob(path: Gitlab::FileDetector::PATTERNS[:gitlab_ci], data: data) }
2021-01-29 00:20:46 +05:30
context 'feature enabled' do
2020-04-08 14:13:33 +05:30
it 'is true' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_truthy
end
context 'file is invalid format' do
let(:data) { 'foo' }
it 'is false' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_falsey
end
end
context 'does not use the default ci config' do
before do
project.ci_config_path = 'something_bad'
end
it 'is false' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_falsey
end
end
context 'does not have the needed cookie' do
before do
helper.request.cookies.delete "suggest_gitlab_ci_yml_commit_#{project.id}"
end
it 'is false' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_falsey
end
end
2020-10-24 23:57:45 +05:30
context 'blob does not have auxiliary view' do
before do
allow(blob).to receive(:auxiliary_viewer).and_return(nil)
end
it 'is false' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_falsey
end
end
2020-04-08 14:13:33 +05:30
end
2021-01-29 00:20:46 +05:30
context 'feature disabled' do
2020-04-08 14:13:33 +05:30
before do
2021-01-29 00:20:46 +05:30
stub_feature_flags(suggest_pipeline: false)
2020-04-08 14:13:33 +05:30
end
it 'is false' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_falsey
end
end
end
context 'when file is not a pipeline config file' do
let(:blob) { fake_blob(path: 'LICENSE') }
2021-01-29 00:20:46 +05:30
context 'feature enabled' do
2020-04-08 14:13:33 +05:30
it 'is false' do
expect(helper.show_suggest_pipeline_creation_celebration?).to be_falsey
end
end
end
end
end
describe 'suggest_pipeline_commit_cookie_name' do
let(:project) { create(:project) }
it 'uses project id to make up the cookie name' do
assign(:project, project)
expect(helper.suggest_pipeline_commit_cookie_name).to eq "suggest_gitlab_ci_yml_commit_#{project.id}"
end
2017-08-17 22:00:37 +05:30
end
2018-05-09 12:01:36 +05:30
2020-10-24 23:57:45 +05:30
describe `#ide_edit_button` do
let_it_be(:namespace) { create(:namespace, name: 'gitlab') }
let_it_be(:project) { create(:project, :repository, namespace: namespace) }
let_it_be(:current_user) { create(:user) }
let(:can_push_code) { true }
let(:blob) { project.repository.blob_at('refs/heads/master', 'README.md') }
subject(:link) { helper.ide_edit_button(project, 'master', 'README.md', blob: blob) }
before do
allow(helper).to receive(:current_user).and_return(current_user)
allow(helper).to receive(:can?).with(current_user, :push_code, project).and_return(can_push_code)
allow(helper).to receive(:can_collaborate_with_project?).and_return(true)
end
it 'returns a link with a Web IDE route' do
expect(Capybara.string(link).find_link('Web IDE')[:href]).to eq("/-/ide/project/#{project.full_path}/edit/master/-/README.md")
end
context 'when edit is the primary button' do
before do
stub_feature_flags(web_ide_primary_edit: false)
end
it 'is rendered as inverted' do
expect(link).to match(/btn-inverted/)
end
it 'passes on secondary tracking attributes' do
parsed_link = Capybara.string(link).find_link('Web IDE')
expect(parsed_link[:'data-track-event']).to eq("click_edit_ide")
expect(parsed_link[:'data-track-label']).to eq("Web IDE")
expect(parsed_link[:'data-track-property']).to eq("secondary")
end
end
context 'when Web IDE is the primary button' do
before do
stub_feature_flags(web_ide_primary_edit: true)
end
it 'is rendered as primary' do
expect(link).not_to match(/btn-inverted/)
end
it 'passes on primary tracking attributes' do
parsed_link = Capybara.string(link).find_link('Web IDE')
expect(parsed_link[:'data-track-event']).to eq("click_edit_ide")
expect(parsed_link[:'data-track-label']).to eq("Web IDE")
expect(parsed_link[:'data-track-property']).to eq(nil)
end
end
end
2018-05-09 12:01:36 +05:30
describe '#ide_edit_path' do
let(:project) { create(:project) }
2019-09-30 21:07:59 +05:30
let(:current_user) { create(:user) }
let(:can_push_code) { true }
before do
allow(helper).to receive(:current_user).and_return(current_user)
allow(helper).to receive(:can?).and_return(can_push_code)
end
2018-05-09 12:01:36 +05:30
around do |example|
old_script_name = Rails.application.routes.default_url_options[:script_name]
begin
example.run
ensure
Rails.application.routes.default_url_options[:script_name] = old_script_name
end
end
it 'returns full IDE path' do
Rails.application.routes.default_url_options[:script_name] = nil
2018-11-08 19:23:39 +05:30
expect(helper.ide_edit_path(project, "master", "")).to eq("/-/ide/project/#{project.namespace.path}/#{project.path}/edit/master")
end
it 'returns full IDE path with second -' do
Rails.application.routes.default_url_options[:script_name] = nil
expect(helper.ide_edit_path(project, "testing/slashes", "readme.md")).to eq("/-/ide/project/#{project.namespace.path}/#{project.path}/edit/testing/slashes/-/readme.md")
2018-05-09 12:01:36 +05:30
end
it 'returns IDE path without relative_url_root' do
Rails.application.routes.default_url_options[:script_name] = "/gitlab"
2018-11-08 19:23:39 +05:30
expect(helper.ide_edit_path(project, "master", "")).to eq("/gitlab/-/ide/project/#{project.namespace.path}/#{project.path}/edit/master")
2018-05-09 12:01:36 +05:30
end
2019-07-07 11:18:12 +05:30
it 'escapes special characters' do
Rails.application.routes.default_url_options[:script_name] = nil
2020-03-13 15:44:24 +05:30
expect(helper.ide_edit_path(project, "testing/#hashes", "readme.md#test")).to eq("/-/ide/project/#{project.full_path}/edit/testing/%23hashes/-/readme.md%23test")
expect(helper.ide_edit_path(project, "testing/#hashes", "src#/readme.md#test")).to eq("/-/ide/project/#{project.full_path}/edit/testing/%23hashes/-/src%23/readme.md%23test")
2019-07-07 11:18:12 +05:30
end
it 'does not escape "/" character' do
Rails.application.routes.default_url_options[:script_name] = nil
expect(helper.ide_edit_path(project, "testing/slashes", "readme.md/")).to eq("/-/ide/project/#{project.namespace.path}/#{project.path}/edit/testing/slashes/-/readme.md/")
end
2019-09-30 21:07:59 +05:30
context 'when user is not logged in' do
let(:current_user) { nil }
it 'returns IDE path inside the project' do
expect(helper.ide_edit_path(project, "master", "")).to eq("/-/ide/project/#{project.namespace.path}/#{project.path}/edit/master")
end
end
context 'when user cannot push to the project' do
let(:can_push_code) { false }
it "returns IDE path with the user's fork" do
expect(helper.ide_edit_path(project, "master", "")).to eq("/-/ide/project/#{current_user.namespace.full_path}/#{project.path}/edit/master")
end
end
2018-05-09 12:01:36 +05:30
end
2019-12-21 20:55:43 +05:30
2021-01-29 00:20:46 +05:30
describe '#ide_merge_request_path' do
let_it_be(:project) { create(:project, :repository) }
let_it_be(:merge_request) { create(:merge_request, source_project: project)}
it 'returns IDE path for the given MR if MR is not merged' do
expect(helper.ide_merge_request_path(merge_request)).to eq("/-/ide/project/#{project.full_path}/merge_requests/#{merge_request.iid}")
end
context 'when the MR comes from a fork' do
include ProjectForksHelper
let(:forked_project) { fork_project(project, nil, repository: true) }
let(:merge_request) { create(:merge_request, source_project: forked_project, target_project: project) }
it 'returns IDE path for MR in the forked repo with target project included as param' do
expect(helper.ide_merge_request_path(merge_request)).to eq("/-/ide/project/#{forked_project.full_path}/merge_requests/#{merge_request.iid}?target_project=#{CGI.escape(project.full_path)}")
end
end
context 'when the MR is merged' do
let(:current_user) { build(:user) }
let_it_be(:merge_request) { create(:merge_request, :merged, source_project: project, source_branch: 'testing-1', target_branch: 'feature-1') }
before do
allow(helper).to receive(:current_user).and_return(current_user)
allow(helper).to receive(:can?).and_return(true)
end
it 'returns default IDE url with master branch' do
expect(helper.ide_merge_request_path(merge_request)).to eq("/-/ide/project/#{project.full_path}/edit/master")
end
it 'includes file path passed' do
expect(helper.ide_merge_request_path(merge_request, 'README.md')).to eq("/-/ide/project/#{project.full_path}/edit/master/-/README.md")
end
context 'when target branch exists' do
before do
allow(merge_request).to receive(:target_branch_exists?).and_return(true)
end
it 'returns IDE edit url with the target branch' do
expect(helper.ide_merge_request_path(merge_request)).to eq("/-/ide/project/#{project.full_path}/edit/feature-1")
end
end
end
end
2019-12-21 20:55:43 +05:30
describe '#ide_fork_and_edit_path' do
2021-01-03 14:25:43 +05:30
let_it_be(:project) { create(:project) }
let_it_be(:user) { create(:user) }
let(:current_user) { user }
2019-12-21 20:55:43 +05:30
before do
allow(helper).to receive(:current_user).and_return(current_user)
2021-01-03 14:25:43 +05:30
allow(helper).to receive(:can?).and_return(true)
2019-12-21 20:55:43 +05:30
end
it 'returns path to fork the repo with a redirect param to the full IDE path' do
uri = URI(helper.ide_fork_and_edit_path(project, "master", ""))
params = CGI.unescape(uri.query)
expect(uri.path).to eq("/#{project.namespace.path}/#{project.path}/-/forks")
expect(params).to include("continue[to]=/-/ide/project/#{project.namespace.path}/#{project.path}/edit/master")
expect(params).to include("namespace_key=#{current_user.namespace.id}")
end
context 'when user is not logged in' do
let(:current_user) { nil }
it 'returns nil' do
expect(helper.ide_fork_and_edit_path(project, "master", "")).to be_nil
end
end
end
2020-11-24 15:15:51 +05:30
2021-01-03 14:25:43 +05:30
describe '#fork_and_edit_path' do
let_it_be(:project) { create(:project) }
let_it_be(:user) { create(:user) }
let(:current_user) { user }
before do
allow(helper).to receive(:current_user).and_return(current_user)
allow(helper).to receive(:can?).and_return(true)
end
it 'returns path to fork the repo with a redirect param to the full edit path' do
uri = URI(helper.fork_and_edit_path(project, "master", ""))
params = CGI.unescape(uri.query)
expect(uri.path).to eq("/#{project.namespace.path}/#{project.path}/-/forks")
expect(params).to include("continue[to]=/#{project.namespace.path}/#{project.path}/-/edit/master/")
expect(params).to include("namespace_key=#{current_user.namespace.id}")
end
context 'when user is not logged in' do
let(:current_user) { nil }
it 'returns nil' do
expect(helper.ide_fork_and_edit_path(project, "master", "")).to be_nil
end
end
end
2020-11-24 15:15:51 +05:30
describe '#editing_ci_config?' do
let(:project) { build(:project) }
subject { helper.editing_ci_config? }
before do
assign(:project, project)
assign(:path, path)
end
context 'when path is nil' do
let(:path) { nil }
it { is_expected.to be_falsey }
end
context 'when path is not a ci file' do
let(:path) { 'some-file.txt' }
it { is_expected.to be_falsey }
end
context 'when path ends is gitlab-ci.yml' do
let(:path) { '.gitlab-ci.yml' }
it { is_expected.to be_truthy }
end
context 'when path ends with gitlab-ci.yml' do
let(:path) { 'template.gitlab-ci.yml' }
it { is_expected.to be_truthy }
end
context 'with custom ci paths' do
let(:path) { 'path/to/ci.yaml' }
before do
project.ci_config_path = 'path/to/ci.yaml'
end
it { is_expected.to be_truthy }
end
context 'with custom ci config and path' do
let(:path) { 'path/to/template.gitlab-ci.yml' }
before do
project.ci_config_path = 'ci/path/.gitlab-ci.yml@another-group/another-project'
end
it { is_expected.to be_truthy }
end
end
2015-09-11 14:41:01 +05:30
end