mirror of
https://github.com/harvester/harvester-ui-extension.git
synced 2025-12-13 21:21:44 +00:00
294 lines
7.3 KiB
Vue
294 lines
7.3 KiB
Vue
<script>
|
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
|
import LabelValue from '@shell/components/LabelValue';
|
|
import { BadgeState } from '@components/BadgeState';
|
|
import { Banner } from '@components/Banner';
|
|
import { RadioGroup, RadioButton } from '@components/Form/Radio';
|
|
import HarvesterDisk from '../../mixins/harvester-disk';
|
|
import Tags from '../../components/DiskTags';
|
|
import { HCI } from '../../types';
|
|
import { LONGHORN_SYSTEM } from './index';
|
|
|
|
export default {
|
|
components: {
|
|
LabeledInput,
|
|
LabelValue,
|
|
BadgeState,
|
|
Banner,
|
|
RadioGroup,
|
|
RadioButton,
|
|
Tags,
|
|
},
|
|
|
|
mixins: [
|
|
HarvesterDisk,
|
|
],
|
|
|
|
props: {
|
|
value: {
|
|
type: Object,
|
|
default: () => {
|
|
return {};
|
|
},
|
|
},
|
|
disks: {
|
|
type: Array,
|
|
default: () => [],
|
|
},
|
|
mode: {
|
|
type: String,
|
|
default: 'edit',
|
|
},
|
|
},
|
|
data() {
|
|
return {};
|
|
},
|
|
computed: {
|
|
allowSchedulingOptions() {
|
|
return [{
|
|
label: this.t('generic.enabled'),
|
|
value: true,
|
|
}, {
|
|
label: this.t('generic.disabled'),
|
|
value: false,
|
|
}];
|
|
},
|
|
|
|
evictionRequestedOptions() {
|
|
return [{
|
|
label: this.t('generic.yes'),
|
|
value: true,
|
|
}, {
|
|
label: this.t('generic.no'),
|
|
value: false,
|
|
}];
|
|
},
|
|
|
|
mountedMessage() {
|
|
const state = this.blockDevice?.metadata?.state || {};
|
|
|
|
if (state?.error) {
|
|
return state?.message;
|
|
} else {
|
|
return '';
|
|
}
|
|
},
|
|
|
|
isProvisioned() {
|
|
return this.blockDevice?.spec.fileSystem.provisioned;
|
|
},
|
|
|
|
forceFormattedDisabled() {
|
|
const lastFormattedAt = this.blockDevice?.status?.deviceStatus?.fileSystem?.LastFormattedAt;
|
|
const fileSystem = this.blockDevice?.status?.deviceStatus?.fileSystem.type;
|
|
|
|
const systems = ['ext4', 'XFS'];
|
|
|
|
if (lastFormattedAt || this.blockDevice?.childParts?.length > 0) {
|
|
return true;
|
|
} else if (systems.includes(fileSystem)) {
|
|
return false;
|
|
} else if (!fileSystem) {
|
|
return true;
|
|
} else {
|
|
return !this.canEditPath;
|
|
}
|
|
},
|
|
|
|
canEditPath() {
|
|
if (this.mountedMessage) {
|
|
return true;
|
|
}
|
|
|
|
if (this.value.isNew && !this.value.originPath) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
isFormatted() {
|
|
return !!this.blockDevice?.status?.deviceStatus?.fileSystem?.LastFormattedAt;
|
|
},
|
|
|
|
formattedBannerLabel() {
|
|
const system = this.blockDevice?.status?.deviceStatus?.fileSystem?.type;
|
|
|
|
const label = this.t('harvester.host.disk.lastFormattedAt.info');
|
|
|
|
if (system) {
|
|
return `${ label } ${ this.t('harvester.host.disk.fileSystem.info', { system }) }`;
|
|
} else {
|
|
return label;
|
|
}
|
|
},
|
|
|
|
provisionPhase() {
|
|
return this.blockDevice?.provisionPhase || {};
|
|
},
|
|
|
|
blockDevice() {
|
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
|
const name = this.value?.name;
|
|
|
|
return this.$store.getters[`${ inStore }/byId`](HCI.BLOCK_DEVICE, `${ LONGHORN_SYSTEM }/${ name }`) || {};
|
|
},
|
|
|
|
isCorrupted() {
|
|
return this.blockDevice?.status?.deviceStatus?.fileSystem?.corrupted;
|
|
},
|
|
|
|
isFormatting() {
|
|
return this.blockDevice.isFormatting;
|
|
},
|
|
},
|
|
methods: {
|
|
update() {
|
|
this.$emit('input', this.value);
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div class="disk" @input="update">
|
|
<div class="mt-10" />
|
|
<Banner
|
|
v-if="mountedMessage && isProvisioned"
|
|
color="error"
|
|
:label="mountedMessage"
|
|
/>
|
|
<Banner
|
|
v-if="isFormatting"
|
|
color="info"
|
|
:label="t('harvester.host.disk.fileSystem.formatting')"
|
|
/>
|
|
<Banner
|
|
v-else-if="isFormatted && !isCorrupted"
|
|
color="info"
|
|
:label="formattedBannerLabel"
|
|
/>
|
|
<div v-if="!value.isNew">
|
|
<div class="row">
|
|
<div class="col span-12">
|
|
<Tags
|
|
v-model="value.tags"
|
|
:label="t('harvester.host.disk.tags.label')"
|
|
:add-label="t('harvester.host.disk.tags.addLabel')"
|
|
:mode="mode"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div class="row mt-10">
|
|
<div class="col span-12">
|
|
<div class="pull-right">
|
|
{{ t('harvester.host.disk.conditions') }}:
|
|
<BadgeState
|
|
v-clean-tooltip="readyCondition.message"
|
|
:color="readyCondition.status === 'True' ? 'bg-success' : 'bg-error' "
|
|
:icon="readyCondition.status === 'True' ? 'icon-checkmark' : 'icon-warning' "
|
|
label="Ready"
|
|
class="mr-10 ml-10 state"
|
|
/>
|
|
<BadgeState
|
|
v-clean-tooltip="schedulableCondition.message"
|
|
:color="schedulableCondition.status === 'True' ? 'bg-success' : 'bg-error' "
|
|
:icon="schedulableCondition.status === 'True' ? 'icon-checkmark' : 'icon-warning' "
|
|
label="Schedulable"
|
|
class="mr-10 state"
|
|
/>
|
|
<BadgeState
|
|
v-if="provisionPhase.label"
|
|
:color="provisionPhase.color"
|
|
:icon="provisionPhase.icon"
|
|
:label="provisionPhase.label"
|
|
class="mr-10 state"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div v-if="!value.isNew" class="row mt-30">
|
|
<div class="col flex span-12">
|
|
<LabelValue
|
|
:name="t('harvester.host.disk.storageAvailable.label')"
|
|
:value="value.storageAvailable"
|
|
/>
|
|
<LabelValue
|
|
:name="t('harvester.host.disk.storageScheduled.label')"
|
|
:value="value.storageScheduled"
|
|
/>
|
|
<LabelValue
|
|
:name="t('harvester.host.disk.storageMaximum.label')"
|
|
:value="value.storageMaximum"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<hr class="mt-10" />
|
|
</div>
|
|
<div class="row mt-10">
|
|
<div class="col span-12">
|
|
<LabeledInput
|
|
v-model="value.displayName"
|
|
:label="t('generic.name')"
|
|
:disabled="true"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div v-if="(value.isNew && !isFormatted) || isCorrupted" class="row mt-10">
|
|
<div class="col span-6">
|
|
<RadioGroup
|
|
v-model="value.forceFormatted"
|
|
:mode="mode"
|
|
name="forceFormatted"
|
|
label-key="harvester.host.disk.forceFormatted.label"
|
|
:labels="[t('generic.no'),t('harvester.host.disk.forceFormatted.yes')]"
|
|
:options="[false, true]"
|
|
:disabled="forceFormattedDisabled"
|
|
tooltip-key="harvester.host.disk.forceFormatted.toolTip"
|
|
>
|
|
<template #1="{option, listeners}">
|
|
<RadioButton
|
|
:label="option.label"
|
|
:val="option.value"
|
|
:value="value.forceFormatted"
|
|
:disabled="forceFormattedDisabled && !value.forceFormatted"
|
|
v-on="listeners"
|
|
/>
|
|
</template>
|
|
</RadioGroup>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
.close {
|
|
top: 10px;
|
|
right: 10px;
|
|
padding:0;
|
|
position: absolute;
|
|
}
|
|
|
|
.disk {
|
|
position: relative;
|
|
|
|
.secret-name {
|
|
height: $input-height;
|
|
}
|
|
|
|
&:not(:last-of-type) {
|
|
padding-bottom: 10px;
|
|
margin-bottom: 30px;
|
|
}
|
|
}
|
|
|
|
.flex {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
}
|
|
|
|
.badge-state {
|
|
padding: 2px 5px;
|
|
}
|
|
</style>
|