Commit b7f8ba97 authored by Paul Slaughter's avatar Paul Slaughter

Fix ide action to handle empty repo with tree

**How?**
Previously the openBranch action (which is called
 on a route change) would recreate the tree
 whenever there was an empty repo. Now we
 only do this if the current file tree does not exist.
parent 8a6b537d
...@@ -83,10 +83,14 @@ export const showBranchNotFoundError = ({ dispatch }, branchId) => { ...@@ -83,10 +83,14 @@ export const showBranchNotFoundError = ({ dispatch }, branchId) => {
}); });
}; };
export const showEmptyState = ({ commit, state, dispatch }, { projectId, branchId }) => { export const loadEmptyBranch = ({ commit, state }, { projectId, branchId }) => {
const treePath = `${projectId}/${branchId}`; const treePath = `${projectId}/${branchId}`;
const currentTree = state.trees[`${projectId}/${branchId}`];
dispatch('setCurrentBranchId', branchId); // If we already have a tree, let's not recreate an empty one
if (currentTree) {
return;
}
commit(types.CREATE_TREE, { treePath }); commit(types.CREATE_TREE, { treePath });
commit(types.TOGGLE_LOADING, { commit(types.TOGGLE_LOADING, {
...@@ -114,8 +118,16 @@ export const loadFile = ({ dispatch, state }, { basePath }) => { ...@@ -114,8 +118,16 @@ export const loadFile = ({ dispatch, state }, { basePath }) => {
} }
}; };
export const loadBranch = ({ dispatch, getters }, { projectId, branchId }) => export const loadBranch = ({ dispatch, getters, state }, { projectId, branchId }) => {
dispatch('getBranchData', { const currentProject = state.projects[projectId];
if (currentProject?.branches?.[branchId]) {
return Promise.resolve();
} else if (getters.emptyRepo) {
return dispatch('loadEmptyBranch', { projectId, branchId });
}
return dispatch('getBranchData', {
projectId, projectId,
branchId, branchId,
}) })
...@@ -137,29 +149,23 @@ export const loadBranch = ({ dispatch, getters }, { projectId, branchId }) => ...@@ -137,29 +149,23 @@ export const loadBranch = ({ dispatch, getters }, { projectId, branchId }) =>
dispatch('showBranchNotFoundError', branchId); dispatch('showBranchNotFoundError', branchId);
throw err; throw err;
}); });
};
export const openBranch = ({ dispatch, state, getters }, { projectId, branchId, basePath }) => { export const openBranch = ({ dispatch }, { projectId, branchId, basePath }) => {
const currentProject = state.projects[projectId]; dispatch('setCurrentBranchId', branchId);
if (getters.emptyRepo) {
return dispatch('showEmptyState', { projectId, branchId }); return dispatch('loadBranch', { projectId, branchId })
} .then(() => dispatch('loadFile', { basePath }))
if (!currentProject || !currentProject.branches[branchId]) { .catch(
dispatch('setCurrentBranchId', branchId); () =>
new Error(
return dispatch('loadBranch', { projectId, branchId }) sprintf(
.then(() => dispatch('loadFile', { basePath })) __('An error occurred while getting files for - %{branchId}'),
.catch( {
() => branchId: `<strong>${esc(projectId)}/${esc(branchId)}</strong>`,
new Error( },
sprintf( false,
__('An error occurred while getting files for - %{branchId}'),
{
branchId: `<strong>${esc(projectId)}/${esc(branchId)}</strong>`,
},
false,
),
), ),
); ),
} );
return Promise.resolve(dispatch('loadFile', { basePath }));
}; };
---
title: Fix some Web IDE bugs with empty projects
merge_request: 25463
author:
type: fixed
...@@ -4,7 +4,7 @@ import { ...@@ -4,7 +4,7 @@ import {
refreshLastCommitData, refreshLastCommitData,
showBranchNotFoundError, showBranchNotFoundError,
createNewBranchFromDefault, createNewBranchFromDefault,
showEmptyState, loadEmptyBranch,
openBranch, openBranch,
loadFile, loadFile,
loadBranch, loadBranch,
...@@ -16,6 +16,8 @@ import router from '~/ide/ide_router'; ...@@ -16,6 +16,8 @@ import router from '~/ide/ide_router';
import { resetStore } from '../../helpers'; import { resetStore } from '../../helpers';
import testAction from '../../../helpers/vuex_action_helper'; import testAction from '../../../helpers/vuex_action_helper';
const TEST_PROJECT_ID = 'abc/def';
describe('IDE store project actions', () => { describe('IDE store project actions', () => {
let mock; let mock;
let store; let store;
...@@ -24,7 +26,7 @@ describe('IDE store project actions', () => { ...@@ -24,7 +26,7 @@ describe('IDE store project actions', () => {
store = createStore(); store = createStore();
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
store.state.projects['abc/def'] = { store.state.projects[TEST_PROJECT_ID] = {
branches: {}, branches: {},
}; };
}); });
...@@ -83,7 +85,7 @@ describe('IDE store project actions', () => { ...@@ -83,7 +85,7 @@ describe('IDE store project actions', () => {
{ {
type: 'SET_BRANCH_COMMIT', type: 'SET_BRANCH_COMMIT',
payload: { payload: {
projectId: 'abc/def', projectId: TEST_PROJECT_ID,
branchId: 'master', branchId: 'master',
commit: { id: '123' }, commit: { id: '123' },
}, },
...@@ -200,17 +202,17 @@ describe('IDE store project actions', () => { ...@@ -200,17 +202,17 @@ describe('IDE store project actions', () => {
}); });
}); });
describe('showEmptyState', () => { describe('loadEmptyBranch', () => {
it('creates a blank tree and sets loading state to false', done => { it('creates a blank tree and sets loading state to false', done => {
testAction( testAction(
showEmptyState, loadEmptyBranch,
{ projectId: 'abc/def', branchId: 'master' }, { projectId: TEST_PROJECT_ID, branchId: 'master' },
store.state, store.state,
[ [
{ type: 'CREATE_TREE', payload: { treePath: 'abc/def/master' } }, { type: 'CREATE_TREE', payload: { treePath: `${TEST_PROJECT_ID}/master` } },
{ {
type: 'TOGGLE_LOADING', type: 'TOGGLE_LOADING',
payload: { entry: store.state.trees['abc/def/master'], forceValue: false }, payload: { entry: store.state.trees[`${TEST_PROJECT_ID}/master`], forceValue: false },
}, },
], ],
jasmine.any(Object), jasmine.any(Object),
...@@ -218,13 +220,15 @@ describe('IDE store project actions', () => { ...@@ -218,13 +220,15 @@ describe('IDE store project actions', () => {
); );
}); });
it('sets the currentBranchId to the branchId that was passed', done => { it('does nothing, if tree already exists', done => {
const trees = { [`${TEST_PROJECT_ID}/master`]: [] };
testAction( testAction(
showEmptyState, loadEmptyBranch,
{ projectId: 'abc/def', branchId: 'master' }, { projectId: TEST_PROJECT_ID, branchId: 'master' },
store.state, { trees },
jasmine.any(Object), [],
[{ type: 'setCurrentBranchId', payload: 'master' }], [],
done, done,
); );
}); });
...@@ -278,10 +282,29 @@ describe('IDE store project actions', () => { ...@@ -278,10 +282,29 @@ describe('IDE store project actions', () => {
}); });
describe('loadBranch', () => { describe('loadBranch', () => {
const projectId = 'abc/def'; const projectId = TEST_PROJECT_ID;
const branchId = '123-lorem'; const branchId = '123-lorem';
const ref = 'abcd2322'; const ref = 'abcd2322';
it('when empty repo, loads empty branch', done => {
const mockGetters = { emptyRepo: true };
testAction(
loadBranch,
{ projectId, branchId },
{ ...store.state, ...mockGetters },
[],
[{ type: 'loadEmptyBranch', payload: { projectId, branchId } }],
done,
);
});
it('when branch already exists, does nothing', done => {
store.state.projects[projectId].branches[branchId] = {};
testAction(loadBranch, { projectId, branchId }, store.state, [], [], done);
});
it('fetches branch data', done => { it('fetches branch data', done => {
const mockGetters = { findBranch: () => ({ commit: { id: ref } }) }; const mockGetters = { findBranch: () => ({ commit: { id: ref } }) };
spyOn(store, 'dispatch').and.returnValue(Promise.resolve()); spyOn(store, 'dispatch').and.returnValue(Promise.resolve());
...@@ -317,7 +340,7 @@ describe('IDE store project actions', () => { ...@@ -317,7 +340,7 @@ describe('IDE store project actions', () => {
}); });
describe('openBranch', () => { describe('openBranch', () => {
const projectId = 'abc/def'; const projectId = TEST_PROJECT_ID;
const branchId = '123-lorem'; const branchId = '123-lorem';
const branch = { const branch = {
...@@ -335,55 +358,6 @@ describe('IDE store project actions', () => { ...@@ -335,55 +358,6 @@ describe('IDE store project actions', () => {
}); });
}); });
it('loads file right away if the branch has already been fetched', done => {
spyOn(store, 'dispatch');
Object.assign(store.state, {
projects: {
[projectId]: {
branches: {
[branchId]: { foo: 'bar' },
},
},
},
});
openBranch(store, branch)
.then(() => {
expect(store.dispatch.calls.allArgs()).toEqual([['loadFile', { basePath: undefined }]]);
})
.then(done)
.catch(done.fail);
});
describe('empty repo', () => {
beforeEach(() => {
spyOn(store, 'dispatch').and.returnValue(Promise.resolve());
Object.assign(store.state, {
currentProjectId: 'abc/def',
projects: {
'abc/def': {
empty_repo: true,
},
},
});
});
afterEach(() => {
resetStore(store);
});
it('dispatches showEmptyState action right away', done => {
openBranch(store, branch)
.then(() => {
expect(store.dispatch.calls.allArgs()).toEqual([['showEmptyState', branch]]);
done();
})
.catch(done.fail);
});
});
describe('existing branch', () => { describe('existing branch', () => {
beforeEach(() => { beforeEach(() => {
spyOn(store, 'dispatch').and.returnValue(Promise.resolve()); spyOn(store, 'dispatch').and.returnValue(Promise.resolve());
...@@ -410,11 +384,17 @@ describe('IDE store project actions', () => { ...@@ -410,11 +384,17 @@ describe('IDE store project actions', () => {
it('dispatches correct branch actions', done => { it('dispatches correct branch actions', done => {
openBranch(store, branch) openBranch(store, branch)
.then(() => { .then(val => {
expect(store.dispatch.calls.allArgs()).toEqual([ expect(store.dispatch.calls.allArgs()).toEqual([
['setCurrentBranchId', branchId], ['setCurrentBranchId', branchId],
['loadBranch', { projectId, branchId }], ['loadBranch', { projectId, branchId }],
]); ]);
expect(val).toEqual(
new Error(
`An error occurred while getting files for - <strong>${projectId}/${branchId}</strong>`,
),
);
}) })
.then(done) .then(done)
.catch(done.fail); .catch(done.fail);
......
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