| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 |
- <template>
- <div class="system-disk">
- <disk
- diskKey="system"
- :max="max"
- :min="min"
- :form="form"
- :decorator="decorator"
- :hypervisor="hypervisor"
- :types-map="typesMap"
- :elements="elements"
- :disabled="disabled"
- :storageParams="storageParams"
- :storageHostParams="storageHostParams"
- :schedtagParams="getSchedtagParams()"
- :size-disabled="sizeDisabled || disabled"
- :storage-status-map="storageStatusMap"
- :simplify="true"
- @showStorageChange="showStorageChange"
- @diskTypeChange="setDiskMedium"
- @storageHostChange="(val) => $emit('storageHostChange', val)" />
- </div>
- </template>
- <script>
- import _ from 'lodash'
- import * as R from 'ramda'
- import Disk from '@Compute/sections/Disk'
- // import { STORAGE_AUTO } from '@Compute/constants'
- import { IMAGES_TYPE_MAP, STORAGE_TYPES } from '@/constants/compute'
- import { HYPERVISORS_MAP } from '@/constants'
- import { findAndUnshift, findAndPush } from '@/utils/utils'
- import { diskSupportTypeMedium, getOriginDiskKey } from '@/utils/common/hypervisor'
- import { MEDIUM_MAP } from '@Compute/constants'
- // 磁盘最小值
- export const DISK_MIN_SIZE = 10
- // let isFirstSetDefaultSize = true
- export default {
- name: 'SystemDisk',
- components: {
- Disk,
- },
- props: {
- form: {
- type: Object,
- required: true,
- validator: val => val.fd && val.fc,
- },
- type: {
- type: String,
- required: true,
- validator: val => ['idc', 'private', 'public'].includes(val),
- },
- hypervisor: {
- },
- sku: {
- type: Object,
- },
- capabilityData: {
- type: Object,
- required: true,
- },
- image: {
- type: Object,
- },
- decorator: {
- type: Object,
- required: true,
- },
- disabled: {
- type: Boolean,
- default: false,
- },
- isHostImageType: {
- type: Boolean,
- default: false,
- },
- domain: {
- type: String,
- default: 'default',
- },
- sizeDisabled: {
- type: Boolean,
- default: false,
- },
- defaultSize: {
- type: Number,
- },
- defaultType: {
- type: Object,
- },
- isServertemplate: {
- type: Boolean,
- default: false,
- },
- storageParams: {
- type: Object,
- },
- storageHostParams: Object,
- ignoreStorageStatus: {
- type: Boolean,
- default: false,
- },
- },
- computed: {
- isPublic () {
- return this.type === 'public'
- },
- isPrivate () {
- return this.type === 'private'
- },
- isIDC () {
- return this.type === 'idc'
- },
- imageMinDisk () {
- const image = this.image
- let minSize = 0
- if (!image) return 0
- if (this.isHostImageType) {
- if (image.root_image) {
- minSize = (image.root_image.min_disk_mb / 1024) || 0
- }
- } else if (image.info) {
- minSize = ((image.info.min_disk_mb || image.info.min_disk) / 1024) || 0
- } else {
- minSize = ((image.min_disk_mb || image.min_disk) / 1024) || 0
- }
- return Math.ceil(minSize)
- },
- elements () {
- const ret = ['disk-select']
- if (this.isIDC && !this.isServertemplate) {
- ret.push('schedtag')
- if (this.form.fd.hypervisor === HYPERVISORS_MAP.esxi.key || this.form.fd.hypervisor === HYPERVISORS_MAP.kvm.key) {
- ret.push('storage') // kvm,vmware支持指定存储
- }
- }
- return ret
- },
- typesMap () {
- const ret = {}
- const hyper = this.getHypervisor()
- if (!hyper) return ret
- const hypervisorDisks = { ...STORAGE_TYPES[hyper] } || {}
- if (!this.capabilityData || !this.capabilityData.storage_types2) return ret
- let currentTypes = this.capabilityData.storage_types2[hyper] || []
- if (!R.isNil(this.sku) && !R.isEmpty(this.sku)) {
- if (this.sku.sys_disk_type && !this.defaultSize) { // 有 defaultSize 表示是调整配置,不需要根据sku信息过滤
- const skuDiskTypes = this.sku.sys_disk_type.split(',')
- if (skuDiskTypes && skuDiskTypes.length) {
- currentTypes = currentTypes.filter(val => {
- const type = val.split('/')[0]
- return skuDiskTypes.includes(type)
- })
- }
- } else {
- for (const obj in hypervisorDisks) {
- if (hypervisorDisks[obj].skuFamily && !hypervisorDisks[obj].skuFamily.includes(this.sku.instance_type_family)) {
- delete hypervisorDisks[obj]
- }
- }
- }
- } else {
- if (this.isPublic) {
- currentTypes = []
- }
- }
- const localIndex = currentTypes.findIndex(item => item.includes('local'))
- const novaIndex = currentTypes.findIndex(item => item.includes('nova'))
- if (localIndex !== -1 && localIndex !== 0) { // 将local放置首位
- currentTypes = findAndUnshift(currentTypes, item => item.includes('local'))
- }
- if (novaIndex !== -1 && novaIndex !== (currentTypes.length - 1)) { // 将nova放置到最后
- currentTypes = findAndPush(currentTypes, item => item.includes('nova'))
- }
- for (let i = 0, len = currentTypes.length; i < len; i++) {
- const typeItemArr = currentTypes[i].split('/')
- const type = typeItemArr[0]
- const medium = typeItemArr[1]
- let opt = hypervisorDisks[type] || this.getExtraDiskOpt(type)
- // 磁盘区分介质
- if (diskSupportTypeMedium(hyper)) {
- opt = {
- ...opt,
- key: `${type}/${medium}`,
- label: `${opt.label}(${MEDIUM_MAP[medium]})`,
- }
- }
- if (opt && !opt.sysUnusable) {
- // 新建ucloud虚拟机时,系统盘类型选择普通本地盘或SSD本地盘,其大小只能是系统镜像min_disk大小
- let max = opt.sysMax
- if (hyper === HYPERVISORS_MAP.ucloud.key && ['LOCAL_NORMAL', 'LOCAL_SSD'].includes(getOriginDiskKey(opt.key))) {
- max = this.imageMinDisk
- }
- // 谷歌云共享核心磁盘最多为3072GB
- if (hyper === HYPERVISORS_MAP.google.key && this.sku && ['e2-micro', 'e2-small', 'e2-medium', 'f1-micro', 'g1-small'].includes(this.sku.name)) {
- max = 3072
- }
- ret[opt.key] = {
- ...opt,
- medium,
- sysMin: Math.max(this.imageMinDisk, opt.sysMin, DISK_MIN_SIZE),
- sysMax: max,
- label: opt.key === 'nova' ? this.$t('compute.text_1141') : opt.label,
- }
- if (this.hypervisor === HYPERVISORS_MAP.google.key) {
- ret[opt.key].sysMin = opt.sysMin
- }
- }
- }
- // if (this.isIDC && this.hypervisor !== HYPERVISORS_MAP.kvm.key) {
- // ret[STORAGE_AUTO.key] = {
- // ...STORAGE_AUTO,
- // sysMin: Math.max(this.imageMinDisk, DISK_MIN_SIZE),
- // sysMax: STORAGE_AUTO.sysMax,
- // }
- // }
- if (this.hypervisor === HYPERVISORS_MAP.google.key) {
- delete ret['local-ssd']
- }
- if (this.hypervisor === HYPERVISORS_MAP.qcloud.key) {
- delete ret.local_nvme
- delete ret.local_pro
- }
- this.$nextTick(this.setDefaultType)
- return ret
- },
- currentTypeObj () {
- if (R.is(Object, this.typesMap) && this.form.fd[this.decorator.type[0]] && this.form.fd[this.decorator.type[0]].key) {
- return this.typesMap[this.form.fd[this.decorator.type[0]].key] || {}
- }
- return {}
- },
- max () {
- return this.currentTypeObj.sysMax || 0
- },
- min () {
- return this.currentTypeObj.sysMin || 0
- },
- storageStatusMap () {
- var statusMap = {
- type: '',
- tooltip: '',
- isError: false,
- }
- if (this.ignoreStorageStatus || !this.form.fd.systemDiskType || !this.form.fd.systemDiskType.key) return statusMap
- if (this.capabilityData.storage_types3 && this.hypervisor && !this.isPublic && this.hypervisor !== HYPERVISORS_MAP.hcso.hypervisor) {
- const storageTypes3 = this.capabilityData.storage_types3[this.hypervisor] || {}
- const storageTypes = Object.keys(storageTypes3)
- for (const item of storageTypes) {
- const key = Array.isArray(item.split('/')) ? item.split('/')[0] : ''
- const storages = storageTypes3[item] || {}
- const isAllEmpty = storages.capacity === 0
- if (key === this.currentTypeObj.key && isAllEmpty) {
- // 没有设置容量:XXX存储的容量没有设置,无法创建虚拟机,请到存储--块存储进行设置,如无法查看请联系管理员设置
- statusMap = { type: 'error', tooltip: this.$t('compute.text_1142', [key]), isError: true }
- break
- }
- if (key === this.currentTypeObj.key && storages.capacity) {
- // 选择磁盘容量不足:XXX存储的容量不足,无法创建虚拟机,请到存储--块存储进行查看,如无法查看请联系管理员查看
- if (storages.free_capacity === 0 || storages.free_capacity / 1024 < this.form.fd.systemDiskSize) {
- statusMap = { type: 'error', tooltip: this.$t('compute.text_1143', [key]), isError: true }
- break
- }
- }
- }
- }
- this.$bus.$emit('VMCreateDisabled', statusMap.isError)
- return statusMap
- },
- },
- created () {
- this.setDefaultType = _.debounce(this.setDefaultType, 300)
- },
- methods: {
- setDefaultType () {
- if (R.isNil(this.typesMap) || R.isEmpty(this.typesMap)) {
- this.form.fc.setFieldsValue({
- [this.decorator.type[0]]: { key: '', label: '' },
- [this.decorator.size[0]]: 0,
- })
- return
- }
- if ([IMAGES_TYPE_MAP.host.key, IMAGES_TYPE_MAP.snapshot.key].includes(this.form.fd.imageType)) return // 主机镜像和主机快照设置默认值交给外层处理
- const keys = Object.keys(this.typesMap)
- const firstKey = keys[0]
- const diskMsg = this.typesMap[firstKey]
- this.form.fc.setFieldsValue(this.defaultType || {
- [this.decorator.type[0]]: { key: diskMsg.key, label: diskMsg.label },
- })
- this.setDiskMedium(diskMsg)
- this.$nextTick(() => { // 解决磁盘大小 inputNumber 第一次点击变为0 的bug
- this.form.fc.setFieldsValue({
- [this.decorator.size[0]]: this.defaultSize || +diskMsg.sysMin,
- })
- })
- },
- getExtraDiskOpt (type) {
- const hyper = this.getHypervisor()
- // 腾讯云过滤掉local_basic和local_ssd类型的盘
- if (hyper === HYPERVISORS_MAP.qcloud.key) {
- if (['local_basic', 'local_ssd'].includes(type)) {
- return
- }
- }
- // VMware过滤掉rbd类型的盘
- if (hyper === HYPERVISORS_MAP.esxi.key) {
- if (['rbd'].includes(type)) {
- return
- }
- }
- return {
- label: `${type}`,
- key: `${type}`,
- min: 1,
- max: 3 * 1024,
- sysMin: 10,
- sysMax: 500,
- }
- },
- getSchedtagParams () {
- const params = {
- with_meta: true,
- cloud_env: 'onpremise',
- resource_type: 'storages',
- limit: 0,
- }
- const scopeParams = {}
- if (this.$store.getters.isAdminMode) {
- scopeParams.project_domain = this.domain
- } else {
- scopeParams.scope = this.$store.getters.scope
- }
- return {
- ...params,
- ...scopeParams,
- }
- },
- getHypervisor () {
- let ret = this.hypervisor
- if (this.isPublic) {
- if (this.sku && this.sku.provider) {
- ret = this.sku.provider.toLowerCase()
- }
- }
- return ret
- },
- showStorageChange (v) {
- if (this.form.fi) {
- this.$set(this.form.fi, 'showStorage', v)
- }
- const decoratorKey = _.get(this.decorator, 'systemDisk.storage[0]') || 'systemDiskStorage'
- if (!v) {
- this.$set(this.form.fd, decoratorKey, undefined)
- }
- },
- setDiskMedium (v) {
- if (this.form.fi) {
- this.$set(this.form.fi, 'systemDiskMedium', _.get(this.typesMap, `[${v.key}].medium`))
- }
- },
- isSomeLocal (types) {
- const localTypes = types.filter(item => item.indexOf('local') !== -1)
- return localTypes.length > 1
- },
- },
- }
- </script>
|