Commit 0a7d50ac authored by Stan Hu's avatar Stan Hu

Merge branch 'ce-to-ee-2018-06-15' into 'master'

CE upstream - 2018-06-15 12:21 UTC

See merge request gitlab-org/gitlab-ee!6154
parents ea5cfebf 32f39fb2
...@@ -9,7 +9,7 @@ export default { ...@@ -9,7 +9,7 @@ export default {
return Vue.http.get(endpoint, { params: { format: 'json' } }); return Vue.http.get(endpoint, { params: { format: 'json' } });
}, },
getFileData(endpoint) { getFileData(endpoint) {
return Vue.http.get(endpoint, { params: { format: 'json' } }); return Vue.http.get(endpoint, { params: { format: 'json', viewer: 'none' } });
}, },
getRawFileData(file) { getRawFileData(file) {
if (file.tempFile) { if (file.tempFile) {
......
...@@ -49,31 +49,6 @@ export const setLastCommitMessage = ({ rootState, commit }, data) => { ...@@ -49,31 +49,6 @@ export const setLastCommitMessage = ({ rootState, commit }, data) => {
commit(rootTypes.SET_LAST_COMMIT_MSG, commitMsg, { root: true }); commit(rootTypes.SET_LAST_COMMIT_MSG, commitMsg, { root: true });
}; };
export const checkCommitStatus = ({ rootState }) =>
service
.getBranchData(rootState.currentProjectId, rootState.currentBranchId)
.then(({ data }) => {
const { id } = data.commit;
const selectedBranch =
rootState.projects[rootState.currentProjectId].branches[rootState.currentBranchId];
if (selectedBranch.workingReference !== id) {
return true;
}
return false;
})
.catch(() =>
flash(
__('Error checking branch data. Please try again.'),
'alert',
document,
null,
false,
true,
),
);
export const updateFilesAfterCommit = ({ commit, dispatch, rootState }, { data }) => { export const updateFilesAfterCommit = ({ commit, dispatch, rootState }, { data }) => {
const selectedProject = rootState.projects[rootState.currentProjectId]; const selectedProject = rootState.projects[rootState.currentProjectId];
const lastCommit = { const lastCommit = {
...@@ -128,24 +103,17 @@ export const updateFilesAfterCommit = ({ commit, dispatch, rootState }, { data } ...@@ -128,24 +103,17 @@ export const updateFilesAfterCommit = ({ commit, dispatch, rootState }, { data }
export const commitChanges = ({ commit, state, getters, dispatch, rootState, rootGetters }) => { export const commitChanges = ({ commit, state, getters, dispatch, rootState, rootGetters }) => {
const newBranch = state.commitAction !== consts.COMMIT_TO_CURRENT_BRANCH; const newBranch = state.commitAction !== consts.COMMIT_TO_CURRENT_BRANCH;
const payload = createCommitPayload(getters.branchName, newBranch, state, rootState); const payload = createCommitPayload({
const getCommitStatus = newBranch ? Promise.resolve(false) : dispatch('checkCommitStatus'); branch: getters.branchName,
newBranch,
state,
rootState,
});
commit(types.UPDATE_LOADING, true); commit(types.UPDATE_LOADING, true);
return getCommitStatus return service
.then( .commit(rootState.currentProjectId, payload)
branchChanged =>
new Promise(resolve => {
if (branchChanged) {
// show the modal with a Bootstrap call
$('#ide-create-branch-modal').modal('show');
} else {
resolve();
}
}),
)
.then(() => service.commit(rootState.currentProjectId, payload))
.then(({ data }) => { .then(({ data }) => {
commit(types.UPDATE_LOADING, false); commit(types.UPDATE_LOADING, false);
...@@ -220,12 +188,16 @@ export const commitChanges = ({ commit, state, getters, dispatch, rootState, roo ...@@ -220,12 +188,16 @@ export const commitChanges = ({ commit, state, getters, dispatch, rootState, roo
); );
}) })
.catch(err => { .catch(err => {
let errMsg = __('Error committing changes. Please try again.'); if (err.response.status === 400) {
if (err.response.data && err.response.data.message) { $('#ide-create-branch-modal').modal('show');
errMsg += ` (${stripHtml(err.response.data.message)})`; } else {
let errMsg = __('Error committing changes. Please try again.');
if (err.response.data && err.response.data.message) {
errMsg += ` (${stripHtml(err.response.data.message)})`;
}
flash(errMsg, 'alert', document, null, false, true);
window.dispatchEvent(new Event('resize'));
} }
flash(errMsg, 'alert', document, null, false, true);
window.dispatchEvent(new Event('resize'));
commit(types.UPDATE_LOADING, false); commit(types.UPDATE_LOADING, false);
}); });
......
...@@ -47,6 +47,7 @@ export default { ...@@ -47,6 +47,7 @@ export default {
baseRaw: null, baseRaw: null,
html: data.html, html: data.html,
size: data.size, size: data.size,
lastCommitSha: data.last_commit_sha,
}); });
}, },
[types.SET_FILE_RAW_DATA](state, { file, raw }) { [types.SET_FILE_RAW_DATA](state, { file, raw }) {
......
...@@ -17,6 +17,7 @@ export const dataStructure = () => ({ ...@@ -17,6 +17,7 @@ export const dataStructure = () => ({
changed: false, changed: false,
staged: false, staged: false,
lastCommitPath: '', lastCommitPath: '',
lastCommitSha: '',
lastCommit: { lastCommit: {
id: '', id: '',
url: '', url: '',
...@@ -104,7 +105,7 @@ export const setPageTitle = title => { ...@@ -104,7 +105,7 @@ export const setPageTitle = title => {
document.title = title; document.title = title;
}; };
export const createCommitPayload = (branch, newBranch, state, rootState) => ({ export const createCommitPayload = ({ branch, newBranch, state, rootState }) => ({
branch, branch,
commit_message: state.commitMessage, commit_message: state.commitMessage,
actions: rootState.stagedFiles.map(f => ({ actions: rootState.stagedFiles.map(f => ({
...@@ -112,6 +113,7 @@ export const createCommitPayload = (branch, newBranch, state, rootState) => ({ ...@@ -112,6 +113,7 @@ export const createCommitPayload = (branch, newBranch, state, rootState) => ({
file_path: f.path, file_path: f.path,
content: f.content, content: f.content,
encoding: f.base64 ? 'base64' : 'text', encoding: f.base64 ? 'base64' : 'text',
last_commit_id: newBranch ? undefined : f.lastCommitSha,
})), })),
start_branch: newBranch ? rootState.currentBranchId : undefined, start_branch: newBranch ? rootState.currentBranchId : undefined,
}); });
......
/* eslint-disable new-cap, comma-dangle, no-new */
import $ from 'jquery'; import $ from 'jquery';
import Vue from 'vue'; import Vue from 'vue';
import Flash from '../flash'; import createFlash from '../flash';
import initIssuableSidebar from '../init_issuable_sidebar'; import initIssuableSidebar from '../init_issuable_sidebar';
import './merge_conflict_store'; import './merge_conflict_store';
import MergeConflictsService from './merge_conflict_service'; import MergeConflictsService from './merge_conflict_service';
import './mixins/line_conflict_utils';
import './mixins/line_conflict_actions';
import './components/diff_file_editor'; import './components/diff_file_editor';
import './components/inline_conflict_lines'; import './components/inline_conflict_lines';
import './components/parallel_conflict_lines'; import './components/parallel_conflict_lines';
...@@ -19,7 +15,7 @@ export default function initMergeConflicts() { ...@@ -19,7 +15,7 @@ export default function initMergeConflicts() {
const mergeConflictsStore = gl.mergeConflicts.mergeConflictsStore; const mergeConflictsStore = gl.mergeConflicts.mergeConflictsStore;
const mergeConflictsService = new MergeConflictsService({ const mergeConflictsService = new MergeConflictsService({
conflictsPath: conflictsEl.dataset.conflictsPath, conflictsPath: conflictsEl.dataset.conflictsPath,
resolveConflictsPath: conflictsEl.dataset.resolveConflictsPath resolveConflictsPath: conflictsEl.dataset.resolveConflictsPath,
}); });
initIssuableSidebar(); initIssuableSidebar();
...@@ -29,17 +25,26 @@ export default function initMergeConflicts() { ...@@ -29,17 +25,26 @@ export default function initMergeConflicts() {
components: { components: {
'diff-file-editor': gl.mergeConflicts.diffFileEditor, 'diff-file-editor': gl.mergeConflicts.diffFileEditor,
'inline-conflict-lines': gl.mergeConflicts.inlineConflictLines, 'inline-conflict-lines': gl.mergeConflicts.inlineConflictLines,
'parallel-conflict-lines': gl.mergeConflicts.parallelConflictLines 'parallel-conflict-lines': gl.mergeConflicts.parallelConflictLines,
}, },
data: mergeConflictsStore.state, data: mergeConflictsStore.state,
computed: { computed: {
conflictsCountText() { return mergeConflictsStore.getConflictsCountText(); }, conflictsCountText() {
readyToCommit() { return mergeConflictsStore.isReadyToCommit(); }, return mergeConflictsStore.getConflictsCountText();
commitButtonText() { return mergeConflictsStore.getCommitButtonText(); }, },
showDiffViewTypeSwitcher() { return mergeConflictsStore.fileTextTypePresent(); } readyToCommit() {
return mergeConflictsStore.isReadyToCommit();
},
commitButtonText() {
return mergeConflictsStore.getCommitButtonText();
},
showDiffViewTypeSwitcher() {
return mergeConflictsStore.fileTextTypePresent();
},
}, },
created() { created() {
mergeConflictsService.fetchConflictsData() mergeConflictsService
.fetchConflictsData()
.then(({ data }) => { .then(({ data }) => {
if (data.type === 'error') { if (data.type === 'error') {
mergeConflictsStore.setFailedRequest(data.message); mergeConflictsStore.setFailedRequest(data.message);
...@@ -87,9 +92,9 @@ export default function initMergeConflicts() { ...@@ -87,9 +92,9 @@ export default function initMergeConflicts() {
}) })
.catch(() => { .catch(() => {
mergeConflictsStore.setSubmitState(false); mergeConflictsStore.setSubmitState(false);
new Flash('Failed to save merge conflicts resolutions. Please try again!'); createFlash('Failed to save merge conflicts resolutions. Please try again!');
}); });
} },
} },
}); });
} }
...@@ -24,6 +24,10 @@ module InternalRedirect ...@@ -24,6 +24,10 @@ module InternalRedirect
nil nil
end end
def sanitize_redirect(url_or_path)
safe_redirect_path(url_or_path) || safe_redirect_path_for_url(url_or_path)
end
def host_allowed?(uri) def host_allowed?(uri)
uri.host == request.host && uri.host == request.host &&
uri.port == request.port uri.port == request.port
......
...@@ -197,15 +197,14 @@ class Projects::BlobController < Projects::ApplicationController ...@@ -197,15 +197,14 @@ class Projects::BlobController < Projects::ApplicationController
end end
def show_json def show_json
json = blob_json(@blob) set_last_commit_sha
return render_404 unless json
path_segments = @path.split('/') path_segments = @path.split('/')
path_segments.pop path_segments.pop
tree_path = path_segments.join('/') tree_path = path_segments.join('/')
render json: json.merge( json = {
id: @blob.id, id: @blob.id,
last_commit_sha: @last_commit_sha,
path: blob.path, path: blob.path,
name: blob.name, name: blob.name,
extension: blob.extension, extension: blob.extension,
...@@ -221,6 +220,10 @@ class Projects::BlobController < Projects::ApplicationController ...@@ -221,6 +220,10 @@ class Projects::BlobController < Projects::ApplicationController
commits_path: project_commits_path(project, @id), commits_path: project_commits_path(project, @id),
tree_path: project_tree_path(project, File.join(@ref, tree_path)), tree_path: project_tree_path(project, File.join(@ref, tree_path)),
permalink: project_blob_path(project, File.join(@commit.id, @path)) permalink: project_blob_path(project, File.join(@commit.id, @path))
) }
json.merge!(blob_json(@blob) || {}) unless params[:viewer] == 'none'
render json: json
end end
end end
...@@ -54,6 +54,31 @@ describe InternalRedirect do ...@@ -54,6 +54,31 @@ describe InternalRedirect do
end end
end end
describe '#sanitize_redirect' do
let(:valid_path) { '/hello/world?hello=world' }
let(:valid_url) { "http://test.host#{valid_path}" }
it 'returns `nil` for invalid paths' do
invalid_path = '//not/valid'
expect(controller.sanitize_redirect(invalid_path)).to eq nil
end
it 'returns `nil` for invalid urls' do
input = 'http://test.host:3000/invalid'
expect(controller.sanitize_redirect(input)).to eq nil
end
it 'returns input for valid paths' do
expect(controller.sanitize_redirect(valid_path)).to eq valid_path
end
it 'returns path for valid urls' do
expect(controller.sanitize_redirect(valid_url)).to eq valid_path
end
end
describe '#host_allowed?' do describe '#host_allowed?' do
it 'allows uris with the same host and port' do it 'allows uris with the same host and port' do
expect(controller.host_allowed?(URI('http://test.host/test'))).to be(true) expect(controller.host_allowed?(URI('http://test.host/test'))).to be(true)
......
...@@ -55,6 +55,25 @@ describe Projects::BlobController do ...@@ -55,6 +55,25 @@ describe Projects::BlobController do
expect(json_response).to have_key 'raw_path' expect(json_response).to have_key 'raw_path'
end end
end end
context "with viewer=none" do
let(:id) { 'master/README.md' }
before do
get(:show,
namespace_id: project.namespace,
project_id: project,
id: id,
format: :json,
viewer: 'none')
end
it do
expect(response).to be_ok
expect(json_response).not_to have_key 'html'
expect(json_response).to have_key 'raw_path'
end
end
end end
context 'with tree path' do context 'with tree path' do
......
...@@ -22,7 +22,8 @@ describe 'Project deploy keys', :js do ...@@ -22,7 +22,8 @@ describe 'Project deploy keys', :js do
accept_confirm { find('.ic-remove').click() } accept_confirm { find('.ic-remove').click() }
expect(page).not_to have_selector('.fa-spinner', count: 0) wait_for_requests
expect(page).to have_selector('.deploy-key', count: 0) expect(page).to have_selector('.deploy-key', count: 0)
end end
end end
......
...@@ -108,77 +108,6 @@ describe('IDE commit module actions', () => { ...@@ -108,77 +108,6 @@ describe('IDE commit module actions', () => {
}); });
}); });
describe('checkCommitStatus', () => {
beforeEach(() => {
store.state.currentProjectId = 'abcproject';
store.state.currentBranchId = 'master';
store.state.projects.abcproject = {
branches: {
master: {
workingReference: '1',
},
},
};
});
it('calls service', done => {
spyOn(service, 'getBranchData').and.returnValue(
Promise.resolve({
data: {
commit: { id: '123' },
},
}),
);
store
.dispatch('commit/checkCommitStatus')
.then(() => {
expect(service.getBranchData).toHaveBeenCalledWith('abcproject', 'master');
done();
})
.catch(done.fail);
});
it('returns true if current ref does not equal returned ID', done => {
spyOn(service, 'getBranchData').and.returnValue(
Promise.resolve({
data: {
commit: { id: '123' },
},
}),
);
store
.dispatch('commit/checkCommitStatus')
.then(val => {
expect(val).toBeTruthy();
done();
})
.catch(done.fail);
});
it('returns false if current ref equals returned ID', done => {
spyOn(service, 'getBranchData').and.returnValue(
Promise.resolve({
data: {
commit: { id: '1' },
},
}),
);
store
.dispatch('commit/checkCommitStatus')
.then(val => {
expect(val).toBeFalsy();
done();
})
.catch(done.fail);
});
});
describe('updateFilesAfterCommit', () => { describe('updateFilesAfterCommit', () => {
const data = { const data = {
id: '123', id: '123',
...@@ -314,6 +243,7 @@ describe('IDE commit module actions', () => { ...@@ -314,6 +243,7 @@ describe('IDE commit module actions', () => {
...file('changed'), ...file('changed'),
type: 'blob', type: 'blob',
active: true, active: true,
lastCommitSha: '123456789',
}; };
store.state.stagedFiles.push(f); store.state.stagedFiles.push(f);
store.state.changedFiles = [ store.state.changedFiles = [
...@@ -366,6 +296,7 @@ describe('IDE commit module actions', () => { ...@@ -366,6 +296,7 @@ describe('IDE commit module actions', () => {
file_path: jasmine.anything(), file_path: jasmine.anything(),
content: jasmine.anything(), content: jasmine.anything(),
encoding: jasmine.anything(), encoding: jasmine.anything(),
last_commit_id: undefined,
}, },
], ],
start_branch: 'master', start_branch: 'master',
...@@ -376,6 +307,32 @@ describe('IDE commit module actions', () => { ...@@ -376,6 +307,32 @@ describe('IDE commit module actions', () => {
.catch(done.fail); .catch(done.fail);
}); });
it('sends lastCommit ID when not creating new branch', done => {
store.state.commit.commitAction = '1';
store
.dispatch('commit/commitChanges')
.then(() => {
expect(service.commit).toHaveBeenCalledWith('abcproject', {
branch: jasmine.anything(),
commit_message: 'testing 123',
actions: [
{
action: 'update',
file_path: jasmine.anything(),
content: jasmine.anything(),
encoding: jasmine.anything(),
last_commit_id: '123456789',
},
],
start_branch: undefined,
});
done();
})
.catch(done.fail);
});
it('sets last Commit Msg', done => { it('sets last Commit Msg', done => {
store store
.dispatch('commit/commitChanges') .dispatch('commit/commitChanges')
......
import * as utils from '~/ide/stores/utils'; import * as utils from '~/ide/stores/utils';
import { file } from '../helpers';
describe('Multi-file store utils', () => { describe('Multi-file store utils', () => {
describe('setPageTitle', () => { describe('setPageTitle', () => {
...@@ -63,4 +64,59 @@ describe('Multi-file store utils', () => { ...@@ -63,4 +64,59 @@ describe('Multi-file store utils', () => {
expect(foundEntry).toBeUndefined(); expect(foundEntry).toBeUndefined();
}); });
}); });
describe('createCommitPayload', () => {
it('returns API payload', () => {
const state = {
commitMessage: 'commit message',
};
const rootState = {
stagedFiles: [
{
...file('staged'),
path: 'staged',
content: 'updated file content',
lastCommitSha: '123456789',
},
{
...file('newFile'),
path: 'added',
tempFile: true,
content: 'new file content',
base64: true,
lastCommitSha: '123456789',
},
],
currentBranchId: 'master',
};
const payload = utils.createCommitPayload({
branch: 'master',
newBranch: false,
state,
rootState,
});
expect(payload).toEqual({
branch: 'master',
commit_message: 'commit message',
actions: [
{
action: 'update',
file_path: 'staged',
content: 'updated file content',
encoding: 'text',
last_commit_id: '123456789',
},
{
action: 'create',
file_path: 'added',
content: 'new file content',
encoding: 'base64',
last_commit_id: '123456789',
},
],
start_branch: undefined,
});
});
});
}); });
...@@ -54,14 +54,6 @@ describe Ci::BuildTraceChunk, :clean_gitlab_redis_shared_state do ...@@ -54,14 +54,6 @@ describe Ci::BuildTraceChunk, :clean_gitlab_redis_shared_state do
it { is_expected.to eq('Sample data in db') } it { is_expected.to eq('Sample data in db') }
end end
context 'when data_store is others' do
before do
build_trace_chunk.send(:write_attribute, :data_store, -1)
end
it { expect { subject }.to raise_error('Unsupported data store') }
end
end end
describe '#set_data' do describe '#set_data' do
......
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