Commit 54bdc736 authored by Rémy Coutable's avatar Rémy Coutable

Merge branch 'ce-to-ee' into 'master'

CE to EE for 8.10.0-rc1 (bis)

I'm merging CE to EE one last time before I tag 8.10.0-rc1!

See merge request !534
parents bc1a911e 997de86a
...@@ -15,6 +15,7 @@ v 8.10.0 (unreleased) ...@@ -15,6 +15,7 @@ v 8.10.0 (unreleased)
- Make images fit to the size of the viewport !4810 - Make images fit to the size of the viewport !4810
- Fix check for New Branch button on Issue page !4630 (winniehell) - Fix check for New Branch button on Issue page !4630 (winniehell)
- Fix MR-auto-close text added to description. !4836 - Fix MR-auto-close text added to description. !4836
- Fix issue, preventing users w/o push access to sort tags !5105 (redetection)
- Add Spring EmojiOne updates. - Add Spring EmojiOne updates.
- Fix pagination when sorting by columns with lots of ties (like priority) - Fix pagination when sorting by columns with lots of ties (like priority)
- Updated project header design - Updated project header design
...@@ -28,6 +29,7 @@ v 8.10.0 (unreleased) ...@@ -28,6 +29,7 @@ v 8.10.0 (unreleased)
- PipelinesFinder uses git cache data - PipelinesFinder uses git cache data
- Throttle the update of `project.pushes_since_gc` to 1 minute. - Throttle the update of `project.pushes_since_gc` to 1 minute.
- Check for conflicts with existing Project's wiki path when creating a new project. - Check for conflicts with existing Project's wiki path when creating a new project.
- Show last push widget in upstream after push to fork
- Don't instantiate a git tree on Projects show default view - Don't instantiate a git tree on Projects show default view
- Bump Rinku to 2.0.0 - Bump Rinku to 2.0.0
- Remove unused front-end variable -> default_issues_tracker - Remove unused front-end variable -> default_issues_tracker
...@@ -44,17 +46,23 @@ v 8.10.0 (unreleased) ...@@ -44,17 +46,23 @@ v 8.10.0 (unreleased)
- More descriptive message for git hooks and file locks - More descriptive message for git hooks and file locks
- Handle custom Git hook result in GitLab UI - Handle custom Git hook result in GitLab UI
- Allow '?', or '&' for label names - Allow '?', or '&' for label names
- Fix importer for GitHub Pull Requests when a branch was reused across Pull Requests
- Add date when user joined the team on the member page
v 8.9.5 (unreleased) v 8.9.5
- Improve the request / withdraw access button. !4860 - Add more debug info to import/export and memory killer. !5108
- Fix assigning shared runners as admins. !4961 - Fixed avatar alignment in new MR view. !5095
- Show "locked" label for locked runners on runners admin. !4961
- Downgrade to Redis 3.2.2 due to massive memory leak with Sidekiq
- Add index on the user and emoji name on AwardEmoji table !5061
- Fixes issues importing events in Import/Export. Import/Export version bumped to 0.1.1
- Fix import button disabled when import process fail due to the namespace already been taken.
- Fix diff comments not showing up in activity feed. !5069 - Fix diff comments not showing up in activity feed. !5069
- Security: Update RedCloth to 4.3.2 (Takuya Noguchi) - Add index on both Award Emoji user and name. !5061
- Downgrade to Redis 3.2.2 due to massive memory leak with Sidekiq. !5056
- Re-enable import button when import process fails due to namespace already being taken. !5053
- Fix snippets comments not displayed. !5045
- Fix emoji paths in relative root configurations. !5027
- Fix issues importing events in Import/Export. !4987
- Fixed 'use shortcuts' button on docs. !4979
- Admin should be able to turn shared runners into specific ones. !4961
- Update RedCloth to 4.3.2 for CVE-2012-6684. !4929 (Takuya Noguchi)
- Improve the request / withdraw access button. !4860
v 8.9.4 v 8.9.4
- Fix privilege escalation issue with OAuth external users. - Fix privilege escalation issue with OAuth external users.
......
...@@ -2,7 +2,7 @@ class @CiBuild ...@@ -2,7 +2,7 @@ class @CiBuild
@interval: null @interval: null
@state: null @state: null
constructor: (@build_url, @build_status, @state) -> constructor: (@page_url, @build_url, @build_status, @state) ->
clearInterval(CiBuild.interval) clearInterval(CiBuild.interval)
# Init breakpoint checker # Init breakpoint checker
...@@ -41,7 +41,7 @@ class @CiBuild ...@@ -41,7 +41,7 @@ class @CiBuild
# Only valid for runnig build when output changes during time # Only valid for runnig build when output changes during time
# #
CiBuild.interval = setInterval => CiBuild.interval = setInterval =>
if window.location.href.split("#").first() is @build_url if window.location.href.split("#").first() is @page_url
@getBuildTrace() @getBuildTrace()
, 4000 , 4000
...@@ -57,7 +57,7 @@ class @CiBuild ...@@ -57,7 +57,7 @@ class @CiBuild
getBuildTrace: -> getBuildTrace: ->
$.ajax $.ajax
url: "#{@build_url}/trace.json?state=#{encodeURIComponent(@state)}" url: "#{@page_url}/trace.json?state=#{encodeURIComponent(@state)}"
dataType: "json" dataType: "json"
success: (log) => success: (log) =>
if log.state if log.state
...@@ -70,7 +70,7 @@ class @CiBuild ...@@ -70,7 +70,7 @@ class @CiBuild
$('.js-build-output').html log.html $('.js-build-output').html log.html
@checkAutoscroll() @checkAutoscroll()
else if log.status isnt @build_status else if log.status isnt @build_status
Turbolinks.visit @build_url Turbolinks.visit @page_url
checkAutoscroll: -> checkAutoscroll: ->
$("html,body").scrollTop $("#build-trace").height() if "enabled" is $("#autoscroll-button").data("state") $("html,body").scrollTop $("#build-trace").height() if "enabled" is $("#autoscroll-button").data("state")
......
...@@ -23,6 +23,9 @@ ...@@ -23,6 +23,9 @@
.dropdown-menu, .dropdown-menu,
.dropdown-menu-nav { .dropdown-menu-nav {
display: block; display: block;
@media (max-width: $screen-xs-max) {
width: 100%;
}
} }
.dropdown-menu-toggle { .dropdown-menu-toggle {
......
...@@ -137,6 +137,15 @@ ul.content-list { ...@@ -137,6 +137,15 @@ ul.content-list {
padding-top: 1px; padding-top: 1px;
float: right; float: right;
> .control-text {
margin-right: $gl-padding-top;
line-height: 40px;
&:last-child {
margin-right: 0;
}
}
> .btn, > .btn,
> .btn-group { > .btn-group {
margin-right: $gl-padding-top; margin-right: $gl-padding-top;
......
...@@ -310,7 +310,11 @@ module ProjectsHelper ...@@ -310,7 +310,11 @@ module ProjectsHelper
end end
def last_push_event def last_push_event
if current_user return unless current_user
if fork = current_user.fork_of(@project)
current_user.recent_push(fork.id)
else
current_user.recent_push(@project.id) current_user.recent_push(@project.id)
end end
end end
......
...@@ -75,7 +75,7 @@ class AuditEventService ...@@ -75,7 +75,7 @@ class AuditEventService
@details = { @details = {
with: @details[:with], with: @details[:with],
target_id: @author.id, target_id: @author.id,
target_type: "User", target_type: 'User',
target_details: @author.name, target_details: @author.name,
} }
......
...@@ -20,9 +20,11 @@ module Auth ...@@ -20,9 +20,11 @@ module Auth
token.issuer = registry.issuer token.issuer = registry.issuer
token.audience = AUDIENCE token.audience = AUDIENCE
token.expire_time = token_expire_at token.expire_time = token_expire_at
token[:access] = names.map do |name| token[:access] = names.map do |name|
{ type: 'repository', name: name, actions: %w(*) } { type: 'repository', name: name, actions: %w(*) }
end end
token.encoded token.encoded
end end
......
...@@ -3,17 +3,20 @@ require_relative 'base_service' ...@@ -3,17 +3,20 @@ require_relative 'base_service'
class CreateBranchService < BaseService class CreateBranchService < BaseService
def execute(branch_name, ref, source_project: @project) def execute(branch_name, ref, source_project: @project)
valid_branch = Gitlab::GitRefValidator.validate(branch_name) valid_branch = Gitlab::GitRefValidator.validate(branch_name)
if valid_branch == false
unless valid_branch
return error('Branch name is invalid') return error('Branch name is invalid')
end end
repository = project.repository repository = project.repository
existing_branch = repository.find_branch(branch_name) existing_branch = repository.find_branch(branch_name)
if existing_branch if existing_branch
return error('Branch already exists') return error('Branch already exists')
end end
new_branch = nil new_branch = nil
if source_project != @project if source_project != @project
repository.with_tmp_ref do |tmp_ref| repository.with_tmp_ref do |tmp_ref|
repository.fetch_ref( repository.fetch_ref(
...@@ -29,7 +32,6 @@ class CreateBranchService < BaseService ...@@ -29,7 +32,6 @@ class CreateBranchService < BaseService
end end
if new_branch if new_branch
# GitPushService handles execution of services and hooks for branch pushes
success(new_branch) success(new_branch)
else else
error('Invalid reference name') error('Invalid reference name')
...@@ -39,8 +41,6 @@ class CreateBranchService < BaseService ...@@ -39,8 +41,6 @@ class CreateBranchService < BaseService
end end
def success(branch) def success(branch)
out = super() super().merge(branch: branch)
out[:branch] = branch
out
end end
end end
...@@ -23,8 +23,6 @@ class CreateReleaseService < BaseService ...@@ -23,8 +23,6 @@ class CreateReleaseService < BaseService
end end
def success(release) def success(release)
out = super() super().merge(release: release)
out[:release] = release
out
end end
end end
class CreateSnippetService < BaseService class CreateSnippetService < BaseService
def execute def execute
if project.nil? snippet = if project
snippet = PersonalSnippet.new(params) project.snippets.build(params)
else else
snippet = project.snippets.build(params) PersonalSnippet.new(params)
end end
unless Gitlab::VisibilityLevel.allowed_for?(current_user, params[:visibility_level]) unless Gitlab::VisibilityLevel.allowed_for?(current_user, params[:visibility_level])
deny_visibility_level(snippet) deny_visibility_level(snippet)
......
...@@ -9,6 +9,7 @@ class CreateTagService < BaseService ...@@ -9,6 +9,7 @@ class CreateTagService < BaseService
message.strip! if message message.strip! if message
new_tag = nil new_tag = nil
begin begin
new_tag = repository.add_tag(current_user, tag_name, target, message) new_tag = repository.add_tag(current_user, tag_name, target, message)
rescue Rugged::TagError rescue Rugged::TagError
...@@ -22,6 +23,7 @@ class CreateTagService < BaseService ...@@ -22,6 +23,7 @@ class CreateTagService < BaseService
CreateReleaseService.new(@project, @current_user). CreateReleaseService.new(@project, @current_user).
execute(tag_name, release_description) execute(tag_name, release_description)
end end
success.merge(tag: new_tag) success.merge(tag: new_tag)
else else
error("Target #{target} is invalid") error("Target #{target} is invalid")
......
...@@ -5,7 +5,6 @@ class DeleteBranchService < BaseService ...@@ -5,7 +5,6 @@ class DeleteBranchService < BaseService
repository = project.repository repository = project.repository
branch = repository.find_branch(branch_name) branch = repository.find_branch(branch_name)
# No such branch
unless branch unless branch
return error('No such branch', 404) return error('No such branch', 404)
end end
...@@ -14,18 +13,15 @@ class DeleteBranchService < BaseService ...@@ -14,18 +13,15 @@ class DeleteBranchService < BaseService
return error('Cannot remove HEAD branch', 405) return error('Cannot remove HEAD branch', 405)
end end
# Dont allow remove of protected branch
if project.protected_branch?(branch_name) if project.protected_branch?(branch_name)
return error('Protected branch cant be removed', 405) return error('Protected branch cant be removed', 405)
end end
# Dont allow user to remove branch if he is not allowed to push
unless current_user.can?(:push_code, project) unless current_user.can?(:push_code, project)
return error('You dont have push access to repo', 405) return error('You dont have push access to repo', 405)
end end
if repository.rm_branch(current_user, branch_name) if repository.rm_branch(current_user, branch_name)
# GitPushService handles execution of services and hooks for branch pushes
success('Branch was removed') success('Branch was removed')
else else
error('Failed to remove branch') error('Failed to remove branch')
...@@ -35,15 +31,11 @@ class DeleteBranchService < BaseService ...@@ -35,15 +31,11 @@ class DeleteBranchService < BaseService
end end
def error(message, return_code = 400) def error(message, return_code = 400)
out = super(message) super(message).merge(return_code: return_code)
out[:return_code] = return_code
out
end end
def success(message) def success(message)
out = super() super().merge(message: message)
out[:message] = message
out
end end
def build_push_data(branch) def build_push_data(branch)
......
...@@ -5,7 +5,6 @@ class DeleteTagService < BaseService ...@@ -5,7 +5,6 @@ class DeleteTagService < BaseService
repository = project.repository repository = project.repository
tag = repository.find_tag(tag_name) tag = repository.find_tag(tag_name)
# No such tag
unless tag unless tag
return error('No such tag', 404) return error('No such tag', 404)
end end
...@@ -26,15 +25,11 @@ class DeleteTagService < BaseService ...@@ -26,15 +25,11 @@ class DeleteTagService < BaseService
end end
def error(message, return_code = 400) def error(message, return_code = 400)
out = super(message) super(message).merge(return_code: return_code)
out[:return_code] = return_code
out
end end
def success(message) def success(message)
out = super() super().merge(message: message)
out[:message] = message
out
end end
def build_push_data(tag) def build_push_data(tag)
......
...@@ -15,7 +15,6 @@ module Files ...@@ -15,7 +15,6 @@ module Files
params[:file_content] params[:file_content]
end end
# Validate parameters
validate validate
# Create new branch if it different from source_branch # Create new branch if it different from source_branch
...@@ -26,7 +25,7 @@ module Files ...@@ -26,7 +25,7 @@ module Files
if commit if commit
success success
else else
error("Something went wrong. Your changes were not committed") error('Something went wrong. Your changes were not committed')
end end
rescue Repository::CommitError, Gitlab::Git::Repository::InvalidBlobName, GitHooksService::PreReceiveError, ValidationError => ex rescue Repository::CommitError, Gitlab::Git::Repository::InvalidBlobName, GitHooksService::PreReceiveError, ValidationError => ex
error(ex.message) error(ex.message)
...@@ -51,12 +50,12 @@ module Files ...@@ -51,12 +50,12 @@ module Files
unless project.empty_repo? unless project.empty_repo?
unless @source_project.repository.branch_names.include?(@source_branch) unless @source_project.repository.branch_names.include?(@source_branch)
raise_error("You can only create or edit files when you are on a branch") raise_error('You can only create or edit files when you are on a branch')
end end
if different_branch? if different_branch?
if repository.branch_names.include?(@target_branch) if repository.branch_names.include?(@target_branch)
raise_error("Branch with such name already exists. You need to switch to this branch in order to make changes") raise_error('Branch with such name already exists. You need to switch to this branch in order to make changes')
end end
end end
end end
......
...@@ -29,7 +29,7 @@ module Files ...@@ -29,7 +29,7 @@ module Files
blob = repository.blob_at_branch(@source_branch, @file_path) blob = repository.blob_at_branch(@source_branch, @file_path)
if blob if blob
raise_error("Your changes could not be committed because a file with the same name already exists") raise_error('Your changes could not be committed because a file with the same name already exists')
end end
end end
end end
......
...@@ -31,6 +31,7 @@ class GitTagPushService < BaseService ...@@ -31,6 +31,7 @@ class GitTagPushService < BaseService
unless Gitlab::Git.blank_ref?(params[:newrev]) unless Gitlab::Git.blank_ref?(params[:newrev])
tag_name = Gitlab::Git.ref_name(params[:ref]) tag_name = Gitlab::Git.ref_name(params[:ref])
tag = project.repository.find_tag(tag_name) tag = project.repository.find_tag(tag_name)
if tag && tag.target == params[:newrev] if tag && tag.target == params[:newrev]
commit = project.commit(tag.target) commit = project.commit(tag.target)
commits = [commit].compact commits = [commit].compact
......
...@@ -9,6 +9,7 @@ module Issues ...@@ -9,6 +9,7 @@ module Issues
end end
issues = Issue.where(id: issues_ids) issues = Issue.where(id: issues_ids)
issues.each do |issue| issues.each do |issue|
next unless can?(current_user, :update_issue, issue) next unless can?(current_user, :update_issue, issue)
......
...@@ -20,6 +20,7 @@ module MergeRequests ...@@ -20,6 +20,7 @@ module MergeRequests
return unless merge_request.target_branch == project.default_branch return unless merge_request.target_branch == project.default_branch
closed_issues = merge_request.closes_issues(current_user) closed_issues = merge_request.closes_issues(current_user)
closed_issues.each do |issue| closed_issues.each do |issue|
if can?(current_user, :update_issue, issue) if can?(current_user, :update_issue, issue)
Issues::CloseService.new(project, current_user, {}).execute(issue, commit: merge_request) Issues::CloseService.new(project, current_user, {}).execute(issue, commit: merge_request)
......
...@@ -157,6 +157,7 @@ class NotificationService ...@@ -157,6 +157,7 @@ class NotificationService
else else
mentioned_users mentioned_users
end end
recipients = recipients.concat(participants) recipients = recipients.concat(participants)
# Merge project watchers # Merge project watchers
...@@ -180,6 +181,7 @@ class NotificationService ...@@ -180,6 +181,7 @@ class NotificationService
# build notify method like 'note_commit_email' # build notify method like 'note_commit_email'
notify_method = "note_#{note.noteable_type.underscore}_email".to_sym notify_method = "note_#{note.noteable_type.underscore}_email".to_sym
recipients.each do |recipient| recipients.each do |recipient|
mailer.send(notify_method, recipient.id, note.id).deliver_later mailer.send(notify_method, recipient.id, note.id).deliver_later
end end
......
...@@ -3,10 +3,11 @@ module Projects ...@@ -3,10 +3,11 @@ module Projects
def execute def execute
# check that user is allowed to set specified visibility_level # check that user is allowed to set specified visibility_level
new_visibility = params[:visibility_level] new_visibility = params[:visibility_level]
if new_visibility && new_visibility.to_i != project.visibility_level if new_visibility && new_visibility.to_i != project.visibility_level
unless can?(current_user, :change_visibility_level, project) && unless can?(current_user, :change_visibility_level, project) &&
Gitlab::VisibilityLevel.allowed_for?(current_user, new_visibility) Gitlab::VisibilityLevel.allowed_for?(current_user, new_visibility)
deny_visibility_level(project, new_visibility) deny_visibility_level(project, new_visibility)
return project return project
end end
......
...@@ -82,7 +82,7 @@ class SystemNoteService ...@@ -82,7 +82,7 @@ class SystemNoteService
end end
body << ' ' << 'label'.pluralize(labels_count) body << ' ' << 'label'.pluralize(labels_count)
body = "#{body.capitalize}" body = body.capitalize
create_note(noteable: noteable, project: project, author: author, note: body) create_note(noteable: noteable, project: project, author: author, note: body)
end end
...@@ -125,7 +125,7 @@ class SystemNoteService ...@@ -125,7 +125,7 @@ class SystemNoteService
# Returns the created Note object # Returns the created Note object
def self.change_status(noteable, project, author, status, source) def self.change_status(noteable, project, author, status, source)
body = "Status changed to #{status}" body = "Status changed to #{status}"
body += " by #{source.gfm_reference(project)}" if source body << " by #{source.gfm_reference(project)}" if source
create_note(noteable: noteable, project: project, author: author, note: body) create_note(noteable: noteable, project: project, author: author, note: body)
end end
...@@ -139,7 +139,7 @@ class SystemNoteService ...@@ -139,7 +139,7 @@ class SystemNoteService
# Called when 'merge when build succeeds' is canceled # Called when 'merge when build succeeds' is canceled
def self.cancel_merge_when_build_succeeds(noteable, project, author) def self.cancel_merge_when_build_succeeds(noteable, project, author)
body = "Canceled the automatic merge" body = 'Canceled the automatic merge'
create_note(noteable: noteable, project: project, author: author, note: body) create_note(noteable: noteable, project: project, author: author, note: body)
end end
...@@ -236,6 +236,7 @@ class SystemNoteService ...@@ -236,6 +236,7 @@ class SystemNoteService
else else
'deleted' 'deleted'
end end
body = "#{verb} #{branch_type.to_s} branch `#{branch}`".capitalize body = "#{verb} #{branch_type.to_s} branch `#{branch}`".capitalize
create_note(noteable: noteable, project: project, author: author, note: body) create_note(noteable: noteable, project: project, author: author, note: body)
end end
......
...@@ -21,8 +21,6 @@ class UpdateReleaseService < BaseService ...@@ -21,8 +21,6 @@ class UpdateReleaseService < BaseService
end end
def success(release) def success(release)
out = super() super().merge(release: release)
out[:release] = release
out
end end
end end
...@@ -9,6 +9,7 @@ class UpdateSnippetService < BaseService ...@@ -9,6 +9,7 @@ class UpdateSnippetService < BaseService
def execute def execute
# check that user is allowed to set specified visibility_level # check that user is allowed to set specified visibility_level
new_visibility = params[:visibility_level] new_visibility = params[:visibility_level]
if new_visibility && new_visibility.to_i != snippet.visibility_level if new_visibility && new_visibility.to_i != snippet.visibility_level
unless Gitlab::VisibilityLevel.allowed_for?(current_user, new_visibility) unless Gitlab::VisibilityLevel.allowed_for?(current_user, new_visibility)
deny_visibility_level(snippet, new_visibility) deny_visibility_level(snippet, new_visibility)
......
...@@ -7,7 +7,9 @@ ...@@ -7,7 +7,9 @@
%span You pushed to %span You pushed to
= link_to namespace_project_commits_path(event.project.namespace, event.project, event.ref_name) do = link_to namespace_project_commits_path(event.project.namespace, event.project, event.ref_name) do
%strong= event.ref_name %strong= event.ref_name
branch - if @project && event.project != @project
%span at
%strong= link_to_project event.project
#{time_ago_with_tooltip(event.created_at)} #{time_ago_with_tooltip(event.created_at)}
.pull-right .pull-right
......
...@@ -67,4 +67,4 @@ ...@@ -67,4 +67,4 @@
= render "sidebar" = render "sidebar"
:javascript :javascript
new CiBuild("#{namespace_project_build_url(@project.namespace, @project, @build, :json)}", "#{@build.status}", "#{trace_with_state[:state]}") new CiBuild("#{namespace_project_build_url(@project.namespace, @project, @build)}", "#{namespace_project_build_url(@project.namespace, @project, @build, :json)}", "#{@build.status}", "#{trace_with_state[:state]}")
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
= ci_icon_for_status(status) = ci_icon_for_status(status)
- if stage - if stage
&nbsp; &nbsp;
= stage.titleize.pluralize = stage.titleize
= render statuses.latest.ordered, coverage: @project.build_coverage_enabled?, stage: false, ref: false, allow_retry: true = render statuses.latest.ordered, coverage: @project.build_coverage_enabled?, stage: false, ref: false, allow_retry: true
= render statuses.retried.ordered, coverage: @project.build_coverage_enabled?, stage: false, ref: false, retried: true = render statuses.retried.ordered, coverage: @project.build_coverage_enabled?, stage: false, ref: false, retried: true
%tr %tr
......
...@@ -50,7 +50,7 @@ ...@@ -50,7 +50,7 @@
- stages.each do |stage| - stages.each do |stage|
%th.stage %th.stage
%span.has-tooltip{ title: "#{stage.titleize}" } %span.has-tooltip{ title: "#{stage.titleize}" }
= stage.titleize.pluralize = stage.titleize
%th Duration %th Duration
%th %th
= render @pipelines, commit_sha: true, stage: true, allow_retry: true, stages: stages = render @pipelines, commit_sha: true, stage: true, allow_retry: true, stages: stages
......
...@@ -7,22 +7,22 @@ ...@@ -7,22 +7,22 @@
.nav-text .nav-text
Tags give the ability to mark specific points in history as being important Tags give the ability to mark specific points in history as being important
- if can? current_user, :push_code, @project .nav-controls
.nav-controls - if can? current_user, :push_code, @project
= link_to new_namespace_project_tag_path(@project.namespace, @project), class: 'btn btn-create new-tag-btn' do = link_to new_namespace_project_tag_path(@project.namespace, @project), class: 'btn btn-create new-tag-btn' do
New tag New tag
.dropdown.inline .dropdown.inline
%button.dropdown-toggle.btn{ type: 'button', data: { toggle: 'dropdown'} } %button.dropdown-toggle.btn{ type: 'button', data: { toggle: 'dropdown'} }
%span.light= @sort.humanize %span.light= @sort.humanize
%b.caret %b.caret
%ul.dropdown-menu.dropdown-menu-align-right %ul.dropdown-menu.dropdown-menu-align-right
%li %li
= link_to namespace_project_tags_path(sort: nil) do = link_to namespace_project_tags_path(sort: nil) do
Name Name
= link_to namespace_project_tags_path(sort: sort_value_recently_updated) do = link_to namespace_project_tags_path(sort: sort_value_recently_updated) do
= sort_title_recently_updated = sort_title_recently_updated
= link_to namespace_project_tags_path(sort: sort_value_oldest_updated) do = link_to namespace_project_tags_path(sort: sort_value_oldest_updated) do
= sort_title_oldest_updated = sort_title_oldest_updated
= render 'projects/commits/mirror_status' = render 'projects/commits/mirror_status'
......
...@@ -3,71 +3,74 @@ ...@@ -3,71 +3,74 @@
- user = member.user - user = member.user
%li.js-toggle-container{ class: dom_class(member), id: dom_id(member) } %li.js-toggle-container{ class: dom_class(member), id: dom_id(member) }
%span{ class: ("list-item-name" if show_controls) }
- if user
= image_tag avatar_icon(user, 24), class: "avatar s24", alt: ''
%strong
= link_to user.name, user_path(user)
%span.cgray= user.username
- if user == current_user
%span.label.label-success It's you
- if user.blocked?
%label.label.label-danger
%strong Blocked
- if member.request?
%span.cgray
– Requested
= time_ago_with_tooltip(member.requested_at)
- else
= image_tag avatar_icon(member.invite_email, 24), class: "avatar s24", alt: ''
%strong= member.invite_email
%span.cgray
– Invited
- if member.created_by
by
= link_to member.created_by.name, user_path(member.created_by)
= time_ago_with_tooltip(member.created_at)
- if show_controls && can?(current_user, action_member_permission(:admin, member), member.source)
= link_to 'Resend invite', polymorphic_path([:resend_invite, member]),
method: :post,
class: 'btn-xs btn'
- if show_roles - if show_roles
%span.pull-right .controls
%strong= member.human_access %strong.control-text= member.human_access
- if show_controls - if show_controls
- if !user && can?(current_user, action_member_permission(:admin, member), member.source)
= link_to 'Resend invite', polymorphic_path([:resend_invite, member]),
method: :post,
class: 'btn'
- if can?(current_user, action_member_permission(:update, member), member) - if can?(current_user, action_member_permission(:update, member), member)
= button_tag icon('pencil'), = button_tag icon('pencil'),
type: 'button', type: 'button',
class: 'btn-xs btn btn-grouped inline js-toggle-button', class: 'btn inline js-toggle-button',
title: 'Edit access level' title: 'Edit access level'
- if member.request? - if member.request?
&nbsp;
= link_to icon('check inverse'), polymorphic_path([:approve_access_request, member]), = link_to icon('check inverse'), polymorphic_path([:approve_access_request, member]),
method: :post, method: :post,
class: 'btn-xs btn btn-success', class: 'btn btn-success',
title: 'Grant access' title: 'Grant access'
- if can?(current_user, action_member_permission(:destroy, member), member) - if can?(current_user, action_member_permission(:destroy, member), member)
&nbsp;
- if current_user == user - if current_user == user
= link_to icon('sign-out', text: 'Leave'), polymorphic_path([:leave, member.source, :members]), = link_to icon('sign-out', text: 'Leave'), polymorphic_path([:leave, member.source, :members]),
method: :delete, method: :delete,
data: { confirm: leave_confirmation_message(member.source) }, data: { confirm: leave_confirmation_message(member.source) },
class: 'btn-xs btn btn-remove' class: 'btn btn-remove'
- else - else
= link_to icon('trash'), member, = link_to icon('trash'), member,
remote: true, remote: true,
method: :delete, method: :delete,
data: { confirm: remove_member_message(member) }, data: { confirm: remove_member_message(member) },
class: 'btn-xs btn btn-remove', class: 'btn btn-remove',
title: remove_member_title(member) title: remove_member_title(member)
%span{ class: ("list-item-name" if show_controls) }
- if user
= image_tag avatar_icon(user, 40), class: "avatar s40", alt: ''
%strong
= link_to user.name, user_path(user)
%span.cgray= user.username
- if user == current_user
%span.label.label-success It's you
- if user.blocked?
%label.label.label-danger
%strong Blocked
.cgray
- if member.request?
Requested
= time_ago_with_tooltip(member.requested_at)
- else
Joined #{time_ago_with_tooltip(member.created_at)}
- else
= image_tag avatar_icon(member.invite_email, 40), class: "avatar s40", alt: ''
%strong= member.invite_email
.cgray
Invited
- if member.created_by
by
= link_to member.created_by.name, user_path(member.created_by)
= time_ago_with_tooltip(member.created_at)
- if show_roles
.edit-member.hide.js-toggle-content .edit-member.hide.js-toggle-content
%br %br
= form_for member, remote: true do |f| = form_for member, remote: true do |f|
......
...@@ -62,7 +62,8 @@ class Spinach::Features::AdminGroups < Spinach::FeatureSteps ...@@ -62,7 +62,8 @@ class Spinach::Features::AdminGroups < Spinach::FeatureSteps
step 'I should see "johndoe@gitlab.com" in team list in every project as "Reporter"' do step 'I should see "johndoe@gitlab.com" in team list in every project as "Reporter"' do
page.within ".group-users-list" do page.within ".group-users-list" do
expect(page).to have_content "johndoe@gitlab.com – Invited by" expect(page).to have_content "johndoe@gitlab.com"
expect(page).to have_content "Invited by"
expect(page).to have_content "Reporter" expect(page).to have_content "Reporter"
end end
end end
......
...@@ -4,7 +4,7 @@ module Gitlab ...@@ -4,7 +4,7 @@ module Gitlab
delegate :repo, :sha, :ref, to: :raw_data delegate :repo, :sha, :ref, to: :raw_data
def exists? def exists?
project.repository.branch_exists?(ref) branch_exists? && commit_exists?
end end
def name def name
...@@ -15,11 +15,15 @@ module Gitlab ...@@ -15,11 +15,15 @@ module Gitlab
repo.present? repo.present?
end end
def valid? private
repo.present?
def branch_exists?
project.repository.branch_exists?(ref)
end end
private def commit_exists?
project.repository.commit(sha).present?
end
def short_id def short_id
sha.to_s[0..7] sha.to_s[0..7]
......
...@@ -131,8 +131,10 @@ module Gitlab ...@@ -131,8 +131,10 @@ module Gitlab
def clean_up_restored_branches(branches) def clean_up_restored_branches(branches)
branches.each do |name, _| branches.each do |name, _|
client.delete_ref(repo, "heads/#{name}") client.delete_ref(repo, "heads/#{name}")
project.repository.rm_branch(project.creator, name) project.repository.delete_branch(name) rescue Rugged::ReferenceError
end end
project.repository.after_remove_branch
end end
def apply_labels(issuable) def apply_labels(issuable)
......
...@@ -123,7 +123,7 @@ describe "Pipelines" do ...@@ -123,7 +123,7 @@ describe "Pipelines" do
before { visit namespace_project_pipeline_path(project.namespace, project, pipeline) } before { visit namespace_project_pipeline_path(project.namespace, project, pipeline) }
it 'showing a list of builds' do it 'showing a list of builds' do
expect(page).to have_content('Tests') expect(page).to have_content('Test')
expect(page).to have_content(@success.id) expect(page).to have_content(@success.id)
expect(page).to have_content('Deploy') expect(page).to have_content('Deploy')
expect(page).to have_content(@failed.id) expect(page).to have_content(@failed.id)
......
...@@ -2,17 +2,18 @@ require 'spec_helper' ...@@ -2,17 +2,18 @@ require 'spec_helper'
describe Gitlab::GithubImport::BranchFormatter, lib: true do describe Gitlab::GithubImport::BranchFormatter, lib: true do
let(:project) { create(:project) } let(:project) { create(:project) }
let(:commit) { create(:commit, project: project) }
let(:repo) { double } let(:repo) { double }
let(:raw) do let(:raw) do
{ {
ref: 'feature', ref: 'feature',
repo: repo, repo: repo,
sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b' sha: commit.id
} }
end end
describe '#exists?' do describe '#exists?' do
it 'returns true when branch exists' do it 'returns true when both branch, and commit exists' do
branch = described_class.new(project, double(raw)) branch = described_class.new(project, double(raw))
expect(branch.exists?).to eq true expect(branch.exists?).to eq true
...@@ -23,6 +24,12 @@ describe Gitlab::GithubImport::BranchFormatter, lib: true do ...@@ -23,6 +24,12 @@ describe Gitlab::GithubImport::BranchFormatter, lib: true do
expect(branch.exists?).to eq false expect(branch.exists?).to eq false
end end
it 'returns false when commit does not exist' do
branch = described_class.new(project, double(raw.merge(sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b')))
expect(branch.exists?).to eq false
end
end end
describe '#name' do describe '#name' do
...@@ -33,7 +40,7 @@ describe Gitlab::GithubImport::BranchFormatter, lib: true do ...@@ -33,7 +40,7 @@ describe Gitlab::GithubImport::BranchFormatter, lib: true do
end end
it 'returns formatted ref when branch does not exist' do it 'returns formatted ref when branch does not exist' do
branch = described_class.new(project, double(raw.merge(ref: 'removed-branch'))) branch = described_class.new(project, double(raw.merge(ref: 'removed-branch', sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b')))
expect(branch.name).to eq 'removed-branch-2e5d3239' expect(branch.name).to eq 'removed-branch-2e5d3239'
end end
...@@ -51,18 +58,18 @@ describe Gitlab::GithubImport::BranchFormatter, lib: true do ...@@ -51,18 +58,18 @@ describe Gitlab::GithubImport::BranchFormatter, lib: true do
it 'returns raw sha' do it 'returns raw sha' do
branch = described_class.new(project, double(raw)) branch = described_class.new(project, double(raw))
expect(branch.sha).to eq '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b' expect(branch.sha).to eq commit.id
end end
end end
describe '#valid?' do describe '#valid?' do
it 'returns true when repository exists' do it 'returns true when raw repo is present' do
branch = described_class.new(project, double(raw)) branch = described_class.new(project, double(raw))
expect(branch.valid?).to eq true expect(branch.valid?).to eq true
end end
it 'returns false when repository does not exist' do it 'returns false when raw repo is blank' do
branch = described_class.new(project, double(raw.merge(repo: nil))) branch = described_class.new(project, double(raw.merge(repo: nil)))
expect(branch.valid?).to eq false expect(branch.valid?).to eq false
......
...@@ -2,11 +2,13 @@ require 'spec_helper' ...@@ -2,11 +2,13 @@ require 'spec_helper'
describe Gitlab::GithubImport::PullRequestFormatter, lib: true do describe Gitlab::GithubImport::PullRequestFormatter, lib: true do
let(:project) { create(:project) } let(:project) { create(:project) }
let(:source_sha) { create(:commit, project: project).id }
let(:target_sha) { create(:commit, project: project, git_commit: RepoHelpers.another_sample_commit).id }
let(:repository) { double(id: 1, fork: false) } let(:repository) { double(id: 1, fork: false) }
let(:source_repo) { repository } let(:source_repo) { repository }
let(:source_branch) { double(ref: 'feature', repo: source_repo, sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b') } let(:source_branch) { double(ref: 'feature', repo: source_repo, sha: source_sha) }
let(:target_repo) { repository } let(:target_repo) { repository }
let(:target_branch) { double(ref: 'master', repo: target_repo, sha: '8ffb3c15a5475e59ae909384297fede4badcb4c7') } let(:target_branch) { double(ref: 'master', repo: target_repo, sha: target_sha) }
let(:octocat) { double(id: 123456, login: 'octocat') } let(:octocat) { double(id: 123456, login: 'octocat') }
let(:created_at) { DateTime.strptime('2011-01-26T19:01:12Z') } let(:created_at) { DateTime.strptime('2011-01-26T19:01:12Z') }
let(:updated_at) { DateTime.strptime('2011-01-27T19:01:12Z') } let(:updated_at) { DateTime.strptime('2011-01-27T19:01:12Z') }
...@@ -41,10 +43,10 @@ describe Gitlab::GithubImport::PullRequestFormatter, lib: true do ...@@ -41,10 +43,10 @@ describe Gitlab::GithubImport::PullRequestFormatter, lib: true do
description: "*Created by: octocat*\n\nPlease pull these awesome changes", description: "*Created by: octocat*\n\nPlease pull these awesome changes",
source_project: project, source_project: project,
source_branch: 'feature', source_branch: 'feature',
head_source_sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b', head_source_sha: source_sha,
target_project: project, target_project: project,
target_branch: 'master', target_branch: 'master',
base_target_sha: '8ffb3c15a5475e59ae909384297fede4badcb4c7', base_target_sha: target_sha,
state: 'opened', state: 'opened',
milestone: nil, milestone: nil,
author_id: project.creator_id, author_id: project.creator_id,
...@@ -68,10 +70,10 @@ describe Gitlab::GithubImport::PullRequestFormatter, lib: true do ...@@ -68,10 +70,10 @@ describe Gitlab::GithubImport::PullRequestFormatter, lib: true do
description: "*Created by: octocat*\n\nPlease pull these awesome changes", description: "*Created by: octocat*\n\nPlease pull these awesome changes",
source_project: project, source_project: project,
source_branch: 'feature', source_branch: 'feature',
head_source_sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b', head_source_sha: source_sha,
target_project: project, target_project: project,
target_branch: 'master', target_branch: 'master',
base_target_sha: '8ffb3c15a5475e59ae909384297fede4badcb4c7', base_target_sha: target_sha,
state: 'closed', state: 'closed',
milestone: nil, milestone: nil,
author_id: project.creator_id, author_id: project.creator_id,
...@@ -95,10 +97,10 @@ describe Gitlab::GithubImport::PullRequestFormatter, lib: true do ...@@ -95,10 +97,10 @@ describe Gitlab::GithubImport::PullRequestFormatter, lib: true do
description: "*Created by: octocat*\n\nPlease pull these awesome changes", description: "*Created by: octocat*\n\nPlease pull these awesome changes",
source_project: project, source_project: project,
source_branch: 'feature', source_branch: 'feature',
head_source_sha: '2e5d3239642f9161dcbbc4b70a211a68e5e45e2b', head_source_sha: source_sha,
target_project: project, target_project: project,
target_branch: 'master', target_branch: 'master',
base_target_sha: '8ffb3c15a5475e59ae909384297fede4badcb4c7', base_target_sha: target_sha,
state: 'merged', state: 'merged',
milestone: nil, milestone: nil,
author_id: project.creator_id, author_id: project.creator_id,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment