Class: Azure::ResourceManagement::ARMInterface

Inherits:
AzureInterface show all
Includes:
ARM::ARMBase, ARM::ARMDeploymentTemplate, ARM::Compute, ARM::Compute::Models, ARM::Network, ARM::Network::Models, ARM::Resources, ARM::Resources::Models, ARM::Storage, ARM::Storage::Models, ARM::VnetConfig
Defined in:
lib/azure/resource_management/ARM_interface.rb

Instance Attribute Summary collapse

Attributes inherited from AzureInterface

#ui

Instance Method Summary collapse

Methods included from ARM::VnetConfig

#add_subnet, #create_vnet_config, #divide_network, #get_vnet, #in_use_network?, #new_subnet_address_prefix, #sort_available_networks, #sort_pools, #sort_subnets_by_cidr_prefix, #sort_used_networks_by_hosts_size, #subnet, #subnet_address_prefix, #subnet_cidr_prefix, #subnets_list, #subnets_list_for_specific_address_space, #vnet_address_spaces

Methods included from ARM::ARMDeploymentTemplate

#create_deployment_parameters, #create_deployment_template, #ohai_hints, #tcp_ports

Methods included from ARM::ARMBase

#get_vm_size

Methods inherited from AzureInterface

#create_affinity_group, #create_internal_lb, #create_vnet, #list_affinity_groups, #list_internal_lb, #list_vnets

Methods included from Helpers

#display_list, #msg_pair, #random_string, #strip_non_ascii

Methods included from CustomErrors

included

Constructor Details

#initialize(params = {}) ⇒ ARMInterface

Returns a new instance of ARMInterface.



46
47
48
49
50
51
52
53
54
55
# File 'lib/azure/resource_management/ARM_interface.rb', line 46

def initialize(params = {})
  if params[:azure_client_secret]
    token_provider = MsRestAzure::ApplicationTokenProvider.new(params[:azure_tenant_id], params[:azure_client_id], params[:azure_client_secret])
  else
    token_provider = MsRest::StringTokenProvider.new(params[:token],params[:tokentype])
  end
  @credentials = MsRest::TokenCredentials.new(token_provider)
  @azure_subscription_id = params[:azure_subscription_id]
  super
end

Instance Attribute Details

#connectionObject

Returns the value of attribute connection.



44
45
46
# File 'lib/azure/resource_management/ARM_interface.rb', line 44

def connection
  @connection
end

Instance Method Details

#common_arm_rescue_block(error) ⇒ Object



486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
# File 'lib/azure/resource_management/ARM_interface.rb', line 486

def common_arm_rescue_block(error)
  if error.class == MsRestAzure::AzureOperationError && error.body
    err_json = JSON.parse(error.response.body)
    err_details = err_json["error"]["details"] if err_json["error"]
    if err_details
      err_details.each do |err|
        begin
          ui.error(JSON.parse(err["message"])["error"]["message"])
        rescue JSON::ParserError => e
          ui.error(err["message"])
        end
      end
    else
      ui.error(err_json["error"]["message"])
    end
    Chef::Log.debug(error.response.body)
  else
    begin
      JSON.parse(error.message)
      Chef::Log.debug("#{error.message}")
    rescue JSON::ParserError => e
      ui.error("#{error.message}")
    end
    ui.error("Something went wrong. Please use -VV option for more details.")
    Chef::Log.debug("#{error.backtrace.join("\n")}")
  end
end

#compute_management_clientObject



65
66
67
68
69
70
71
# File 'lib/azure/resource_management/ARM_interface.rb', line 65

def compute_management_client
  @compute_management_client ||= begin
     compute_management_client = ComputeManagementClient.new(@credentials)
     compute_management_client.subscription_id = @azure_subscription_id
     compute_management_client
  end
end

#create_resource_group(params = {}) ⇒ Object



404
405
406
407
408
409
410
411
412
413
414
415
416
417
# File 'lib/azure/resource_management/ARM_interface.rb', line 404

def create_resource_group(params = {})
  resource_group = ResourceGroup.new()
  resource_group.name = params[:azure_resource_group_name]
  resource_group.location = params[:azure_service_location]

  begin
    resource_group = resource_management_client.resource_groups.create_or_update(resource_group.name, resource_group)
  rescue Exception => e
    Chef::Log.error("Failed to create the Resource Group -- exception being rescued: #{e.to_s}")
    common_arm_rescue_block(e)
  end

  resource_group
