mirror of
https://github.com/harvester/harvester-ui-extension.git
synced 2025-12-13 21:21:44 +00:00
1027 lines
28 KiB
Vue
1027 lines
28 KiB
Vue
<script>
|
|
import { isEqual } from 'lodash';
|
|
import { mapGetters } from 'vuex';
|
|
import Tabbed from '@shell/components/Tabbed';
|
|
import { clone, set } from '@shell/utils/object';
|
|
import Tab from '@shell/components/Tabbed/Tab';
|
|
import { Checkbox } from '@components/Form/Checkbox';
|
|
import CruResource from '@shell/components/CruResource';
|
|
import { RadioGroup } from '@components/Form/Radio';
|
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
|
import UnitInput from '@shell/components/form/UnitInput';
|
|
import NodeScheduling from '@shell/components/form/NodeScheduling';
|
|
import PodAffinity from '@shell/components/form/PodAffinity';
|
|
import VGpuDevices from './VirtualMachineVGpuDevices/index';
|
|
import UsbDevices from './VirtualMachineUSBDevices/index';
|
|
import KeyValue from '@shell/components/form/KeyValue';
|
|
import { clear } from '@shell/utils/array';
|
|
import { saferDump } from '@shell/utils/create-yaml';
|
|
import { exceptionToErrorsArray } from '@shell/utils/error';
|
|
import { HCI as HCI_ANNOTATIONS } from '@pkg/harvester/config/labels-annotations';
|
|
import { BEFORE_SAVE_HOOKS, AFTER_SAVE_HOOKS } from '@shell/mixins/child-hook';
|
|
|
|
import CreateEditView from '@shell/mixins/create-edit-view';
|
|
|
|
import { parseVolumeClaimTemplates } from '@pkg/utils/vm';
|
|
import VM_MIXIN from '../../mixins/harvester-vm';
|
|
import { HCI } from '../../types';
|
|
import RestartVMDialog from '../../dialog/RestartVMDialog';
|
|
import PciDevices from './VirtualMachinePciDevices/index';
|
|
import AccessCredentials from './VirtualMachineAccessCredentials';
|
|
import CloudConfig from './VirtualMachineCloudConfig';
|
|
import CpuMemory from './VirtualMachineCpuMemory';
|
|
import Network from './VirtualMachineNetwork';
|
|
import Volume from './VirtualMachineVolume';
|
|
import SSHKey from './VirtualMachineSSHKey';
|
|
import Reserved from './VirtualMachineReserved';
|
|
import { Banner } from '@components/Banner';
|
|
import MessageLink from '@shell/components/MessageLink';
|
|
|
|
export default {
|
|
name: 'HarvesterEditVM',
|
|
|
|
emits: ['update:value'],
|
|
|
|
components: {
|
|
Tab,
|
|
Tabbed,
|
|
Checkbox,
|
|
RadioGroup,
|
|
CruResource,
|
|
LabeledInput,
|
|
LabeledSelect,
|
|
NameNsDescription,
|
|
Volume,
|
|
SSHKey,
|
|
Network,
|
|
CpuMemory,
|
|
CloudConfig,
|
|
NodeScheduling,
|
|
PodAffinity,
|
|
AccessCredentials,
|
|
Reserved,
|
|
PciDevices,
|
|
RestartVMDialog,
|
|
UnitInput,
|
|
VGpuDevices,
|
|
KeyValue,
|
|
Banner,
|
|
MessageLink,
|
|
UsbDevices,
|
|
},
|
|
|
|
mixins: [CreateEditView, VM_MIXIN],
|
|
|
|
inheritAttrs: false,
|
|
|
|
props: {
|
|
value: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
},
|
|
|
|
data() {
|
|
const cloneVM = clone(this.value);
|
|
const isRestartImmediately = this.value.actualState === 'Running';
|
|
|
|
const hostname = this.value.spec.template.spec.hostname || '';
|
|
|
|
return {
|
|
cloneVM,
|
|
count: 2,
|
|
templateId: '',
|
|
templateVersionId: '',
|
|
namePrefix: '',
|
|
isSingle: true,
|
|
useTemplate: false,
|
|
isOpen: false,
|
|
hostname,
|
|
isRestartImmediately,
|
|
};
|
|
},
|
|
|
|
computed: {
|
|
...mapGetters({ t: 'i18n/t' }),
|
|
|
|
to() {
|
|
return {
|
|
name: 'harvester-c-cluster-resource',
|
|
params: { cluster: this.$store.getters['clusterId'], resource: HCI.HOST },
|
|
};
|
|
},
|
|
|
|
machineTypeOptions() {
|
|
return this.machineTypes.map((type) => {
|
|
if (!type) return { label: 'None', value: '' };
|
|
|
|
return { label: type, value: type };
|
|
});
|
|
},
|
|
|
|
templateOptions() {
|
|
return this.templates.map( (T) => {
|
|
const isReady = this.hasAvailableVersion(T.id);
|
|
|
|
return {
|
|
label: T.id,
|
|
value: T.id,
|
|
disabled: !isReady
|
|
};
|
|
});
|
|
},
|
|
|
|
versionOptions() {
|
|
const defaultVersion = this.curTemplateResource?.defaultVersion;
|
|
|
|
return this.versions.filter( (O) => O.templateId === this.templateId).map( (T) => {
|
|
const version = T.version;
|
|
|
|
const label = defaultVersion === version ? `${ version } (${ this.t('generic.default') })` : version;
|
|
const value = T.id;
|
|
|
|
return {
|
|
label,
|
|
value,
|
|
disabled: !T.isReady
|
|
};
|
|
});
|
|
},
|
|
|
|
curTemplateResource() {
|
|
return this.templates.find( (O) => O.id === this.templateId);
|
|
},
|
|
|
|
nameLabel() {
|
|
return this.isSingle ? 'harvester.virtualMachine.instance.single.nameLabel' : 'harvester.virtualMachine.instance.multiple.nameLabel';
|
|
},
|
|
|
|
hostnameLabel() {
|
|
return this.isSingle ? 'harvester.virtualMachine.instance.single.host.label' : 'harvester.virtualMachine.instance.multiple.host.label';
|
|
},
|
|
|
|
hostPlaceholder() {
|
|
return this.isSingle ? this.t('harvester.virtualMachine.instance.single.host.placeholder') : this.t('harvester.virtualMachine.instance.multiple.host.placeholder');
|
|
},
|
|
|
|
secretNamePrefix() {
|
|
return this.value?.metadata?.name;
|
|
},
|
|
|
|
isQemuInstalled() {
|
|
return this.value.isQemuInstalled;
|
|
},
|
|
|
|
hasRestartAction() {
|
|
return this.value.hasAction('restart');
|
|
},
|
|
|
|
hasStartAction() {
|
|
return this.value.hasAction('start');
|
|
},
|
|
|
|
enableCpuPinningCheckbox() {
|
|
if (this.mode === 'create') {
|
|
return this.nodes.some((node) => node.isCPUManagerEnabled); // any one of nodes has label cpuManager=true
|
|
}
|
|
|
|
return true;
|
|
},
|
|
isValidationPassed() {
|
|
// check if any disk hasDiskError is true
|
|
const hasError = this.diskRows.some((disk) => disk.hasDiskError === true);
|
|
|
|
return !hasError;
|
|
},
|
|
showCpuPinningBanner() {
|
|
if (!this.value.cpuPinningFeatureEnabled) {
|
|
return false;
|
|
}
|
|
|
|
if (this.mode === 'edit') {
|
|
return this.cpuPinning !== !!this.cloneVM.spec.template.spec.domain.cpu.dedicatedCpuPlacement;
|
|
}
|
|
|
|
if (this.mode === 'create') {
|
|
return this.nodes.every((node) => !node.isCPUManagerEnabled); // no node enabled CPU manager
|
|
}
|
|
|
|
return false;
|
|
},
|
|
usbPassthroughEnabled() {
|
|
return this.$store.getters['harvester-common/getFeatureEnabled']('usbPassthrough');
|
|
},
|
|
},
|
|
|
|
watch: {
|
|
templateId: {
|
|
async handler(id, old) {
|
|
if (!id) {
|
|
return;
|
|
}
|
|
if (id !== old && !this.templateVersionId) {
|
|
const templates = await this.$store.dispatch('harvester/findAll', { type: HCI.VM_TEMPLATE });
|
|
|
|
this.templateVersionId = templates.find( (O) => O.id === id)?.spec?.defaultVersionId;
|
|
}
|
|
},
|
|
immediate: false
|
|
},
|
|
|
|
templateVersionId: {
|
|
async handler(id) {
|
|
if (!id) {
|
|
return;
|
|
}
|
|
const versions = await this.$store.dispatch('harvester/findAll', { type: HCI.VM_VERSION });
|
|
const curVersion = versions.find( (V) => V.id === id);
|
|
|
|
if (!curVersion?.spec?.vm) return;
|
|
|
|
const cloneVersionVM = clone(curVersion.spec.vm);
|
|
|
|
delete cloneVersionVM.spec?.template?.spec?.accessCredentials;
|
|
delete cloneVersionVM.spec?.template?.metadata?.annotations?.[HCI_ANNOTATIONS.DYNAMIC_SSHKEYS_NAMES];
|
|
delete cloneVersionVM.spec?.template?.metadata?.annotations?.[HCI_ANNOTATIONS.DYNAMIC_SSHKEYS_USERS];
|
|
|
|
const claimTemplate = parseVolumeClaimTemplates(cloneVersionVM);
|
|
const deleteDataSource = claimTemplate.map((volume) => {
|
|
if (volume?.spec?.dataSource) {
|
|
delete volume.spec.dataSource;
|
|
}
|
|
|
|
return volume;
|
|
});
|
|
|
|
cloneVersionVM.metadata.annotations[HCI_ANNOTATIONS.VOLUME_CLAIM_TEMPLATE] = JSON.stringify(deleteDataSource);
|
|
|
|
// Update instance labels, labels and annotations
|
|
this.value.spec.template.metadata.labels = cloneVersionVM.spec.template.metadata.labels || {};
|
|
this.value.metadata.labels = {
|
|
...(curVersion.metadata?.labels || {}),
|
|
...(cloneVersionVM.metadata?.labels || {})
|
|
};
|
|
this.value.metadata.annotations = {
|
|
...(curVersion.metadata?.annotations || {}),
|
|
...(cloneVersionVM.metadata?.annotations || {})
|
|
};
|
|
|
|
this.getInitConfig({
|
|
value: cloneVersionVM, existUserData: true, fromTemplate: true
|
|
});
|
|
this['hasCreateVolumes'] = []; // When using the template, all volume names need to be newly created
|
|
}
|
|
},
|
|
|
|
useTemplate(neu) {
|
|
if (neu === false) {
|
|
this.templateId = '';
|
|
this.templateVersionId = '';
|
|
this.value.applyDefaults();
|
|
this.getInitConfig({ value: this.value, init: this.isCreate });
|
|
}
|
|
},
|
|
},
|
|
|
|
created() {
|
|
this.registerAfterHook(async() => {
|
|
await this.restartVM();
|
|
const id = `${ this.value.metadata.namespace }/${ this.value.metadata.name }`;
|
|
|
|
const res = this.$store.getters['harvester/byId'](HCI.VM, id);
|
|
|
|
try {
|
|
await this.saveSecret(res);
|
|
await this.saveAccessCredentials(res);
|
|
} catch (e) {
|
|
this.errors.push(...exceptionToErrorsArray(e));
|
|
}
|
|
|
|
if (!this.errors.length && this.isSingle) {
|
|
this.done();
|
|
}
|
|
});
|
|
|
|
if (this.registerBeforeHook) {
|
|
this.registerBeforeHook(this.updateBeforeSave);
|
|
}
|
|
},
|
|
|
|
mounted() {
|
|
this.imageId = this.$route.query?.image || this.imageId;
|
|
|
|
const diskRows = this.getDiskRows(this.value);
|
|
|
|
this['diskRows'] = diskRows;
|
|
const templateId = this.$route.query.templateId;
|
|
const templateVersionId = this.$route.query.versionId;
|
|
|
|
if (templateId && templateVersionId) {
|
|
this.templateId = templateId;
|
|
this.templateVersionId = templateVersionId;
|
|
this.useTemplate = true;
|
|
}
|
|
},
|
|
|
|
methods: {
|
|
cancelAction() {
|
|
const { fromPage = HCI.VM } = this.$route?.query; // default back to VM list page
|
|
const cancelOverride = {
|
|
name: this.doneRoute,
|
|
params: { resource: fromPage }
|
|
};
|
|
|
|
this.$router.replace(cancelOverride);
|
|
},
|
|
saveVM(buttonCb) {
|
|
clear(this.errors);
|
|
|
|
if (this.isSingle) {
|
|
this.saveSingle(buttonCb);
|
|
} else {
|
|
this.saveMultiple(buttonCb);
|
|
}
|
|
},
|
|
|
|
async saveSingle(buttonCb) {
|
|
this.parseVM();
|
|
this.value.spec.template.spec.hostname = this.hostname ? this.hostname : this.value.metadata.name;
|
|
await this._save(this.value, buttonCb);
|
|
if (!this.errors.length) {
|
|
buttonCb(true);
|
|
} else {
|
|
buttonCb(false);
|
|
}
|
|
},
|
|
|
|
async saveMultiple(buttonCb) {
|
|
this.namePrefix = this.value.metadata.name || '';
|
|
const join = this.namePrefix.endsWith('-') ? '' : '-';
|
|
const baseHostname = this.hostname ? this.hostname : this.namePrefix;
|
|
|
|
if (this.count < 1) {
|
|
this.errors = [this.t('harvester.virtualMachine.instance.multiple.countTip')];
|
|
buttonCb(false);
|
|
|
|
return;
|
|
}
|
|
|
|
if (!this.value.metadata.name) {
|
|
this.errors.push(this.t('validation.required', { key: this.t('generic.name') }, true));
|
|
buttonCb(false);
|
|
|
|
return;
|
|
}
|
|
|
|
const cloneValue = clone(this.value);
|
|
const cloneSpec = clone(this.spec);
|
|
|
|
for (let i = 1; i <= this.count; i++) {
|
|
this.value['spec'] = cloneValue.spec;
|
|
this['spec'] = cloneSpec;
|
|
const suffix = i < 10 ? `0${ i }` : i;
|
|
|
|
this.value.cleanForNew();
|
|
this.value.metadata.name = `${ this.namePrefix }${ join }${ suffix }`;
|
|
this.value.spec.template.spec['hostname'] = `${ baseHostname }${ join }${ suffix }`;
|
|
this.secretName = '';
|
|
await this.parseVM();
|
|
const basicValue = await this.$store.dispatch('harvester/clone', { resource: this.value });
|
|
|
|
await this._save(basicValue, buttonCb);
|
|
|
|
if (i === this.count && !this.errors.length) {
|
|
buttonCb(true);
|
|
this.done();
|
|
} else if (i === this.count) {
|
|
this.value.metadata.name = this.namePrefix;
|
|
buttonCb(false);
|
|
}
|
|
}
|
|
},
|
|
|
|
async _save(value, buttonCb) {
|
|
try {
|
|
await this.applyHooks(BEFORE_SAVE_HOOKS);
|
|
await value.save();
|
|
await this.applyHooks(AFTER_SAVE_HOOKS);
|
|
} catch (e) {
|
|
this.errors.push(...exceptionToErrorsArray(e));
|
|
buttonCb(false);
|
|
}
|
|
},
|
|
|
|
restartVM() {
|
|
if (this.mode !== 'edit') {
|
|
return;
|
|
}
|
|
if (!this.value.isRunning) {
|
|
return;
|
|
}
|
|
const cloneDeepNewVM = clone(this.value);
|
|
|
|
// new VM
|
|
delete cloneDeepNewVM?.metadata;
|
|
delete cloneDeepNewVM?.__clone;
|
|
|
|
// old VM
|
|
delete this.cloneVM?.metadata;
|
|
delete this.cloneVM?.__clone;
|
|
|
|
// add empty hostDevices to old VM as CRD does not have it.
|
|
const devicesObj = this.cloneVM?.spec?.template?.spec?.domain?.devices;
|
|
|
|
if (devicesObj && devicesObj.hostDevices === undefined) {
|
|
set(devicesObj, 'hostDevices', []);
|
|
}
|
|
|
|
const oldVM = JSON.parse(JSON.stringify(this.cloneVM));
|
|
const newVM = JSON.parse(JSON.stringify(cloneDeepNewVM));
|
|
|
|
// we won't show restart dialog in yaml page as we don't have a way to detect change in yaml editor.
|
|
// only check VM is changed in form page.
|
|
if (isEqual(oldVM, newVM)) {
|
|
return;
|
|
}
|
|
|
|
return new Promise((resolve) => {
|
|
this.isOpen = true;
|
|
|
|
this.$nextTick(() => {
|
|
if (this?.$refs?.restartDialog) {
|
|
this.$refs.restartDialog.resolve = resolve;
|
|
} else {
|
|
return resolve();
|
|
}
|
|
});
|
|
});
|
|
},
|
|
|
|
updateBeforeSave() {
|
|
if (this.isSingle) {
|
|
if (!this.value.spec.template.spec.hostname) {
|
|
this.value.spec.template.spec['hostname'] = this.value.metadata.name;
|
|
}
|
|
}
|
|
|
|
const errors = this.getAccessCredentialsValidation();
|
|
|
|
if (errors.length > 0) {
|
|
return Promise.reject(errors);
|
|
} else {
|
|
return Promise.resolve();
|
|
}
|
|
},
|
|
|
|
validateCount(count) {
|
|
if (count > 10) {
|
|
this['count'] = 10;
|
|
}
|
|
},
|
|
|
|
updateTemplateId() {
|
|
this.templateVersionId = '';
|
|
},
|
|
|
|
onTabChanged({ tab }) {
|
|
if (tab.name === 'advanced') {
|
|
this.$refs.yamlEditor?.refresh();
|
|
}
|
|
},
|
|
|
|
hasAvailableVersion(templateId) {
|
|
let hasAvailableVersion = false;
|
|
|
|
this.versions.filter( (O) => O.templateId === templateId).find( (T) => {
|
|
if (T.isReady) {
|
|
hasAvailableVersion = true;
|
|
}
|
|
});
|
|
|
|
return hasAvailableVersion;
|
|
},
|
|
|
|
generateYaml() {
|
|
this.showYaml = this?.$refs?.vmCruResource?.showYaml || false;
|
|
this.parseVM();
|
|
const out = saferDump(this.value);
|
|
|
|
return out;
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<CruResource
|
|
v-if="spec"
|
|
id="vm"
|
|
ref="vmCruResource"
|
|
:done-route="doneRoute"
|
|
:resource="value"
|
|
:cancel-event="true"
|
|
:mode="mode"
|
|
:validation-passed="isValidationPassed"
|
|
:can-yaml="isSingle ? true : false"
|
|
:errors="errors"
|
|
:generate-yaml="generateYaml"
|
|
:apply-hooks="applyHooks"
|
|
@finish="saveVM"
|
|
@cancel="cancelAction"
|
|
@error="e=>errors=e"
|
|
>
|
|
<RadioGroup
|
|
v-if="isCreate"
|
|
v-model:value="isSingle"
|
|
class="mb-20 vm-radio-group"
|
|
name="createInstanceMode"
|
|
:options="[true,false]"
|
|
:labels="[t('harvester.virtualMachine.instance.single.label'), t('harvester.virtualMachine.instance.multiple.label')]"
|
|
/>
|
|
|
|
<NameNsDescription
|
|
:value="value"
|
|
:mode="mode"
|
|
:has-extra="!isSingle"
|
|
:name-label="nameLabel"
|
|
:namespaced="true"
|
|
:name-placeholder="isSingle ? 'nameNsDescription.name.placeholder' : 'harvester.virtualMachine.instance.multiple.nameNsDescription'"
|
|
:extra-columns="isSingle ? [] :['type']"
|
|
@update:value="$emit('update:value', $event)"
|
|
>
|
|
<template #type>
|
|
<LabeledInput
|
|
v-if="!isSingle"
|
|
v-model:value.number="count"
|
|
:min="2"
|
|
type="number"
|
|
:label="t('harvester.virtualMachine.instance.multiple.count')"
|
|
required
|
|
@update:value="validateCount"
|
|
/>
|
|
</template>
|
|
</NameNsDescription>
|
|
|
|
<Checkbox
|
|
v-if="isCreate"
|
|
v-model:value="useTemplate"
|
|
class="check mb-20"
|
|
type="checkbox"
|
|
label-key="harvester.virtualMachine.useTemplate.label"
|
|
/>
|
|
|
|
<div
|
|
v-if="useTemplate"
|
|
class="row mb-20"
|
|
>
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="templateId"
|
|
label-key="harvester.virtualMachine.useTemplate.template.label"
|
|
:options="templateOptions"
|
|
@update:value="updateTemplateId"
|
|
/>
|
|
</div>
|
|
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="templateVersionId"
|
|
label-key="harvester.virtualMachine.useTemplate.version.label"
|
|
:options="versionOptions"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<Tabbed
|
|
:side-tabs="true"
|
|
@changed="onTabChanged"
|
|
>
|
|
<Tab
|
|
name="basics"
|
|
:label="t('harvester.virtualMachine.detail.tabs.basics')"
|
|
>
|
|
<CpuMemory
|
|
:cpu="cpu"
|
|
:memory="memory"
|
|
:mode="mode"
|
|
@updateCpuMemory="updateCpuMemory"
|
|
/>
|
|
|
|
<SSHKey
|
|
v-model:value="sshKey"
|
|
class="mb-20"
|
|
:create-namespace="true"
|
|
:namespace="value.metadata.namespace"
|
|
:mode="mode"
|
|
:disabled="isWindows || isEdit"
|
|
@update:sshKey="updateSSHKey"
|
|
@register-after-hook="registerAfterHook"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="Volume"
|
|
:label="t('harvester.tab.volume')"
|
|
:weight="-1"
|
|
>
|
|
<Volume
|
|
v-model:value="diskRows"
|
|
:mode="mode"
|
|
:custom-volume-mode="customVolumeMode"
|
|
:namespace="value.metadata.namespace"
|
|
:resource-type="value.type"
|
|
:vm="value"
|
|
:validate-required="true"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="Network"
|
|
:label="t('harvester.tab.network')"
|
|
:weight="-2"
|
|
>
|
|
<Network
|
|
v-model:value="networkRows"
|
|
:mode="mode"
|
|
:is-single="isSingle"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="nodeScheduling"
|
|
:label="t('workload.container.titles.nodeScheduling')"
|
|
:weight="-3"
|
|
>
|
|
<template #default>
|
|
<NodeScheduling
|
|
:mode="mode"
|
|
:value="spec.template.spec"
|
|
:nodes="nodesIdOptions"
|
|
/>
|
|
</template>
|
|
</Tab>
|
|
|
|
<Tab
|
|
:label="t('harvester.tab.vmScheduling')"
|
|
name="vmScheduling"
|
|
:weight="-4"
|
|
>
|
|
<template #default>
|
|
<PodAffinity
|
|
:mode="mode"
|
|
:value="spec.template.spec"
|
|
:nodes="nodes"
|
|
:all-namespaces-option-available="true"
|
|
:namespaces="filteredNamespaces"
|
|
:overwrite-labels="affinityLabels"
|
|
/>
|
|
</template>
|
|
</Tab>
|
|
|
|
<Tab
|
|
v-if="enabledPCI"
|
|
:label="t('harvester.tab.pciDevices')"
|
|
name="pciDevices"
|
|
:weight="-5"
|
|
>
|
|
<PciDevices
|
|
:mode="mode"
|
|
:value="spec.template.spec"
|
|
:vm="value"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
v-if="enabledSriovgpu"
|
|
:label="t('harvester.tab.vGpuDevices')"
|
|
name="vGpuDevices"
|
|
:weight="-6"
|
|
>
|
|
<VGpuDevices
|
|
:mode="mode"
|
|
:value="spec.template.spec"
|
|
:vm="value"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
v-if="enabledPCI && usbPassthroughEnabled"
|
|
:label="t('harvester.tab.usbDevices')"
|
|
name="usbDevices"
|
|
:weight="-7"
|
|
>
|
|
<UsbDevices
|
|
:mode="mode"
|
|
:value="spec.template.spec"
|
|
:vm="value"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
v-if="isEdit"
|
|
:label="t('harvester.tab.accessCredentials')"
|
|
name="accessCredentials"
|
|
:weight="-8"
|
|
>
|
|
<AccessCredentials
|
|
v-model:value="accessCredentials"
|
|
:mode="mode"
|
|
:resource-type="value"
|
|
:is-qemu-installed="isQemuInstalled"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="labels"
|
|
:label="t('generic.labels')"
|
|
:weight="-9"
|
|
>
|
|
<Banner color="info">
|
|
<t k="harvester.virtualMachine.labels.banner" />
|
|
</Banner>
|
|
<KeyValue
|
|
key="labels"
|
|
:value="value.labels"
|
|
:add-label="t('labels.addLabel')"
|
|
:mode="mode"
|
|
:read-allowed="false"
|
|
:value-can-be-empty="true"
|
|
@update:value="value.setLabels($event)"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="instanceLabel"
|
|
:label="t('harvester.tab.instanceLabel')"
|
|
:weight="-10"
|
|
>
|
|
<Banner color="info">
|
|
<t k="harvester.virtualMachine.instanceLabels.banner" />
|
|
</Banner>
|
|
<KeyValue
|
|
key="instance-labels"
|
|
:value="value.instanceLabels"
|
|
:protected-keys="value.systemLabels || []"
|
|
:toggle-filter="toggler"
|
|
:add-label="t('labels.addLabel')"
|
|
:mode="mode"
|
|
:read-allowed="false"
|
|
:value-can-be-empty="true"
|
|
@update:value="value.setInstanceLabels($event)"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="annotations"
|
|
:label="t('harvester.tab.annotations')"
|
|
:weight="-11"
|
|
>
|
|
<Banner color="info">
|
|
<t k="harvester.virtualMachine.annotations.banner" />
|
|
</Banner>
|
|
<KeyValue
|
|
key="annotations"
|
|
:value="value.annotations"
|
|
:protected-keys="value.systemAnnotations || []"
|
|
:toggle-filter="toggler"
|
|
:add-label="t('labels.addAnnotation')"
|
|
:mode="mode"
|
|
:read-allowed="false"
|
|
:value-can-be-empty="true"
|
|
@update:value="value.setAnnotations($event)"
|
|
/>
|
|
</Tab>
|
|
|
|
<Tab
|
|
name="advanced"
|
|
:label="t('harvester.tab.advanced')"
|
|
:weight="-12"
|
|
>
|
|
<div class="row mb-20">
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="runStrategy"
|
|
label-key="harvester.virtualMachine.runStrategy"
|
|
:options="runStrategies"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="osType"
|
|
label-key="harvester.virtualMachine.osType"
|
|
:options="OS"
|
|
:disabled="!isCreate"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="row mb-20">
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="maintenanceStrategy"
|
|
label-key="harvester.virtualMachine.maintenanceStrategy.label"
|
|
:options="maintenanceStrategies"
|
|
:get-option-label="getMaintenanceStrategyOptionLabel"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
<div class="col span-6">
|
|
<Reserved
|
|
:reserved-memory="reservedMemory"
|
|
:mode="mode"
|
|
@updateReserved="updateReserved"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="row mb-20">
|
|
<a
|
|
v-if="showAdvanced"
|
|
v-t="'harvester.generic.showMore'"
|
|
role="button"
|
|
@click="toggleAdvanced"
|
|
/>
|
|
<a
|
|
v-else
|
|
v-t="'harvester.generic.showMore'"
|
|
role="button"
|
|
@click="toggleAdvanced"
|
|
/>
|
|
</div>
|
|
|
|
<div
|
|
v-if="showAdvanced"
|
|
class="mb-20"
|
|
>
|
|
<div class="row mb-20">
|
|
<div class="col span-6">
|
|
<LabeledInput
|
|
v-model:value="hostname"
|
|
:label-key="hostnameLabel"
|
|
:placeholder="hostPlaceholder"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
|
|
<div class="col span-6">
|
|
<LabeledSelect
|
|
v-model:value="machineType"
|
|
label-key="harvester.virtualMachine.input.MachineType"
|
|
:options="machineTypeOptions"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="row mb-20">
|
|
<div class="col span-6">
|
|
<UnitInput
|
|
v-model:value="terminationGracePeriodSeconds"
|
|
:suffix="terminationGracePeriodSeconds == 1 ? 'Second' : 'Seconds'"
|
|
:label="t('harvester.virtualMachine.terminationGracePeriodSeconds.label')"
|
|
:mode="mode"
|
|
positive
|
|
@update:value="updateTerminationGracePeriodSeconds"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<CloudConfig
|
|
ref="yamlEditor"
|
|
:user-script="userScript"
|
|
:mode="mode"
|
|
:os-type="osType"
|
|
:view-code="isWindows"
|
|
:namespace="value.metadata.namespace"
|
|
:network-script="networkScript"
|
|
@updateUserData="updateUserData"
|
|
@updateNetworkData="updateNetworkData"
|
|
@updateDataTemplateId="updateDataTemplateId"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-if="value.cpuPinningFeatureEnabled"
|
|
v-model:value="cpuPinning"
|
|
:disabled="!enableCpuPinningCheckbox"
|
|
class="check"
|
|
type="checkbox"
|
|
tooltip-key="harvester.virtualMachine.cpuPinning.tooltip"
|
|
label-key="harvester.virtualMachine.cpuPinning.label"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-model:value="installUSBTablet"
|
|
class="check mt-20"
|
|
type="checkbox"
|
|
tooltip-key="harvester.virtualMachine.usbTip"
|
|
label-key="harvester.virtualMachine.enableUsb"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-model:value="installAgent"
|
|
class="check"
|
|
type="checkbox"
|
|
:disabled="isWindows"
|
|
label-key="harvester.virtualMachine.installAgent"
|
|
:mode="mode"
|
|
@update:value="updateAgent"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-model:value="tpmEnabled"
|
|
class="check"
|
|
type="checkbox"
|
|
label-key="harvester.virtualMachine.advancedOptions.tpm"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-if="value.tpmPersistentStateFeatureEnabled && tpmEnabled"
|
|
v-model:value="tpmPersistentStateEnabled"
|
|
class="check"
|
|
type="checkbox"
|
|
:label="t('harvester.virtualMachine.advancedOptions.tpmPersistentState')"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-model:value="efiEnabled"
|
|
class="check"
|
|
type="checkbox"
|
|
:label="t('harvester.virtualMachine.advancedOptions.efiEnabled')"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-if="value.efiPersistentStateFeatureEnabled && efiEnabled"
|
|
v-model:value="efiPersistentStateEnabled"
|
|
class="check"
|
|
type="checkbox"
|
|
:label="t('harvester.virtualMachine.advancedOptions.efiPersistentState')"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Checkbox
|
|
v-if="efiEnabled"
|
|
v-model:value="secureBoot"
|
|
class="check"
|
|
type="checkbox"
|
|
:label="t('harvester.virtualMachine.advancedOptions.secureBoot')"
|
|
:mode="mode"
|
|
/>
|
|
|
|
<Banner
|
|
v-if="showCpuPinningBanner"
|
|
color="warning"
|
|
>
|
|
<MessageLink
|
|
v-if="mode === 'create'"
|
|
:to="to"
|
|
prefix-label="harvester.virtualMachine.advancedOptions.cpuManager.prefix"
|
|
middle-label="harvester.virtualMachine.advancedOptions.cpuManager.middle"
|
|
suffix-label="harvester.virtualMachine.advancedOptions.cpuManager.suffix"
|
|
/>
|
|
<span v-if="mode==='edit'">
|
|
{{ t('harvester.virtualMachine.cpuPinning.restartVMMessage') }}
|
|
</span>
|
|
</Banner>
|
|
</Tab>
|
|
</Tabbed>
|
|
|
|
<RestartVMDialog
|
|
v-if="isOpen"
|
|
ref="restartDialog"
|
|
:vm="value"
|
|
/>
|
|
</CruResource>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
#vm {
|
|
:deep() .vm-radio-group .radio-group {
|
|
display: flex;
|
|
.radio-container {
|
|
margin-right: 30px;
|
|
}
|
|
}
|
|
|
|
.restart {
|
|
display: flex;
|
|
justify-content: flex-end;
|
|
}
|
|
|
|
.banner-right {
|
|
width: auto;
|
|
display: flex;
|
|
justify-items: center;
|
|
}
|
|
}
|
|
</style>
|