Commit 892da007 authored by Andrew Fontaine's avatar Andrew Fontaine

Merge branch '33098-distribute-daily-cron-schedules-out-over-the-hour' into 'master'

Resolve "Distribute daily cron schedules out over the hour"

See merge request gitlab-org/gitlab!30729
parents f524c3a5 f0354bb1
...@@ -56,6 +56,19 @@ export const getMonthNames = abbreviated => { ...@@ -56,6 +56,19 @@ export const getMonthNames = abbreviated => {
export const pad = (val, len = 2) => `0${val}`.slice(-len); export const pad = (val, len = 2) => `0${val}`.slice(-len);
/**
* Returns i18n weekday names array.
*/
export const getWeekdayNames = () => [
__('Sunday'),
__('Monday'),
__('Tuesday'),
__('Wednesday'),
__('Thursday'),
__('Friday'),
__('Saturday'),
];
/** /**
* Given a date object returns the day of the week in English * Given a date object returns the day of the week in English
* @param {date} date * @param {date} date
......
<script> <script>
import { GlSprintf, GlLink } from '@gitlab/ui';
import { s__, sprintf } from '~/locale';
import { getWeekdayNames } from '~/lib/utils/datetime_utility';
export default { export default {
components: {
GlSprintf,
GlLink,
},
props: { props: {
initialCronInterval: { initialCronInterval: {
type: String, type: String,
...@@ -9,25 +17,51 @@ export default { ...@@ -9,25 +17,51 @@ export default {
}, },
data() { data() {
return { return {
isEditingCustom: false,
randomHour: this.generateRandomHour(),
randomWeekDayIndex: this.generateRandomWeekDayIndex(),
randomDay: this.generateRandomDay(),
inputNameAttribute: 'schedule[cron]', inputNameAttribute: 'schedule[cron]',
cronInterval: this.initialCronInterval, cronInterval: this.initialCronInterval,
cronIntervalPresets: {
everyDay: '0 4 * * *',
everyWeek: '0 4 * * 0',
everyMonth: '0 4 1 * *',
},
cronSyntaxUrl: 'https://en.wikipedia.org/wiki/Cron', cronSyntaxUrl: 'https://en.wikipedia.org/wiki/Cron',
customInputEnabled: false,
}; };
}, },
computed: { computed: {
cronIntervalPresets() {
return {
everyDay: `0 ${this.randomHour} * * *`,
everyWeek: `0 ${this.randomHour} * * ${this.randomWeekDayIndex}`,
everyMonth: `0 ${this.randomHour} ${this.randomDay} * *`,
};
},
intervalIsPreset() { intervalIsPreset() {
return Object.values(this.cronIntervalPresets).includes(this.cronInterval); return Object.values(this.cronIntervalPresets).includes(this.cronInterval);
}, },
// The text input is editable when there's a custom interval, or when it's formattedTime() {
// a preset interval and the user clicks the 'custom' radio button if (this.randomHour > 12) {
isEditable() { return `${this.randomHour - 12}:00pm`;
return Boolean(this.customInputEnabled || !this.intervalIsPreset); } else if (this.randomHour === 12) {
return `12:00pm`;
}
return `${this.randomHour}:00am`;
},
weekday() {
return getWeekdayNames()[this.randomWeekDayIndex];
},
everyDayText() {
return sprintf(s__(`Every day (at %{time})`), { time: this.formattedTime });
},
everyWeekText() {
return sprintf(s__('Every week (%{weekday} at %{time})'), {
weekday: this.weekday,
time: this.formattedTime,
});
},
everyMonthText() {
return sprintf(s__('Every month (Day %{day} at %{time})'), {
day: this.randomDay,
time: this.formattedTime,
});
}, },
}, },
watch: { watch: {
...@@ -39,14 +73,31 @@ export default { ...@@ -39,14 +73,31 @@ export default {
}); });
}, },
}, },
created() { // If at the mounting stage the default is still an empty string, we
if (this.intervalIsPreset) { // know we are not editing an existing field so we update it so
this.enableCustomInput = false; // that the default is the first radio option
mounted() {
if (this.cronInterval === '') {
this.cronInterval = this.cronIntervalPresets.everyDay;
} }
}, },
methods: { methods: {
setCustomInput(e) {
if (!this.isEditingCustom) {
this.isEditingCustom = true;
this.$refs.customInput.click();
// Because we need to manually trigger the click on the radio btn,
// it will add a space to update the v-model. If the user is typing
// and the space is added, it will feel very unituitive so we reset
// the value to the original
this.cronInterval = e.target.value;
}
if (this.intervalIsPreset) {
this.isEditingCustom = false;
}
},
toggleCustomInput(shouldEnable) { toggleCustomInput(shouldEnable) {
this.customInputEnabled = shouldEnable; this.isEditingCustom = shouldEnable;
if (shouldEnable) { if (shouldEnable) {
// We need to change the value so other radios don't remain selected // We need to change the value so other radios don't remain selected
...@@ -54,30 +105,21 @@ export default { ...@@ -54,30 +105,21 @@ export default {
this.cronInterval = `${this.cronInterval} `; this.cronInterval = `${this.cronInterval} `;
} }
}, },
generateRandomHour() {
return Math.floor(Math.random() * 23);
},
generateRandomWeekDayIndex() {
return Math.floor(Math.random() * 6);
},
generateRandomDay() {
return Math.floor(Math.random() * 28);
},
}, },
}; };
</script> </script>
<template> <template>
<div class="interval-pattern-form-group"> <div class="interval-pattern-form-group">
<div class="cron-preset-radio-input">
<input
id="custom"
:name="inputNameAttribute"
:value="cronInterval"
:checked="isEditable"
class="label-bold"
type="radio"
@click="toggleCustomInput(true)"
/>
<label for="custom"> {{ s__('PipelineSheduleIntervalPattern|Custom') }} </label>
<span class="cron-syntax-link-wrap">
(<a :href="cronSyntaxUrl" target="_blank"> {{ __('Cron syntax') }} </a>)
</span>
</div>
<div class="cron-preset-radio-input"> <div class="cron-preset-radio-input">
<input <input
id="every-day" id="every-day"
...@@ -89,7 +131,9 @@ export default { ...@@ -89,7 +131,9 @@ export default {
@click="toggleCustomInput(false)" @click="toggleCustomInput(false)"
/> />
<label class="label-bold" for="every-day"> {{ __('Every day (at 4:00am)') }} </label> <label class="label-bold" for="every-day">
{{ everyDayText }}
</label>
</div> </div>
<div class="cron-preset-radio-input"> <div class="cron-preset-radio-input">
...@@ -104,7 +148,7 @@ export default { ...@@ -104,7 +148,7 @@ export default {
/> />
<label class="label-bold" for="every-week"> <label class="label-bold" for="every-week">
{{ __('Every week (Sundays at 4:00am)') }} {{ everyWeekText }}
</label> </label>
</div> </div>
...@@ -120,20 +164,43 @@ export default { ...@@ -120,20 +164,43 @@ export default {
/> />
<label class="label-bold" for="every-month"> <label class="label-bold" for="every-month">
{{ __('Every month (on the 1st at 4:00am)') }} {{ everyMonthText }}
</label> </label>
</div> </div>
<div class="cron-preset-radio-input">
<input
id="custom"
ref="customInput"
v-model="cronInterval"
:name="inputNameAttribute"
:value="cronInterval"
class="label-bold"
type="radio"
@click="toggleCustomInput(true)"
/>
<label for="custom"> {{ s__('PipelineSheduleIntervalPattern|Custom') }} </label>
<gl-sprintf :message="__('(%{linkStart}Cron syntax%{linkEnd})')">
<template #link="{content}">
<gl-link :href="cronSyntaxUrl" target="_blank" class="gl-font-sm">
{{ content }}
</gl-link>
</template>
</gl-sprintf>
</div>
<div class="cron-interval-input-wrapper"> <div class="cron-interval-input-wrapper">
<input <input
id="schedule_cron" id="schedule_cron"
v-model="cronInterval" v-model="cronInterval"
:placeholder="__('Define a custom pattern with cron syntax')" :placeholder="__('Define a custom pattern with cron syntax')"
:name="inputNameAttribute" :name="inputNameAttribute"
:disabled="!isEditable"
class="form-control inline cron-interval-input" class="form-control inline cron-interval-input"
type="text" type="text"
required="true" required="true"
@input="setCustomInput"
/> />
</div> </div>
</div> </div>
......
...@@ -11,9 +11,7 @@ Vue.use(Translate); ...@@ -11,9 +11,7 @@ Vue.use(Translate);
function initIntervalPatternInput() { function initIntervalPatternInput() {
const intervalPatternMount = document.getElementById('interval-pattern-input'); const intervalPatternMount = document.getElementById('interval-pattern-input');
const initialCronInterval = intervalPatternMount const initialCronInterval = intervalPatternMount?.dataset?.initialInterval;
? intervalPatternMount.dataset.initialInterval
: '';
return new Vue({ return new Vue({
el: intervalPatternMount, el: intervalPatternMount,
......
...@@ -21,11 +21,6 @@ ...@@ -21,11 +21,6 @@
.cron-interval-input { .cron-interval-input {
margin: 10px 10px 0 0; margin: 10px 10px 0 0;
} }
.cron-syntax-link-wrap {
margin-right: 10px;
font-size: 12px;
}
} }
.pipeline-schedule-table-row { .pipeline-schedule-table-row {
......
---
title: Update cron job schedule to have a random time generated on page load
merge_request: 30729
author:
type: changed
...@@ -586,6 +586,9 @@ msgid_plural "(%d closed)" ...@@ -586,6 +586,9 @@ msgid_plural "(%d closed)"
msgstr[0] "" msgstr[0] ""
msgstr[1] "" msgstr[1] ""
msgid "(%{linkStart}Cron syntax%{linkEnd})"
msgstr ""
msgid "(%{mrCount} merged)" msgid "(%{mrCount} merged)"
msgstr "" msgstr ""
...@@ -6417,9 +6420,6 @@ msgstr "" ...@@ -6417,9 +6420,6 @@ msgstr ""
msgid "Cron Timezone" msgid "Cron Timezone"
msgstr "" msgstr ""
msgid "Cron syntax"
msgstr ""
msgid "Crossplane" msgid "Crossplane"
msgstr "" msgstr ""
...@@ -8711,13 +8711,13 @@ msgstr "" ...@@ -8711,13 +8711,13 @@ msgstr ""
msgid "Every day" msgid "Every day"
msgstr "" msgstr ""
msgid "Every day (at 4:00am)" msgid "Every day (at %{time})"
msgstr "" msgstr ""
msgid "Every month" msgid "Every month"
msgstr "" msgstr ""
msgid "Every month (on the 1st at 4:00am)" msgid "Every month (Day %{day} at %{time})"
msgstr "" msgstr ""
msgid "Every three months" msgid "Every three months"
...@@ -8729,7 +8729,7 @@ msgstr "" ...@@ -8729,7 +8729,7 @@ msgstr ""
msgid "Every week" msgid "Every week"
msgstr "" msgstr ""
msgid "Every week (Sundays at 4:00am)" msgid "Every week (%{weekday} at %{time})"
msgstr "" msgstr ""
msgid "Everyone" msgid "Everyone"
......
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import IntervalPatternInput from '~/pages/projects/pipeline_schedules/shared/components/interval_pattern_input.vue'; import IntervalPatternInput from '~/pages/projects/pipeline_schedules/shared/components/interval_pattern_input.vue';
const cronIntervalPresets = {
everyDay: '0 4 * * *',
everyWeek: '0 4 * * 0',
everyMonth: '0 4 1 * *',
};
describe('Interval Pattern Input Component', () => { describe('Interval Pattern Input Component', () => {
let oldWindowGl; let oldWindowGl;
let wrapper; let wrapper;
const mockHour = 4;
const mockWeekDayIndex = 1;
const mockDay = 1;
const cronIntervalPresets = {
everyDay: `0 ${mockHour} * * *`,
everyWeek: `0 ${mockHour} * * ${mockWeekDayIndex}`,
everyMonth: `0 ${mockHour} ${mockDay} * *`,
};
const findEveryDayRadio = () => wrapper.find('#every-day'); const findEveryDayRadio = () => wrapper.find('#every-day');
const findEveryWeekRadio = () => wrapper.find('#every-week'); const findEveryWeekRadio = () => wrapper.find('#every-week');
const findEveryMonthRadio = () => wrapper.find('#every-month'); const findEveryMonthRadio = () => wrapper.find('#every-month');
...@@ -21,13 +25,20 @@ describe('Interval Pattern Input Component', () => { ...@@ -21,13 +25,20 @@ describe('Interval Pattern Input Component', () => {
const selectEveryMonthRadio = () => findEveryMonthRadio().setChecked(); const selectEveryMonthRadio = () => findEveryMonthRadio().setChecked();
const selectCustomRadio = () => findCustomRadio().trigger('click'); const selectCustomRadio = () => findCustomRadio().trigger('click');
const createWrapper = (props = {}) => { const createWrapper = (props = {}, data = {}) => {
if (wrapper) { if (wrapper) {
throw new Error('A wrapper already exists'); throw new Error('A wrapper already exists');
} }
wrapper = shallowMount(IntervalPatternInput, { wrapper = shallowMount(IntervalPatternInput, {
propsData: { ...props }, propsData: { ...props },
data() {
return {
randomHour: data?.hour || mockHour,
randomWeekDayIndex: mockWeekDayIndex,
randomDay: mockDay,
};
},
}); });
}; };
...@@ -47,39 +58,64 @@ describe('Interval Pattern Input Component', () => { ...@@ -47,39 +58,64 @@ describe('Interval Pattern Input Component', () => {
window.gl = oldWindowGl; window.gl = oldWindowGl;
}); });
describe('when prop initialCronInterval is passed', () => { describe('the input field defaults', () => {
describe('and prop initialCronInterval is custom', () => { beforeEach(() => {
beforeEach(() => { createWrapper();
createWrapper({ initialCronInterval: '1 2 3 4 5' }); });
});
it('the input is enabled', () => { it('to a non empty string when no initial value is not passed', () => {
expect(findCustomInput().attributes('disabled')).toBeUndefined(); expect(findCustomInput()).not.toBe('');
});
}); });
});
describe('and prop initialCronInterval is a preset', () => { describe('the input field', () => {
beforeEach(() => { const initialCron = '0 * * * *';
createWrapper({ initialCronInterval: cronIntervalPresets.everyDay });
});
it('the input is disabled', () => { beforeEach(() => {
expect(findCustomInput().attributes('disabled')).toBe('disabled'); createWrapper({ initialCronInterval: initialCron });
}); });
it('is equal to the prop `initialCronInterval` when passed', () => {
expect(findCustomInput().element.value).toBe(initialCron);
}); });
}); });
describe('when prop initialCronInterval is not passed', () => { describe('The input field is enabled', () => {
beforeEach(() => { beforeEach(() => {
createWrapper(); createWrapper();
}); });
it('the input is enabled since custom is default value', () => { it('when a default option is selected', () => {
expect(findCustomInput().attributes('disabled')).toBeUndefined(); selectEveryDayRadio();
return wrapper.vm.$nextTick().then(() => {
expect(findCustomInput().attributes('disabled')).toBeUndefined();
});
});
it('when the custom option is selected', () => {
selectCustomRadio();
return wrapper.vm.$nextTick().then(() => {
expect(findCustomInput().attributes('disabled')).toBeUndefined();
});
}); });
}); });
describe('User Actions', () => { describe('formattedTime computed property', () => {
it.each`
desc | hour | expectedValue
${'returns a time in the afternoon if the value of `random time` is higher than 12'} | ${13} | ${'1:00pm'}
${'returns a time in the morning if the value of `random time` is lower than 12'} | ${11} | ${'11:00am'}
${'returns "12:00pm" if the value of `random time` is exactly 12'} | ${12} | ${'12:00pm'}
`('$desc', ({ hour, expectedValue }) => {
createWrapper({}, { hour });
expect(wrapper.vm.formattedTime).toBe(expectedValue);
});
});
describe('User Actions with radio buttons', () => {
it.each` it.each`
desc | initialCronInterval | act | expectedValue desc | initialCronInterval | act | expectedValue
${'when everyday is selected, update value'} | ${'1 2 3 4 5'} | ${selectEveryDayRadio} | ${cronIntervalPresets.everyDay} ${'when everyday is selected, update value'} | ${'1 2 3 4 5'} | ${selectEveryDayRadio} | ${cronIntervalPresets.everyDay}
...@@ -96,4 +132,23 @@ describe('Interval Pattern Input Component', () => { ...@@ -96,4 +132,23 @@ describe('Interval Pattern Input Component', () => {
}); });
}); });
}); });
describe('User actions with input field for Cron syntax', () => {
beforeEach(() => {
createWrapper();
});
it('when editing the cron input it selects the custom radio button', () => {
const newValue = '0 * * * *';
findCustomInput().setValue(newValue);
expect(wrapper.vm.cronInterval).toBe(newValue);
});
it('when value of input is one of the defaults, it selects the corresponding radio button', () => {
findCustomInput().setValue(cronIntervalPresets.everyWeek);
expect(wrapper.vm.cronInterval).toBe(cronIntervalPresets.everyWeek);
});
});
}); });
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