Commit 7b3b0024 authored by Natalia Tepluhina's avatar Natalia Tepluhina

Merge branch 'afontaine/new-environments-polling' into 'master'

Move Environments App Query to Use Polling

See merge request gitlab-org/gitlab!75259
parents c4a14b1e fafaf43f
<script> <script>
import { GlBadge, GlTab, GlTabs } from '@gitlab/ui'; import { GlBadge, GlTab, GlTabs } from '@gitlab/ui';
import environmentAppQuery from '../graphql/queries/environmentApp.query.graphql'; import environmentAppQuery from '../graphql/queries/environment_app.query.graphql';
import pollIntervalQuery from '../graphql/queries/poll_interval.query.graphql';
import EnvironmentFolder from './new_environment_folder.vue'; import EnvironmentFolder from './new_environment_folder.vue';
export default { export default {
...@@ -13,8 +14,17 @@ export default { ...@@ -13,8 +14,17 @@ export default {
apollo: { apollo: {
environmentApp: { environmentApp: {
query: environmentAppQuery, query: environmentAppQuery,
pollInterval() {
return this.interval;
}, },
}, },
interval: {
query: pollIntervalQuery,
},
},
data() {
return { interval: undefined };
},
computed: { computed: {
folders() { folders() {
return this.environmentApp?.environments.filter((e) => e.size > 1) ?? []; return this.environmentApp?.environments.filter((e) => e.size > 1) ?? [];
......
import VueApollo from 'vue-apollo'; import VueApollo from 'vue-apollo';
import createDefaultClient from '~/lib/graphql'; import createDefaultClient from '~/lib/graphql';
import environmentApp from './queries/environmentApp.query.graphql'; import environmentApp from './queries/environment_app.query.graphql';
import { resolvers } from './resolvers'; import { resolvers } from './resolvers';
import typeDefs from './typedefs.graphql'; import typeDefs from './typedefs.graphql';
......
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import { s__ } from '~/locale';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils'; import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import pollIntervalQuery from './queries/poll_interval.query.graphql';
const buildErrors = (errors = []) => ({
errors,
__typename: 'LocalEnvironmentErrors',
});
const mapNestedEnvironment = (env) => ({ const mapNestedEnvironment = (env) => ({
...convertObjectPropsToCamelCase(env, { deep: true }), ...convertObjectPropsToCamelCase(env, { deep: true }),
...@@ -12,8 +19,17 @@ const mapEnvironment = (env) => ({ ...@@ -12,8 +19,17 @@ const mapEnvironment = (env) => ({
export const resolvers = (endpoint) => ({ export const resolvers = (endpoint) => ({
Query: { Query: {
environmentApp() { environmentApp(_context, _variables, { cache }) {
return axios.get(endpoint, { params: { nested: true } }).then((res) => ({ return axios.get(endpoint, { params: { nested: true } }).then((res) => {
const interval = res.headers['poll-interval'];
if (interval) {
cache.writeQuery({ query: pollIntervalQuery, data: { interval } });
} else {
cache.writeQuery({ query: pollIntervalQuery, data: { interval: undefined } });
}
return {
availableCount: res.data.available_count, availableCount: res.data.available_count,
environments: res.data.environments.map(mapNestedEnvironment), environments: res.data.environments.map(mapNestedEnvironment),
reviewApp: { reviewApp: {
...@@ -22,7 +38,8 @@ export const resolvers = (endpoint) => ({ ...@@ -22,7 +38,8 @@ export const resolvers = (endpoint) => ({
}, },
stoppedCount: res.data.stopped_count, stoppedCount: res.data.stopped_count,
__typename: 'LocalEnvironmentApp', __typename: 'LocalEnvironmentApp',
})); };
});
}, },
folder(_, { environment: { folderPath } }) { folder(_, { environment: { folderPath } }) {
return axios.get(folderPath, { params: { per_page: 3 } }).then((res) => ({ return axios.get(folderPath, { params: { per_page: 3 } }).then((res) => ({
...@@ -32,19 +49,51 @@ export const resolvers = (endpoint) => ({ ...@@ -32,19 +49,51 @@ export const resolvers = (endpoint) => ({
__typename: 'LocalEnvironmentFolder', __typename: 'LocalEnvironmentFolder',
})); }));
}, },
isLastDeployment(_, { environment }) {
// eslint-disable-next-line @gitlab/require-i18n-strings
return environment?.lastDeployment?.['last?'];
},
}, },
Mutations: { Mutation: {
stopEnvironment(_, { environment: { stopPath } }) { stopEnvironment(_, { environment }) {
return axios.post(stopPath); return axios
.post(environment.stopPath)
.then(() => buildErrors())
.catch(() => {
return buildErrors([
s__('Environments|An error occurred while stopping the environment, please try again'),
]);
});
}, },
deleteEnvironment(_, { environment: { deletePath } }) { deleteEnvironment(_, { environment: { deletePath } }) {
return axios.delete(deletePath); return axios.delete(deletePath);
}, },
rollbackEnvironment(_, { environment: { retryUrl } }) { rollbackEnvironment(_, { environment, isLastDeployment }) {
return axios.post(retryUrl); return axios
.post(environment?.retryUrl)
.then(() => buildErrors())
.catch(() => {
buildErrors([
isLastDeployment
? s__(
'Environments|An error occurred while re-deploying the environment, please try again',
)
: s__(
'Environments|An error occurred while rolling back the environment, please try again',
),
]);
});
}, },
cancelAutoStop(_, { environment: { autoStopPath } }) { cancelAutoStop(_, { environment: { autoStopPath } }) {
return axios.post(autoStopPath); return axios
.post(autoStopPath)
.then(() => buildErrors())
.catch((err) =>
buildErrors([
err?.response?.data?.message ||
s__('Environments|An error occurred while canceling the auto stop, please try again'),
]),
);
}, },
}, },
}); });
...@@ -33,3 +33,20 @@ type LocalEnvironmentApp { ...@@ -33,3 +33,20 @@ type LocalEnvironmentApp {
environments: [NestedLocalEnvironment!]! environments: [NestedLocalEnvironment!]!
reviewApp: ReviewApp! reviewApp: ReviewApp!
} }
type LocalErrors {
errors: [String!]!
}
extend type Query {
environmentApp: LocalEnvironmentApp
folder(environment: NestedLocalEnvironment): LocalEnvironmentFolder
isLastDeployment: Boolean
}
extend type Mutation {
stopEnvironment(environment: LocalEnvironment): LocalErrors
deleteEnvironment(environment: LocalEnvironment): LocalErrors
rollbackEnvironment(environment: LocalEnvironment): LocalErrors
cancelAutoStop(environment: LocalEnvironment): LocalErrors
}
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import { resolvers } from '~/environments/graphql/resolvers'; import { resolvers } from '~/environments/graphql/resolvers';
import pollIntervalQuery from '~/environments/graphql/queries/poll_interval.query.graphql';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import { environmentsApp, resolvedEnvironmentsApp, folder, resolvedFolder } from './mock_data'; import { environmentsApp, resolvedEnvironmentsApp, folder, resolvedFolder } from './mock_data';
...@@ -21,10 +22,27 @@ describe('~/frontend/environments/graphql/resolvers', () => { ...@@ -21,10 +22,27 @@ describe('~/frontend/environments/graphql/resolvers', () => {
describe('environmentApp', () => { describe('environmentApp', () => {
it('should fetch environments and map them to frontend data', async () => { it('should fetch environments and map them to frontend data', async () => {
mock.onGet(ENDPOINT, { params: { nested: true } }).reply(200, environmentsApp); const cache = { writeQuery: jest.fn() };
mock.onGet(ENDPOINT, { params: { nested: true } }).reply(200, environmentsApp, {});
const app = await mockResolvers.Query.environmentApp(); const app = await mockResolvers.Query.environmentApp(null, null, { cache });
expect(app).toEqual(resolvedEnvironmentsApp); expect(app).toEqual(resolvedEnvironmentsApp);
expect(cache.writeQuery).toHaveBeenCalledWith({
query: pollIntervalQuery,
data: { interval: undefined },
});
});
it('should set the poll interval when there is one', async () => {
const cache = { writeQuery: jest.fn() };
mock
.onGet(ENDPOINT, { params: { nested: true } })
.reply(200, environmentsApp, { 'poll-interval': 3000 });
await mockResolvers.Query.environmentApp(null, null, { cache });
expect(cache.writeQuery).toHaveBeenCalledWith({
query: pollIntervalQuery,
data: { interval: 3000 },
});
}); });
}); });
describe('folder', () => { describe('folder', () => {
...@@ -42,7 +60,7 @@ describe('~/frontend/environments/graphql/resolvers', () => { ...@@ -42,7 +60,7 @@ describe('~/frontend/environments/graphql/resolvers', () => {
it('should post to the stop environment path', async () => { it('should post to the stop environment path', async () => {
mock.onPost(ENDPOINT).reply(200); mock.onPost(ENDPOINT).reply(200);
await mockResolvers.Mutations.stopEnvironment(null, { environment: { stopPath: ENDPOINT } }); await mockResolvers.Mutation.stopEnvironment(null, { environment: { stopPath: ENDPOINT } });
expect(mock.history.post).toContainEqual( expect(mock.history.post).toContainEqual(
expect.objectContaining({ url: ENDPOINT, method: 'post' }), expect.objectContaining({ url: ENDPOINT, method: 'post' }),
...@@ -53,7 +71,7 @@ describe('~/frontend/environments/graphql/resolvers', () => { ...@@ -53,7 +71,7 @@ describe('~/frontend/environments/graphql/resolvers', () => {
it('should post to the retry environment path', async () => { it('should post to the retry environment path', async () => {
mock.onPost(ENDPOINT).reply(200); mock.onPost(ENDPOINT).reply(200);
await mockResolvers.Mutations.rollbackEnvironment(null, { await mockResolvers.Mutation.rollbackEnvironment(null, {
environment: { retryUrl: ENDPOINT }, environment: { retryUrl: ENDPOINT },
}); });
...@@ -66,7 +84,7 @@ describe('~/frontend/environments/graphql/resolvers', () => { ...@@ -66,7 +84,7 @@ describe('~/frontend/environments/graphql/resolvers', () => {
it('should DELETE to the delete environment path', async () => { it('should DELETE to the delete environment path', async () => {
mock.onDelete(ENDPOINT).reply(200); mock.onDelete(ENDPOINT).reply(200);
await mockResolvers.Mutations.deleteEnvironment(null, { await mockResolvers.Mutation.deleteEnvironment(null, {
environment: { deletePath: ENDPOINT }, environment: { deletePath: ENDPOINT },
}); });
...@@ -79,7 +97,7 @@ describe('~/frontend/environments/graphql/resolvers', () => { ...@@ -79,7 +97,7 @@ describe('~/frontend/environments/graphql/resolvers', () => {
it('should post to the auto stop path', async () => { it('should post to the auto stop path', async () => {
mock.onPost(ENDPOINT).reply(200); mock.onPost(ENDPOINT).reply(200);
await mockResolvers.Mutations.cancelAutoStop(null, { await mockResolvers.Mutation.cancelAutoStop(null, {
environment: { autoStopPath: ENDPOINT }, environment: { autoStopPath: ENDPOINT },
}); });
......
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