jormungandr-bite/src/client/app/admin/views/queue.vue

312 lines
8 KiB
Vue
Raw Normal View History

2019-02-05 23:24:59 -07:00
<template>
<div>
<ui-card>
2019-03-14 22:48:17 -06:00
<template #title><fa :icon="faChartBar"/> {{ $t('title') }}</template>
2019-03-12 02:11:06 -06:00
<section class="wptihjuy">
2019-03-12 02:20:40 -06:00
<header><fa :icon="faPaperPlane"/> Deliver</header>
2019-03-14 22:09:05 -06:00
<ui-info warn v-if="latestStats && latestStats.deliver.waiting > 0">The queue is jammed.</ui-info>
2019-03-12 02:11:06 -06:00
<ui-horizon-group inputs v-if="latestStats" class="fit-bottom">
2019-03-12 06:53:36 -06:00
<ui-input :value="latestStats.deliver.activeSincePrevTick | number" type="text" readonly>
<span>Process</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="fasPlayCircle"/></template>
2019-03-14 22:09:19 -06:00
<template #suffix>jobs/tick</template>
2019-03-12 06:53:36 -06:00
</ui-input>
<ui-input :value="latestStats.deliver.active | number" type="text" readonly>
<span>Active</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="farPlayCircle"/></template>
2019-03-12 06:53:36 -06:00
<template #suffix>jobs</template>
</ui-input>
2019-03-12 02:11:06 -06:00
<ui-input :value="latestStats.deliver.waiting | number" type="text" readonly>
<span>Waiting</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="faStopCircle"/></template>
2019-03-12 06:53:36 -06:00
<template #suffix>jobs</template>
</ui-input>
2019-03-12 02:11:06 -06:00
<ui-input :value="latestStats.deliver.delayed | number" type="text" readonly>
<span>Delayed</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="faStopwatch"/></template>
2019-03-12 06:53:36 -06:00
<template #suffix>jobs</template>
2019-03-07 23:27:06 -07:00
</ui-input>
</ui-horizon-group>
2019-03-12 02:11:06 -06:00
<div ref="deliverChart" class="chart"></div>
</section>
2019-03-12 02:11:06 -06:00
<section class="wptihjuy">
2019-03-12 02:20:40 -06:00
<header><fa :icon="faInbox"/> Inbox</header>
2019-03-14 22:09:05 -06:00
<ui-info warn v-if="latestStats && latestStats.inbox.waiting > 0">The queue is jammed.</ui-info>
2019-03-12 02:11:06 -06:00
<ui-horizon-group inputs v-if="latestStats" class="fit-bottom">
2019-03-12 06:53:36 -06:00
<ui-input :value="latestStats.inbox.activeSincePrevTick | number" type="text" readonly>
<span>Process</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="fasPlayCircle"/></template>
2019-03-14 22:09:19 -06:00
<template #suffix>jobs/tick</template>
2019-03-12 06:53:36 -06:00
</ui-input>
<ui-input :value="latestStats.inbox.active | number" type="text" readonly>
<span>Active</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="farPlayCircle"/></template>
2019-03-12 06:53:36 -06:00
<template #suffix>jobs</template>
</ui-input>
2019-03-12 02:11:06 -06:00
<ui-input :value="latestStats.inbox.waiting | number" type="text" readonly>
<span>Waiting</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="faStopCircle"/></template>
2019-03-12 06:53:36 -06:00
<template #suffix>jobs</template>
</ui-input>
2019-03-12 02:11:06 -06:00
<ui-input :value="latestStats.inbox.delayed | number" type="text" readonly>
<span>Delayed</span>
2019-03-12 08:30:44 -06:00
<template #prefix><fa :icon="faStopwatch"/></template>
2019-03-12 06:53:36 -06:00
<template #suffix>jobs</template>
2019-03-07 23:27:06 -07:00
</ui-input>
</ui-horizon-group>
2019-03-12 02:11:06 -06:00
<div ref="inboxChart" class="chart"></div>
</section>
2019-02-05 23:24:59 -07:00
<section>
<ui-button @click="removeAllJobs">{{ $t('remove-all-jobs') }}</ui-button>
</section>
</ui-card>
2019-03-14 22:48:17 -06:00
<ui-card>
<template #title><fa :icon="faTasks"/> {{ $t('jobs') }}</template>
<section class="fit-top">
<ui-horizon-group inputs>
<ui-select v-model="domain">
<template #label>{{ $t('queue') }}</template>
<option value="deliver">{{ $t('domains.deliver') }}</option>
<option value="inbox">{{ $t('domains.inbox') }}</option>
</ui-select>
<ui-select v-model="state">
<template #label>{{ $t('state') }}</template>
<option value="delayed">{{ $t('states.delayed') }}</option>
</ui-select>
</ui-horizon-group>
<sequential-entrance animation="entranceFromTop" delay="25">
<div class="xvvuvgsv" v-for="job in jobs">
<b>{{ job.id }}</b>
<template v-if="domain === 'deliver'">
<span>{{ job.data.to }}</span>
</template>
<template v-if="domain === 'inbox'">
<span>{{ job.activity.id }}</span>
</template>
</div>
</sequential-entrance>
<ui-info v-if="jobs.length == jobsLimit">{{ $t('result-is-truncated', { n: jobsLimit }) }}</ui-info>
</section>
</ui-card>
2019-02-05 23:24:59 -07:00
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import i18n from '../../i18n';
2019-03-12 02:11:06 -06:00
import ApexCharts from 'apexcharts';
import * as tinycolor from 'tinycolor2';
2019-03-12 08:30:44 -06:00
import { faTasks, faInbox, faStopwatch, faPlayCircle as fasPlayCircle } from '@fortawesome/free-solid-svg-icons';
2019-03-14 22:48:17 -06:00
import { faPaperPlane, faStopCircle, faPlayCircle as farPlayCircle, faChartBar } from '@fortawesome/free-regular-svg-icons';
2019-02-05 23:24:59 -07:00
2019-03-13 08:27:11 -06:00
const limit = 150;
2019-02-05 23:24:59 -07:00
export default Vue.extend({
i18n: i18n('admin/views/queue.vue'),
data() {
return {
2019-03-12 02:11:06 -06:00
stats: [],
deliverChart: null,
inboxChart: null,
2019-03-14 22:48:17 -06:00
jobs: [],
jobsLimit: 50,
domain: 'deliver',
state: 'delayed',
faTasks, faPaperPlane, faInbox, faStopwatch, faStopCircle, farPlayCircle, fasPlayCircle, faChartBar
2019-02-05 23:24:59 -07:00
};
},
2019-03-12 02:11:06 -06:00
computed: {
latestStats(): any {
return this.stats[this.stats.length - 1];
}
},
watch: {
stats(stats) {
this.inboxChart.updateSeries([{
2019-03-12 06:53:36 -06:00
name: 'Process',
2019-03-13 08:27:11 -06:00
type: 'area',
2019-03-12 02:11:06 -06:00
data: stats.map((x, i) => ({ x: i, y: x.inbox.activeSincePrevTick }))
2019-03-12 06:53:36 -06:00
}, {
name: 'Active',
2019-03-13 08:27:11 -06:00
type: 'area',
2019-03-12 06:53:36 -06:00
data: stats.map((x, i) => ({ x: i, y: x.inbox.active }))
2019-03-12 02:11:06 -06:00
}, {
name: 'Waiting',
2019-03-13 08:27:11 -06:00
type: 'line',
2019-03-12 02:11:06 -06:00
data: stats.map((x, i) => ({ x: i, y: x.inbox.waiting }))
}, {
name: 'Delayed',
2019-03-13 08:27:11 -06:00
type: 'line',
2019-03-12 02:11:06 -06:00
data: stats.map((x, i) => ({ x: i, y: x.inbox.delayed }))
}]);
this.deliverChart.updateSeries([{
2019-03-12 06:53:36 -06:00
name: 'Process',
2019-03-13 08:27:11 -06:00
type: 'area',
2019-03-12 02:11:06 -06:00
data: stats.map((x, i) => ({ x: i, y: x.deliver.activeSincePrevTick }))
2019-03-12 06:53:36 -06:00
}, {
name: 'Active',
2019-03-13 08:27:11 -06:00
type: 'area',
2019-03-12 06:53:36 -06:00
data: stats.map((x, i) => ({ x: i, y: x.deliver.active }))
2019-03-12 02:11:06 -06:00
}, {
name: 'Waiting',
2019-03-13 08:27:11 -06:00
type: 'line',
2019-03-12 02:11:06 -06:00
data: stats.map((x, i) => ({ x: i, y: x.deliver.waiting }))
}, {
name: 'Delayed',
2019-03-13 08:27:11 -06:00
type: 'line',
2019-03-12 02:11:06 -06:00
data: stats.map((x, i) => ({ x: i, y: x.deliver.delayed }))
}]);
2019-03-14 22:48:17 -06:00
},
domain() {
this.jobs = [];
this.fetchJobs();
},
state() {
this.jobs = [];
this.fetchJobs();
},
2019-03-12 02:11:06 -06:00
},
mounted() {
2019-03-14 22:48:17 -06:00
this.fetchJobs();
2019-03-14 22:09:19 -06:00
const chartOpts = id => ({
2019-03-12 02:11:06 -06:00
chart: {
2019-03-14 22:09:19 -06:00
id,
group: 'queue',
2019-03-12 02:11:06 -06:00
type: 'area',
height: 200,
animations: {
dynamicAnimation: {
enabled: false
}
},
toolbar: {
show: false
},
zoom: {
enabled: false
}
},
dataLabels: {
enabled: false
},
grid: {
clipMarkers: false,
2019-03-14 10:43:11 -06:00
borderColor: 'rgba(0, 0, 0, 0.1)',
xaxis: {
lines: {
show: true,
}
},
2019-03-12 02:11:06 -06:00
},
stroke: {
curve: 'straight',
width: 2
},
tooltip: {
enabled: false
},
legend: {
labels: {
colors: tinycolor(getComputedStyle(document.documentElement).getPropertyValue('--text')).toRgbString()
},
},
series: [] as any,
2019-03-12 18:26:38 -06:00
colors: ['#00E396', '#00BCD4', '#FFB300', '#e53935'],
2019-03-12 02:11:06 -06:00
xaxis: {
type: 'numeric',
labels: {
show: false
},
tooltip: {
enabled: false
}
},
yaxis: {
show: false,
min: 0,
}
2019-03-14 22:09:19 -06:00
});
2019-03-07 21:10:38 -07:00
2019-03-14 22:09:19 -06:00
this.inboxChart = new ApexCharts(this.$refs.inboxChart, chartOpts('a'));
this.deliverChart = new ApexCharts(this.$refs.deliverChart, chartOpts('b'));
2019-03-07 21:10:38 -07:00
2019-03-12 02:11:06 -06:00
this.inboxChart.render();
this.deliverChart.render();
const connection = this.$root.stream.useSharedConnection('queueStats');
connection.on('stats', this.onStats);
connection.on('statsLog', this.onStatsLog);
connection.send('requestLog', {
id: Math.random().toString().substr(2, 8),
2019-03-13 08:27:11 -06:00
length: limit
2019-03-12 02:11:06 -06:00
});
2019-03-07 21:10:38 -07:00
this.$once('hook:beforeDestroy', () => {
2019-03-12 02:11:06 -06:00
connection.dispose();
this.inboxChart.destroy();
this.deliverChart.destroy();
});
},
2019-02-05 23:24:59 -07:00
methods: {
async removeAllJobs() {
const process = async () => {
await this.$root.api('admin/queue/clear');
this.$root.dialog({
type: 'success',
splash: true
});
};
await process().catch(e => {
this.$root.dialog({
type: 'error',
text: e.toString()
});
});
},
2019-03-12 02:11:06 -06:00
onStats(stats) {
this.stats.push(stats);
2019-03-13 08:27:11 -06:00
if (this.stats.length > limit) this.stats.shift();
2019-03-12 02:11:06 -06:00
},
onStatsLog(statsLog) {
for (const stats of statsLog.reverse()) {
this.onStats(stats);
}
2019-03-14 22:48:17 -06:00
},
fetchJobs() {
this.$root.api('admin/queue/jobs', {
domain: this.domain,
state: this.state,
limit: this.jobsLimit
}).then(jobs => {
this.jobs = jobs;
});
},
2019-02-05 23:24:59 -07:00
}
});
</script>
2019-03-12 02:11:06 -06:00
<style lang="stylus" scoped>
.wptihjuy
> .chart
min-height 200px !important
2019-03-14 22:14:50 -06:00
margin 0 -8px
2019-03-12 02:11:06 -06:00
2019-03-14 22:48:17 -06:00
.xvvuvgsv
> b
margin-right 16px
2019-03-12 02:11:06 -06:00
</style>