2018-12-05 23:21:45 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
module TreeHelper
|
2018-03-17 18:26:18 +05:30
|
|
|
FILE_LIMIT = 1_000
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
# Sorts a repository's tree so that folders are before files and renders
|
|
|
|
# their corresponding partials
|
|
|
|
#
|
2018-03-17 18:26:18 +05:30
|
|
|
# tree - A `Tree` object for the current tree
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2014-09-02 18:07:02 +05:30
|
|
|
def render_tree(tree)
|
2016-09-13 17:45:13 +05:30
|
|
|
# Sort submodules and folders together by name ahead of files
|
2014-09-02 18:07:02 +05:30
|
|
|
folders, files, submodules = tree.trees, tree.blobs, tree.submodules
|
2018-12-05 23:21:45 +05:30
|
|
|
tree = []
|
2016-09-13 17:45:13 +05:30
|
|
|
items = (folders + submodules).sort_by(&:name) + files
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
if items.size > FILE_LIMIT
|
|
|
|
tree << render(partial: 'projects/tree/truncated_notice_tree_row',
|
|
|
|
locals: { limit: FILE_LIMIT, total: items.size })
|
|
|
|
items = items.take(FILE_LIMIT)
|
|
|
|
end
|
|
|
|
|
|
|
|
tree << render(partial: 'projects/tree/tree_row', collection: items) if items.present?
|
2018-12-05 23:21:45 +05:30
|
|
|
tree.join.html_safe
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
# Return an image icon depending on the file type and mode
|
2014-09-02 18:07:02 +05:30
|
|
|
#
|
|
|
|
# type - String type of the tree item; either 'folder' or 'file'
|
2015-04-26 12:48:37 +05:30
|
|
|
# mode - File unix mode
|
|
|
|
# name - File name
|
|
|
|
def tree_icon(type, mode, name)
|
2018-12-13 13:39:08 +05:30
|
|
|
icon([file_type_icon_class(type, mode, name), 'fw'])
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
# Using Rails `*_path` methods can be slow, especially when generating
|
|
|
|
# many paths, as with a repository tree that has thousands of items.
|
|
|
|
def fast_project_blob_path(project, blob_path)
|
|
|
|
ActionDispatch::Journey::Router::Utils.escape_path(
|
2020-03-13 15:44:24 +05:30
|
|
|
File.join(relative_url_root, project.path_with_namespace, '-', 'blob', blob_path)
|
2018-12-13 13:39:08 +05:30
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def fast_project_tree_path(project, tree_path)
|
|
|
|
ActionDispatch::Journey::Router::Utils.escape_path(
|
2020-03-13 15:44:24 +05:30
|
|
|
File.join(relative_url_root, project.path_with_namespace, '-', 'tree', tree_path)
|
2018-12-13 13:39:08 +05:30
|
|
|
)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# Simple shortcut to File.join
|
|
|
|
def tree_join(*args)
|
|
|
|
File.join(*args)
|
|
|
|
end
|
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
def on_top_of_branch?(project = @project, ref = @ref)
|
2017-08-17 22:00:37 +05:30
|
|
|
project.repository.branch_exists?(ref)
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
|
|
|
|
2016-01-14 18:37:52 +05:30
|
|
|
def can_edit_tree?(project = nil, ref = nil)
|
2015-04-26 12:48:37 +05:30
|
|
|
project ||= @project
|
|
|
|
ref ||= @ref
|
2016-01-14 18:37:52 +05:30
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
return false unless on_top_of_branch?(project, ref)
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
can_collaborate_with_project?(project, ref: ref)
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
def tree_edit_branch(project = @project, ref = @ref)
|
2016-01-14 18:37:52 +05:30
|
|
|
return unless can_edit_tree?(project, ref)
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
if user_access(project).can_push_to_branch?(ref)
|
2016-01-14 18:37:52 +05:30
|
|
|
ref
|
|
|
|
else
|
|
|
|
project = tree_edit_project(project)
|
2016-06-02 11:05:42 +05:30
|
|
|
project.repository.next_branch('patch')
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def tree_edit_project(project = @project)
|
|
|
|
if can?(current_user, :push_code, project)
|
|
|
|
project
|
|
|
|
elsif current_user && current_user.already_forked?(project)
|
|
|
|
current_user.fork_of(project)
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2016-01-14 18:37:52 +05:30
|
|
|
def edit_in_new_fork_notice_now
|
2019-07-07 11:18:12 +05:30
|
|
|
_("You're not allowed to make changes to this project directly. "\
|
|
|
|
"A fork of this project is being created that you can make changes in, so you can submit a merge request.")
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def edit_in_new_fork_notice
|
2019-07-07 11:18:12 +05:30
|
|
|
_("You're not allowed to make changes to this project directly. "\
|
|
|
|
"A fork of this project has been created that you can make changes in, so you can submit a merge request.")
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
def edit_in_new_fork_notice_action(action)
|
2019-07-07 11:18:12 +05:30
|
|
|
edit_in_new_fork_notice + _(" Try to %{action} this file again.") % { action: action }
|
2018-03-27 19:54:05 +05:30
|
|
|
end
|
|
|
|
|
2016-01-14 18:37:52 +05:30
|
|
|
def commit_in_fork_help
|
2018-03-27 19:54:05 +05:30
|
|
|
_("A new branch will be created in your fork and a new merge request will be started.")
|
|
|
|
end
|
|
|
|
|
|
|
|
def commit_in_single_accessible_branch
|
2018-10-15 14:42:47 +05:30
|
|
|
branch_name = ERB::Util.html_escape(selected_branch)
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
message = _("Your changes can be committed to %{branch_name} because a merge "\
|
|
|
|
"request is open.") % { branch_name: "<strong>#{branch_name}</strong>" }
|
|
|
|
|
|
|
|
message.html_safe
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
def path_breadcrumbs(max_links = 6)
|
2014-09-02 18:07:02 +05:30
|
|
|
if @path.present?
|
|
|
|
part_path = ""
|
2015-04-26 12:48:37 +05:30
|
|
|
parts = @path.split('/')
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
yield('..', File.join(*parts.first(parts.count - 2))) if parts.count > max_links
|
2014-09-02 18:07:02 +05:30
|
|
|
|
|
|
|
parts.each do |part|
|
|
|
|
part_path = File.join(part_path, part) unless part_path.empty?
|
|
|
|
part_path = part if part_path.empty?
|
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
next if parts.count > max_links && !parts.last(2).include?(part)
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
yield(part, part_path)
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
def up_dir_path
|
2014-09-02 18:07:02 +05:30
|
|
|
file = File.join(@path, "..")
|
|
|
|
tree_join(@ref, file)
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
# returns the relative path of the first subdir that doesn't have only one directory descendant
|
2018-03-17 18:26:18 +05:30
|
|
|
def flatten_tree(root_path, tree)
|
2019-05-03 19:53:19 +05:30
|
|
|
tree.flat_path.sub(%r{\A#{Regexp.escape(root_path)}/}, '')
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
def selected_branch
|
|
|
|
@branch_name || tree_edit_branch
|
|
|
|
end
|
2018-12-13 13:39:08 +05:30
|
|
|
|
|
|
|
def relative_url_root
|
|
|
|
Gitlab.config.gitlab.relative_url_root.presence || '/'
|
|
|
|
end
|
2019-09-30 21:07:59 +05:30
|
|
|
|
|
|
|
# project and path are used on the EE version
|
|
|
|
def tree_content_data(logs_path, project, path)
|
|
|
|
{
|
|
|
|
"logs-path" => logs_path
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def breadcrumb_data_attributes
|
|
|
|
attrs = {
|
|
|
|
can_collaborate: can_collaborate_with_project?(@project).to_s,
|
2020-01-01 13:55:28 +05:30
|
|
|
new_blob_path: project_new_blob_path(@project, @ref),
|
|
|
|
upload_path: project_create_blob_path(@project, @ref),
|
|
|
|
new_dir_path: project_create_dir_path(@project, @ref),
|
2019-09-30 21:07:59 +05:30
|
|
|
new_branch_path: new_project_branch_path(@project),
|
|
|
|
new_tag_path: new_project_tag_path(@project),
|
|
|
|
can_edit_tree: can_edit_tree?.to_s
|
|
|
|
}
|
|
|
|
|
|
|
|
if can?(current_user, :fork_project, @project) && can?(current_user, :create_merge_request_in, @project)
|
|
|
|
continue_param = {
|
|
|
|
to: project_new_blob_path(@project, @id),
|
|
|
|
notice: edit_in_new_fork_notice,
|
|
|
|
notice_now: edit_in_new_fork_notice_now
|
|
|
|
}
|
|
|
|
|
|
|
|
attrs.merge!(
|
|
|
|
fork_new_blob_path: project_forks_path(@project, namespace_key: current_user.namespace.id, continue: continue_param),
|
|
|
|
fork_new_directory_path: project_forks_path(@project, namespace_key: current_user.namespace.id, continue: continue_param.merge({
|
|
|
|
to: request.fullpath,
|
|
|
|
notice: _("%{edit_in_new_fork_notice} Try to create a new directory again.") % { edit_in_new_fork_notice: edit_in_new_fork_notice }
|
|
|
|
})),
|
|
|
|
fork_upload_blob_path: project_forks_path(@project, namespace_key: current_user.namespace.id, continue: continue_param.merge({
|
|
|
|
to: request.fullpath,
|
|
|
|
notice: _("%{edit_in_new_fork_notice} Try to upload a file again.") % { edit_in_new_fork_notice: edit_in_new_fork_notice }
|
|
|
|
}))
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
attrs
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
def vue_file_list_data(project, ref)
|
|
|
|
{
|
|
|
|
project_path: project.full_path,
|
|
|
|
project_short_path: project.path,
|
|
|
|
ref: ref,
|
2020-05-05 14:28:15 +05:30
|
|
|
escaped_ref: ActionDispatch::Journey::Router::Utils.escape_path(ref),
|
2019-12-26 22:10:19 +05:30
|
|
|
full_name: project.name_with_namespace
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
def ide_base_path(project)
|
|
|
|
can_push_code = current_user&.can?(:push_code, project)
|
|
|
|
fork_path = current_user&.fork_of(project)&.full_path
|
|
|
|
|
|
|
|
if can_push_code
|
|
|
|
project.full_path
|
|
|
|
else
|
|
|
|
fork_path || project.full_path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def vue_ide_link_data(project, ref)
|
|
|
|
can_collaborate = can_collaborate_with_project?(project)
|
|
|
|
can_create_mr_from_fork = can?(current_user, :fork_project, project) && can?(current_user, :create_merge_request_in, project)
|
|
|
|
show_web_ide_button = (can_collaborate || current_user&.already_forked?(project) || can_create_mr_from_fork)
|
|
|
|
|
|
|
|
{
|
|
|
|
ide_base_path: ide_base_path(project),
|
|
|
|
needs_to_fork: !can_collaborate && !current_user&.already_forked?(project),
|
|
|
|
show_web_ide_button: show_web_ide_button,
|
|
|
|
show_gitpod_button: show_web_ide_button && Gitlab::Gitpod.feature_and_settings_enabled?(project),
|
|
|
|
gitpod_url: full_gitpod_url(project, ref),
|
|
|
|
gitpod_enabled: current_user&.gitpod_enabled
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def full_gitpod_url(project, ref)
|
|
|
|
return "" unless Gitlab::Gitpod.feature_and_settings_enabled?(project)
|
|
|
|
|
|
|
|
"#{Gitlab::CurrentSettings.gitpod_url}##{project_tree_url(project, tree_join(ref, @path || ''))}"
|
|
|
|
end
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
def directory_download_links(project, ref, archive_prefix)
|
|
|
|
Gitlab::Workhorse::ARCHIVE_FORMATS.map do |fmt|
|
|
|
|
{
|
|
|
|
text: fmt,
|
|
|
|
path: project_archive_path(project, id: tree_join(ref, archive_prefix), format: fmt)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
2019-12-04 20:38:33 +05:30
|
|
|
|
|
|
|
TreeHelper.prepend_if_ee('::EE::TreeHelper')
|