end

#create_server(params = {}) ⇒ Object



322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
# File 'lib/azure/resource_management/ARM_interface.rb', line 322

def create_server(params = {})
  platform(params[:azure_image_reference_offer])
  # resource group creation
  if resource_group_exist?(params[:azure_resource_group_name])
    ui.log("INFO:Resource Group #{params[:azure_resource_group_name]} already exist. Skipping its creation.")
    ui.log("INFO:Adding new VM #{params[:azure_vm_name]} to this resource group.")
  else
    ui.log("Creating ResourceGroup....\n\n")
    resource_group = create_resource_group(params)
    Chef::Log.info("ResourceGroup creation successfull.")
    Chef::Log.info("Resource Group name is: #{resource_group.name}")
    Chef::Log.info("Resource Group ID is: #{resource_group.id}")
  end

  # virtual machine creation
  if virtual_machine_exist?(params[:azure_resource_group_name], params[:azure_vm_name])
    ui.log("INFO:Virtual Machine #{params[:azure_vm_name]} already exist under the Resource Group #{params[:azure_resource_group_name]}. Exiting for now.")
  else
    params[:chef_extension_version] = params[:chef_extension_version].nil? ? get_latest_chef_extension_version(params) : params[:chef_extension_version]
    params[:vm_size] = get_vm_size(params[:azure_vm_size])
    params[:vnet_config] = create_vnet_config(
      params[:azure_resource_group_name],
      params[:azure_vnet_name],
      params[:azure_vnet_subnet_name]
    )
    if params[:tcp_endpoints]
      if @platform == 'Windows'
        params[:tcp_endpoints] = params[:tcp_endpoints] + ",3389"
      else
        params[:tcp_endpoints] = params[:tcp_endpoints] + ",22,16001"
      end
      random_no = rand(100..1000)
      params[:azure_sec_group_name] = params[:azure_vm_name] + '_sec_grp_' + random_no.to_s
      if security_group_exist?(params[:azure_resource_group_name], params[:azure_sec_group_name])
        random_no = rand(100..1000)
        params[:azure_sec_group_name] = params[:azure_vm_name] + '_sec_grp_' + random_no.to_s
      end
    end

    ui.log("Creating Virtual Machine....")
    deployment = create_virtual_machine_using_template(params)
    ui.log("Virtual Machine creation successfull.") unless deployment.nil?

    unless deployment.nil?
      ui.log("Deployment name is: #{deployment.name}")
      ui.log("Deployment ID is: #{deployment.id}")
      deployment.properties.dependencies.each do |deploy|
        if deploy.resource_type == "Microsoft.Compute/virtualMachines"
          if params[:chef_extension_public_param][:extendedLogs] == "true"
            print "\n\nWaiting for the first chef-client run on virtual machine #{deploy.resource_name}"
            fetch_chef_client_logs(params[:azure_resource_group_name],
              deploy.resource_name,
              params[:chef_extension],
              Time.now
            )
          end

          ui.log("VM Details ...")
          ui.log("-------------------------------")
          ui.log("Virtual Machine name is: #{deploy.resource_name}")
          ui.log("Virtual Machine ID is: #{deploy.id}")
          show_server(deploy.resource_name, params[:azure_resource_group_name])
        end
      end
    end
  end
end

#create_virtual_machine_using_template(params) ⇒ Object



419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
# File 'lib/azure/resource_management/ARM_interface.rb', line 419

def create_virtual_machine_using_template(params)
  template = create_deployment_template(params)
  parameters = create_deployment_parameters(params, @platform)

  deploy_prop = DeploymentProperties.new
  deploy_prop.template = template
  deploy_prop.parameters = parameters
  deploy_prop.mode = 'Incremental'

  deploy_params = Deployment.new
  deploy_params.properties = deploy_prop

  deployment = resource_management_client.deployments.create_or_update(params[:azure_resource_group_name], "#{params[:azure_vm_name]}_deploy", deploy_params)
  deployment
end

#create_vm_extension(params) ⇒ Object



435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
# File 'lib/azure/resource_management/ARM_interface.rb', line 435

