-
Notifications
You must be signed in to change notification settings - Fork 900
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add TransformationMapping#search_vms_and_validate method
- Loading branch information
Showing
3 changed files
with
260 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
144 changes: 144 additions & 0 deletions
144
app/models/transformation_mapping/vm_migration_validator.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,144 @@ | ||
class TransformationMapping::VmMigrationValidator | ||
VM_CONFLICT = "conflict".freeze | ||
VM_EMPTY_NAME = "empty_name".freeze | ||
VM_IN_OTHER_PLAN = "in_other_plan".freeze | ||
VM_MIGRATED = "migrated".freeze | ||
VM_NOT_EXIST = "not_exist".freeze | ||
VM_VALID = "ok".freeze | ||
|
||
def initialize(mapping, vm_list = nil) | ||
@mapping = mapping | ||
@vm_list = vm_list | ||
end | ||
|
||
def validate | ||
@vm_list.present? ? identify_vms : select_vms | ||
end | ||
|
||
def select_vms | ||
valid_list = [] | ||
|
||
@mapping.transformation_mapping_items.where(:source_type => EmsCluster).collect(&:source).collect(&:vms).flatten.each do |vm| | ||
reason = validate_vm(vm, true) | ||
valid_list << VmMigrateStruct.new(vm, reason) if reason == VM_VALID | ||
end | ||
|
||
{"valid" => valid_list} | ||
end | ||
|
||
def identify_vms | ||
valid_list = [] | ||
invalid_list = [] | ||
conflict_list = [] | ||
|
||
@vm_list.each do |row| | ||
vm_name = row['name'] | ||
|
||
if vm_name.blank? | ||
invalid_list << VmMigrateStruct.new(nil, '') | ||
next | ||
end | ||
|
||
query = Vm.where(:name => vm_name) | ||
query = query.where(:uid_ems => row['uid_ems']) if row['uid_ems'].present? | ||
query = query.joins(:host).where(:hosts => {:name => row['host']}) if row['host'].present? | ||
query = query.joins(:ext_management_system).where(:ext_management_systems => {:name => row['provider']}) if row['provider'].present? | ||
|
||
vms = query.to_a | ||
if vms.size.zero? | ||
invalid_list << VmMigrateStruct.new(nil, vm_name) | ||
elsif vms.size == 1 | ||
reason = validate_vm(vms.first, false) | ||
(reason == VM_VALID ? valid_list : invalid_list) << VmMigrateStruct.new(vms.first, reason) | ||
else | ||
vms.each { |vm| conflict_list << VmMigrateStruct.new(vm, VM_CONFLICT) } | ||
end | ||
end | ||
|
||
{ | ||
"valid" => valid_list, | ||
"invalid" => invalid_list, | ||
"conflicted" => conflict_list | ||
} | ||
end | ||
|
||
def validate_vm(vm, quick = true) | ||
# a valid vm must find all resources in the mapping and has never been migrated | ||
invalid_list = [] | ||
|
||
unless valid_cluster?(vm) | ||
invalid_list << "cluster: %{name}" % {:name => vm.ems_cluster.name} | ||
return no_mapping_msg(invalid_list) if quick | ||
end | ||
|
||
invalid_storages = unmapped_storages(vm) | ||
if invalid_storages.present? | ||
invalid_list << "storages: %{list}" % {:list => invalid_storages.collect(&:name).join(", ")} | ||
return no_mapping_msg(invalid_list) if quick | ||
end | ||
|
||
invalid_lans = unmapped_lans(vm) | ||
if invalid_lans.present? | ||
invalid_list << "lans: %{list}" % {:list => invalid_lans.collect(&:name).join(', ')} | ||
return no_mapping_msg(invalid_list) if quick | ||
end | ||
|
||
return no_mapping_msg(invalid_list) if invalid_list.present? | ||
vm_migration_status(vm) | ||
end | ||
|
||
def vm_migration_status(vm) | ||
vm_as_resources = ServiceResource.where(:resource => vm).includes(:service_template).where(:service_templates => {:type => "ServiceTemplateTransformationPlan"}) | ||
|
||
# VM has not been migrated before | ||
return VM_VALID if vm_as_resources.blank? | ||
|
||
return VM_MIGRATED if vm_as_resources.any? { |rsc| rsc.status == ServiceResource::STATUS_COMPLETED } | ||
|
||
# VM failed in previous migration | ||
vm_as_resources.all? { |rsc| rsc.status == ServiceResource::STATUS_FAILED } ? VM_VALID : VM_IN_OTHER_PLAN | ||
end | ||
|
||
def no_mapping_msg(list) | ||
("Mapping source not found - %{list}") % {:list => list.join('. ')} | ||
end | ||
|
||
def valid_cluster?(vm) | ||
@mapping.transformation_mapping_items.where(:source => vm.ems_cluster).exists? | ||
end | ||
|
||
# return an empty array if all storages are valid for transformation | ||
# otherwise return an array of invalid datastores | ||
def unmapped_storages(vm) | ||
vm.datastores - @mapping.transformation_mapping_items.where(:source => vm.datastores).collect(&:source) | ||
end | ||
|
||
# return an empty array if all lans are valid for transformation | ||
# otherwise return an array of invalid lans | ||
def unmapped_lans(vm) | ||
vm.lans - @mapping.transformation_mapping_items.where(:source => vm.lans).collect(&:source) | ||
end | ||
|
||
class VmMigrateStruct < MiqHashStruct | ||
def initialize(vm, reason) | ||
options = | ||
if vm.nil? | ||
{ | ||
"name" => reason, | ||
"reason" => reason.blank? ? VM_EMPTY_NAME : VM_NOT_EXIST | ||
} | ||
else | ||
{ | ||
"name" => vm.name, | ||
"cluster" => vm.ems_cluster.try(:name), | ||
"path" => "#{vm.ext_management_system.name}/#{vm.parent_blue_folder_path(:exclude_non_display_folders => true)}", | ||
"allocated_size" => vm.allocated_disk_storage, | ||
"id" => vm.id, | ||
"reason" => reason | ||
} | ||
end | ||
|
||
super(options) | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters