blob: a55b66ccdba3206aa6a719f7b11ea84c21090603 [file] [log] [blame]
<template>
<b-container fluid="xl">
<page-title />
<b-row>
<b-col xl="11">
<!-- Expired certificates banner -->
<alert :show="expiredCertificateTypes.length > 0" variant="danger">
<template v-if="expiredCertificateTypes.length > 1">
{{ $t('pageCertificates.alert.certificatesExpiredMessage') }}
</template>
<template v-else>
{{
$t('pageCertificates.alert.certificateExpiredMessage', {
certificate: expiredCertificateTypes[0],
})
}}
</template>
</alert>
<!-- Expiring certificates banner -->
<alert :show="expiringCertificateTypes.length > 0" variant="warning">
<template v-if="expiringCertificateTypes.length > 1">
{{ $t('pageCertificates.alert.certificatesExpiringMessage') }}
</template>
<template v-else>
{{
$t('pageCertificates.alert.certificateExpiringMessage', {
certificate: expiringCertificateTypes[0],
})
}}
</template>
</alert>
</b-col>
</b-row>
<b-row>
<b-col xl="11" class="text-right">
<b-button
v-b-modal.generate-csr
data-test-id="certificates-button-generateCsr"
variant="link"
>
<icon-add />
{{ $t('pageCertificates.generateCsr') }}
</b-button>
<b-button
variant="primary"
:disabled="certificatesForUpload.length === 0"
@click="initModalUploadCertificate(null)"
>
<icon-add />
{{ $t('pageCertificates.addNewCertificate') }}
</b-button>
</b-col>
</b-row>
<b-row>
<b-col xl="11">
<b-table
responsive="md"
show-empty
hover
:busy="isBusy"
:fields="fields"
:items="tableItems"
:empty-text="$t('global.table.emptyMessage')"
>
<template #cell(validFrom)="{ value }">
{{ $filters.formatDate(value) }}
</template>
<template #cell(validUntil)="{ value }">
<status-icon
v-if="getDaysUntilExpired(value) < 31"
:status="getIconStatus(value)"
/>
{{ $filters.formatDate(value) }}
</template>
<template #cell(actions)="{ value, item }">
<table-row-action
v-for="(action, index) in value"
:key="index"
:value="action.value"
:title="action.title"
:enabled="action.enabled"
@click-table-action="onTableRowAction($event, item)"
>
<template #icon>
<icon-replace v-if="action.value === 'replace'" />
<icon-trashcan v-if="action.value === 'delete'" />
</template>
</table-row-action>
</template>
</b-table>
</b-col>
</b-row>
<!-- Modals -->
<modal-upload-certificate :certificate="modalCertificate" @ok="onModalOk" />
<modal-generate-csr />
</b-container>
</template>
<script>
import IconAdd from '@carbon/icons-vue/es/add--alt/20';
import IconReplace from '@carbon/icons-vue/es/renew/20';
import IconTrashcan from '@carbon/icons-vue/es/trash-can/20';
import ModalGenerateCsr from './ModalGenerateCsr';
import ModalUploadCertificate from './ModalUploadCertificate';
import PageTitle from '@/components/Global/PageTitle';
import TableRowAction from '@/components/Global/TableRowAction';
import StatusIcon from '@/components/Global/StatusIcon';
import Alert from '@/components/Global/Alert';
import BVToastMixin from '@/components/Mixins/BVToastMixin';
import LoadingBarMixin from '@/components/Mixins/LoadingBarMixin';
import { useI18n } from 'vue-i18n';
import i18n from '@/i18n';
export default {
name: 'Certificates',
components: {
Alert,
IconAdd,
IconReplace,
IconTrashcan,
ModalGenerateCsr,
ModalUploadCertificate,
PageTitle,
StatusIcon,
TableRowAction,
},
mixins: [BVToastMixin, LoadingBarMixin],
beforeRouteLeave(to, from, next) {
this.hideLoader();
next();
},
data() {
return {
$t: useI18n().t,
isBusy: true,
modalCertificate: null,
fileTypeCorrect: undefined,
fields: [
{
key: 'certificate',
label: i18n.global.t('pageCertificates.table.certificate'),
},
{
key: 'issuedBy',
label: i18n.global.t('pageCertificates.table.issuedBy'),
},
{
key: 'issuedTo',
label: i18n.global.t('pageCertificates.table.issuedTo'),
},
{
key: 'validFrom',
label: i18n.global.t('pageCertificates.table.validFrom'),
},
{
key: 'validUntil',
label: i18n.global.t('pageCertificates.table.validUntil'),
},
{
key: 'actions',
label: '',
tdClass: 'text-right text-nowrap',
},
],
};
},
computed: {
certificates() {
return this.$store.getters['certificates/allCertificates'];
},
tableItems() {
return this.certificates.map((certificate) => {
return {
...certificate,
actions: [
{
value: 'replace',
title: i18n.global.t('pageCertificates.replaceCertificate'),
},
{
value: 'delete',
title: i18n.global.t('pageCertificates.deleteCertificate'),
enabled:
certificate.type === 'TrustStore Certificate' ? true : false,
},
],
};
});
},
certificatesForUpload() {
return this.$store.getters['certificates/availableUploadTypes'];
},
bmcTime() {
return this.$store.getters['global/bmcTime'];
},
expiredCertificateTypes() {
return this.certificates.reduce((acc, val) => {
const daysUntilExpired = this.getDaysUntilExpired(val.validUntil);
if (daysUntilExpired < 1) {
acc.push(val.certificate);
}
return acc;
}, []);
},
expiringCertificateTypes() {
return this.certificates.reduce((acc, val) => {
const daysUntilExpired = this.getDaysUntilExpired(val.validUntil);
if (daysUntilExpired < 31 && daysUntilExpired > 0) {
acc.push(val.certificate);
}
return acc;
}, []);
},
},
async created() {
this.startLoader();
await this.$store.dispatch('global/getBmcTime');
this.$store.dispatch('certificates/getCertificates').finally(() => {
this.endLoader();
this.isBusy = false;
});
},
methods: {
onTableRowAction(event, rowItem) {
switch (event) {
case 'replace':
this.initModalUploadCertificate(rowItem);
break;
case 'delete':
this.initModalDeleteCertificate(rowItem);
break;
default:
break;
}
},
initModalUploadCertificate(certificate = null) {
this.modalCertificate = certificate;
this.$bvModal.show('upload-certificate');
},
initModalDeleteCertificate(certificate) {
this.$bvModal
.msgBoxConfirm(
i18n.global.t('pageCertificates.modal.deleteConfirmMessage', {
issuedBy: certificate.issuedBy,
certificate: certificate.certificate,
}),
{
title: i18n.global.t('pageCertificates.deleteCertificate'),
okTitle: i18n.global.t('global.action.delete'),
cancelTitle: i18n.global.t('global.action.cancel'),
autoFocusButton: 'ok',
},
)
.then((deleteConfirmed) => {
if (deleteConfirmed) this.deleteCertificate(certificate);
});
},
onModalOk({ addNew, file, type, location }) {
if (addNew) {
// Upload a new certificate
this.fileTypeCorrect = this.getIsFileTypeCorrect(file);
if (this.fileTypeCorrect) {
this.addNewCertificate(file, type);
} else {
this.errorToast(
i18n.global.t(
'pageCertificates.alert.incorrectCertificateFileType',
),
{
title: i18n.global.t(
'pageCertificates.toast.errorAddCertificate',
),
},
);
}
} else {
// Replace an existing certificate
this.replaceCertificate(file, type, location);
}
},
addNewCertificate(file, type) {
if (this.fileTypeCorrect === true) {
this.startLoader();
this.$store
.dispatch('certificates/addNewCertificate', { file, type })
.then((success) => this.successToast(success))
.catch(({ message }) => this.errorToast(message))
.finally(() => this.endLoader());
}
},
replaceCertificate(file, type, location) {
this.startLoader();
const reader = new FileReader();
reader.readAsBinaryString(file);
reader.onloadend = (event) => {
const certificateString = event.target.result;
this.$store
.dispatch('certificates/replaceCertificate', {
certificateString,
type,
location,
})
.then((success) => this.successToast(success))
.catch(({ message }) => this.errorToast(message))
.finally(() => this.endLoader());
};
},
deleteCertificate({ type, location }) {
this.startLoader();
this.$store
.dispatch('certificates/deleteCertificate', {
type,
location,
})
.then((success) => this.successToast(success))
.catch(({ message }) => this.errorToast(message))
.finally(() => this.endLoader());
},
getDaysUntilExpired(date) {
if (this.bmcTime) {
const validUntilMs = date.getTime();
const currentBmcTimeMs = this.bmcTime.getTime();
const oneDayInMs = 24 * 60 * 60 * 1000;
return Math.round((validUntilMs - currentBmcTimeMs) / oneDayInMs);
}
return new Date();
},
getIconStatus(date) {
const daysUntilExpired = this.getDaysUntilExpired(date);
if (daysUntilExpired < 1) {
return 'danger';
} else if (daysUntilExpired < 31) {
return 'warning';
}
},
getIsFileTypeCorrect(file) {
const fileTypeExtension = file.name.split('.').pop();
return fileTypeExtension === 'pem';
},
},
};
</script>