// Licensed to the Apache Software Foundation (ASF) under one // or more contributor license agreements. See the NOTICE file // distributed with this work for additional information // regarding copyright ownership. The ASF licenses this file // to you under the Apache License, Version 2.0 (the // "License"); you may not use this file except in compliance // with the License. You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, // software distributed under the License is distributed on an // "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. See the License for the // specific language governing permissions and limitations // under the License. import kubernetes from '@/assets/icons/kubernetes.svg?inline' import store from '@/store' export default { name: 'compute', title: 'label.compute', icon: 'cloud', children: [ { name: 'vm', title: 'label.instances', icon: 'desktop', docHelp: 'adminguide/virtual_machines.html', permission: ['listVirtualMachinesMetrics'], resourceType: 'UserVm', filters: () => { const filters = ['running', 'stopped'] if (!(store.getters.project && store.getters.project.id)) { filters.unshift('self') } return filters }, columns: () => { const fields = ['displayname', 'name', 'state', 'ipaddress'] const metricsFields = ['cpunumber', 'cpuused', 'cputotal', { memoryused: (record) => { return record.memorykbs && record.memoryintfreekbs ? parseFloat(100.0 * (record.memorykbs - record.memoryintfreekbs) / record.memorykbs).toFixed(2) + '%' : '0.0%' } }, 'memorytotal', 'networkread', 'networkwrite', 'diskkbsread', 'diskkbswrite', 'diskiopstotal' ] if (store.getters.metrics) { fields.push(...metricsFields) } if (store.getters.userInfo.roletype === 'Admin') { fields.splice(2, 0, 'instancename') fields.push('account') fields.push('hostname') fields.push('zonename') } else if (store.getters.userInfo.roletype === 'DomainAdmin') { fields.splice(2, 0, 'displayname') fields.push('account') fields.push('zonename') } else { fields.splice(2, 0, 'displayname') fields.push('zonename') } return fields }, searchFilters: ['name', 'zoneid', 'domainid', 'account', 'tags'], details: ['displayname', 'name', 'id', 'state', 'ipaddress', 'templatename', 'ostypename', 'serviceofferingname', 'isdynamicallyscalable', 'haenable', 'hypervisor', 'boottype', 'bootmode', 'account', 'domain', 'zonename'], tabs: [{ component: () => import('@/views/compute/InstanceTab.vue') }], actions: [ { api: 'deployVirtualMachine', icon: 'plus', label: 'label.vm.add', docHelp: 'adminguide/virtual_machines.html#creating-vms', listView: true, component: () => import('@/views/compute/DeployVM.vue') }, { api: 'updateVirtualMachine', icon: 'edit', label: 'label.action.edit.instance', docHelp: 'adminguide/virtual_machines.html#changing-the-vm-name-os-or-group', dataView: true, args: ['name', 'displayname', 'ostypeid', 'isdynamicallyscalable', 'haenable', 'group'], show: (record) => { return ['Stopped'].includes(record.state) } }, { api: 'startVirtualMachine', icon: 'caret-right', label: 'label.action.start.instance', message: 'message.action.start.instance', docHelp: 'adminguide/virtual_machines.html#stopping-and-starting-vms', dataView: true, groupAction: true, popup: true, groupMap: (selection) => { return selection.map(x => { return { id: x } }) }, show: (record) => { return ['Stopped'].includes(record.state) }, component: () => import('@/views/compute/StartVirtualMachine.vue') }, { api: 'stopVirtualMachine', icon: 'poweroff', label: 'label.action.stop.instance', message: 'message.action.stop.instance', docHelp: 'adminguide/virtual_machines.html#stopping-and-starting-vms', dataView: true, groupAction: true, groupMap: (selection, values) => { return selection.map(x => { return { id: x, forced: values.forced } }) }, args: ['forced'], show: (record) => { return ['Running'].includes(record.state) } }, { api: 'rebootVirtualMachine', icon: 'reload', label: 'label.action.reboot.instance', message: 'message.action.reboot.instance', docHelp: 'adminguide/virtual_machines.html#stopping-and-starting-vms', dataView: true, show: (record) => { return ['Running'].includes(record.state) }, args: (record, store) => { var fields = [] fields.push('forced') if (record.hypervisor === 'VMware') { if (store.apis.rebootVirtualMachine.params.filter(x => x.name === 'bootintosetup').length > 0) { fields.push('bootintosetup') } } return fields } }, { api: 'restoreVirtualMachine', icon: 'sync', label: 'label.reinstall.vm', message: 'message.reinstall.vm', dataView: true, args: ['virtualmachineid', 'templateid'], show: (record) => { return ['Running', 'Stopped'].includes(record.state) }, mapping: { virtualmachineid: { value: (record) => { return record.id } } } }, { api: 'createVMSnapshot', icon: 'camera', label: 'label.action.vmsnapshot.create', docHelp: 'adminguide/virtual_machines.html#virtual-machine-snapshots', dataView: true, args: ['virtualmachineid', 'name', 'description', 'snapshotmemory', 'quiescevm'], show: (record) => { return ((['Running'].includes(record.state) && record.hypervisor !== 'LXC') || (['Stopped'].includes(record.state) && ((record.hypervisor !== 'KVM' && record.hypervisor !== 'LXC') || (record.hypervisor === 'KVM' && record.pooltype === 'PowerFlex')))) }, mapping: { virtualmachineid: { value: (record, params) => { return record.id } } } }, { api: 'createSnapshot', icon: ['fas', 'camera-retro'], label: 'label.action.vmstoragesnapshot.create', docHelp: 'adminguide/virtual_machines.html#virtual-machine-snapshots', dataView: true, popup: true, component: () => import('@/views/compute/CreateSnapshotWizard.vue') }, { api: 'assignVirtualMachineToBackupOffering', icon: 'folder-add', label: 'label.backup.offering.assign', message: 'label.backup.offering.assign', docHelp: 'adminguide/virtual_machines.html#backup-offerings', dataView: true, args: ['virtualmachineid', 'backupofferingid'], show: (record) => { return !record.backupofferingid }, mapping: { virtualmachineid: { value: (record, params) => { return record.id } } } }, { api: 'createBackup', icon: 'cloud-upload', label: 'label.create.backup', message: 'message.backup.create', docHelp: 'adminguide/virtual_machines.html#creating-vm-backups', dataView: true, args: ['virtualmachineid'], show: (record) => { return record.backupofferingid }, mapping: { virtualmachineid: { value: (record, params) => { return record.id } } } }, { api: 'createBackupSchedule', icon: 'schedule', label: 'Configure Backup Schedule', docHelp: 'adminguide/virtual_machines.html#creating-vm-backups', dataView: true, popup: true, show: (record) => { return record.backupofferingid }, component: () => import('@/views/compute/BackupScheduleWizard.vue'), mapping: { virtualmachineid: { value: (record, params) => { return record.id } }, intervaltype: { options: ['HOURLY', 'DAILY', 'WEEKLY', 'MONTHLY'] } } }, { api: 'removeVirtualMachineFromBackupOffering', icon: 'scissor', label: 'label.backup.offering.remove', message: 'label.backup.offering.remove', docHelp: 'adminguide/virtual_machines.html#restoring-vm-backups', dataView: true, args: ['virtualmachineid', 'forced'], show: (record) => { return record.backupofferingid }, mapping: { virtualmachineid: { value: (record, params) => { return record.id } } } }, { api: 'attachIso', icon: 'paper-clip', label: 'label.action.attach.iso', docHelp: 'adminguide/templates.html#attaching-an-iso-to-a-vm', dataView: true, popup: true, show: (record) => { return ['Running', 'Stopped'].includes(record.state) && !record.isoid }, component: () => import('@/views/compute/AttachIso.vue') }, { api: 'detachIso', icon: 'link', label: 'label.action.detach.iso', message: 'message.detach.iso.confirm', dataView: true, args: ['virtualmachineid'], show: (record) => { return ['Running', 'Stopped'].includes(record.state) && 'isoid' in record && record.isoid }, mapping: { virtualmachineid: { value: (record, params) => { return record.id } } } }, { api: 'updateVMAffinityGroup', icon: 'swap', label: 'label.change.affinity', docHelp: 'adminguide/virtual_machines.html#change-affinity-group-for-an-existing-vm', dataView: true, args: ['affinitygroupids'], show: (record) => { return ['Stopped'].includes(record.state) }, component: () => import('@/views/compute/ChangeAffinity'), popup: true }, { api: 'scaleVirtualMachine', icon: 'arrows-alt', label: 'label.scale.vm', docHelp: 'adminguide/virtual_machines.html#how-to-dynamically-scale-cpu-and-ram', dataView: true, show: (record) => { return ['Stopped'].includes(record.state) || (['Running'].includes(record.state) && record.hypervisor !== 'KVM' && record.hypervisor !== 'LXC') }, popup: true, component: () => import('@/views/compute/ScaleVM.vue') }, { api: 'migrateVirtualMachine', icon: 'drag', label: 'label.migrate.instance.to.host', docHelp: 'adminguide/virtual_machines.html#moving-vms-between-hosts-manual-live-migration', dataView: true, show: (record, store) => { return ['Running'].includes(record.state) && ['Admin'].includes(store.userInfo.roletype) }, popup: true, component: () => import('@/views/compute/MigrateWizard') }, { api: 'migrateVirtualMachine', icon: 'drag', label: 'label.migrate.instance.to.ps', message: 'message.migrate.instance.to.ps', docHelp: 'adminguide/virtual_machines.html#moving-vms-between-hosts-manual-live-migration', dataView: true, show: (record, store) => { return ['Stopped'].includes(record.state) && ['Admin'].includes(store.userInfo.roletype) }, component: () => import('@/views/compute/MigrateVMStorage'), popup: true }, { api: 'resetPasswordForVirtualMachine', icon: 'key', label: 'label.action.reset.password', message: 'message.action.instance.reset.password', dataView: true, show: (record) => { return ['Running', 'Stopped'].includes(record.state) && record.passwordenabled }, response: (result) => { return result.virtualmachine && result.virtualmachine.password ? `Password of the VM is ${result.virtualmachine.password}` : null } }, { api: 'resetSSHKeyForVirtualMachine', icon: 'lock', label: 'label.reset.ssh.key.pair', message: 'message.desc.reset.ssh.key.pair', docHelp: 'adminguide/virtual_machines.html#resetting-ssh-keys', dataView: true, args: ['keypair', 'account', 'domainid'], show: (record) => { return ['Stopped'].includes(record.state) }, mapping: { keypair: { api: 'listSSHKeyPairs', params: (record) => { return { account: record.account, domainid: record.domainid } } }, account: { value: (record) => { return record.account } }, domainid: { value: (record) => { return record.domainid } } } }, { api: 'assignVirtualMachine', icon: 'user-add', label: 'label.assign.instance.another', dataView: true, component: () => import('@/views/compute/AssignInstance'), popup: true, show: (record) => { return ['Stopped'].includes(record.state) } }, { api: 'recoverVirtualMachine', icon: 'medicine-box', label: 'label.recover.vm', message: 'message.recover.vm', dataView: true, show: (record, store) => { return ['Destroyed'].includes(record.state) && store.features.allowuserexpungerecovervm } }, { api: 'unmanageVirtualMachine', icon: 'disconnect', label: 'label.action.unmanage.virtualmachine', message: 'message.action.unmanage.virtualmachine', dataView: true, show: (record) => { return ['Running', 'Stopped'].includes(record.state) && record.hypervisor === 'VMware' } }, { api: 'expungeVirtualMachine', icon: 'delete', label: 'label.action.expunge.instance', message: 'message.action.expunge.instance', docHelp: 'adminguide/virtual_machines.html#deleting-vms', dataView: true, show: (record, store) => { return ['Destroyed', 'Expunging'].includes(record.state) && store.features.allowuserexpungerecovervm } }, { api: 'destroyVirtualMachine', icon: 'delete', label: 'label.action.destroy.instance', message: 'message.action.destroy.instance', docHelp: 'adminguide/virtual_machines.html#deleting-vms', dataView: true, groupAction: true, args: (record, store, group) => { return (['Admin'].includes(store.userInfo.roletype) || store.features.allowuserexpungerecovervm) ? ['expunge'] : [] }, popup: true, groupMap: (selection, values) => { return selection.map(x => { return { id: x, expunge: values.expunge } }) }, show: (record) => { return ['Running', 'Stopped', 'Error'].includes(record.state) }, component: () => import('@/views/compute/DestroyVM.vue') } ] }, { name: 'kubernetes', title: 'label.kubernetes', icon: kubernetes, docHelp: 'plugins/cloudstack-kubernetes-service.html', permission: ['listKubernetesClusters'], columns: () => { var fields = ['name', 'state', 'size', 'cpunumber', 'memory'] if (['Admin', 'DomainAdmin'].includes(store.getters.userInfo.roletype)) { fields.push('account') } fields.push('zonename') return fields }, details: ['name', 'description', 'zonename', 'kubernetesversionname', 'size', 'masternodes', 'cpunumber', 'memory', 'keypair', 'associatednetworkname', 'account', 'domain', 'zonename'], tabs: [{ name: 'k8s', component: () => import('@/views/compute/KubernetesServiceTab.vue') }], actions: [ { api: 'createKubernetesCluster', icon: 'plus', label: 'label.kubernetes.cluster.create', docHelp: 'plugins/cloudstack-kubernetes-service.html#creating-a-new-kubernetes-cluster', listView: true, popup: true, component: () => import('@/views/compute/CreateKubernetesCluster.vue') }, { api: 'startKubernetesCluster', icon: 'caret-right', label: 'label.kubernetes.cluster.start', message: 'message.kubernetes.cluster.start', docHelp: 'plugins/cloudstack-kubernetes-service.html#starting-a-stopped-kubernetes-cluster', dataView: true, show: (record) => { return ['Stopped'].includes(record.state) } }, { api: 'stopKubernetesCluster', icon: 'poweroff', label: 'label.kubernetes.cluster.stop', message: 'message.kubernetes.cluster.stop', docHelp: 'plugins/cloudstack-kubernetes-service.html#stopping-kubernetes-cluster', dataView: true, show: (record) => { return !['Stopped', 'Destroyed', 'Destroying'].includes(record.state) } }, { api: 'scaleKubernetesCluster', icon: 'swap', label: 'label.kubernetes.cluster.scale', message: 'message.kubernetes.cluster.scale', docHelp: 'plugins/cloudstack-kubernetes-service.html#scaling-kubernetes-cluster', dataView: true, show: (record) => { return ['Created', 'Running'].includes(record.state) }, popup: true, component: () => import('@/views/compute/ScaleKubernetesCluster.vue') }, { api: 'upgradeKubernetesCluster', icon: 'plus-circle', label: 'label.kubernetes.cluster.upgrade', message: 'message.kubernetes.cluster.upgrade', docHelp: 'plugins/cloudstack-kubernetes-service.html#upgrading-kubernetes-cluster', dataView: true, show: (record) => { return ['Created', 'Running'].includes(record.state) }, popup: true, component: () => import('@/views/compute/UpgradeKubernetesCluster.vue') }, { api: 'deleteKubernetesCluster', icon: 'delete', label: 'label.kubernetes.cluster.delete', message: 'message.kubernetes.cluster.delete', docHelp: 'plugins/cloudstack-kubernetes-service.html#deleting-kubernetes-cluster', dataView: true, show: (record) => { return !['Destroyed', 'Destroying'].includes(record.state) } } ] }, { name: 'vmgroup', title: 'label.instance.groups', icon: 'gold', docHelp: 'adminguide/virtual_machines.html#changing-the-vm-name-os-or-group', permission: ['listInstanceGroups'], columns: ['name', 'account'], details: ['name', 'id', 'account', 'domain', 'created'], related: [{ name: 'vm', title: 'label.instances', param: 'groupid' }], actions: [ { api: 'createInstanceGroup', icon: 'plus', label: 'label.new.instance.group', listView: true, args: ['name'] }, { api: 'updateInstanceGroup', icon: 'edit', label: 'label.update.instance.group', dataView: true, args: ['name'] }, { api: 'deleteInstanceGroup', icon: 'delete', label: 'label.delete.instance.group', dataView: true } ] }, { name: 'ssh', title: 'label.ssh.key.pairs', icon: 'key', docHelp: 'adminguide/virtual_machines.html#using-ssh-keys-for-authentication', permission: ['listSSHKeyPairs'], columns: () => { var fields = ['name', 'fingerprint'] if (['Admin', 'DomainAdmin'].includes(store.getters.userInfo.roletype)) { fields.push('account') } return fields }, details: ['name', 'fingerprint', 'account', 'domain'], related: [{ name: 'vm', title: 'label.instances', param: 'keypair' }], actions: [ { api: 'createSSHKeyPair', icon: 'plus', label: 'label.create.ssh.key.pair', docHelp: 'adminguide/virtual_machines.html#creating-the-ssh-keypair', listView: true, popup: true, component: () => import('@/views/compute/CreateSSHKeyPair.vue') }, { api: 'deleteSSHKeyPair', icon: 'delete', label: 'label.remove.ssh.key.pair', dataView: true, args: ['name', 'account', 'domainid'], mapping: { name: { value: (record, params) => { return record.name } }, account: { value: (record, params) => { return record.account } }, domainid: { value: (record, params) => { return record.domainid } } } } ] }, { name: 'affinitygroup', title: 'label.affinity.groups', icon: 'swap', docHelp: 'adminguide/virtual_machines.html#affinity-groups', permission: ['listAffinityGroups'], columns: () => { var fields = ['name', 'type', 'description'] if (['Admin', 'DomainAdmin'].includes(store.getters.userInfo.roletype)) { fields.push('account') } return fields }, details: ['name', 'id', 'description', 'type', 'account', 'domain'], related: [{ name: 'vm', title: 'label.instances', param: 'affinitygroupid' }], actions: [ { api: 'createAffinityGroup', icon: 'plus', label: 'label.add.affinity.group', docHelp: 'adminguide/virtual_machines.html#creating-a-new-affinity-group', listView: true, args: ['name', 'description', 'type'], mapping: { type: { options: ['host anti-affinity', 'host affinity'] } } }, { api: 'deleteAffinityGroup', icon: 'delete', label: 'label.delete.affinity.group', docHelp: 'adminguide/virtual_machines.html#delete-an-affinity-group', message: 'message.delete.affinity.group', dataView: true } ] } ] }