Commit a7f879ba authored by Illya Klymov's avatar Illya Klymov Committed by Enrique Alcántara

Ensure git url validation is always performed

* early clicking submit button when importing project
* separate import page inside project

Changelog: fixed
parent 68755b38
import initProjectNew from '~/projects/project_new';
initProjectNew.bindEvents();
import $ from 'jquery'; import $ from 'jquery';
import { debounce } from 'lodash'; import { debounce } from 'lodash';
import DEFAULT_PROJECT_TEMPLATES from 'ee_else_ce/projects/default_project_templates'; import DEFAULT_PROJECT_TEMPLATES from 'ee_else_ce/projects/default_project_templates';
import { DEFAULT_DEBOUNCE_AND_THROTTLE_MS } from '../lib/utils/constants';
import axios from '../lib/utils/axios_utils'; import axios from '../lib/utils/axios_utils';
import { import {
convertToTitleCase, convertToTitleCase,
...@@ -13,20 +14,26 @@ let hasUserDefinedProjectPath = false; ...@@ -13,20 +14,26 @@ let hasUserDefinedProjectPath = false;
let hasUserDefinedProjectName = false; let hasUserDefinedProjectName = false;
const invalidInputClass = 'gl-field-error-outline'; const invalidInputClass = 'gl-field-error-outline';
const cancelSource = axios.CancelToken.source();
const endpoint = `${gon.relative_url_root}/import/url/validate`;
let importCredentialsValidationPromise = null;
const validateImportCredentials = (url, user, password) => { const validateImportCredentials = (url, user, password) => {
const endpoint = `${gon.relative_url_root}/import/url/validate`; cancelSource.cancel();
return axios importCredentialsValidationPromise = axios
.post(endpoint, { .post(endpoint, { url, user, password }, { cancelToken: cancelSource.cancel() })
url,
user,
password,
})
.then(({ data }) => data) .then(({ data }) => data)
.catch(() => ({ .catch((thrown) =>
// intentionally reporting success in case of validation error axios.isCancel(thrown)
// we do not want to block users from trying import in case of validation exception ? {
success: true, cancelled: true,
})); }
: {
// intentionally reporting success in case of validation error
// we do not want to block users from trying import in case of validation exception
success: true,
},
);
return importCredentialsValidationPromise;
}; };
const onProjectNameChange = ($projectNameInput, $projectPathInput) => { const onProjectNameChange = ($projectNameInput, $projectPathInput) => {
...@@ -72,7 +79,7 @@ const deriveProjectPathFromUrl = ($projectImportUrl) => { ...@@ -72,7 +79,7 @@ const deriveProjectPathFromUrl = ($projectImportUrl) => {
.parents('.toggle-import-form') .parents('.toggle-import-form')
.find('#project_path'); .find('#project_path');
if (hasUserDefinedProjectPath) { if (hasUserDefinedProjectPath || $currentProjectPath.length === 0) {
return; return;
} }
...@@ -114,7 +121,7 @@ const bindEvents = () => { ...@@ -114,7 +121,7 @@ const bindEvents = () => {
const $projectImportUrlUser = $('#project_import_url_user'); const $projectImportUrlUser = $('#project_import_url_user');
const $projectImportUrlPassword = $('#project_import_url_password'); const $projectImportUrlPassword = $('#project_import_url_password');
const $projectImportUrlError = $('.js-import-url-error'); const $projectImportUrlError = $('.js-import-url-error');
const $projectImportForm = $('.project-import form'); const $projectImportForm = $('form.js-project-import');
const $projectPath = $('.tab-pane.active #project_path'); const $projectPath = $('.tab-pane.active #project_path');
const $useTemplateBtn = $('.template-button > input'); const $useTemplateBtn = $('.template-button > input');
const $projectFieldsForm = $('.project-fields-form'); const $projectFieldsForm = $('.project-fields-form');
...@@ -124,7 +131,7 @@ const bindEvents = () => { ...@@ -124,7 +131,7 @@ const bindEvents = () => {
const $projectTemplateButtons = $('.project-templates-buttons'); const $projectTemplateButtons = $('.project-templates-buttons');
const $projectName = $('.tab-pane.active #project_name'); const $projectName = $('.tab-pane.active #project_name');
if ($newProjectForm.length !== 1) { if ($newProjectForm.length !== 1 && $projectImportForm.length !== 1) {
return; return;
} }
...@@ -168,20 +175,28 @@ const bindEvents = () => { ...@@ -168,20 +175,28 @@ const bindEvents = () => {
$projectPath.val($projectPath.val().trim()); $projectPath.val($projectPath.val().trim());
}); });
const updateUrlPathWarningVisibility = debounce(async () => { const updateUrlPathWarningVisibility = async () => {
const { success: isUrlValid } = await validateImportCredentials( const { success: isUrlValid, cancelled } = await validateImportCredentials(
$projectImportUrl.val(), $projectImportUrl.val(),
$projectImportUrlUser.val(), $projectImportUrlUser.val(),
$projectImportUrlPassword.val(), $projectImportUrlPassword.val(),
); );
if (cancelled) {
return;
}
$projectImportUrl.toggleClass(invalidInputClass, !isUrlValid); $projectImportUrl.toggleClass(invalidInputClass, !isUrlValid);
$projectImportUrlError.toggleClass('hide', isUrlValid); $projectImportUrlError.toggleClass('hide', isUrlValid);
}, 500); };
const debouncedUpdateUrlPathWarningVisibility = debounce(
updateUrlPathWarningVisibility,
DEFAULT_DEBOUNCE_AND_THROTTLE_MS,
);
let isProjectImportUrlDirty = false; let isProjectImportUrlDirty = false;
$projectImportUrl.on('blur', () => { $projectImportUrl.on('blur', () => {
isProjectImportUrlDirty = true; isProjectImportUrlDirty = true;
updateUrlPathWarningVisibility(); debouncedUpdateUrlPathWarningVisibility();
}); });
$projectImportUrl.on('keyup', () => { $projectImportUrl.on('keyup', () => {
deriveProjectPathFromUrl($projectImportUrl); deriveProjectPathFromUrl($projectImportUrl);
...@@ -190,17 +205,33 @@ const bindEvents = () => { ...@@ -190,17 +205,33 @@ const bindEvents = () => {
[$projectImportUrl, $projectImportUrlUser, $projectImportUrlPassword].forEach(($f) => { [$projectImportUrl, $projectImportUrlUser, $projectImportUrlPassword].forEach(($f) => {
$f.on('input', () => { $f.on('input', () => {
if (isProjectImportUrlDirty) { if (isProjectImportUrlDirty) {
updateUrlPathWarningVisibility(); debouncedUpdateUrlPathWarningVisibility();
} }
}); });
}); });
$projectImportForm.on('submit', (e) => { $projectImportForm.on('submit', async (e) => {
e.preventDefault();
if (importCredentialsValidationPromise === null) {
// we didn't validate credentials yet
debouncedUpdateUrlPathWarningVisibility.cancel();
updateUrlPathWarningVisibility();
}
const submitBtn = $projectImportForm.find('input[type="submit"]');
submitBtn.disable();
await importCredentialsValidationPromise;
submitBtn.enable();
const $invalidFields = $projectImportForm.find(`.${invalidInputClass}`); const $invalidFields = $projectImportForm.find(`.${invalidInputClass}`);
if ($invalidFields.length > 0) { if ($invalidFields.length > 0) {
$invalidFields[0].focus(); $invalidFields[0].focus();
e.preventDefault(); } else {
e.stopPropagation(); // calling .submit() on HTMLFormElement does not trigger 'submit' event
// We are using this behavior to bypass this handler and avoid infinite loop
$projectImportForm[0].submit();
} }
}); });
......
...@@ -82,7 +82,7 @@ ...@@ -82,7 +82,7 @@
.js-toggle-content.toggle-import-form{ class: ('hide' if active_tab != 'import') } .js-toggle-content.toggle-import-form{ class: ('hide' if active_tab != 'import') }
= form_for @project, html: { class: 'new_project gl-show-field-errors' } do |f| = form_for @project, html: { class: 'new_project gl-show-field-errors js-project-import' } do |f|
%hr %hr
= render "shared/import_form", f: f = render "shared/import_form", f: f
= render 'projects/new_project_fields', f: f, project_name_id: "import-url-name", hide_init_with_readme: true, track_label: track_label = render 'projects/new_project_fields', f: f, project_name_id: "import-url-name", hide_init_with_readme: true, track_label: track_label
...@@ -12,8 +12,8 @@ ...@@ -12,8 +12,8 @@
:preserve :preserve
#{h(@project.import_state.last_error)} #{h(@project.import_state.last_error)}
= form_for @project, url: project_import_path(@project), method: :post do |f| = form_for @project, url: project_import_path(@project), method: :post, html: { class: 'js-project-import' } do |f|
= render "shared/import_form", f: f = render "shared/import_form", f: f
.form-actions .form-actions
= f.submit 'Start import', class: "gl-button btn btn-confirm" = f.submit 'Start import', class: "gl-button btn btn-confirm", data: { disable_with: false }
...@@ -68,7 +68,7 @@ RSpec.describe 'New project', :js do ...@@ -68,7 +68,7 @@ RSpec.describe 'New project', :js do
end end
it '"Import project" tab creates projects with features enabled' do it '"Import project" tab creates projects with features enabled' do
stub_request(:get, "http://foo.git/info/refs?service=git-upload-pack").to_return(status: 200, body: "001e# servdice=git-upload-pack") stub_request(:get, "http://foo.git/info/refs?service=git-upload-pack").to_return(status: 200, body: "001e# service=git-upload-pack", headers: { 'Content-Type': 'application/x-git-upload-pack-advertisement' })
visit new_project_path visit new_project_path
click_link 'Import project' click_link 'Import project'
...@@ -84,6 +84,7 @@ RSpec.describe 'New project', :js do ...@@ -84,6 +84,7 @@ RSpec.describe 'New project', :js do
fill_in 'project_path', with: 'import-project-with-features1' fill_in 'project_path', with: 'import-project-with-features1'
choose 'project_visibility_level_20' choose 'project_visibility_level_20'
click_button 'Create project' click_button 'Create project'
wait_for_requests
created_project = Project.last created_project = Project.last
......
...@@ -306,10 +306,24 @@ RSpec.describe 'New project', :js do ...@@ -306,10 +306,24 @@ RSpec.describe 'New project', :js do
expect(page).to have_text('There is not a valid Git repository at this URL') expect(page).to have_text('There is not a valid Git repository at this URL')
end end
it 'reports error if repo URL is not a valid Git repository and submit button is clicked immediately' do
stub_request(:get, "http://foo/bar/info/refs?service=git-upload-pack").to_return(status: 200, body: "not-a-git-repo")
fill_in 'project_import_url', with: 'http://foo/bar'
click_on 'Create project'
wait_for_requests
expect(page).to have_text('There is not a valid Git repository at this URL')
end
it 'keeps "Import project" tab open after form validation error' do it 'keeps "Import project" tab open after form validation error' do
collision_project = create(:project, name: 'test-name-collision', namespace: user.namespace) collision_project = create(:project, name: 'test-name-collision', namespace: user.namespace)
stub_request(:get, "http://foo/bar/info/refs?service=git-upload-pack").to_return({ status: 200,
body: '001e# service=git-upload-pack',
headers: { 'Content-Type': 'application/x-git-upload-pack-advertisement' } })
fill_in 'project_import_url', with: collision_project.http_url_to_repo fill_in 'project_import_url', with: 'http://foo/bar'
fill_in 'project_name', with: collision_project.name fill_in 'project_name', with: collision_project.name
click_on 'Create project' click_on 'Create project'
...@@ -319,6 +333,38 @@ RSpec.describe 'New project', :js do ...@@ -319,6 +333,38 @@ RSpec.describe 'New project', :js do
end end
end end
context 'when import is initiated from project page' do
before do
project_without_repo = create(:project, name: 'project-without-repo', namespace: user.namespace)
visit project_path(project_without_repo)
click_on 'Import repository'
end
it 'reports error when invalid url is provided' do
stub_request(:get, "http://foo/bar/info/refs?service=git-upload-pack").to_return(status: 200, body: "not-a-git-repo")
fill_in 'project_import_url', with: 'http://foo/bar'
click_on 'Start import'
wait_for_requests
expect(page).to have_text('There is not a valid Git repository at this URL')
end
it 'initiates import when valid repo url is provided' do
stub_request(:get, "http://foo/bar/info/refs?service=git-upload-pack").to_return({ status: 200,
body: '001e# service=git-upload-pack',
headers: { 'Content-Type': 'application/x-git-upload-pack-advertisement' } })
fill_in 'project_import_url', with: 'http://foo/bar'
click_on 'Start import'
wait_for_requests
expect(page).to have_text('Import in progress')
end
end
context 'from GitHub' do context 'from GitHub' do
before do before do
first('.js-import-github').click first('.js-import-github').click
......
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