def create_vm_extension(params)
  vm_ext = VirtualMachineExtension.new
  vm_ext.name = params[:chef_extension]
  vm_ext.location = params[:azure_service_location]
  vm_ext.publisher = params[:chef_extension_publisher]
  vm_ext.virtual_machine_extension_type = params[:chef_extension]
  vm_ext.type_handler_version = params[:chef_extension_version].nil? ? get_latest_chef_extension_version(params) : params[:chef_extension_version]
  vm_ext.auto_upgrade_minor_version = false
  vm_ext.settings = params[:chef_extension_public_param]
  vm_ext.protected_settings = params[:chef_extension_private_param]
  begin
    vm_extension = compute_management_client.virtual_machine_extensions.create_or_update(
      params[:azure_resource_group_name],
      params[:azure_vm_name],
      vm_ext.name,
      vm_ext
    )
  rescue Exception => error
    Chef::Log.error("Failed to create the Virtual Machine Extension -- exception being rescued.")
    common_arm_rescue_block(error)
  end

  vm_extension
end

#delete_resource_group(resource_group_name) ⇒ Object



477
478
479
480
481
482
483
484
# File 'lib/azure/resource_management/ARM_interface.rb', line 477

def delete_resource_group(resource_group_name)
  ui.info 'Resource group deletion takes some time. Please wait ...'
  
  begin
    server = resource_management_client.resource_groups.delete(resource_group_name)
  end until server.nil?
  puts "\n"
end

#delete_server(resource_group_name, vm_name) ⇒ Object



122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
# File 'lib/azure/resource_management/ARM_interface.rb', line 122

def delete_server(resource_group_name, vm_name)
  server = compute_management_client.virtual_machines.get(resource_group_name, vm_name)
  if server && server.name == vm_name
    puts "\n\n"
    msg_pair(ui, 'VM Name', server.name)
    msg_pair(ui, 'VM Size', server.hardware_profile.vm_size)
    msg_pair(ui, 'VM OS', server.storage_profile.os_disk.os_type)
    puts "\n"

    begin
      ui.confirm('Do you really want to delete this server')
    rescue SystemExit   # Need to handle this as confirming with N/n raises SystemExit exception
      server = nil      # Cleanup is implicitly performed in other cloud plugins
      exit!
    end

    ui.info 'Deleting ..'

    begin
      server_detail = compute_management_client.virtual_machines.delete(resource_group_name, vm_name)
    end until server_detail.nil?

    puts "\n"
    ui.warn "Deleted server #{vm_name}"
  end
end

#extension_already_installed?(server) ⇒ Boolean

Returns:

  • (Boolean)


460
461
462
463
464
465
# File 'lib/azure/resource_management/ARM_interface.rb', line 460

def extension_already_installed?(server)
  server.resources.each do |extension|
    return true if (extension.virtual_machine_extension_type == "ChefClient" || extension.virtual_machine_extension_type == "LinuxChefClient")
  end if server.resources
  false
end

#fetch_chef_client_logs(resource_group_name, virtual_machine_name, chef_extension_name, fetch_process_start_time, fetch_process_wait_timeout = 30) ⇒ Object



282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
# File 'lib/azure/resource_management/ARM_interface.rb', line 282

def fetch_chef_client_logs(resource_group_name, virtual_machine_name, chef_extension_name, fetch_process_start_time, fetch_process_wait_timeout = 30)
  ## fetch substatus field which contains the chef-client run logs ##
  substatus = fetch_substatus(resource_group_name, virtual_machine_name, chef_extension_name)

  unless substatus.nil?
    ## chef-client run logs becomes available ##
    status = parse_substatus_code(substatus.code, 2)
    message = substatus.message

    puts "\n\n******** Please find the chef-client run details below ********\n\n"
    print "----> chef-client run status: "
    case status
      when 'succeeded'
        ## chef-client run succeeded ##
        color = :green
      when 'failed'
        ## chef-client run failed ##
        color = :red
      when 'transitioning'
        ## chef-client run did not complete within maximum timeout of 30 minutes ##
        ## fetch whatever logs available under the chef-client.log file ##
        color = :yellow
      end
      puts "#{ui.color(status, color, :bold)}"
      puts "----> chef-client run logs: "
      puts "\n#{message}\n"  ## message field of substatus contains the chef-client run logs ##
  else
    ## unavailability of the substatus field indicates that chef-client run is not completed yet on the server ##
    fetch_process_wait_time = ((Time.now - fetch_process_start_time) / 60).round
    if fetch_process_wait_time <= fetch_process_wait_timeout
      print "#{ui.color('.', :bold)}"
      sleep 30
      fetch_chef_client_logs(resource_group_name, virtual_machine_name, chef_extension_name, fetch_process_start_time, fetch_process_wait_timeout)
    else
      ## wait time exceeded 30 minutes timeout ##
      ui.error "\nchef-client run logs could not be fetched since fetch process exceeded wait timeout of #{fetch_process_wait_timeout} minutes.\n"
    end
  end
