/** * Common code between environmets app and folder view */ import _ from 'underscore'; import Visibility from 'visibilityjs'; import EnvironmentsStore from 'ee_else_ce/environments/stores/environments_store'; import Poll from '../../lib/utils/poll'; import { getParameterByName } from '../../lib/utils/common_utils'; import { s__ } from '../../locale'; import Flash from '../../flash'; import eventHub from '../event_hub'; import EnvironmentsService from '../services/environments_service'; import tablePagination from '../../vue_shared/components/pagination/table_pagination.vue'; import environmentTable from '../components/environments_table.vue'; import tabs from '../../vue_shared/components/navigation_tabs.vue'; import container from '../components/container.vue'; export default { components: { environmentTable, container, tabs, tablePagination, }, data() { const store = new EnvironmentsStore(); return { store, state: store.state, isLoading: false, isMakingRequest: false, scope: getParameterByName('scope') || 'available', page: getParameterByName('page') || '1', requestData: {}, environmentInStopModal: {}, environmentInRollbackModal: {}, }; }, methods: { saveData(resp) { this.isLoading = false; // Prevent the absence of the nested flag from causing mismatches const response = this.filterNilValues(resp.config.params); const request = this.filterNilValues(this.requestData); if (_.isEqual(response, request)) { this.store.storeAvailableCount(resp.data.available_count); this.store.storeStoppedCount(resp.data.stopped_count); this.store.storeEnvironments(resp.data.environments); this.store.setReviewAppDetails(resp.data.review_app); this.store.setPagination(resp.headers); } }, filterNilValues(obj) { return _.omit(obj, value => _.isUndefined(value) || _.isNull(value)); }, /** * Handles URL and query parameter changes. * When the user uses the pagination or the tabs, * - update URL * - Make API request to the server with new parameters * - Update the polling function * - Update the internal state */ updateContent(parameters) { this.updateInternalState(parameters); // fetch new data return this.service .fetchEnvironments(this.requestData) .then(response => { this.successCallback(response); this.poll.enable({ data: this.requestData, response }); }) .catch(() => { this.errorCallback(); // restart polling this.poll.restart(); }); }, errorCallback() { this.isLoading = false; Flash(s__('Environments|An error occurred while fetching the environments.')); }, postAction({ endpoint, errorMessage = s__('Environments|An error occurred while making the request.'), }) { if (!this.isMakingRequest) { this.isLoading = true; this.service .postAction(endpoint) .then(() => this.fetchEnvironments()) .catch(err => { this.isLoading = false; Flash(_.isFunction(errorMessage) ? errorMessage(err.response.data) : errorMessage); }); } }, fetchEnvironments() { this.isLoading = true; return this.service .fetchEnvironments(this.requestData) .then(this.successCallback) .catch(this.errorCallback); }, updateStopModal(environment) { this.environmentInStopModal = environment; }, updateRollbackModal(environment) { this.environmentInRollbackModal = environment; }, stopEnvironment(environment) { const endpoint = environment.stop_path; const errorMessage = s__( 'Environments|An error occurred while stopping the environment, please try again', ); this.postAction({ endpoint, errorMessage }); }, rollbackEnvironment(environment) { const { retryUrl, isLastDeployment } = environment; const errorMessage = 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', ); this.postAction({ endpoint: retryUrl, errorMessage }); }, cancelAutoStop(autoStopPath) { const errorMessage = ({ message }) => message || s__('Environments|An error occurred while canceling the auto stop, please try again'); this.postAction({ endpoint: autoStopPath, errorMessage }); }, }, computed: { tabs() { return [ { name: s__('Available'), scope: 'available', count: this.state.availableCounter, isActive: this.scope === 'available', }, { name: s__('Stopped'), scope: 'stopped', count: this.state.stoppedCounter, isActive: this.scope === 'stopped', }, ]; }, }, /** * Fetches all the environments and stores them. * Toggles loading property. */ created() { this.service = new EnvironmentsService(this.endpoint); this.requestData = { page: this.page, scope: this.scope, nested: true }; this.poll = new Poll({ resource: this.service, method: 'fetchEnvironments', data: this.requestData, successCallback: this.successCallback, errorCallback: this.errorCallback, notificationCallback: isMakingRequest => { this.isMakingRequest = isMakingRequest; }, }); if (!Visibility.hidden()) { this.isLoading = true; this.poll.makeRequest(); } else { this.fetchEnvironments(); } Visibility.change(() => { if (!Visibility.hidden()) { this.poll.restart(); } else { this.poll.stop(); } }); eventHub.$on('postAction', this.postAction); eventHub.$on('requestStopEnvironment', this.updateStopModal); eventHub.$on('stopEnvironment', this.stopEnvironment); eventHub.$on('requestRollbackEnvironment', this.updateRollbackModal); eventHub.$on('rollbackEnvironment', this.rollbackEnvironment); eventHub.$on('cancelAutoStop', this.cancelAutoStop); }, beforeDestroy() { eventHub.$off('postAction', this.postAction); eventHub.$off('requestStopEnvironment', this.updateStopModal); eventHub.$off('stopEnvironment', this.stopEnvironment); eventHub.$off('requestRollbackEnvironment', this.updateRollbackModal); eventHub.$off('rollbackEnvironment', this.rollbackEnvironment); eventHub.$off('cancelAutoStop', this.cancelAutoStop); }, };