2019-12-21 20:55:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
RSpec.describe TreeHelper do
|
2018-03-17 18:26:18 +05:30
|
|
|
let(:project) { create(:project, :repository) }
|
|
|
|
let(:repository) { project.repository }
|
2018-12-13 13:39:08 +05:30
|
|
|
let(:sha) { 'c1c67abbaf91f624347bb3ae96eabe3a1b742478' }
|
|
|
|
|
|
|
|
def create_file(filename)
|
|
|
|
project.repository.create_file(
|
|
|
|
project.creator,
|
|
|
|
filename,
|
|
|
|
'test this',
|
|
|
|
message: "Automatically created file #{filename}",
|
|
|
|
branch_name: 'master'
|
|
|
|
)
|
|
|
|
end
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
describe '.render_tree' do
|
2015-09-11 14:41:01 +05:30
|
|
|
before do
|
2018-03-17 18:26:18 +05:30
|
|
|
@id = sha
|
2018-05-09 12:01:36 +05:30
|
|
|
@path = ""
|
2018-03-17 18:26:18 +05:30
|
|
|
@project = project
|
|
|
|
@lfs_blob_ids = []
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'displays all entries without a warning' do
|
|
|
|
tree = repository.tree(sha, 'files')
|
|
|
|
|
|
|
|
html = render_tree(tree)
|
|
|
|
|
|
|
|
expect(html).not_to have_selector('.tree-truncated-warning')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'truncates entries and adds a warning' do
|
|
|
|
stub_const('TreeHelper::FILE_LIMIT', 1)
|
|
|
|
tree = repository.tree(sha, 'files')
|
|
|
|
|
|
|
|
html = render_tree(tree)
|
|
|
|
|
|
|
|
expect(html).to have_selector('.tree-truncated-warning', count: 1)
|
|
|
|
expect(html).to have_selector('.tree-item-file-name', count: 1)
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
describe '.fast_project_blob_path' do
|
|
|
|
it 'generates the same path as project_blob_path' do
|
|
|
|
blob_path = repository.tree(sha, 'with space').entries.first.path
|
|
|
|
fast_path = fast_project_blob_path(project, blob_path)
|
|
|
|
std_path = project_blob_path(project, blob_path)
|
|
|
|
|
|
|
|
expect(fast_path).to eq(std_path)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'generates the same path with encoded file names' do
|
|
|
|
tree = repository.tree(sha, 'encoding')
|
|
|
|
blob_path = tree.entries.find { |entry| entry.path == 'encoding/テスト.txt' }.path
|
|
|
|
fast_path = fast_project_blob_path(project, blob_path)
|
|
|
|
std_path = project_blob_path(project, blob_path)
|
|
|
|
|
|
|
|
expect(fast_path).to eq(std_path)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'respects a configured relative URL' do
|
|
|
|
allow(Gitlab.config.gitlab).to receive(:relative_url_root).and_return('/gitlab/root')
|
|
|
|
blob_path = repository.tree(sha, '').entries.first.path
|
|
|
|
fast_path = fast_project_blob_path(project, blob_path)
|
|
|
|
|
|
|
|
expect(fast_path).to start_with('/gitlab/root')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'encodes files starting with #' do
|
|
|
|
filename = '#test-file'
|
|
|
|
create_file(filename)
|
|
|
|
|
|
|
|
fast_path = fast_project_blob_path(project, filename)
|
|
|
|
|
|
|
|
expect(fast_path).to end_with('%23test-file')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.fast_project_tree_path' do
|
|
|
|
let(:tree_path) { repository.tree(sha, 'with space').path }
|
|
|
|
let(:fast_path) { fast_project_tree_path(project, tree_path) }
|
|
|
|
let(:std_path) { project_tree_path(project, tree_path) }
|
|
|
|
|
|
|
|
it 'generates the same path as project_tree_path' do
|
|
|
|
expect(fast_path).to eq(std_path)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'respects a configured relative URL' do
|
|
|
|
allow(Gitlab.config.gitlab).to receive(:relative_url_root).and_return('/gitlab/root')
|
|
|
|
|
|
|
|
expect(fast_path).to start_with('/gitlab/root')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'encodes files starting with #' do
|
|
|
|
filename = '#test-file'
|
|
|
|
create_file(filename)
|
|
|
|
|
|
|
|
fast_path = fast_project_tree_path(project, filename)
|
|
|
|
|
|
|
|
expect(fast_path).to end_with('%23test-file')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
describe 'flatten_tree' do
|
|
|
|
let(:tree) { repository.tree(sha, 'files') }
|
|
|
|
let(:root_path) { 'files' }
|
|
|
|
let(:tree_item) { tree.entries.find { |entry| entry.path == path } }
|
|
|
|
|
|
|
|
subject { flatten_tree(root_path, tree_item) }
|
2015-04-26 12:48:37 +05:30
|
|
|
|
|
|
|
context "on a directory containing more than one file/directory" do
|
2018-03-17 18:26:18 +05:30
|
|
|
let(:path) { 'files/html' }
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
it "returns the directory name" do
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(subject).to match('html')
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "on a directory containing only one directory" do
|
2018-03-17 18:26:18 +05:30
|
|
|
let(:path) { 'files/flat' }
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
it "returns the flattened path" do
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(subject).to match('flat/path/correct')
|
|
|
|
end
|
|
|
|
|
|
|
|
context "with a nested root path" do
|
|
|
|
let(:root_path) { 'files/flat' }
|
|
|
|
|
|
|
|
it "returns the flattened path with the root path suffix removed" do
|
|
|
|
expect(subject).to match('path/correct')
|
|
|
|
end
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
end
|
2018-05-09 12:01:36 +05:30
|
|
|
|
|
|
|
context 'when the root path contains a plus character' do
|
|
|
|
let(:root_path) { 'gtk/C++' }
|
|
|
|
let(:tree_item) { double(flat_path: 'gtk/C++/glade') }
|
|
|
|
|
|
|
|
it 'returns the flattened path' do
|
|
|
|
expect(subject).to eq('glade')
|
|
|
|
end
|
|
|
|
end
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
describe '#commit_in_single_accessible_branch' do
|
|
|
|
it 'escapes HTML from the branch name' do
|
|
|
|
helper.instance_variable_set(:@branch_name, "<script>alert('escape me!');</script>")
|
|
|
|
escaped_branch_name = '<script>alert('escape me!');</script>'
|
|
|
|
|
|
|
|
expect(helper.commit_in_single_accessible_branch).to include(escaped_branch_name)
|
|
|
|
end
|
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
|
|
|
|
describe '#vue_file_list_data' do
|
|
|
|
before do
|
|
|
|
allow(helper).to receive(:current_user).and_return(nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns a list of attributes related to the project' do
|
|
|
|
expect(helper.vue_file_list_data(project, sha)).to include(
|
|
|
|
can_push_code: nil,
|
|
|
|
fork_path: nil,
|
|
|
|
escaped_ref: sha,
|
|
|
|
ref: sha,
|
|
|
|
project_path: project.full_path,
|
|
|
|
project_short_path: project.path,
|
|
|
|
full_name: project.name_with_namespace
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'user does not have write access but a personal fork exists' do
|
|
|
|
include ProjectForksHelper
|
|
|
|
|
|
|
|
let_it_be(:user) { create(:user) }
|
|
|
|
let!(:forked_project) { create(:project, :repository, namespace: user.namespace) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
project.add_guest(user)
|
|
|
|
fork_project(project, nil, target_project: forked_project)
|
|
|
|
|
|
|
|
allow(helper).to receive(:current_user).and_return(user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'includes fork_path too' do
|
|
|
|
expect(helper.vue_file_list_data(project, sha)).to include(
|
|
|
|
fork_path: forked_project.full_path
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'user has write access' do
|
|
|
|
let_it_be(:user) { create(:user) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
project.add_developer(user)
|
|
|
|
|
|
|
|
allow(helper).to receive(:current_user).and_return(user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'includes can_push_code: true' do
|
|
|
|
expect(helper.vue_file_list_data(project, sha)).to include(
|
|
|
|
can_push_code: "true"
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|