2014-06-07 08:46:58 -04:00
module BlobHelper
2016-06-15 14:52:23 -04:00
def highlight ( blob_name , blob_content , repository : nil , plain : false )
2018-01-15 10:00:58 -05:00
plain || = blob_content . length > Blob :: MAXIMUM_TEXT_HIGHLIGHT_SIZE
2016-06-15 14:52:23 -04:00
highlighted = Gitlab :: Highlight . highlight ( blob_name , blob_content , plain : plain , repository : repository )
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 ,
2017-04-06 12:36:38 -04:00
tree_join ( ref , path ) ,
options [ :link_opts ] )
end
def edit_blob_link ( project = @project , ref = @ref , path = @path , options = { } )
2016-08-17 11:05:00 -04:00
blob = options . delete ( :blob )
blob || = project . repository . blob_at ( ref , path ) rescue nil
2015-12-18 04:03:34 -05:00
2017-05-08 19:58:54 -04:00
return unless blob && blob . readable_text?
2015-12-18 04:03:34 -05:00
2017-04-06 12:36:38 -04:00
common_classes = " btn js-edit-blob #{ options [ :extra_class ] } "
2015-12-18 04:03:34 -05:00
2016-01-21 16:46:49 -05:00
if ! on_top_of_branch? ( project , ref )
2017-04-06 12:36:38 -04:00
button_tag 'Edit' , class : " #{ common_classes } disabled has-tooltip " , title : " You can only edit files when you are on a branch " , data : { container : 'body' }
# This condition applies to anonymous or users who can edit directly
2017-04-17 15:03:17 -04:00
elsif ! current_user || ( current_user && can_modify_blob? ( blob , project , ref ) )
2017-12-21 10:05:47 -05:00
link_to 'Edit' , edit_blob_path ( project , ref , path , options ) , class : " #{ common_classes } btn-sm "
2017-04-06 12:36:38 -04:00
elsif current_user && can? ( current_user , :fork_project , project )
2017-04-10 16:51:24 -04:00
continue_params = {
2017-12-21 10:05:47 -05:00
to : edit_blob_path ( project , ref , path , options ) ,
2017-04-10 16:51:24 -04:00
notice : edit_in_new_fork_notice ,
notice_now : edit_in_new_fork_notice_now
}
2017-06-29 13:06:35 -04:00
fork_path = project_forks_path ( project , namespace_key : current_user . namespace . id , continue : continue_params )
2017-04-10 16:51:24 -04:00
button_tag 'Edit' ,
class : " #{ common_classes } js-edit-blob-link-fork-toggler " ,
data : { action : 'edit' , fork_path : fork_path }
2015-12-18 04:03:34 -05:00
end
end
2017-12-21 10:05:47 -05:00
def ide_edit_path ( project = @project , ref = @ref , path = @path , options = { } )
" #{ ide_path } /project #{ edit_blob_path ( project , ref , path , options ) } "
end
def ide_edit_text
2018-01-11 03:23:44 -05:00
" #{ _ ( 'Web IDE' ) } "
2017-12-21 10:05:47 -05:00
end
def ide_blob_link ( project = @project , ref = @ref , path = @path , options = { } )
return unless show_new_ide?
blob = options . delete ( :blob )
blob || = project . repository . blob_at ( ref , path ) rescue nil
return unless blob && blob . readable_text?
common_classes = " btn js-edit-ide #{ options [ :extra_class ] } "
if ! on_top_of_branch? ( project , ref )
button_tag ide_edit_text , class : " #{ common_classes } disabled has-tooltip " , title : _ ( 'You can only edit files when you are on a branch' ) , data : { container : 'body' }
# This condition applies to anonymous or users who can edit directly
elsif current_user && can_modify_blob? ( blob , project , ref )
link_to ide_edit_text , ide_edit_path ( project , ref , path , options ) , class : " #{ common_classes } btn-sm "
elsif current_user && can? ( current_user , :fork_project , project )
continue_params = {
to : ide_edit_path ( project , ref , path , options ) ,
notice : edit_in_new_fork_notice ,
notice_now : edit_in_new_fork_notice_now
}
fork_path = project_forks_path ( project , namespace_key : current_user . namespace . id , continue : continue_params )
button_tag ide_edit_text ,
class : common_classes ,
data : { fork_path : fork_path }
end
end
2015-12-18 04:03:34 -05:00
def modify_file_link ( project = @project , ref = @ref , path = @path , label : , action : , btn_class : , modal_type : )
return unless current_user
blob = project . repository . blob_at ( ref , path ) rescue nil
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'
2015-12-18 04:03:34 -05:00
elsif can? ( current_user , :fork_project , project )
continue_params = {
2017-04-10 16:51:24 -04:00
to : request . fullpath ,
2015-12-18 04:03:34 -05:00
notice : edit_in_new_fork_notice + " Try to #{ action } this file again. " ,
notice_now : edit_in_new_fork_notice_now
}
2017-06-29 13:06:35 -04:00
fork_path = project_forks_path ( project , namespace_key : current_user . namespace . id , continue : continue_params )
2015-12-18 04:03:34 -05:00
2017-04-10 16:51:24 -04:00
button_tag label ,
class : " #{ common_classes } js-edit-blob-link-fork-toggler " ,
data : { action : action , fork_path : fork_path }
2015-12-18 04:03:34 -05:00
end
end
def replace_blob_link ( project = @project , ref = @ref , path = @path )
modify_file_link (
project ,
ref ,
path ,
label : " Replace " ,
action : " replace " ,
btn_class : " default " ,
modal_type : " upload "
)
end
def delete_blob_link ( project = @project , ref = @ref , path = @path )
modify_file_link (
project ,
ref ,
path ,
label : " Delete " ,
action : " delete " ,
btn_class : " remove " ,
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
" Leave edit mode? \n All unsaved changes will be lost. "
end
def editing_preview_title ( filename )
2015-05-12 19:40:11 -04:00
if Gitlab :: MarkupHelper . previewable? ( filename )
2015-01-26 18:03:14 -05:00
'Preview'
else
2017-04-06 09:55:11 -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
2017-12-11 11:23:29 -05:00
def blob_raw_url ( only_path : false )
2017-05-02 18:42:37 -04:00
if @build && @entry
2017-12-11 11:23:29 -05:00
raw_project_job_artifacts_url ( @project , @build , path : @entry . path , only_path : only_path )
2017-05-02 18:42:37 -04:00
elsif @snippet
2017-04-13 12:47:28 -04:00
if @snippet . project_id
2017-12-11 11:23:29 -05:00
raw_project_snippet_url ( @project , @snippet , only_path : only_path )
2017-04-13 12:47:28 -04:00
else
2017-12-11 11:23:29 -05:00
raw_snippet_url ( @snippet , only_path : only_path )
2017-04-13 12:47:28 -04:00
end
elsif @blob
2017-12-11 11:23:29 -05:00
project_raw_url ( @project , @id , only_path : only_path )
2017-04-13 12:47:28 -04:00
end
2017-04-13 13:21:07 -04:00
end
2017-12-11 11:23:29 -05:00
def blob_raw_path
blob_raw_url ( only_path : true )
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
# If we blindly set the 'real' content type when serving a Git blob we
# are enabling XSS attacks. An attacker could upload e.g. a Javascript
# file to a Git repository, trick the browser of a victim into
# downloading the blob, and then the 'application/javascript' content
# type would tell the browser to execute the attacker's Javascript. By
# overriding the content type and setting it to 'text/plain' (in the
# example of Javascript) we tell the browser of the victim not to
# execute untrusted data.
def safe_content_type ( blob )
if blob . text?
'text/plain; charset=utf-8'
elsif blob . image?
blob . content_type
else
'application/octet-stream'
end
end
2016-03-03 11:59:47 -05:00
2016-03-07 08:27:53 -05:00
def cached_blob?
stale = stale? ( etag : @blob . id ) # The #stale? method sets cache headers.
# Because we are opionated we set the cache headers ourselves.
2016-03-07 10:49:46 -05:00
response . cache_control [ :public ] = @project . public?
2016-03-03 11:59:47 -05:00
2017-02-22 10:10:32 -05:00
response . cache_control [ :max_age ] =
if @ref && @commit && @ref == @commit . id
# This is a link to a commit by its commit SHA. That means that the blob
# is immutable. The only reason to invalidate the cache is if the commit
# was deleted or if the user lost access to the repository.
Blob :: CACHE_TIME_IMMUTABLE
else
# A branch or tag points at this blob. That means that the expected blob
# value may change over time.
Blob :: CACHE_TIME
end
2016-03-03 11:59:47 -05:00
2016-03-07 10:49:46 -05:00
response . etag = @blob . id
2016-03-07 08:27:53 -05:00
! stale
2016-03-03 11:59:47 -05:00
end
2016-04-11 09:49:25 -04:00
def licenses_for_select
return @licenses_for_select if defined? ( @licenses_for_select )
licenses = Licensee :: License . all
@licenses_for_select = {
2016-06-15 03:12:42 -04:00
Popular : licenses . select ( & :featured ) . map { | license | { name : license . name , id : license . key } } ,
Other : licenses . reject ( & :featured ) . map { | license | { name : license . name , id : license . key } }
2016-04-11 09:49:25 -04:00
}
end
2016-04-29 10:25:03 -04:00
2016-06-24 15:43:46 -04:00
def ref_project
@ref_project || = @target_project || @project
end
2016-05-13 11:57:03 -04:00
def gitignore_names
2016-06-24 15:43:46 -04:00
@gitignore_names || = Gitlab :: Template :: GitignoreTemplate . dropdown_names
2016-06-02 12:20:08 -04:00
end
2016-05-11 20:38:43 -04:00
2016-06-02 12:20:08 -04:00
def gitlab_ci_ymls
2016-12-21 10:21:55 -05:00
@gitlab_ci_ymls || = Gitlab :: Template :: GitlabCiYmlTemplate . dropdown_names ( params [ :context ] )
2016-04-29 10:25:03 -04:00
end
2016-08-19 11:17:14 -04:00
2016-11-02 11:41:32 -04:00
def dockerfile_names
@dockerfile_names || = Gitlab :: Template :: DockerfileTemplate . dropdown_names
end
2016-08-19 11:17:14 -04:00
def blob_editor_paths
{
'relative-url-root' = > Rails . application . config . relative_url_root ,
'assets-prefix' = > Gitlab :: Application . config . assets . prefix ,
'blob-language' = > @blob && @blob . language . try ( :ace_mode )
}
end
2017-03-14 13:58:52 -04:00
def copy_file_path_button ( file_path )
2017-04-06 17:10:14 -04:00
clipboard_button ( text : file_path , gfm : " ` #{ file_path } ` " , class : 'btn-clipboard btn-transparent prepend-left-5' , title : 'Copy file path to clipboard' )
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 )
2017-04-13 13:13:31 -04:00
clipboard_button ( target : " .blob-content[data-blob-id=' #{ blob . id } '] " , class : " btn btn-sm js-copy-blob-source-btn " , title : " Copy source to clipboard " )
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?
2017-05-15 11:03:34 -04:00
2017-05-02 18:45:50 -04:00
if blob . raw_binary? || blob . stored_externally?
2017-11-16 14:34:15 -05:00
icon = sprite_icon ( 'download' )
2017-04-13 13:11:52 -04:00
title = 'Download'
else
icon = icon ( 'file-code-o' )
title = 'Open raw'
end
2017-08-03 08:29:35 -04:00
link_to icon , blob_raw_path , class : 'btn btn-sm has-tooltip' , target : '_blank' , rel : 'noopener noreferrer' , title : title , data : { container : 'body' }
2017-03-14 13:58:52 -04:00
end
2017-04-13 13:08:39 -04:00
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
options << link_to ( 'load it anyway' , url_for ( 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
merge_project = can? ( current_user , :create_merge_request , project ) ? project : ( current_user && current_user . fork_of ( project ) )
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
2014-06-07 08:46:58 -04:00
end