2018-08-18 07:19:57 -04:00
# frozen_string_literal: true
2014-06-07 08:46:58 -04:00
module BlobHelper
2018-09-06 00:34:25 -04:00
def highlight ( file_name , file_content , language : nil , plain : false )
highlighted = Gitlab :: Highlight . highlight ( file_name , file_content , plain : plain , language : language )
2018-01-15 10:00:58 -05:00
2016-07-15 01:43:49 -04:00
raw %( <pre class="code highlight"><code> #{ highlighted } </code></pre> )
2014-06-07 08:46:58 -04:00
end
def no_highlight_files
2014-12-03 06:50:00 -05:00
%w( credits changelog news copying copyright license authors )
2014-06-07 08:46:58 -04:00
end
2015-01-26 18:03:14 -05:00
2017-12-21 10:05:47 -05:00
def edit_blob_path ( project = @project , ref = @ref , path = @path , options = { } )
2017-06-29 13:06:35 -04:00
project_edit_blob_path ( project ,
2018-02-09 02:59:03 -05:00
tree_join ( ref , path ) ,
options [ :link_opts ] )
2017-04-06 12:36:38 -04:00
end
2020-04-27 05:09:51 -04:00
def ide_edit_path ( project = @project , ref = @ref , path = @path )
2019-07-11 13:27:21 -04:00
project_path =
if ! current_user || can? ( current_user , :push_code , project )
project . full_path
else
# We currently always fork to the user's namespace
# in edit_fork_button_tag
" #{ current_user . namespace . full_path } / #{ project . path } "
end
2020-02-12 10:09:37 -05:00
segments = [ ide_path , 'project' , project_path , 'edit' , encode_ide_path ( ref ) ]
2019-04-03 07:38:46 -04:00
segments . concat ( [ '-' , encode_ide_path ( path ) ] ) if path . present?
2018-05-23 05:36:41 -04:00
File . join ( segments )
2018-02-23 10:52:18 -05:00
end
2019-10-15 23:06:12 -04:00
def ide_fork_and_edit_path ( project = @project , ref = @ref , path = @path , options = { } )
if current_user
project_forks_path ( project ,
namespace_key : current_user & . namespace & . id ,
continue : edit_blob_fork_params ( ide_edit_path ( project , ref , path ) ) )
end
end
2019-04-03 07:38:46 -04:00
def encode_ide_path ( path )
url_encode ( path ) . gsub ( '%2F' , '/' )
end
2018-02-22 07:54:19 -05:00
def edit_blob_button ( project = @project , ref = @ref , path = @path , options = { } )
2018-02-08 08:12:44 -05:00
return unless blob = readable_blob ( options , path , project , ref )
2015-12-18 04:03:34 -05:00
2020-01-24 13:09:00 -05:00
common_classes = " btn btn-primary js-edit-blob ml-2 #{ options [ :extra_class ] } "
2015-12-18 04:03:34 -05:00
2018-02-23 03:09:32 -05:00
edit_button_tag ( blob ,
common_classes ,
2018-02-23 10:04:31 -05:00
_ ( 'Edit' ) ,
2020-04-27 05:09:51 -04:00
Feature . enabled? ( :web_ide_default ) ? ide_edit_path ( project , ref , path ) : edit_blob_path ( project , ref , path , options ) ,
2018-02-23 03:09:32 -05:00
project ,
ref )
2015-12-18 04:03:34 -05:00
end
2020-04-27 05:09:51 -04:00
def ide_edit_button ( project = @project , ref = @ref , path = @path , blob : )
2019-02-27 07:44:24 -05:00
return if Feature . enabled? ( :web_ide_default )
2020-04-27 05:09:51 -04:00
return unless blob
2018-03-20 13:00:56 -04:00
edit_button_tag ( blob ,
2020-01-24 13:09:00 -05:00
'btn btn-inverted btn-primary ide-edit-button ml-2' ,
2018-03-20 13:00:56 -04:00
_ ( 'Web IDE' ) ,
2020-04-27 05:09:51 -04:00
ide_edit_path ( project , ref , path ) ,
2018-03-20 13:00:56 -04:00
project ,
ref )
end
2020-04-27 05:09:51 -04:00
def modify_file_button ( project = @project , ref = @ref , path = @path , blob : , label : , action : , btn_class : , modal_type : )
2015-12-18 04:03:34 -05:00
return unless current_user
return unless blob
2017-04-10 16:51:24 -04:00
common_classes = " btn btn- #{ btn_class } "
2016-01-21 16:46:49 -05:00
if ! on_top_of_branch? ( project , ref )
2017-04-10 16:51:24 -04:00
button_tag label , class : " #{ common_classes } disabled has-tooltip " , title : " You can only #{ action } files when you are on a branch " , data : { container : 'body' }
2017-05-02 18:45:50 -04:00
elsif blob . stored_externally?
2017-04-10 16:51:24 -04:00
button_tag label , class : " #{ common_classes } disabled has-tooltip " , title : " It is not possible to #{ action } files that are stored in LFS using the web interface " , data : { container : 'body' }
2017-04-17 15:03:17 -04:00
elsif can_modify_blob? ( blob , project , ref )
2017-04-10 16:51:24 -04:00
button_tag label , class : " #{ common_classes } " , 'data-target' = > " # modal- #{ modal_type } -blob " , 'data-toggle' = > 'modal'
2018-04-06 12:27:12 -04:00
elsif can? ( current_user , :fork_project , project ) && can? ( current_user , :create_merge_request_in , project )
2018-02-15 06:03:37 -05:00
edit_fork_button_tag ( common_classes , project , label , edit_modify_file_fork_params ( action ) , action )
2015-12-18 04:03:34 -05:00
end
end
2020-04-27 05:09:51 -04:00
def replace_blob_link ( project = @project , ref = @ref , path = @path , blob : )
2018-02-23 10:04:31 -05:00
modify_file_button (
2015-12-18 04:03:34 -05:00
project ,
ref ,
path ,
2020-04-27 05:09:51 -04:00
blob : blob ,
2019-04-04 13:05:25 -04:00
label : _ ( " Replace " ) ,
2015-12-18 04:03:34 -05:00
action : " replace " ,
btn_class : " default " ,
modal_type : " upload "
)
end
2020-04-27 05:09:51 -04:00
def delete_blob_link ( project = @project , ref = @ref , path = @path , blob : )
2018-02-23 10:04:31 -05:00
modify_file_button (
2015-12-18 04:03:34 -05:00
project ,
ref ,
path ,
2020-04-27 05:09:51 -04:00
blob : blob ,
2019-04-04 13:05:25 -04:00
label : _ ( " Delete " ) ,
2015-12-18 04:03:34 -05:00
action : " delete " ,
2019-11-07 10:06:33 -05:00
btn_class : " default " ,
2015-12-18 04:03:34 -05:00
modal_type : " remove "
)
end
2017-04-17 15:03:17 -04:00
def can_modify_blob? ( blob , project = @project , ref = @ref )
2017-05-02 18:45:50 -04:00
! blob . stored_externally? && can_edit_tree? ( project , ref )
2015-01-26 18:03:14 -05:00
end
def leave_edit_message
2019-04-04 13:05:25 -04:00
_ ( " Leave edit mode? All unsaved changes will be lost. " )
2015-01-26 18:03:14 -05:00
end
def editing_preview_title ( filename )
2015-05-12 19:40:11 -04:00
if Gitlab :: MarkupHelper . previewable? ( filename )
2019-04-04 13:05:25 -04:00
_ ( 'Preview' )
2015-01-26 18:03:14 -05:00
else
2019-04-04 13:05:25 -04:00
_ ( 'Preview changes' )
2015-01-26 18:03:14 -05:00
end
end
2014-10-04 06:29:18 -04:00
# Return an image icon depending on the file mode and extension
#
# mode - File unix mode
# mode - File name
def blob_icon ( mode , name )
icon ( " #{ file_type_icon_class ( 'file' , mode , name ) } fw " )
end
2015-12-07 09:03:50 -05:00
2018-07-12 21:24:11 -04:00
def blob_raw_url ( ** kwargs )
2017-05-02 18:42:37 -04:00
if @build && @entry
2018-07-12 21:24:11 -04:00
raw_project_job_artifacts_url ( @project , @build , path : @entry . path , ** kwargs )
2017-05-02 18:42:37 -04:00
elsif @snippet
2019-12-10 16:08:01 -05:00
gitlab_raw_snippet_url ( @snippet )
2017-04-13 12:47:28 -04:00
elsif @blob
2018-07-12 21:24:11 -04:00
project_raw_url ( @project , @id , ** kwargs )
2017-04-13 12:47:28 -04:00
end
2017-04-13 13:21:07 -04:00
end
2018-07-12 21:24:11 -04:00
def blob_raw_path ( ** kwargs )
blob_raw_url ( ** kwargs , only_path : true )
2017-12-11 11:23:29 -05:00
end
2015-09-12 23:54:06 -04:00
# SVGs can contain malicious JavaScript; only include whitelisted
# elements and attributes. Note that this whitelist is by no means complete
# and may omit some elements.
2017-04-13 13:21:07 -04:00
def sanitize_svg_data ( data )
Gitlab :: Sanitizers :: SVG . clean ( data )
2015-09-12 23:54:06 -04:00
end
2016-02-24 05:53:30 -05:00
2016-06-24 15:43:46 -04:00
def ref_project
@ref_project || = @target_project || @project
end
2018-09-05 09:49:14 -04:00
def template_dropdown_names ( items )
2018-08-28 09:14:39 -04:00
grouped = items . group_by ( & :category )
categories = grouped . keys
categories . each_with_object ( { } ) do | category , hash |
hash [ category ] = grouped [ category ] . map do | item |
2018-10-02 19:00:38 -04:00
{ name : item . name , id : item . key }
2018-08-28 09:14:39 -04:00
end
end
end
2018-09-05 09:49:14 -04:00
private :template_dropdown_names
2018-08-28 09:14:39 -04:00
2018-10-05 12:21:21 -04:00
def licenses_for_select ( project )
2018-10-02 19:00:38 -04:00
@licenses_for_select || = template_dropdown_names ( TemplateFinder . build ( :licenses , project ) . execute )
end
2018-10-05 12:21:21 -04:00
def gitignore_names ( project )
2018-10-02 19:00:38 -04:00
@gitignore_names || = template_dropdown_names ( TemplateFinder . build ( :gitignores , project ) . execute )
2016-06-02 12:20:08 -04:00
end
2016-05-11 20:38:43 -04:00
2018-10-05 12:21:21 -04:00
def gitlab_ci_ymls ( project )
2018-10-02 19:00:38 -04:00
@gitlab_ci_ymls || = template_dropdown_names ( TemplateFinder . build ( :gitlab_ci_ymls , project ) . execute )
2016-04-29 10:25:03 -04:00
end
2016-08-19 11:17:14 -04:00
2018-10-05 12:21:21 -04:00
def dockerfile_names ( project )
2018-10-02 19:00:38 -04:00
@dockerfile_names || = template_dropdown_names ( TemplateFinder . build ( :dockerfiles , project ) . execute )
2016-11-02 11:41:32 -04:00
end
2018-10-05 12:21:21 -04:00
def blob_editor_paths ( project )
2016-08-19 11:17:14 -04:00
{
'relative-url-root' = > Rails . application . config . relative_url_root ,
'assets-prefix' = > Gitlab :: Application . config . assets . prefix ,
2018-08-03 09:24:26 -04:00
'blob-filename' = > @blob && @blob . path ,
2018-12-14 12:56:25 -05:00
'project-id' = > project . id ,
'is-markdown' = > @blob && @blob . path && Gitlab :: MarkupHelper . gitlab_markdown? ( @blob . path )
2016-08-19 11:17:14 -04:00
}
end
2017-03-14 13:58:52 -04:00
def copy_file_path_button ( file_path )
2019-10-09 08:06:13 -04:00
clipboard_button ( text : file_path , gfm : " ` #{ file_path } ` " , class : 'btn-clipboard btn-transparent' , title : _ ( 'Copy file path' ) )
2017-03-14 13:58:52 -04:00
end
2017-04-13 13:13:31 -04:00
def copy_blob_source_button ( blob )
2017-04-13 13:11:52 -04:00
return unless blob . rendered_as_text? ( ignore_errors : false )
2019-10-09 08:06:13 -04:00
clipboard_button ( target : " .blob-content[data-blob-id=' #{ blob . id } '] " , class : " btn btn-sm js-copy-blob-source-btn " , title : _ ( " Copy file contents " ) )
2017-03-14 13:58:52 -04:00
end
2017-04-13 13:11:52 -04:00
def open_raw_blob_button ( blob )
2017-05-02 18:45:50 -04:00
return if blob . empty?
2018-12-13 12:49:05 -05:00
return if blob . binary? || blob . stored_externally?
2018-07-17 21:19:53 -04:00
2019-04-04 13:05:25 -04:00
title = _ ( 'Open raw' )
2019-12-06 13:07:44 -05:00
link_to sprite_icon ( 'doc-code' ) ,
external_storage_url_or_path ( blob_raw_path ) ,
class : 'btn btn-sm has-tooltip' ,
target : '_blank' ,
rel : 'noopener noreferrer' ,
aria : { label : title } ,
title : title ,
data : { container : 'body' }
2017-03-14 13:58:52 -04:00
end
2017-04-13 13:08:39 -04:00
2018-07-11 22:23:00 -04:00
def download_blob_button ( blob )
2018-07-08 21:55:36 -04:00
return if blob . empty?
2018-07-17 23:16:30 -04:00
2019-04-04 13:05:25 -04:00
title = _ ( 'Download' )
2019-12-06 13:07:44 -05:00
link_to sprite_icon ( 'download' ) ,
external_storage_url_or_path ( blob_raw_path ( inline : false ) ) ,
download : @path ,
class : 'btn btn-sm has-tooltip' ,
target : '_blank' ,
rel : 'noopener noreferrer' ,
aria : { label : title } ,
title : title ,
data : { container : 'body' }
2018-07-08 21:55:36 -04:00
end
2017-04-26 16:48:49 -04:00
def blob_render_error_reason ( viewer )
case viewer . render_error
2017-05-26 19:27:30 -04:00
when :collapsed
" it is larger than #{ number_to_human_size ( viewer . collapse_limit ) } "
2017-04-13 13:08:39 -04:00
when :too_large
2017-05-26 19:27:30 -04:00
" it is larger than #{ number_to_human_size ( viewer . size_limit ) } "
2017-05-02 18:45:50 -04:00
when :server_side_but_stored_externally
case viewer . blob . external_storage
when :lfs
'it is stored in LFS'
2017-05-02 18:42:37 -04:00
when :build_artifact
'it is stored as a job artifact'
2017-05-02 18:45:50 -04:00
else
'it is stored externally'
end
2017-04-13 13:08:39 -04:00
end
end
2017-04-21 14:22:04 -04:00
2017-04-26 16:48:49 -04:00
def blob_render_error_options ( viewer )
2017-05-02 18:45:50 -04:00
error = viewer . render_error
2017-04-21 14:22:04 -04:00
options = [ ]
2017-05-26 19:27:30 -04:00
if error == :collapsed
2018-04-08 00:35:30 -04:00
options << link_to ( 'load it anyway' , url_for ( safe_params . merge ( viewer : viewer . type , expanded : true , format : nil ) ) )
2017-04-21 14:22:04 -04:00
end
2017-05-02 18:45:50 -04:00
# If the error is `:server_side_but_stored_externally`, the simple viewer will show the same error,
# so don't bother switching.
if viewer . rich? && viewer . blob . rendered_as_text? && error != :server_side_but_stored_externally
2017-04-21 14:33:48 -04:00
options << link_to ( 'view the source' , '#' , class : 'js-blob-viewer-switch-btn' , data : { viewer : 'simple' } )
2017-04-21 14:22:04 -04:00
end
2017-08-03 08:29:35 -04:00
options << link_to ( 'download it' , blob_raw_path , target : '_blank' , rel : 'noopener noreferrer' )
2017-04-21 14:22:04 -04:00
options
end
2017-05-13 13:06:51 -04:00
def contribution_options ( project )
options = [ ]
if can? ( current_user , :create_issue , project )
2017-06-29 13:06:35 -04:00
options << link_to ( " submit an issue " , new_project_issue_path ( project ) )
2017-05-13 13:06:51 -04:00
end
2018-04-06 12:27:12 -04:00
merge_project = merge_request_source_project_for_project ( @project )
2017-05-13 13:06:51 -04:00
if merge_project
2017-06-29 13:06:35 -04:00
options << link_to ( " create a merge request " , project_new_merge_request_path ( project ) )
2017-05-13 13:06:51 -04:00
end
options
end
2018-02-05 10:25:31 -05:00
2018-02-08 08:12:44 -05:00
def readable_blob ( options , path , project , ref )
2018-02-05 10:25:31 -05:00
blob = options . delete ( :blob )
blob || = project . repository . blob_at ( ref , path ) rescue nil
2018-02-08 08:12:44 -05:00
blob if blob & . readable_text?
2018-02-05 10:25:31 -05:00
end
2018-02-15 03:44:08 -05:00
def edit_blob_fork_params ( path )
{
2018-02-22 07:54:19 -05:00
to : path ,
notice : edit_in_new_fork_notice ,
notice_now : edit_in_new_fork_notice_now
2018-02-05 10:25:31 -05:00
}
2018-02-09 04:32:28 -05:00
end
2018-02-15 03:44:08 -05:00
def edit_modify_file_fork_params ( action )
{
2018-02-22 07:54:19 -05:00
to : request . fullpath ,
notice : edit_in_new_fork_notice_action ( action ) ,
notice_now : edit_in_new_fork_notice_now
2018-02-09 04:32:28 -05:00
}
2018-02-15 03:44:08 -05:00
end
2018-02-15 06:03:37 -05:00
def edit_fork_button_tag ( common_classes , project , label , params , action = 'edit' )
2018-02-15 03:44:08 -05:00
fork_path = project_forks_path ( project , namespace_key : current_user . namespace . id , continue : params )
2018-02-09 04:32:28 -05:00
2018-02-15 06:03:37 -05:00
button_tag label ,
2018-02-05 10:25:31 -05:00
class : " #{ common_classes } js-edit-blob-link-fork-toggler " ,
2018-02-15 06:03:37 -05:00
data : { action : action , fork_path : fork_path }
2018-02-05 10:25:31 -05:00
end
2018-02-08 05:58:47 -05:00
2018-02-22 07:54:19 -05:00
def edit_disabled_button_tag ( button_text , common_classes )
2018-02-08 05:58:47 -05:00
button_tag ( button_text , class : " #{ common_classes } disabled has-tooltip " , title : _ ( 'You can only edit files when you are on a branch' ) , data : { container : 'body' } )
end
def edit_link_tag ( link_text , edit_path , common_classes )
link_to link_text , edit_path , class : " #{ common_classes } btn-sm "
end
2018-02-23 03:09:32 -05:00
def edit_button_tag ( blob , common_classes , text , edit_path , project , ref )
if ! on_top_of_branch? ( project , ref )
edit_disabled_button_tag ( text , common_classes )
# This condition only applies to users who are logged in
2018-02-23 10:52:18 -05:00
elsif ! current_user || ( current_user && can_modify_blob? ( blob , project , ref ) )
2018-02-23 03:09:32 -05:00
edit_link_tag ( text , edit_path , common_classes )
2018-04-06 08:18:58 -04:00
elsif can? ( current_user , :fork_project , project ) && can? ( current_user , :create_merge_request_in , project )
2018-02-23 07:58:27 -05:00
edit_fork_button_tag ( common_classes , project , text , edit_blob_fork_params ( edit_path ) )
2018-02-23 03:09:32 -05:00
end
end
2020-02-24 16:09:08 -05:00
def show_suggest_pipeline_creation_celebration?
experiment_enabled? ( :suggest_pipeline ) &&
@blob . path == Gitlab :: FileDetector :: PATTERNS [ :gitlab_ci ] &&
2020-03-18 08:09:13 -04:00
@blob . auxiliary_viewer . valid? ( project : @project , sha : @commit . sha , user : current_user ) &&
2020-02-24 16:09:08 -05:00
@project . uses_default_ci_config? &&
cookies [ suggest_pipeline_commit_cookie_name ] . present?
end
def suggest_pipeline_commit_cookie_name
" suggest_gitlab_ci_yml_commit_ #{ @project . id } "
end
2020-03-12 17:09:45 -04:00
def human_access
@project . team . human_max_access ( current_user & . id ) . try ( :downcase )
end
2014-06-07 08:46:58 -04:00
end