diff --git a/app/controllers/api/base_controller/generic.rb b/app/controllers/api/base_controller/generic.rb index 47808cd78d..9b7bb713ff 100644 --- a/app/controllers/api/base_controller/generic.rb +++ b/app/controllers/api/base_controller/generic.rb @@ -246,9 +246,12 @@ def model_ident(model, type = nil) end def validate_id(id, key_id, klass) - if id.nil? || (key_id == "id" && !id.integer?) - raise BadRequestError, "Invalid #{klass} #{key_id} #{id || "nil"} specified" + id.split(',').map(&:to_i).each do |id| + if id.nil? || (key_id == "id" && !id.integer?) + raise BadRequestError, "Invalid #{klass} #{key_id} #{id || "nil"} specified" + end end + id.split(/\s*,\s*/) end end end diff --git a/app/controllers/api/base_controller/renderer.rb b/app/controllers/api/base_controller/renderer.rb index 21a32498ac..95cb87a15e 100644 --- a/app/controllers/api/base_controller/renderer.rb +++ b/app/controllers/api/base_controller/renderer.rb @@ -1,4 +1,5 @@ require 'jbuilder' +require 'byebug' module Api class BaseController @@ -14,7 +15,7 @@ def render_collection(type, resources, opts = {}) # Helper proc to render a single resource # def render_resource(type, resource, opts = {}) - render :json => resource_to_jbuilder(type, gen_reftype(type, opts), resource, opts).target!, :status => status_from_resource(resource) + render :json => resource_to_jbuilder(type, gen_reftype(type, opts), resource, opts).target!, :status => status_from_resource(resource) end # @@ -132,6 +133,9 @@ def resource_search(id, type, klass = nil, key_id = nil) target = if respond_to?("find_#{type}") public_send("find_#{type}", id) + elsif type == "vm_infras" + vm_infra_reconfigure_form(type,klass,id) + #find_resource(klass, key_id, id) else find_resource(klass, key_id, id) end @@ -149,6 +153,185 @@ def filter_resource(target, type, klass) res end + def vm_infra_reconfigure_form(type, klass, id) + @request_id = "new" + request_data = id + reconfigure_ids = request_data.split(/\s*,\s*/).map(&:to_i) + request_hash = build_reconfigure_hash(reconfigure_ids) + return request_hash + end + + def build_reconfigure_hash(reconfigure_ids) + #@req = nil + @reconfig_values = {} + if @request_id == 'new' + @reconfig_values = get_reconfig_info(reconfigure_ids) + else + @req = MiqRequest.find_by(:id => @request_id) + @reconfig_values[:src_ids] = @req.options[:src_ids] + @reconfig_values[:memory], @reconfig_values[:memory_type] = @req.options[:vm_memory] ? reconfigure_calculations(@req.options[:vm_memory]) : ['', ''] + @reconfig_values[:cores_per_socket_count] = @req.options[:cores_per_socket] ? @req.options[:cores_per_socket].to_s : '' + @reconfig_values[:socket_count] = @req.options[:number_of_sockets] ? @req.options[:number_of_sockets].to_s : '' + # check if there is only one VM that supports disk reconfiguration + + @reconfig_values[:disk_add] = @req.options[:disk_add] + @reconfig_values[:disk_resize] = @req.options[:disk_resize] + @reconfig_values[:cdrom_connect] = @req.options[:cdrom_connect] + @reconfig_values[:cdrom_disconnect] = @req.options[:cdrom_disconnect] + vmdisks = [] + vmcdroms = [] + @req.options[:disk_add]&.each do |disk| + adsize, adunit = reconfigure_calculations(disk[:disk_size_in_mb]) + vmdisks << {:hdFilename => disk[:disk_name], + :hdType => disk[:thin_provisioned] ? 'thin' : 'thick', + :hdMode => disk[:persistent] ? 'persistent' : 'nonpersistent', + :hdSize => adsize.to_s, + :hdUnit => adunit, + :new_controller_type => disk[:new_controller_type].to_s, + :cb_dependent => disk[:dependent], + :cb_bootable => disk[:bootable], + :add_remove => 'add'} + end + + reconfig_item = Vm.find(reconfigure_ids) + if reconfig_item + reconfig_item.first.hardware.disks.each do |disk| + next if disk.device_type != 'disk' + + removing = '' + delbacking = false + if disk.filename && @req.options[:disk_remove] + @req.options[:disk_remove].each do |remdisk| + if remdisk[:disk_name] == disk.filename + removing = 'remove' + delbacking = remdisk[:delete_backing] + end + end + end + dsize, dunit = reconfigure_calculations(disk.size / (1024 * 1024)) + vmdisk = {:hdFilename => disk.filename, + :hdType => disk.disk_type.to_s, + :hdMode => disk.mode.to_s, + :hdSize => dsize.to_s, + :hdUnit => dunit.to_s, + :delete_backing => delbacking, + :cb_bootable => disk.bootable, + :add_remove => removing} + vmdisks << vmdisk + end + cdroms = reconfig_item.first.hardware.cdroms + if cdroms.present? + vmcdroms = build_request_cdroms_list(cdroms) + end + end + @reconfig_values[:disks] = vmdisks + @reconfig_values[:cdroms] = vmcdroms + end + + @reconfig_values[:cb_memory] = false #if @request_id != new we need to change false + @reconfig_values[:cb_cpu] = false #if @request_id != new we need to change false + @reconfig_values + end + + def get_reconfig_info(reconfigure_ids) + @reconfigureitems = Vm.find(reconfigure_ids).sort_by(&:name) + # set memory to nil if multiple items were selected with different mem_cpu values + memory = @reconfigureitems.first.mem_cpu + memory = nil unless @reconfigureitems.all? { |vm| vm.mem_cpu == memory } + + socket_count = @reconfigureitems.first.num_cpu + socket_count = '' unless @reconfigureitems.all? { |vm| vm.num_cpu == socket_count } + + cores_per_socket = @reconfigureitems.first.cpu_cores_per_socket + cores_per_socket = '' unless @reconfigureitems.all? { |vm| vm.cpu_cores_per_socket == cores_per_socket } + memory, memory_type = reconfigure_calculations(memory) + # if only one vm that supports disk reconfiguration is selected, get the disks information + vmdisks = [] + @reconfigureitems.first.hardware.disks.order(:filename).each do |disk| + next if disk.device_type != 'disk' + + dsize, dunit = reconfigure_calculations(disk.size / (1024 * 1024)) + vmdisks << {:hdFilename => disk.filename, + :hdType => disk.disk_type, + :hdMode => disk.mode, + :hdSize => dsize, + :hdUnit => dunit, + :add_remove => '', + :cb_bootable => disk.bootable, + :vm => @reconfigureitems} + end + + # reconfiguring network adapters is only supported when one vm was selected + network_adapters = [] + vmcdroms = [] + if @reconfigureitems.size == 1 + vm = @reconfigureitems.first + + if vm.supports?(:reconfigure_network_adapters) + network_adapters = build_network_adapters_list(vm) + end + + if vm.supports?(:reconfigure_cdroms) + # CD-ROMS + vmcdroms = build_vmcdrom_list(vm) + end + end + + {:objectIds => reconfigure_ids, + :memory => memory, + :memory_type => memory_type, + :socket_count => socket_count.to_s, + :cores_per_socket_count => cores_per_socket.to_s, + :disks => vmdisks, + :network_adapters => network_adapters, + :cdroms => vmcdroms, + :vm_vendor => @reconfigureitems.first.vendor, + :vm_type => @reconfigureitems.first.class.name, + :orchestration_stack_id => @reconfigureitems.first.try(:orchestration_stack_id), + :disk_default_type => @reconfigureitems.first.try(:disk_default_type) || 'thin'} + end + + def reconfigure_calculations(mbsize) + humansize = mbsize + fmt = "MB" + if mbsize.to_i > 1024 && (mbsize.to_i % 1024).zero? + humansize = mbsize.to_i / 1024 + fmt = "GB" + end + return humansize.to_s, fmt + end + + def build_network_adapters_list(vm) + network_adapters = [] + vm.hardware.guest_devices.order(:device_name => 'asc').each do |guest_device| + lan = Lan.find_by(:id => guest_device.lan_id) + network_adapters << {:name => guest_device.device_name, :vlan => lan.name, :mac => guest_device.address, :add_remove => ''} unless lan.nil? + end + + if vm.kind_of?(ManageIQ::Providers::Vmware::CloudManager::Vm) + vm.network_ports.order(:name).each do |port| + network_adapters << { :name => port.name, :network => port.cloud_subnets.try(:first).try(:name) || _('None'), :mac => port.mac_address, :add_remove => '' } + end + end + network_adapters + end + + def build_vmcdrom_list(vm) + vmcdroms = [] + cdroms = vm.hardware.cdroms + if cdroms.present? + cdroms.map do |cd| + id = cd.id + device_name = cd.device_name + type = cd.device_type + filename = filename_string(cd.filename) + storage_id = cd.storage_id || '' + vmcdroms << {:id => id, :device_name => device_name, :filename => filename, :type => type, :storage_id => storage_id} + end + vmcdroms + end + end + def collection_search(is_subcollection, type, klass) res = if is_subcollection diff --git a/app/controllers/api/vm_infras_controller.rb b/app/controllers/api/vm_infras_controller.rb new file mode 100644 index 0000000000..502584585c --- /dev/null +++ b/app/controllers/api/vm_infras_controller.rb @@ -0,0 +1,5 @@ +module Api + class VmInfrasController < BaseProviderController + end + + end \ No newline at end of file diff --git a/config/api.yml b/config/api.yml index d9f35c7921..81c1a0dbb2 100644 --- a/config/api.yml +++ b/config/api.yml @@ -801,6 +801,17 @@ :post: - :name: delete :identifier: cloud_volume_snapshot_delete + :vm_infras: + :description: Vm Infras + :identifier: vm_infra + :options: + - :collection + :verbs: *gpppd + :klass: Vm + :resource_actions: + :get: + - :name: read + :identifier: vm_infra_view :cloud_volume_types: :description: Cloud Volume Types :identifier: cloud_volume_type @@ -2647,7 +2658,7 @@ :identifier: picture_new :policies: :description: Policies - :identifier: miq_policy + :identifier: miq_policy,, :options: - :collection - :subcollection