267 lines
7.3 KiB
Vue
267 lines
7.3 KiB
Vue
<script>
|
|
import { GlAlert, GlButton, GlLoadingIcon, GlModal, GlModalDirective } from '@gitlab/ui';
|
|
import { __ } from '~/locale';
|
|
import ciHeader from '~/vue_shared/components/header_ci_component.vue';
|
|
import { setUrlFragment, redirectTo } from '~/lib/utils/url_utility';
|
|
import getPipelineQuery from '../graphql/queries/get_pipeline_header_data.query.graphql';
|
|
import deletePipelineMutation from '../graphql/mutations/delete_pipeline.mutation.graphql';
|
|
import retryPipelineMutation from '../graphql/mutations/retry_pipeline.mutation.graphql';
|
|
import cancelPipelineMutation from '../graphql/mutations/cancel_pipeline.mutation.graphql';
|
|
import { LOAD_FAILURE, POST_FAILURE, DELETE_FAILURE, DEFAULT } from '../constants';
|
|
|
|
const DELETE_MODAL_ID = 'pipeline-delete-modal';
|
|
const POLL_INTERVAL = 10000;
|
|
|
|
export default {
|
|
name: 'PipelineHeaderSection',
|
|
pipelineCancel: 'pipelineCancel',
|
|
pipelineRetry: 'pipelineRetry',
|
|
finishedStatuses: ['FAILED', 'SUCCESS', 'CANCELED'],
|
|
components: {
|
|
ciHeader,
|
|
GlAlert,
|
|
GlButton,
|
|
GlLoadingIcon,
|
|
GlModal,
|
|
},
|
|
directives: {
|
|
GlModal: GlModalDirective,
|
|
},
|
|
errorTexts: {
|
|
[LOAD_FAILURE]: __('We are currently unable to fetch data for the pipeline header.'),
|
|
[POST_FAILURE]: __('An error occurred while making the request.'),
|
|
[DELETE_FAILURE]: __('An error occurred while deleting the pipeline.'),
|
|
[DEFAULT]: __('An unknown error occurred.'),
|
|
},
|
|
inject: {
|
|
// Receive `fullProject` and `pipelinesPath`
|
|
paths: {
|
|
default: {},
|
|
},
|
|
pipelineId: {
|
|
default: '',
|
|
},
|
|
pipelineIid: {
|
|
default: '',
|
|
},
|
|
},
|
|
apollo: {
|
|
pipeline: {
|
|
query: getPipelineQuery,
|
|
variables() {
|
|
return {
|
|
fullPath: this.paths.fullProject,
|
|
iid: this.pipelineIid,
|
|
};
|
|
},
|
|
update: data => data.project.pipeline,
|
|
error() {
|
|
this.reportFailure(LOAD_FAILURE);
|
|
},
|
|
pollInterval: POLL_INTERVAL,
|
|
watchLoading(isLoading) {
|
|
if (!isLoading) {
|
|
// To ensure apollo has updated the cache,
|
|
// we only remove the loading state in sync with GraphQL
|
|
this.isCanceling = false;
|
|
this.isRetrying = false;
|
|
}
|
|
},
|
|
},
|
|
},
|
|
data() {
|
|
return {
|
|
pipeline: null,
|
|
failureType: null,
|
|
isCanceling: false,
|
|
isRetrying: false,
|
|
isDeleting: false,
|
|
};
|
|
},
|
|
computed: {
|
|
deleteModalConfirmationText() {
|
|
return __(
|
|
'Are you sure you want to delete this pipeline? Doing so will expire all pipeline caches and delete all related objects, such as builds, logs, artifacts, and triggers. This action cannot be undone.',
|
|
);
|
|
},
|
|
hasError() {
|
|
return this.failureType;
|
|
},
|
|
hasPipelineData() {
|
|
return Boolean(this.pipeline);
|
|
},
|
|
isLoadingInitialQuery() {
|
|
return this.$apollo.queries.pipeline.loading && !this.hasPipelineData;
|
|
},
|
|
status() {
|
|
return this.pipeline?.status;
|
|
},
|
|
isFinished() {
|
|
return this.$options.finishedStatuses.includes(this.status);
|
|
},
|
|
shouldRenderContent() {
|
|
return !this.isLoadingInitialQuery && this.hasPipelineData;
|
|
},
|
|
failure() {
|
|
switch (this.failureType) {
|
|
case LOAD_FAILURE:
|
|
return {
|
|
text: this.$options.errorTexts[LOAD_FAILURE],
|
|
variant: 'danger',
|
|
};
|
|
case POST_FAILURE:
|
|
return {
|
|
text: this.$options.errorTexts[POST_FAILURE],
|
|
variant: 'danger',
|
|
};
|
|
case DELETE_FAILURE:
|
|
return {
|
|
text: this.$options.errorTexts[DELETE_FAILURE],
|
|
variant: 'danger',
|
|
};
|
|
default:
|
|
return {
|
|
text: this.$options.errorTexts[DEFAULT],
|
|
variant: 'danger',
|
|
};
|
|
}
|
|
},
|
|
},
|
|
watch: {
|
|
isFinished(finished) {
|
|
if (finished) {
|
|
this.$apollo.queries.pipeline.stopPolling();
|
|
}
|
|
},
|
|
},
|
|
methods: {
|
|
reportFailure(errorType) {
|
|
this.failureType = errorType;
|
|
},
|
|
async postPipelineAction(name, mutation) {
|
|
try {
|
|
const {
|
|
data: {
|
|
[name]: { errors },
|
|
},
|
|
} = await this.$apollo.mutate({
|
|
mutation,
|
|
variables: { id: this.pipeline.id },
|
|
});
|
|
|
|
if (errors.length > 0) {
|
|
this.reportFailure(POST_FAILURE);
|
|
} else {
|
|
await this.$apollo.queries.pipeline.refetch();
|
|
if (!this.isFinished) {
|
|
this.$apollo.queries.pipeline.startPolling(POLL_INTERVAL);
|
|
}
|
|
}
|
|
} catch {
|
|
this.reportFailure(POST_FAILURE);
|
|
}
|
|
},
|
|
cancelPipeline() {
|
|
this.isCanceling = true;
|
|
this.postPipelineAction(this.$options.pipelineCancel, cancelPipelineMutation);
|
|
},
|
|
retryPipeline() {
|
|
this.isRetrying = true;
|
|
this.postPipelineAction(this.$options.pipelineRetry, retryPipelineMutation);
|
|
},
|
|
async deletePipeline() {
|
|
this.isDeleting = true;
|
|
this.$apollo.queries.pipeline.stopPolling();
|
|
|
|
try {
|
|
const {
|
|
data: {
|
|
pipelineDestroy: { errors },
|
|
},
|
|
} = await this.$apollo.mutate({
|
|
mutation: deletePipelineMutation,
|
|
variables: {
|
|
id: this.pipeline.id,
|
|
},
|
|
});
|
|
|
|
if (errors.length > 0) {
|
|
this.reportFailure(DELETE_FAILURE);
|
|
this.isDeleting = false;
|
|
} else {
|
|
redirectTo(setUrlFragment(this.paths.pipelinesPath, 'delete_success'));
|
|
}
|
|
} catch {
|
|
this.$apollo.queries.pipeline.startPolling(POLL_INTERVAL);
|
|
this.reportFailure(DELETE_FAILURE);
|
|
this.isDeleting = false;
|
|
}
|
|
},
|
|
},
|
|
DELETE_MODAL_ID,
|
|
};
|
|
</script>
|
|
<template>
|
|
<div class="pipeline-header-container">
|
|
<gl-alert v-if="hasError" :variant="failure.variant">{{ failure.text }}</gl-alert>
|
|
<ci-header
|
|
v-if="shouldRenderContent"
|
|
:status="pipeline.detailedStatus"
|
|
:time="pipeline.createdAt"
|
|
:user="pipeline.user"
|
|
:item-id="Number(pipelineId)"
|
|
item-name="Pipeline"
|
|
>
|
|
<gl-button
|
|
v-if="pipeline.retryable"
|
|
:loading="isRetrying"
|
|
:disabled="isRetrying"
|
|
category="secondary"
|
|
variant="info"
|
|
data-testid="retryPipeline"
|
|
class="js-retry-button"
|
|
@click="retryPipeline()"
|
|
>
|
|
{{ __('Retry') }}
|
|
</gl-button>
|
|
|
|
<gl-button
|
|
v-if="pipeline.cancelable"
|
|
:loading="isCanceling"
|
|
:disabled="isCanceling"
|
|
class="gl-ml-3"
|
|
variant="danger"
|
|
data-testid="cancelPipeline"
|
|
@click="cancelPipeline()"
|
|
>
|
|
{{ __('Cancel running') }}
|
|
</gl-button>
|
|
|
|
<gl-button
|
|
v-if="pipeline.userPermissions.destroyPipeline"
|
|
v-gl-modal="$options.DELETE_MODAL_ID"
|
|
:loading="isDeleting"
|
|
:disabled="isDeleting"
|
|
class="gl-ml-3"
|
|
variant="danger"
|
|
category="secondary"
|
|
data-testid="deletePipeline"
|
|
>
|
|
{{ __('Delete') }}
|
|
</gl-button>
|
|
</ci-header>
|
|
<gl-loading-icon v-if="isLoadingInitialQuery" size="lg" class="gl-mt-3 gl-mb-3" />
|
|
|
|
<gl-modal
|
|
:modal-id="$options.DELETE_MODAL_ID"
|
|
:title="__('Delete pipeline')"
|
|
:ok-title="__('Delete pipeline')"
|
|
ok-variant="danger"
|
|
@ok="deletePipeline()"
|
|
>
|
|
<p>
|
|
{{ deleteModalConfirmationText }}
|
|
</p>
|
|
</gl-modal>
|
|
</div>
|
|
</template>
|