Commit b342c852 authored by GitLab Bot's avatar GitLab Bot

Automatic merge of gitlab-org/gitlab-ce master

parents 34d56e9f f4101aea
...@@ -3,12 +3,12 @@ ...@@ -3,12 +3,12 @@
class Admin::RequestsProfilesController < Admin::ApplicationController class Admin::RequestsProfilesController < Admin::ApplicationController
def index def index
@profile_token = Gitlab::RequestProfiler.profile_token @profile_token = Gitlab::RequestProfiler.profile_token
@profiles = Gitlab::RequestProfiler::Profile.all.group_by(&:request_path) @profiles = Gitlab::RequestProfiler.all.group_by(&:request_path)
end end
def show def show
clean_name = Rack::Utils.clean_path_info(params[:name]) clean_name = Rack::Utils.clean_path_info(params[:name])
profile = Gitlab::RequestProfiler::Profile.find(clean_name) profile = Gitlab::RequestProfiler.find(clean_name)
unless profile && profile.content_type unless profile && profile.content_type
return redirect_to admin_requests_profiles_path, alert: 'Profile not found' return redirect_to admin_requests_profiles_path, alert: 'Profile not found'
......
...@@ -117,14 +117,8 @@ module MergeRequests ...@@ -117,14 +117,8 @@ module MergeRequests
collect_errors_from_merge_request(merge_request) unless merge_request.valid? collect_errors_from_merge_request(merge_request) unless merge_request.valid?
end end
def create_params(branch) def base_params
params = { params = {}
assignees: [current_user],
source_branch: branch,
source_project: project,
target_branch: push_options[:target] || target_project.default_branch,
target_project: target_project
}
if push_options.key?(:merge_when_pipeline_succeeds) if push_options.key?(:merge_when_pipeline_succeeds)
params.merge!( params.merge!(
...@@ -133,26 +127,36 @@ module MergeRequests ...@@ -133,26 +127,36 @@ module MergeRequests
) )
end end
if push_options.key?(:remove_source_branch)
params[:force_remove_source_branch] = push_options[:remove_source_branch]
end
if push_options.key?(:target)
params[:target_branch] = push_options[:target]
end
params params
end end
def update_params def create_params(branch)
params = {} params = base_params
if push_options.key?(:merge_when_pipeline_succeeds)
params.merge!( params.merge!(
merge_when_pipeline_succeeds: push_options[:merge_when_pipeline_succeeds], assignees: [current_user],
merge_user: current_user source_branch: branch,
source_project: project,
target_project: target_project
) )
end
if push_options.key?(:target) params[:target_branch] ||= target_project.default_branch
params[:target_branch] = push_options[:target]
end
params params
end end
def update_params
base_params
end
def collect_errors_from_merge_request(merge_request) def collect_errors_from_merge_request(merge_request)
merge_request.errors.full_messages.each do |error| merge_request.errors.full_messages.each do |error|
errors << error errors << error
......
---
title: Support remove source branch on merge w/ push options
merge_request: 30728
author:
type: added
...@@ -286,6 +286,7 @@ as pushing changes: ...@@ -286,6 +286,7 @@ as pushing changes:
- Create a new merge request for the pushed branch. - Create a new merge request for the pushed branch.
- Set the target of the merge request to a particular branch. - Set the target of the merge request to a particular branch.
- Set the merge request to merge when its pipeline succeeds. - Set the merge request to merge when its pipeline succeeds.
- Set the merge request to remove the source branch when it's merged.
### Create a new merge request using git push options ### Create a new merge request using git push options
...@@ -329,6 +330,19 @@ pipeline succeeds at the same time using a `-o` flag per push option: ...@@ -329,6 +330,19 @@ pipeline succeeds at the same time using a `-o` flag per push option:
git push -o merge_request.create -o merge_request.merge_when_pipeline_succeeds git push -o merge_request.create -o merge_request.merge_when_pipeline_succeeds
``` ```
### Set removing the source branch using git push options
To set an existing merge request to remove the source branch when the
merge request is merged, the
`merge_request.remove_source_branch` push option can be used:
```sh
git push -o merge_request.remove_source_branch
```
You can also use this push option in addition to the
`merge_request.create` push option.
## Find the merge request that introduced a change ## Find the merge request that introduced a change
> [Introduced](https://gitlab.com/gitlab-org/gitlab-ce/issues/2383) in GitLab 10.5. > [Introduced](https://gitlab.com/gitlab-org/gitlab-ce/issues/2383) in GitLab 10.5.
......
...@@ -4,7 +4,12 @@ module Gitlab ...@@ -4,7 +4,12 @@ module Gitlab
class PushOptions class PushOptions
VALID_OPTIONS = HashWithIndifferentAccess.new({ VALID_OPTIONS = HashWithIndifferentAccess.new({
merge_request: { merge_request: {
keys: [:create, :merge_when_pipeline_succeeds, :target] keys: [
:create,
:merge_when_pipeline_succeeds,
:remove_source_branch,
:target
]
}, },
ci: { ci: {
keys: [:skip] keys: [:skip]
......
...@@ -6,6 +6,21 @@ module Gitlab ...@@ -6,6 +6,21 @@ module Gitlab
module RequestProfiler module RequestProfiler
PROFILES_DIR = "#{Gitlab.config.shared.path}/tmp/requests_profiles".freeze PROFILES_DIR = "#{Gitlab.config.shared.path}/tmp/requests_profiles".freeze
def all
Dir["#{PROFILES_DIR}/*.{html,txt}"].map do |path|
Profile.new(File.basename(path))
end.select(&:valid?)
end
module_function :all
def find(name)
file_path = File.join(PROFILES_DIR, name)
return unless File.exist?(file_path)
Profile.new(name)
end
module_function :find
def profile_token def profile_token
Rails.cache.fetch('profile-token') do Rails.cache.fetch('profile-token') do
Devise.friendly_token Devise.friendly_token
......
...@@ -7,19 +7,6 @@ module Gitlab ...@@ -7,19 +7,6 @@ module Gitlab
alias_method :to_param, :name alias_method :to_param, :name
def self.all
Dir["#{PROFILES_DIR}/*.{html,txt}"].map do |path|
new(File.basename(path))
end
end
def self.find(name)
file_path = File.join(PROFILES_DIR, name)
return unless File.exist?(file_path)
new(name)
end
def initialize(name) def initialize(name)
@name = name @name = name
@file_path = File.join(PROFILES_DIR, name) @file_path = File.join(PROFILES_DIR, name)
...@@ -27,8 +14,8 @@ module Gitlab ...@@ -27,8 +14,8 @@ module Gitlab
set_attributes set_attributes
end end
def content def valid?
File.read("#{PROFILES_DIR}/#{name}") @request_path.present?
end end
def content_type def content_type
...@@ -43,11 +30,13 @@ module Gitlab ...@@ -43,11 +30,13 @@ module Gitlab
private private
def set_attributes def set_attributes
_, path, timestamp, profile_mode, type = name.split(/(.*)_(\d+)_(.*)\.(html|txt)$/) matches = name.match(/^(?<path>.*)_(?<timestamp>\d+)(_(?<profile_mode>\w+))?\.(?<type>html|txt)$/)
@request_path = path.tr('|', '/') return unless matches
@time = Time.at(timestamp.to_i).utc
@profile_mode = profile_mode @request_path = matches[:path].tr('|', '/')
@type = type @time = Time.at(matches[:timestamp].to_i).utc
@profile_mode = matches[:profile_mode] || 'unknown'
@type = matches[:type]
end end
end end
end end
......
...@@ -23,7 +23,7 @@ describe Admin::RequestsProfilesController do ...@@ -23,7 +23,7 @@ describe Admin::RequestsProfilesController do
end end
after do after do
File.unlink(test_file) FileUtils.rm_rf(tmpdir)
end end
context 'when loading HTML profile' do context 'when loading HTML profile' do
......
require 'spec_helper' require 'spec_helper'
describe 'Admin::RequestsProfilesController' do describe 'Admin::RequestsProfilesController' do
let(:tmpdir) { Dir.mktmpdir('profiler-test') }
before do before do
FileUtils.mkdir_p(Gitlab::RequestProfiler::PROFILES_DIR) stub_const('Gitlab::RequestProfiler::PROFILES_DIR', tmpdir)
sign_in(create(:admin)) sign_in(create(:admin))
end end
after do after do
Gitlab::RequestProfiler.remove_all_profiles FileUtils.rm_rf(tmpdir)
end end
describe 'GET /admin/requests_profiles' do describe 'GET /admin/requests_profiles' do
...@@ -60,6 +62,12 @@ describe 'Admin::RequestsProfilesController' do ...@@ -60,6 +62,12 @@ describe 'Admin::RequestsProfilesController' do
name: "|gitlab-org|infrastructure_#{time2.to_i}_memory.html", name: "|gitlab-org|infrastructure_#{time2.to_i}_memory.html",
created: time2, created: time2,
profile_mode: 'Memory' profile_mode: 'Memory'
},
{
request_path: '/gitlab-org/infrastructure',
name: "|gitlab-org|infrastructure_#{time2.to_i}.html",
created: time2,
profile_mode: 'Unknown'
} }
] ]
end end
......
require 'fast_spec_helper'
describe Gitlab::RequestProfiler::Profile do
let(:profile) { described_class.new(filename) }
describe '.new' do
context 'using old filename' do
let(:filename) { '|api|v4|version.txt_1562854738.html' }
it 'returns valid data' do
expect(profile).to be_valid
expect(profile.request_path).to eq('/api/v4/version.txt')
expect(profile.time).to eq(Time.at(1562854738).utc)
expect(profile.type).to eq('html')
end
end
context 'using new filename' do
let(:filename) { '|api|v4|version.txt_1563547949_execution.html' }
it 'returns valid data' do
expect(profile).to be_valid
expect(profile.request_path).to eq('/api/v4/version.txt')
expect(profile.profile_mode).to eq('execution')
expect(profile.time).to eq(Time.at(1563547949).utc)
expect(profile.type).to eq('html')
end
end
end
describe '#content_type' do
context 'when using html file' do
let(:filename) { '|api|v4|version.txt_1562854738_memory.html' }
it 'returns valid data' do
expect(profile).to be_valid
expect(profile.content_type).to eq('text/html')
end
end
context 'when using text file' do
let(:filename) { '|api|v4|version.txt_1562854738_memory.txt' }
it 'returns valid data' do
expect(profile).to be_valid
expect(profile.content_type).to eq('text/plain')
end
end
context 'when file is unknown' do
let(:filename) { '|api|v4|version.txt_1562854738_memory.xxx' }
it 'returns valid data' do
expect(profile).not_to be_valid
expect(profile.content_type).to be_nil
end
end
end
end
...@@ -13,15 +13,42 @@ describe Gitlab::RequestProfiler do ...@@ -13,15 +13,42 @@ describe Gitlab::RequestProfiler do
end end
end end
context 'with temporary PROFILES_DIR' do
let(:tmpdir) { Dir.mktmpdir('profiler-test') }
let(:profile_name) { '|api|v4|version.txt_1562854738_memory.html' }
let(:profile_path) { File.join(tmpdir, profile_name) }
before do
stub_const('Gitlab::RequestProfiler::PROFILES_DIR', tmpdir)
FileUtils.touch(profile_path)
end
after do
FileUtils.rm_rf(tmpdir)
end
describe '.remove_all_profiles' do describe '.remove_all_profiles' do
it 'removes Gitlab::RequestProfiler::PROFILES_DIR directory' do it 'removes Gitlab::RequestProfiler::PROFILES_DIR directory' do
dir = described_class::PROFILES_DIR described_class.remove_all_profiles
FileUtils.mkdir_p(dir)
expect(Dir.exist?(dir)).to be true expect(Dir.exist?(tmpdir)).to be false
end
end
described_class.remove_all_profiles describe '.all' do
expect(Dir.exist?(dir)).to be false subject { described_class.all }
it 'returns all profiles' do
expect(subject.map(&:name)).to contain_exactly(profile_name)
end
end
describe '.find' do
subject { described_class.find(profile_name) }
it 'returns all profiles' do
expect(subject.name).to eq(profile_name)
end
end end
end end
end end
...@@ -90,6 +90,16 @@ describe MergeRequests::PushOptionsHandlerService do ...@@ -90,6 +90,16 @@ describe MergeRequests::PushOptionsHandlerService do
end end
end end
shared_examples_for 'a service that can remove the source branch when it is merged' do
subject(:last_mr) { MergeRequest.last }
it 'returns true to force_remove_source_branch?' do
service.execute
expect(last_mr.force_remove_source_branch?).to eq(true)
end
end
shared_examples_for 'a service that does not create a merge request' do shared_examples_for 'a service that does not create a merge request' do
it do it do
expect { service.execute }.not_to change { MergeRequest.count } expect { service.execute }.not_to change { MergeRequest.count }
...@@ -208,6 +218,72 @@ describe MergeRequests::PushOptionsHandlerService do ...@@ -208,6 +218,72 @@ describe MergeRequests::PushOptionsHandlerService do
end end
end end
describe '`remove_source_branch` push option' do
let(:push_options) { { remove_source_branch: true } }
context 'with a new branch' do
let(:changes) { new_branch_changes }
it_behaves_like 'a service that does not create a merge request'
it 'adds an error to the service' do
error = "A merge_request.create push option is required to create a merge request for branch #{source_branch}"
service.execute
expect(service.errors).to include(error)
end
context 'when coupled with the `create` push option' do
let(:push_options) { { create: true, remove_source_branch: true } }
it_behaves_like 'a service that can create a merge request'
it_behaves_like 'a service that can remove the source branch when it is merged'
end
end
context 'with an existing branch but no open MR' do
let(:changes) { existing_branch_changes }
it_behaves_like 'a service that does not create a merge request'
it 'adds an error to the service' do
error = "A merge_request.create push option is required to create a merge request for branch #{source_branch}"
service.execute
expect(service.errors).to include(error)
end
context 'when coupled with the `create` push option' do
let(:push_options) { { create: true, remove_source_branch: true } }
it_behaves_like 'a service that can create a merge request'
it_behaves_like 'a service that can remove the source branch when it is merged'
end
end
context 'with an existing branch that has a merge request open' do
let(:changes) { existing_branch_changes }
let!(:merge_request) { create(:merge_request, source_project: project, source_branch: source_branch)}
it_behaves_like 'a service that does not create a merge request'
it_behaves_like 'a service that can remove the source branch when it is merged'
end
context 'with a deleted branch' do
let(:changes) { deleted_branch_changes }
it_behaves_like 'a service that does nothing'
end
context 'with the project default branch' do
let(:changes) { default_branch_changes }
it_behaves_like 'a service that does nothing'
end
end
describe '`target` push option' do describe '`target` push option' do
let(:push_options) { { target: target_branch } } let(:push_options) { { target: target_branch } }
......
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