projects_helper.rb 11.3 KB
Newer Older
R
randx 已提交
1
module ProjectsHelper
2
  def link_to_project(project)
J
Escape!  
Josh Frye 已提交
3
    link_to [project.namespace.becomes(Namespace), project], title: h(project.name) do
4
      title = content_tag(:span, project.name, class: 'project-name')
5 6

      if project.namespace
D
Dmitriy Zaporozhets 已提交
7
        namespace = content_tag(:span, "#{project.namespace.human_name} / ", class: 'namespace-name')
8 9 10 11 12
        title = namespace + title
      end

      title
    end
13
  end
R
randx 已提交
14

J
Jacob Schatz 已提交
15 16 17
  def link_to_member_avatar(author, opts = {})
    default_opts = { avatar: true, name: true, size: 16, author_class: 'author', title: ":name" }
    opts = default_opts.merge(opts)
18
    image_tag(avatar_icon(author, opts[:size]), width: opts[:size], class: "avatar avatar-inline #{"s#{opts[:size]}" if opts[:size]}", alt: '') if opts[:avatar]
J
Jacob Schatz 已提交
19 20
  end

P
Phil Hughes 已提交
21
  def link_to_member(project, author, opts = {}, &block)
22
    default_opts = { avatar: true, name: true, size: 16, author_class: 'author', title: ":name", tooltip: false }
23 24
    opts = default_opts.merge(opts)

25 26
    return "(deleted)" unless author

27 28
    author_html =  ""

29
    # Build avatar image tag
30
    author_html << image_tag(avatar_icon(author, opts[:size]), width: opts[:size], class: "avatar avatar-inline #{"s#{opts[:size]}" if opts[:size]}", alt: '') if opts[:avatar]
31

32
    # Build name span tag
33 34 35
    if opts[:by_username]
      author_html << content_tag(:span, sanitize("@#{author.username}"), class: opts[:author_class]) if opts[:name]
    else
36
      tooltip_data = { placement: 'top' }
P
Phil Hughes 已提交
37
      author_html << content_tag(:span, sanitize(author.name), class: [opts[:author_class], ('has-tooltip' if opts[:tooltip])], title: (author.to_reference if opts[:tooltip]), data: (tooltip_data if opts[:tooltip])) if opts[:name]
38
    end
39

P
Phil Hughes 已提交
40 41
    author_html << capture(&block) if block

42
    author_html = author_html.html_safe
43

44
    if opts[:name]
45
      link_to(author_html, user_path(author), class: "author_link #{"#{opts[:extra_class]}" if opts[:extra_class]} #{"#{opts[:mobile_classes]}" if opts[:mobile_classes]}").html_safe
46
    else
47
      title = opts[:title].sub(":name", sanitize(author.name))
48
      link_to(author_html, user_path(author), class: "author_link has-tooltip", title: title, data: { container: 'body' } ).html_safe
49
    end
R
randx 已提交
50
  end
51

52 53 54 55 56 57 58
  def project_title(project, name = nil, url = nil)
    namespace_link =
      if project.group
        link_to(simple_sanitize(project.group.name), group_path(project.group))
      else
        owner = project.namespace.owner
        link_to(simple_sanitize(owner.name), user_path(owner))
59
      end
60

P
Phil Hughes 已提交
61
    project_link = link_to simple_sanitize(project.name), project_path(project), { class: "project-item-select-holder" }
62

P
Phil Hughes 已提交
63
    if current_user
64 65 66
      project_link << button_tag(type: 'button', class: "dropdown-toggle-caret js-projects-dropdown-toggle", aria: { label: "Toggle switch project dropdown" }, data: { target: ".js-dropdown-menu-projects", toggle: "dropdown" }) do
        icon("chevron-down")
      end
P
Phil Hughes 已提交
67 68
    end

P
Phil Hughes 已提交
69
    full_title = "#{namespace_link} / #{project_link}".html_safe
P
Phil Hughes 已提交
70
    full_title << ' &middot; '.html_safe << link_to(simple_sanitize(name), url) if name
71

72
    full_title
73
  end
74 75 76 77

  def remove_project_message(project)
    "You are going to remove #{project.name_with_namespace}.\n Removed project CANNOT be restored!\n Are you ABSOLUTELY sure?"
  end
78

79 80 81 82
  def transfer_project_message(project)
    "You are going to transfer #{project.name_with_namespace} to another owner. Are you ABSOLUTELY sure?"
  end

83
  def remove_fork_project_message(project)
D
Douwe Maan 已提交
84
    "You are going to remove the fork relationship to source project #{@project.forked_from_project.name_with_namespace}.  Are you ABSOLUTELY sure?"
85 86
  end

87 88 89 90 91 92 93 94
  def project_nav_tabs
    @nav_tabs ||= get_project_nav_tabs(@project, current_user)
  end

  def project_nav_tab?(name)
    project_nav_tabs.include? name
  end

D
Douwe Maan 已提交
95 96 97 98 99 100 101 102
  def project_for_deploy_key(deploy_key)
    if deploy_key.projects.include?(@project)
      @project
    else
      deploy_key.projects.find { |project| can?(current_user, :read_project, project) }
    end
  end

V
Valery Sizov 已提交
103 104 105 106 107 108 109 110 111 112
  def can_change_visibility_level?(project, current_user)
    return false unless can?(current_user, :change_visibility_level, project)

    if project.forked?
      project.forked_from_project.visibility_level > Gitlab::VisibilityLevel::PRIVATE
    else
      true
    end
  end

113
  def license_short_name(project)
114
    return 'LICENSE' if project.repository.license_key.nil?
115 116 117 118 119 120

    license = Licensee::License.new(project.repository.license_key)

    license.nickname || license.name
  end

121 122 123 124 125 126 127 128 129 130 131
  def last_push_event
    return unless current_user

    project_ids = [@project.id]
    if fork = current_user.fork_of(@project)
      project_ids << fork.id
    end

    current_user.recent_push(project_ids)
  end

132 133 134
  private

  def get_project_nav_tabs(project, current_user)
135
    nav_tabs = [:home]
136

137
    if !project.empty_repo? && can?(current_user, :download_code, project)
138
      nav_tabs << [:files, :commits, :network, :graphs, :forks]
139 140
    end

141
    if project.repo_exists? && can?(current_user, :read_merge_request, project)
142 143 144
      nav_tabs << :merge_requests
    end

145 146 147 148
    if can?(current_user, :read_pipeline, project)
      nav_tabs << :pipelines
    end

149
    if can?(current_user, :read_build, project)
K
Kamil Trzcinski 已提交
150 151 152
      nav_tabs << :builds
    end

153
    if Gitlab.config.registry.enabled && can?(current_user, :read_container_image, project)
154
      nav_tabs << :container_registry
K
Kamil Trzcinski 已提交
155 156
    end

157 158 159 160
    if can?(current_user, :read_environment, project)
      nav_tabs << :environments
    end

161 162 163 164
    if can?(current_user, :admin_project, project)
      nav_tabs << :settings
    end

165
    if can?(current_user, :read_project_member, project)
166 167 168
      nav_tabs << :team
    end

169 170
    if can?(current_user, :read_issue, project)
      nav_tabs << :issues
171 172
    end

173 174 175 176 177 178 179 180
    if can?(current_user, :read_wiki, project)
      nav_tabs << :wiki
    end

    if can?(current_user, :read_project_snippet, project)
      nav_tabs << :snippets
    end

181 182 183 184
    if can?(current_user, :read_label, project)
      nav_tabs << :labels
    end

185
    if can?(current_user, :read_milestone, project)
186
      nav_tabs << :milestones
187 188
    end

189 190
    nav_tabs.flatten
  end
191

192 193
  def project_lfs_status(project)
    if project.lfs_enabled?
194
      content_tag(:span, class: 'lfs-enabled') do
195 196 197
        'Enabled'
      end
    else
198
      content_tag(:span, class: 'lfs-disabled') do
199 200 201 202 203
        'Disabled'
      end
    end
  end

204 205 206 207 208 209 210 211 212 213 214 215 216 217 218
  def git_user_name
    if current_user
      current_user.name
    else
      "Your name"
    end
  end

  def git_user_email
    if current_user
      current_user.email
    else
      "your@email.com"
    end
  end
219

D
Douwe Maan 已提交
220
  def repository_size(project = @project)
221 222
    size_in_bytes = project.repository_size * 1.megabyte
    number_to_human_size(size_in_bytes, delimiter: ',', precision: 2)
223
  end
224

225
  def default_url_to_repo(project = @project)
226 227
    case default_clone_protocol
    when 'ssh'
228 229 230 231
      project.ssh_url_to_repo
    else
      project.http_url_to_repo
    end
232
  end
233

234
  def default_clone_protocol
235 236
    if allowed_protocols_present?
      enabled_protocol
237
    else
238 239 240 241 242
      if !current_user || current_user.require_ssh_key?
        gitlab_config.protocol
      else
        'ssh'
      end
243
    end
244
  end
245 246 247

  def project_last_activity(project)
    if project.last_activity_at
248
      time_ago_with_tooltip(project.last_activity_at, placement: 'bottom', html_class: 'last_activity_time_ago')
249 250 251 252
    else
      "Never"
    end
  end
D
Dmitriy Zaporozhets 已提交
253

254 255 256 257 258 259 260 261
  def add_special_file_path(project, file_name:, commit_message: nil)
    namespace_project_new_blob_path(
      project.namespace,
      project,
      project.default_branch || 'master',
      file_name:      file_name,
      commit_message: commit_message || "Add #{file_name.downcase}"
    )
262 263
  end

264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
  def add_koding_stack_path(project)
    namespace_project_new_blob_path(
      project.namespace,
      project,
      project.default_branch || 'master',
      file_name:      '.koding.yml',
      commit_message: "Add Koding stack script",
      content: <<-CONTENT.strip_heredoc
        provider:
          aws:
            access_key: '${var.aws_access_key}'
            secret_key: '${var.aws_secret_key}'
        resource:
          aws_instance:
            #{project.path}-vm:
              instance_type: t2.nano
              user_data: |-

                # Created by GitLab UI for :>

                echo _KD_NOTIFY_@Installing Base packages...@

                apt-get update -y
                apt-get install git -y

                echo _KD_NOTIFY_@Cloning #{project.name}...@

                export KODING_USER=${var.koding_user_username}
                export REPO_URL=#{root_url}${var.koding_queryString_repo}.git
                export BRANCH=${var.koding_queryString_branch}

                sudo -i -u $KODING_USER git clone $REPO_URL -b $BRANCH

                echo _KD_NOTIFY_@#{project.name} cloned.@
      CONTENT
    )
  end

  def koding_project_url(project = nil, branch = nil, sha = nil)
    if project
      import_path = "/Home/Stacks/import"

      repo = project.path_with_namespace
      branch ||= project.default_branch
      sha ||= project.commit.short_id

      path = "#{import_path}?repo=#{repo}&branch=#{branch}&sha=#{sha}"

      return URI.join(current_application_settings.koding_url, path).to_s
    end

    current_application_settings.koding_url
  end

318
  def contribution_guide_path(project)
319
    if project && contribution_guide = project.repository.contribution_guide
V
Vinnie Okada 已提交
320 321 322 323
      namespace_project_blob_path(
        project.namespace,
        project,
        tree_join(project.default_branch,
324 325 326 327 328
                  contribution_guide.name)
      )
    end
  end

329 330 331 332
  def readme_path(project)
    filename_path(project, :readme)
  end

333
  def changelog_path(project)
334
    filename_path(project, :changelog)
335 336
  end

337
  def license_path(project)
338
    filename_path(project, :license_blob)
339 340
  end

341
  def version_path(project)
342
    filename_path(project, :version)
D
Dmitriy Zaporozhets 已提交
343
  end
344

345 346 347 348
  def ci_configuration_path(project)
    filename_path(project, :gitlab_ci_yml)
  end

349 350
  def project_wiki_path_with_version(proj, page, version, is_newest)
    url_params = is_newest ? {} : { version_id: version }
V
Vinnie Okada 已提交
351
    namespace_project_wiki_path(proj.namespace, proj, page, url_params)
352
  end
353

V
Valery Sizov 已提交
354 355 356 357 358 359 360 361 362 363
  def project_status_css_class(status)
    case status
    when "started"
      "active"
    when "failed"
      "danger"
    when "finished"
      "success"
    end
  end
364

365 366 367 368 369 370
  def new_readme_path
    ref = @repository.root_ref if @repository
    ref ||= 'master'

    namespace_project_new_blob_path(@project.namespace, @project, tree_join(ref), file_name: 'README.md')
  end
371

372 373 374 375 376 377 378
  def new_license_path
    ref = @repository.root_ref if @repository
    ref ||= 'master'

    namespace_project_new_blob_path(@project.namespace, @project, tree_join(ref), file_name: 'LICENSE')
  end

379
  def readme_cache_key
380
    sha = @project.commit.try(:sha) || 'nil'
381
    [@project.path_with_namespace, sha, "readme"].join('-')
382
  end
383 384 385 386 387 388 389 390 391 392 393 394 395 396

  def round_commit_count(project)
    count = project.commit_count

    if count > 10000
      '10000+'
    elsif count > 5000
      '5000+'
    elsif count > 1000
      '1000+'
    else
      count
    end
  end
397

398 399 400
  def current_ref
    @ref || @repository.try(:root_ref)
  end
401

402 403 404
  def filename_path(project, filename)
    if project && blob = project.repository.send(filename)
      namespace_project_blob_path(
G
Gabriel Mazetto 已提交
405 406 407
        project.namespace,
        project,
        tree_join(project.default_branch, blob.name)
408 409 410
      )
    end
  end
411

412
  def sanitize_repo_path(project, message)
413 414
    return '' unless message.present?

415
    message.strip.gsub(project.repository_storage_path.chomp('/'), "[REPOS PATH]")
416
  end
R
randx 已提交
417
end