# This Model contains code modified as per azure-sdk # and removed dependencies from fog-azure-rm. require 'base64' module ForemanAzureRm class AzureRm < ComputeResource include VMExtensions::ManagedVM alias_attribute :sub_id, :user alias_attribute :secret_key, :password alias_attribute :region, :url alias_attribute :tenant, :uuid validates :user, :password, :url, :uuid, :app_ident, :presence => true has_one :key_pair, :foreign_key => :compute_resource_id, :dependent => :destroy before_create :test_connection, :setup_key_pair class VMContainer attr_accessor :virtualmachines delegate :each, to: :virtualmachines def initialize @virtualmachines = [] end def all(_options = {}) @virtualmachines end end def app_ident attrs[:app_ident] end def app_ident=(name) attrs[:app_ident] = name end def sdk @sdk ||= ForemanAzureRm::AzureSdkAdapter.new(tenant, app_ident, secret_key, sub_id) end def to_label "#{name} (#{provider_friendly_name})" end def self.model_name ComputeResource.model_name end def self.provider_friendly_name 'Azure Resource Manager' end def capabilities [:image] end def self.regions [ ['West Europe', 'westeurope'], ['Central US', 'centralus'], ['South Central US', 'southcentralus'], ['North Central US', 'northcentralus'], ['West Central US', 'westcentralus'], ['East US', 'eastus'], ['East US 2', 'eastus2'], ['West US', 'westus'], ['West US 2', 'westus2'] ] end validates :region, inclusion: { in: regions.collect(&:second), message: "%{value} must be lowercase eg. 'eastus'. No special characters allowed." } def resource_groups sdk.rgs end def test_connection(options = {}) sdk.rgs.each do |rg| puts "#{rg}" end super(options) end def new_vm(args = {}) return AzureRmCompute.new(sdk: sdk) if args.empty? opts = vm_instance_defaults.merge(args.to_h).deep_symbolize_keys # convert rails nested_attributes into a plain hash nested_args = opts.delete(:interfaces_attributes) opts[:interfaces] = nested_attributes_for(:interfaces, nested_args) if nested_args opts.reject! { |k, v| v.nil? } raw_vm = initialize_vm(location: region, resource_group: opts[:resource_group], vm_size: opts[:vm_size], username: opts[:username], password: opts[:password], platform: opts[:platform], ssh_key_data: opts[:ssh_key_data], os_disk_caching: opts[:os_disk_caching], premium_os_disk: opts[:premium_os_disk] ) if opts[:interfaces].present? ifaces = [] opts[:interfaces].each_with_index do |iface_attrs, i| ifaces << new_interface(iface_attrs) end end AzureRmCompute.new(azure_vm: raw_vm ,sdk: sdk, resource_group: opts[:resource_group], nics: ifaces) end def provided_attributes super.merge({ :ip => :provisioning_ip_address }) end def available_vnets(attr = {}) virtual_networks end def available_networks(attr = {}) subnets end def virtual_networks @virtual_networks ||= sdk.vnets.select { |vnet| vnet.location == region } end def subnets vnets = virtual_networks subnets = [] vnets.each do |vnet| subnets.concat(sdk.subnets(vnet.resource_group, vnet.name)) end subnets end alias_method :available_subnets, :subnets def new_interface(attrs = {}) args = { :network => "", :public_ip => "", :private_ip => false, 'persisted?' => false }.merge(attrs.to_h) OpenStruct.new(args) end def editable_network_interfaces? true end def vm_sizes sdk.list_vm_sizes(region) end def associated_host(vm) associate_by("ip", [vm.public_ip_address, vm.private_ip_address]) end def vm_instance_defaults super.deep_merge( interfaces: [new_interface] ) end def vm_nics(vm) ifaces = [] vm.network_profile.network_interfaces.each do |nic| nic_rg = (split_nic_id = nic.id.split('/'))[4] nic_name = split_nic_id[-1] ifaces << sdk.vm_nic(nic_rg, nic_name) end ifaces end def vms(attrs = {}) container = VMContainer.new # Load all vms of the region sdk.list_vms(region).each do |vm| container.virtualmachines << AzureRmCompute.new(azure_vm: vm, sdk:sdk, nics: vm_nics(vm)) end container end def setup_key_pair require 'sshkey' name = "foreman-#{id}#{Foreman.uuid}" key = ::SSHKey.generate build_key_pair :name => name, :secret => key.private_key, :public => key.ssh_public_key end def find_vm_by_uuid(uuid) vm = vms.all.find { |vm| vm.name == uuid } raise ActiveRecord::RecordNotFound unless vm.present? vm end # user data support def user_data_supported? true end def create_vm(args = {}) args = args.to_h.deep_symbolize_keys args[:vm_name] = args[:name].split('.')[0] nics = create_nics(region, args) if args[:password].present? && !args[:ssh_key_data].present? sudoers_cmd = "$echo #{args[:password]} | sudo -S echo '\"#{args[:username]}\" ALL=(ALL) NOPASSWD: ALL' > /etc/sudoers.d/waagent" if args[:script_command].present? # to run the script_cmd given through form as username args[:script_command] = sudoers_cmd + " ; su - \"#{args[:username]}\" -c \"#{args[:script_command]}\"" else args[:script_command] = sudoers_cmd end disable_password_auth = false elsif args[:ssh_key_data].present? && !args[:password].present? disable_password_auth = true else disable_password_auth = false end vm = create_managed_virtual_machine( name: args[:vm_name], location: region, resource_group: args[:resource_group], vm_size: args[:vm_size], username: args[:username], password: args[:password], ssh_key_data: args[:ssh_key_data], disable_password_authentication: disable_password_auth, network_interface_card_ids: nics.map(&:id), platform: args[:platform], vhd_path: args[:image_id], os_disk_caching: args[:os_disk_caching], premium_os_disk: args[:premium_os_disk], custom_data: args[:user_data], script_command: args[:script_command], script_uris: args[:script_uris], ) logger.debug "Virtual Machine #{args[:vm_name]} Created Successfully." create_vm_extension(region, args) # return the vm object using azure_vm AzureRmCompute.new(azure_vm: vm, sdk: sdk, resource_group: args[:resource_group], nics: vm_nics(vm)) rescue RuntimeError => e Foreman::Logging.exception('Unhandled AzureRm error', e) destroy_vm vm.id if vm raise e end def destroy_vm(uuid) vm = find_vm_by_uuid(uuid) rg_name = vm.resource_group os_disk = vm.azure_vm.storage_profile.os_disk data_disks = vm.azure_vm.storage_profile.data_disks nic_ids = vm.network_interface_card_ids sdk.delete_vm(rg_name, vm.name) nic_ids.each do |nic_id| nic = sdk.vm_nic(rg_name, nic_id.split('/')[-1]) if nic.present? public_ip = nic.ip_configurations.first.public_ipaddress sdk.delete_nic(rg_name, nic_id.split('/')[-1]) if public_ip.present? ip_id = public_ip.id sdk.delete_pip(rg_name, ip_id.split('/')[-1]) end end end if os_disk.present? sdk.delete_disk(rg_name, os_disk.name) end true rescue ActiveRecord::RecordNotFound logger.info "Could not find the selected vm." true end end end