mirror of
https://github.com/harvester/harvester-ui-extension.git
synced 2025-12-13 21:21:44 +00:00
370 lines
9.2 KiB
Vue
370 lines
9.2 KiB
Vue
<script>
|
|
import LabeledSelect from '@shell/components/form/LabeledSelect.vue';
|
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
|
import { RadioGroup } from '@components/Form/Radio';
|
|
import { Checkbox } from '@components/Form/Checkbox';
|
|
import { SECRET } from '@shell/config/types';
|
|
import ModalWithCard from '@shell/components/ModalWithCard';
|
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
|
import { Banner } from '@components/Banner';
|
|
|
|
import { base64Encode, base64Decode } from '@shell/utils/crypto';
|
|
import { exceptionToErrorsArray } from '@shell/utils/error';
|
|
|
|
const _NEW = '_NEW';
|
|
|
|
export default {
|
|
name: 'HarvesterSeeder',
|
|
|
|
components: {
|
|
Checkbox,
|
|
LabeledInput,
|
|
LabeledSelect,
|
|
RadioGroup,
|
|
ModalWithCard,
|
|
NameNsDescription,
|
|
Banner,
|
|
},
|
|
|
|
props: {
|
|
mode: {
|
|
type: String,
|
|
required: true
|
|
},
|
|
|
|
node: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
|
|
registerAfterHook: {
|
|
type: Function,
|
|
required: true,
|
|
},
|
|
|
|
inventory: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
},
|
|
|
|
data() {
|
|
const enableInventory = !!this.inventory?.id;
|
|
|
|
return {
|
|
enableInventory,
|
|
value: this.inventory,
|
|
secret: {},
|
|
errors: [],
|
|
newSecretSelected: false,
|
|
isOpen: false,
|
|
};
|
|
},
|
|
|
|
created() {
|
|
this.registerAfterHook(this.saveInventory, 'saveInventory');
|
|
},
|
|
|
|
async fetch() {
|
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
|
|
|
this.secret = await this.$store.dispatch(`${ inStore }/create`, {
|
|
type: SECRET,
|
|
data: {
|
|
username: '',
|
|
password: '',
|
|
},
|
|
metadata: {
|
|
namespace: '',
|
|
name: '',
|
|
describe: '',
|
|
},
|
|
});
|
|
},
|
|
|
|
computed: {
|
|
secretOption() {
|
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
|
|
|
const out = this.$store.getters[`${ inStore }/all`](SECRET).filter((s) => {
|
|
return s.data?.username && s.data?.password;
|
|
}).map( (s) => {
|
|
return {
|
|
label: s.id,
|
|
value: s.id
|
|
};
|
|
});
|
|
|
|
// if (!(this.disableCreate || this.mode === _VIEW) && this.isCreatable) {
|
|
out.unshift({
|
|
label: this.t('harvester.virtualMachine.createSSHKey'),
|
|
value: _NEW,
|
|
});
|
|
// }
|
|
|
|
return out;
|
|
},
|
|
|
|
selectedSecret: {
|
|
get() {
|
|
const namespace = this.value.spec?.baseboardSpec?.connection?.authSecretRef?.namespace;
|
|
const name = this.value?.spec?.baseboardSpec?.connection?.authSecretRef?.name;
|
|
|
|
if (namespace && name) {
|
|
return `${ namespace }/${ name }`;
|
|
} else {
|
|
return '';
|
|
}
|
|
},
|
|
|
|
set(value) {
|
|
if (value === _NEW) {
|
|
this.newSecretSelected = true;
|
|
} else {
|
|
const [namespace, name] = value.split('/');
|
|
|
|
this.value.spec.baseboardSpec.connection.authSecretRef['namespace'] = namespace;
|
|
this.value.spec.baseboardSpec.connection.authSecretRef['name'] = name;
|
|
}
|
|
},
|
|
},
|
|
|
|
username: {
|
|
get() {
|
|
return base64Decode(this.secret?.data?.username);
|
|
},
|
|
|
|
set(value) {
|
|
this.secret.data['username'] = base64Encode(value);
|
|
}
|
|
},
|
|
|
|
password: {
|
|
get() {
|
|
return base64Decode(this.secret?.data?.password);
|
|
},
|
|
|
|
set(value) {
|
|
this.secret.data['password'] = base64Encode(value);
|
|
}
|
|
},
|
|
},
|
|
|
|
methods: {
|
|
async saveInventory() {
|
|
if (this.enableInventory) {
|
|
const errors = [];
|
|
|
|
if (!this.value.spec.baseboardSpec.connection.host) {
|
|
errors.push(this.t('validation.required', { key: this.t('harvester.seeder.inventory.host.label') }, true));
|
|
}
|
|
|
|
if (!this.value.spec.baseboardSpec.connection.port) {
|
|
errors.push(this.t('validation.required', { key: this.t('harvester.seeder.inventory.port.label') }, true));
|
|
}
|
|
|
|
if (!this.selectedSecret) {
|
|
errors.push(this.t('validation.required', { key: this.t('harvester.seeder.inventory.secret.label') }, true));
|
|
}
|
|
|
|
if (errors.length > 0) {
|
|
return Promise.reject(exceptionToErrorsArray(errors));
|
|
}
|
|
|
|
if (!this.value.id) {
|
|
this.value.metadata.annotations['metal.harvesterhci.io/local-node-name'] = this.node.id;
|
|
}
|
|
|
|
this.value.metadata.annotations['metal.harvesterhci.io/local-inventory'] = 'true';
|
|
|
|
return await this.value.save();
|
|
} else if (this.value.id) {
|
|
return await this.value.remove();
|
|
} else {
|
|
return Promise.resolve();
|
|
}
|
|
},
|
|
|
|
show() {
|
|
this.isOpen = true;
|
|
},
|
|
|
|
hide() {
|
|
this.isOpen = false;
|
|
this.newSecretSelected = false;
|
|
},
|
|
|
|
cancel() {
|
|
this.hide();
|
|
},
|
|
|
|
async saveSecret(buttonCb) {
|
|
this.errors = [];
|
|
|
|
if (!this.username) {
|
|
this.errors.push(this.t('validation.required', { key: this.t('harvester.virtualMachine.input.username') }, true));
|
|
}
|
|
|
|
if (!this.password) {
|
|
this.errors.push(this.t('validation.required', { key: this.t('harvester.virtualMachine.input.password') }, true));
|
|
}
|
|
|
|
if (this.errors.length > 0) {
|
|
buttonCb(false);
|
|
|
|
return;
|
|
}
|
|
|
|
try {
|
|
const res = await this.secret.save();
|
|
|
|
if (res.id) {
|
|
this.secretOption.push({
|
|
label: res.id,
|
|
value: res.id
|
|
});
|
|
}
|
|
|
|
this.selectedSecret = res.id;
|
|
|
|
buttonCb(true);
|
|
this.cancel();
|
|
} catch (err) {
|
|
this.errors = [err.message];
|
|
buttonCb(false);
|
|
}
|
|
}
|
|
},
|
|
|
|
watch: {
|
|
newSecretSelected(val) {
|
|
if (val) {
|
|
this.show();
|
|
}
|
|
}
|
|
}
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div>
|
|
<div v-if="inventory.warningMessages.length > 0">
|
|
<Banner
|
|
v-for="(msg, i) in inventory.warningMessages" :key="i" color="error"
|
|
:label="msg.text"
|
|
/>
|
|
</div>
|
|
<div class="row">
|
|
<div class="col span-6">
|
|
<RadioGroup
|
|
v-model:value="enableInventory"
|
|
:options="[
|
|
{ label: t('generic.enabled'), value: true },
|
|
{ label: t('generic.disabled'), value: false }
|
|
]"
|
|
:mode="mode"
|
|
name="enableInventory"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div v-if="enableInventory">
|
|
<div class="row mt-10">
|
|
<div class="col span-6">
|
|
<LabeledInput
|
|
v-model:value="value.spec.baseboardSpec.connection.host"
|
|
:label="t('harvester.seeder.inventory.host.label')"
|
|
:placeholder="t('harvester.seeder.inventory.host.placeholder')"
|
|
:mode="mode"
|
|
required
|
|
/>
|
|
<Checkbox
|
|
v-model:value="value.spec.baseboardSpec.connection.insecureTLS"
|
|
class="mt-5"
|
|
:mode="mode"
|
|
:label="t('harvester.seeder.inventory.insecureTLS.label')"
|
|
/>
|
|
</div>
|
|
<div class="col span-6">
|
|
<LabeledInput
|
|
v-model.number="value.spec.baseboardSpec.connection.port"
|
|
:label="t('harvester.seeder.inventory.port.label')"
|
|
:placeholder="t('harvester.seeder.inventory.port.placeholder')"
|
|
:mode="mode"
|
|
required
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div class="row mt-20">
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="selectedSecret"
|
|
:label="t('harvester.seeder.inventory.secret.label')"
|
|
:mode="mode"
|
|
:options="secretOption"
|
|
required
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div class="row mt-20">
|
|
<div class="col span-6">
|
|
<RadioGroup
|
|
v-model:value="value.spec.events.enabled"
|
|
name="enabled"
|
|
:options="[true, false]"
|
|
:label="t('harvester.seeder.inventory.event.label')"
|
|
:labels="[t('generic.enabled'), t('generic.disabled')]"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
<div
|
|
v-if="value.spec.events.enabled"
|
|
class="col span-6"
|
|
>
|
|
<LabeledInput
|
|
v-model:value="value.spec.events.pollingInterval"
|
|
:label="t('harvester.seeder.inventory.pollingInterval.label')"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<ModalWithCard
|
|
v-if="isOpen"
|
|
width="80%"
|
|
:errors="errors"
|
|
name="secretModal"
|
|
@finish="saveSecret"
|
|
@close="cancel"
|
|
>
|
|
<template #title>
|
|
{{ t('harvester.seeder.inventory.secret.create.title') }}
|
|
</template>
|
|
|
|
<template #content>
|
|
<NameNsDescription
|
|
:value="value"
|
|
@update:value="$emit('update:value', $event)"
|
|
:namespaced="true"
|
|
mode="create"
|
|
/>
|
|
|
|
<LabeledInput
|
|
v-model:value="username"
|
|
:label="t('harvester.virtualMachine.input.username')"
|
|
class="mb-20"
|
|
required
|
|
/>
|
|
|
|
<LabeledInput
|
|
v-model:value="password"
|
|
type="password"
|
|
:label="t('harvester.virtualMachine.input.password')"
|
|
class="mb-20"
|
|
required
|
|
/>
|
|
</template>
|
|
</ModalWithCard>
|
|
</div>
|
|
</div>
|
|
</template>
|