blob: d76f9fe14a154dd8c6be792a92c433f5dd908643 [file] [log] [blame]
<template>
<div>
<b-modal
id="generate-csr"
ref="modal"
size="lg"
no-stacking
:title="$t('pageCertificates.modal.generateACertificateSigningRequest')"
@ok="onOkGenerateCsrModal"
@cancel="resetForm"
@hidden="$v.$reset()"
>
<b-form id="generate-csr-form" novalidate @submit.prevent="handleSubmit">
<b-container fluid>
<b-row>
<b-col lg="9">
<b-row>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.certificateType')"
label-for="certificate-type"
>
<b-form-select
id="certificate-type"
v-model="form.certificateType"
data-test-id="modalGenerateCsr-select-certificateType"
:options="certificateOptions"
:state="getValidationState($v.form.certificateType)"
@input="$v.form.certificateType.$touch()"
>
<template #first>
<b-form-select-option :value="null" disabled>
{{ $t('global.form.selectAnOption') }}
</b-form-select-option>
</template>
</b-form-select>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.country')"
label-for="country"
>
<b-form-select
id="country"
v-model="form.country"
data-test-id="modalGenerateCsr-select-country"
:options="countryOptions"
:state="getValidationState($v.form.country)"
@input="$v.form.country.$touch()"
>
<template #first>
<b-form-select-option :value="null" disabled>
{{ $t('global.form.selectAnOption') }}
</b-form-select-option>
</template>
</b-form-select>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
</b-row>
<b-row>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.state')"
label-for="state"
>
<b-form-input
id="state"
v-model="form.state"
type="text"
data-test-id="modalGenerateCsr-input-state"
:state="getValidationState($v.form.state)"
/>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.city')"
label-for="city"
>
<b-form-input
id="city"
v-model="form.city"
type="text"
data-test-id="modalGenerateCsr-input-city"
:state="getValidationState($v.form.city)"
/>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
</b-row>
<b-row>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.companyName')"
label-for="company-name"
>
<b-form-input
id="company-name"
v-model="form.companyName"
type="text"
data-test-id="modalGenerateCsr-input-companyName"
:state="getValidationState($v.form.companyName)"
/>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.companyUnit')"
label-for="company-unit"
>
<b-form-input
id="company-unit"
v-model="form.companyUnit"
type="text"
data-test-id="modalGenerateCsr-input-companyUnit"
:state="getValidationState($v.form.companyUnit)"
/>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
</b-row>
<b-row>
<b-col lg="6">
<b-form-group
:label="$t('pageCertificates.modal.commonName')"
label-for="common-name"
>
<b-form-input
id="common-name"
v-model="form.commonName"
type="text"
data-test-id="modalGenerateCsr-input-commonName"
:state="getValidationState($v.form.commonName)"
/>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
<b-col lg="6">
<b-form-group label-for="challenge-password">
<template #label>
{{ $t('pageCertificates.modal.challengePassword') }} -
<span class="form-text d-inline">
{{ $t('global.form.optional') }}
</span>
</template>
<b-form-input
id="challenge-password"
v-model="form.challengePassword"
type="text"
data-test-id="modalGenerateCsr-input-challengePassword"
/>
</b-form-group>
</b-col>
</b-row>
<b-row>
<b-col lg="6">
<b-form-group label-for="contact-person">
<template #label>
{{ $t('pageCertificates.modal.contactPerson') }} -
<span class="form-text d-inline">
{{ $t('global.form.optional') }}
</span>
</template>
<b-form-input
id="contact-person"
v-model="form.contactPerson"
type="text"
data-test-id="modalGenerateCsr-input-contactPerson"
/>
</b-form-group>
</b-col>
<b-col lg="6">
<b-form-group label-for="email-address">
<template #label>
{{ $t('pageCertificates.modal.emailAddress') }} -
<span class="form-text d-inline">
{{ $t('global.form.optional') }}
</span>
</template>
<b-form-input
id="email-address"
v-model="form.emailAddress"
type="text"
data-test-id="modalGenerateCsr-input-emailAddress"
/>
</b-form-group>
</b-col>
</b-row>
<b-row>
<b-col lg="12">
<b-form-group label-for="alternate-name">
<template #label>
{{ $t('pageCertificates.modal.alternateName') }} -
<span class="form-text d-inline">
{{ $t('global.form.optional') }}
</span>
</template>
<b-form-text id="alternate-name-help-block">
{{ $t('pageCertificates.modal.alternateNameHelperText') }}
</b-form-text>
<b-form-tags
v-model="form.alternateName"
:remove-on-delete="true"
:tag-pills="true"
input-id="alternate-name"
size="lg"
separator=" "
:input-attrs="{
'aria-describedby': 'alternate-name-help-block',
}"
:duplicate-tag-text="
$t('pageCertificates.modal.duplicateAlternateName')
"
placeholder=""
data-test-id="modalGenerateCsr-input-alternateName"
>
<template #add-button-text>
<icon-add /> {{ $t('global.action.add') }}
</template>
</b-form-tags>
</b-form-group>
</b-col>
</b-row>
</b-col>
<b-col lg="3">
<b-row>
<b-col lg="12">
<p class="col-form-label">
{{ $t('pageCertificates.modal.privateKey') }}
</p>
<b-form-group
:label="$t('pageCertificates.modal.keyPairAlgorithm')"
label-for="key-pair-algorithm"
>
<b-form-select
id="key-pair-algorithm"
v-model="form.keyPairAlgorithm"
data-test-id="modalGenerateCsr-select-keyPairAlgorithm"
:options="keyPairAlgorithmOptions"
:state="getValidationState($v.form.keyPairAlgorithm)"
@input="$v.form.keyPairAlgorithm.$touch()"
>
<template #first>
<b-form-select-option :value="null" disabled>
{{ $t('global.form.selectAnOption') }}
</b-form-select-option>
</template>
</b-form-select>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</b-col>
</b-row>
<b-row>
<b-col lg="12">
<template v-if="$v.form.keyPairAlgorithm.$model === 'EC'">
<b-form-group
:label="$t('pageCertificates.modal.keyCurveId')"
label-for="key-curve-id"
>
<b-form-select
id="key-curve-id"
v-model="form.keyCurveId"
data-test-id="modalGenerateCsr-select-keyCurveId"
:options="keyCurveIdOptions"
:state="getValidationState($v.form.keyCurveId)"
@input="$v.form.keyCurveId.$touch()"
>
<template #first>
<b-form-select-option :value="null" disabled>
{{ $t('global.form.selectAnOption') }}
</b-form-select-option>
</template>
</b-form-select>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</template>
<template v-if="$v.form.keyPairAlgorithm.$model === 'RSA'">
<b-form-group
:label="$t('pageCertificates.modal.keyBitLength')"
label-for="key-bit-length"
>
<b-form-select
id="key-bit-length"
v-model="form.keyBitLength"
data-test-id="modalGenerateCsr-select-keyBitLength"
:options="keyBitLengthOptions"
:state="getValidationState($v.form.keyBitLength)"
@input="$v.form.keyBitLength.$touch()"
>
<template #first>
<b-form-select-option :value="null" disabled>
{{ $t('global.form.selectAnOption') }}
</b-form-select-option>
</template>
</b-form-select>
<b-form-invalid-feedback role="alert">
{{ $t('global.form.fieldRequired') }}
</b-form-invalid-feedback>
</b-form-group>
</template>
</b-col>
</b-row>
</b-col>
</b-row>
</b-container>
</b-form>
<template #modal-footer="{ ok, cancel }">
<b-button variant="secondary" @click="cancel()">
{{ $t('global.action.cancel') }}
</b-button>
<b-button
form="generate-csr-form"
type="submit"
variant="primary"
data-test-id="modalGenerateCsr-button-ok"
@click="ok()"
>
{{ $t('pageCertificates.generateCsr') }}
</b-button>
</template>
</b-modal>
<b-modal
id="csr-string"
no-stacking
size="lg"
:title="$t('pageCertificates.modal.certificateSigningRequest')"
@hidden="onHiddenCsrStringModal"
>
{{ csrString }}
<template #modal-footer>
<b-btn variant="secondary" @click="copyCsrString">
<template v-if="csrStringCopied">
<icon-checkmark />
{{ $t('global.status.copied') }}
</template>
<template v-else>
{{ $t('global.action.copy') }}
</template>
</b-btn>
<a
:href="`data:text/json;charset=utf-8,${csrString}`"
download="certificate.txt"
class="btn btn-primary"
>
{{ $t('global.action.download') }}
</a>
</template>
</b-modal>
</div>
</template>
<script>
import IconAdd from '@carbon/icons-vue/es/add--alt/20';
import IconCheckmark from '@carbon/icons-vue/es/checkmark/20';
import { required, requiredIf } from 'vuelidate/lib/validators';
import { COUNTRY_LIST } from './CsrCountryCodes';
import { CERTIFICATE_TYPES } from '@/store/modules/SecurityAndAccess/CertificatesStore';
import BVToastMixin from '@/components/Mixins/BVToastMixin';
import VuelidateMixin from '@/components/Mixins/VuelidateMixin.js';
export default {
name: 'ModalGenerateCsr',
components: { IconAdd, IconCheckmark },
mixins: [BVToastMixin, VuelidateMixin],
data() {
return {
form: {
certificateType: null,
country: null,
state: null,
city: null,
companyName: null,
companyUnit: null,
commonName: null,
challengePassword: null,
contactPerson: null,
emailAddress: null,
alternateName: [],
keyPairAlgorithm: null,
keyCurveId: null,
keyBitLength: null,
},
certificateOptions: CERTIFICATE_TYPES.reduce((arr, cert) => {
if (cert.type === 'TrustStore Certificate') return arr;
arr.push({
text: cert.label,
value: cert.type,
});
return arr;
}, []),
countryOptions: COUNTRY_LIST.map((country) => ({
text: country.label,
value: country.code,
})),
keyPairAlgorithmOptions: ['EC', 'RSA'],
keyCurveIdOptions: ['prime256v1', 'secp521r1', 'secp384r1'],
keyBitLengthOptions: [2048],
csrString: '',
csrStringCopied: false,
};
},
validations: {
form: {
certificateType: { required },
country: { required },
state: { required },
city: { required },
companyName: { required },
companyUnit: { required },
commonName: { required },
challengePassword: {},
contactPerson: {},
emailAddress: {},
alternateName: {},
keyPairAlgorithm: { required },
keyCurveId: {
reuired: requiredIf(function (form) {
return form.keyPairAlgorithm === 'EC';
}),
},
keyBitLength: {
reuired: requiredIf(function (form) {
return form.keyPairAlgorithm === 'RSA';
}),
},
},
},
methods: {
handleSubmit() {
this.$v.$touch();
if (this.$v.$invalid) return;
this.$store
.dispatch('certificates/generateCsr', this.form)
.then(({ data: { CSRString } }) => {
this.csrString = CSRString;
this.$bvModal.show('csr-string');
this.$v.$reset();
});
},
resetForm() {
for (let key of Object.keys(this.form)) {
if (key === 'alternateName') {
this.form[key] = [];
} else {
this.form[key] = null;
}
}
},
onOkGenerateCsrModal(bvModalEvt) {
// prevent modal close
bvModalEvt.preventDefault();
this.handleSubmit();
},
onHiddenCsrStringModal() {
this.csrString = '';
this.resetForm();
},
copyCsrString(bvModalEvt) {
// prevent modal close
bvModalEvt.preventDefault();
navigator.clipboard.writeText(this.csrString).then(() => {
// Show copied text for 5 seconds
this.csrStringCopied = true;
setTimeout(() => {
this.csrStringCopied = false;
}, 5000 /*5 seconds*/);
});
},
},
};
</script>