end

#fetch_substatus(resource_group_name, virtual_machine_name, chef_extension_name) ⇒ Object



263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
# File 'lib/azure/resource_management/ARM_interface.rb', line 263

def fetch_substatus(resource_group_name, virtual_machine_name, chef_extension_name)
  substatuses = compute_management_client.virtual_machine_extensions.get(
    resource_group_name,
    virtual_machine_name,
    chef_extension_name,
    'instanceView'
  ).instance_view.substatuses

  return nil if substatuses.nil?

  substatuses.each do |substatus|
    if parse_substatus_code(substatus.code, 1) == 'Chef Client run logs'
      return substatus
    end
  end

  return nil
end

#find_server(resource_group, name) ⇒ Object



208
209
210
# File 'lib/azure/resource_management/ARM_interface.rb', line 208

def find_server(resource_group, name)
  compute_management_client.virtual_machines.get(resource_group, name)
end

#get_latest_chef_extension_version(params) ⇒ Object



467
468
469
470
471
472
473
474
475
# File 'lib/azure/resource_management/ARM_interface.rb', line 467

def get_latest_chef_extension_version(params)
  ext_version = compute_management_client.virtual_machine_extension_images.list_versions(
    params[:azure_service_location],
    params[:chef_extension_publisher],
    params[:chef_extension]).last.name
  ext_version_split_values = ext_version.split(".")
  ext_version = ext_version_split_values[0] + "." + ext_version_split_values[1]
  ext_version
end

#list_imagesObject



89
90
# File 'lib/azure/resource_management/ARM_interface.rb', line 89

def list_images
end

#list_servers(resource_group_name = nil) ⇒ Object



92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
# File 'lib/azure/resource_management/ARM_interface.rb', line 92

def list_servers(resource_group_name = nil)
  if resource_group_name.nil?
    servers = compute_management_client.virtual_machines.list_all
  else
    servers = compute_management_client.virtual_machines.list(resource_group_name)
  end

  cols = ['VM Name', 'Resource Group Name', 'Location', 'Provisioning State', 'OS Type']
  rows =  []

  servers.each do |server|
    rows << server.name.to_s
    rows << server.id.split('/')[4].downcase
    rows << server.location.to_s
    rows << begin
                     state = server.provisioning_state.to_s.downcase
                     case state
                     when 'failed'
                       ui.color(state, :red)
                     when 'succeeded'
                       ui.color(state, :green)
                     else
                       ui.color(state, :yellow)
                     end
                   end
    rows << server.storage_profile.os_disk.os_type.to_s
  end
  display_list(ui, cols, rows)
end

#network_resource_clientObject



81
82
83
84
85
86
87
# File 'lib/azure/resource_management/ARM_interface.rb', line 81

def network_resource_client
  @network_resource_client ||= begin
    network_resource_client = NetworkManagementClient.new(@credentials)
    network_resource_client.subscription_id = @azure_subscription_id
    network_resource_client
  end
end

#parse_substatus_code(code, index) ⇒ Object



259
260
261
# File 'lib/azure/resource_management/ARM_interface.rb', line 259

def parse_substatus_code(code, index)
  code.split('/')[index]
end

#platform(image_reference) ⇒ Object



248
249
250
251
252
253
254
255
256
257
# File 'lib/azure/resource_management/ARM_interface.rb', line 248

def platform(image_reference)
  @platform ||= begin
    if image_reference =~ /WindowsServer.*/
      platform = 'Windows'
    else
      platform = 'Linux'
    end
    platform
  end
end

#resource_group_exist?(resource_group_name) ⇒ Boolean

Returns:

  • (Boolean)


244
245
246
# File 'lib/azure/resource_management/ARM_interface.rb', line 244

def resource_group_exist?(resource_group_name)
  resource_management_client.resource_groups.check_existence(resource_group_name)
end

#resource_management_clientObject



57
58
59
60
61
62
63
# File 'lib/azure/resource_management/ARM_interface.rb', line 57

def resource_management_client
  @resource_management_client ||= begin
    resource_management_client = ResourceManagementClient.new(@credentials)
    resource_management_client.subscription_id = @azure_subscription_id
    resource_management_client
  end
