308 lines
8.6 KiB
Vue
308 lines
8.6 KiB
Vue
<script>
|
|
import { isEmpty, findKey } from 'lodash';
|
|
import Vue from 'vue';
|
|
import {
|
|
GlLink,
|
|
GlDeprecatedButton,
|
|
GlButtonGroup,
|
|
GlFormGroup,
|
|
GlFormInput,
|
|
GlDropdown,
|
|
GlDropdownItem,
|
|
GlModal,
|
|
GlTooltipDirective,
|
|
} from '@gitlab/ui';
|
|
import { __, s__ } from '~/locale';
|
|
import Translate from '~/vue_shared/translate';
|
|
import TrackEventDirective from '~/vue_shared/directives/track_event';
|
|
import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
|
|
import Icon from '~/vue_shared/components/icon.vue';
|
|
import { alertsValidator, queriesValidator } from '../validators';
|
|
import { OPERATORS } from '../constants';
|
|
|
|
Vue.use(Translate);
|
|
|
|
const SUBMIT_ACTION_TEXT = {
|
|
create: __('Add'),
|
|
update: __('Save'),
|
|
delete: __('Delete'),
|
|
};
|
|
|
|
const SUBMIT_BUTTON_CLASS = {
|
|
create: 'btn-success',
|
|
update: 'btn-success',
|
|
delete: 'btn-remove',
|
|
};
|
|
|
|
export default {
|
|
components: {
|
|
GlDeprecatedButton,
|
|
GlButtonGroup,
|
|
GlFormGroup,
|
|
GlFormInput,
|
|
GlDropdown,
|
|
GlDropdownItem,
|
|
GlModal,
|
|
GlLink,
|
|
Icon,
|
|
},
|
|
directives: {
|
|
GlTooltip: GlTooltipDirective,
|
|
TrackEvent: TrackEventDirective,
|
|
},
|
|
mixins: [glFeatureFlagsMixin()],
|
|
props: {
|
|
disabled: {
|
|
type: Boolean,
|
|
required: true,
|
|
},
|
|
errorMessage: {
|
|
type: String,
|
|
required: false,
|
|
default: '',
|
|
},
|
|
configuredAlert: {
|
|
type: String,
|
|
required: false,
|
|
default: '',
|
|
},
|
|
alertsToManage: {
|
|
type: Object,
|
|
required: false,
|
|
default: () => ({}),
|
|
validator: alertsValidator,
|
|
},
|
|
relevantQueries: {
|
|
type: Array,
|
|
required: true,
|
|
validator: queriesValidator,
|
|
},
|
|
modalId: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
},
|
|
data() {
|
|
return {
|
|
operators: OPERATORS,
|
|
operator: null,
|
|
threshold: null,
|
|
prometheusMetricId: null,
|
|
selectedAlert: {},
|
|
alertQuery: '',
|
|
};
|
|
},
|
|
computed: {
|
|
isValidQuery() {
|
|
// TODO: Add query validation check (most likely via http request)
|
|
return this.alertQuery.length ? true : null;
|
|
},
|
|
currentQuery() {
|
|
return this.relevantQueries.find(query => query.metricId === this.prometheusMetricId) || {};
|
|
},
|
|
formDisabled() {
|
|
// We need a prometheusMetricId to determine whether we're
|
|
// creating/updating/deleting
|
|
return this.disabled || !(this.prometheusMetricId || this.isValidQuery);
|
|
},
|
|
supportsComputedAlerts() {
|
|
return this.glFeatures.prometheusComputedAlerts;
|
|
},
|
|
queryDropdownLabel() {
|
|
return this.currentQuery.label || s__('PrometheusAlerts|Select query');
|
|
},
|
|
haveValuesChanged() {
|
|
return (
|
|
this.operator &&
|
|
this.threshold === Number(this.threshold) &&
|
|
(this.operator !== this.selectedAlert.operator ||
|
|
this.threshold !== this.selectedAlert.threshold)
|
|
);
|
|
},
|
|
submitAction() {
|
|
if (isEmpty(this.selectedAlert)) return 'create';
|
|
if (this.haveValuesChanged) return 'update';
|
|
return 'delete';
|
|
},
|
|
submitActionText() {
|
|
return SUBMIT_ACTION_TEXT[this.submitAction];
|
|
},
|
|
submitButtonClass() {
|
|
return SUBMIT_BUTTON_CLASS[this.submitAction];
|
|
},
|
|
isSubmitDisabled() {
|
|
return this.disabled || (this.submitAction === 'create' && !this.haveValuesChanged);
|
|
},
|
|
dropdownTitle() {
|
|
return this.submitAction === 'create'
|
|
? s__('PrometheusAlerts|Add alert')
|
|
: s__('PrometheusAlerts|Edit alert');
|
|
},
|
|
},
|
|
watch: {
|
|
alertsToManage() {
|
|
this.resetAlertData();
|
|
},
|
|
submitAction() {
|
|
this.$emit('setAction', this.submitAction);
|
|
},
|
|
},
|
|
methods: {
|
|
selectQuery(queryId) {
|
|
const existingAlertPath = findKey(this.alertsToManage, alert => alert.metricId === queryId);
|
|
const existingAlert = this.alertsToManage[existingAlertPath];
|
|
|
|
if (existingAlert) {
|
|
this.selectedAlert = existingAlert;
|
|
this.operator = existingAlert.operator;
|
|
this.threshold = existingAlert.threshold;
|
|
} else {
|
|
this.selectedAlert = {};
|
|
this.operator = this.operators.greaterThan;
|
|
this.threshold = null;
|
|
}
|
|
|
|
this.prometheusMetricId = queryId;
|
|
},
|
|
handleHidden() {
|
|
this.resetAlertData();
|
|
this.$emit('cancel');
|
|
},
|
|
handleSubmit(e) {
|
|
e.preventDefault();
|
|
this.$emit(this.submitAction, {
|
|
alert: this.selectedAlert.alert_path,
|
|
operator: this.operator,
|
|
threshold: this.threshold,
|
|
prometheus_metric_id: this.prometheusMetricId,
|
|
});
|
|
},
|
|
handleShown() {
|
|
if (this.configuredAlert) {
|
|
this.selectQuery(this.configuredAlert);
|
|
} else if (this.relevantQueries.length === 1) {
|
|
this.selectQuery(this.relevantQueries[0].metricId);
|
|
}
|
|
},
|
|
resetAlertData() {
|
|
this.operator = null;
|
|
this.threshold = null;
|
|
this.prometheusMetricId = null;
|
|
this.selectedAlert = {};
|
|
},
|
|
getAlertFormActionTrackingOption() {
|
|
const label = `${this.submitAction}_alert`;
|
|
return {
|
|
category: document.body.dataset.page,
|
|
action: 'click_button',
|
|
label,
|
|
};
|
|
},
|
|
},
|
|
alertQueryText: {
|
|
label: __('Query'),
|
|
validFeedback: __('Query is valid'),
|
|
invalidFeedback: __('Invalid query'),
|
|
descriptionTooltip: __(
|
|
'Example: Usage = single query. (Requested) / (Capacity) = multiple queries combined into a formula.',
|
|
),
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<gl-modal
|
|
ref="alertModal"
|
|
:title="dropdownTitle"
|
|
:modal-id="modalId"
|
|
:ok-variant="submitAction === 'delete' ? 'danger' : 'success'"
|
|
:ok-disabled="formDisabled"
|
|
@ok="handleSubmit"
|
|
@hidden="handleHidden"
|
|
@shown="handleShown"
|
|
>
|
|
<div v-if="errorMessage" class="alert-modal-message danger_message">{{ errorMessage }}</div>
|
|
<div class="alert-form">
|
|
<gl-form-group
|
|
v-if="supportsComputedAlerts"
|
|
:label="$options.alertQueryText.label"
|
|
label-for="alert-query-input"
|
|
:valid-feedback="$options.alertQueryText.validFeedback"
|
|
:invalid-feedback="$options.alertQueryText.invalidFeedback"
|
|
:state="isValidQuery"
|
|
>
|
|
<gl-form-input id="alert-query-input" v-model.trim="alertQuery" :state="isValidQuery" />
|
|
<template #description>
|
|
<div class="d-flex align-items-center">
|
|
{{ __('Single or combined queries') }}
|
|
<icon
|
|
v-gl-tooltip="$options.alertQueryText.descriptionTooltip"
|
|
name="question"
|
|
class="gl-ml-2"
|
|
/>
|
|
</div>
|
|
</template>
|
|
</gl-form-group>
|
|
<gl-form-group v-else label-for="alert-query-dropdown" :label="$options.alertQueryText.label">
|
|
<gl-dropdown
|
|
id="alert-query-dropdown"
|
|
:text="queryDropdownLabel"
|
|
toggle-class="dropdown-menu-toggle qa-alert-query-dropdown"
|
|
>
|
|
<gl-dropdown-item
|
|
v-for="query in relevantQueries"
|
|
:key="query.metricId"
|
|
data-qa-selector="alert_query_option"
|
|
@click="selectQuery(query.metricId)"
|
|
>
|
|
{{ query.label }}
|
|
</gl-dropdown-item>
|
|
</gl-dropdown>
|
|
</gl-form-group>
|
|
<gl-button-group class="mb-2" :label="s__('PrometheusAlerts|Operator')">
|
|
<gl-deprecated-button
|
|
:class="{ active: operator === operators.greaterThan }"
|
|
:disabled="formDisabled"
|
|
type="button"
|
|
@click="operator = operators.greaterThan"
|
|
>
|
|
{{ operators.greaterThan }}
|
|
</gl-deprecated-button>
|
|
<gl-deprecated-button
|
|
:class="{ active: operator === operators.equalTo }"
|
|
:disabled="formDisabled"
|
|
type="button"
|
|
@click="operator = operators.equalTo"
|
|
>
|
|
{{ operators.equalTo }}
|
|
</gl-deprecated-button>
|
|
<gl-deprecated-button
|
|
:class="{ active: operator === operators.lessThan }"
|
|
:disabled="formDisabled"
|
|
type="button"
|
|
@click="operator = operators.lessThan"
|
|
>
|
|
{{ operators.lessThan }}
|
|
</gl-deprecated-button>
|
|
</gl-button-group>
|
|
<gl-form-group :label="s__('PrometheusAlerts|Threshold')" label-for="alerts-threshold">
|
|
<gl-form-input
|
|
id="alerts-threshold"
|
|
v-model.number="threshold"
|
|
:disabled="formDisabled"
|
|
type="number"
|
|
data-qa-selector="alert_threshold_field"
|
|
/>
|
|
</gl-form-group>
|
|
</div>
|
|
<template #modal-ok>
|
|
<gl-link
|
|
v-track-event="getAlertFormActionTrackingOption()"
|
|
class="text-reset text-decoration-none"
|
|
>
|
|
{{ submitActionText }}
|
|
</gl-link>
|
|
</template>
|
|
</gl-modal>
|
|
</template>
|