end

#security_group_exist?(resource_group_name, security_group_name) ⇒ Boolean

Returns:

  • (Boolean)


228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
# File 'lib/azure/resource_management/ARM_interface.rb', line 228

def security_group_exist?(resource_group_name, security_group_name)
  begin
    network_resource_client.network_security_groups.get(resource_group_name, security_group_name)
    return true
  rescue MsRestAzure::AzureOperationError => error
    if error.body
      err_json = JSON.parse(error.response.body)
      if err_json['error']['code'] == "ResourceNotFound"
        return false
      else
        raise error
      end
    end
  end
end

#show_server(name, resource_group) ⇒ Object



149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
# File 'lib/azure/resource_management/ARM_interface.rb', line 149

def show_server(name, resource_group)
  server = find_server(resource_group, name)
  if server
    network_interface_name = server.network_profile.network_interfaces[0].id.split('/')[-1]
    network_interface_data = network_resource_client.network_interfaces.get(resource_group, network_interface_name)
    public_ip_id_data = network_interface_data.ip_configurations[0].public_ipaddress
    unless public_ip_id_data.nil?
      public_ip_name = public_ip_id_data.id.split('/')[-1]
      public_ip_data = network_resource_client.public_ipaddresses.get(resource_group, public_ip_name)
    else
      public_ip_data = nil
    end

    details = Array.new
    details << ui.color('Server Name', :bold, :cyan)
    details << server.name

    details << ui.color('Size', :bold, :cyan)
    details << server.hardware_profile.vm_size

    details << ui.color('Provisioning State', :bold, :cyan)
    details << server.provisioning_state

    details << ui.color('Location', :bold, :cyan)
    details << server.location

    details << ui.color('Publisher', :bold, :cyan)
    details << server.storage_profile.image_reference.publisher

    details << ui.color('Offer', :bold, :cyan)
    details << server.storage_profile.image_reference.offer

    details << ui.color('Sku', :bold, :cyan)
    details << server.storage_profile.image_reference.sku

    details << ui.color('Version', :bold, :cyan)
    details << server.storage_profile.image_reference.version

    details << ui.color('OS Type', :bold, :cyan)
    details << server.storage_profile.os_disk.os_type

    details << ui.color('Public IP address', :bold, :cyan)
    unless public_ip_data.nil?
      details << public_ip_data.ip_address
    else
      details << ' -- '
    end

    details << ui.color('FQDN', :bold, :cyan)
    unless public_ip_data.nil? or public_ip_data.dns_settings.nil?
      details << public_ip_data.dns_settings.fqdn
    else
      details << ' -- '
    end

    puts ui.list(details, :columns_across, 2)
  end
end

#storage_management_clientObject



73
74
75
76
77
78
79
# File 'lib/azure/resource_management/ARM_interface.rb', line 73

def storage_management_client
  @storage_management_client ||= begin
    storage_management_client = StorageManagementClient.new(@credentials)
    storage_management_client.subscription_id = @azure_subscription_id
    storage_management_client
  end
end

#virtual_machine_exist?(resource_group_name, vm_name) ⇒ Boolean

Returns:

  • (Boolean)


212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
# File 'lib/azure/resource_management/ARM_interface.rb', line 212

def virtual_machine_exist?(resource_group_name, vm_name)
  begin
    compute_management_client.virtual_machines.get(resource_group_name, vm_name)
    return true
  rescue MsRestAzure::AzureOperationError => error
    if error.body
      err_json = JSON.parse(error.response.body)
      if err_json['error']['code'] == "ResourceNotFound"
        return false
      else
        raise error
      end
    end
  end
end

#vm_default_port(params = {}) ⇒ Object



397
398
399
400
401
402
# File 'lib/azure/resource_management/ARM_interface.rb', line 397

def vm_default_port(params = {})
  network_resource_client.network_security_groups.get(
    params[:azure_resource_group_name],
    params[:azure_vm_name]
  ).value!.body.properties.security_rules[0].properties.destination_port_range
end

#vm_public_ip(params = {}) ⇒ Object



390
391
392
393
394
395
# File 'lib/azure/resource_management/ARM_interface.rb', line 390

def vm_public_ip(params = {})
  network_resource_client.public_ipaddresses.get(
    params[:azure_resource_group_name],
    params[:azure_vm_name]
  ).value!.body.properties.ip_address
end