require 'fog/vcloudng' require 'fog/compute' require 'fog/vcloudng/requests/compute/helper' class VcloudngParser < Fog::Parsers::Base def extract_attributes(attributes_xml) attributes = {} until attributes_xml.empty? if attributes_xml.first.is_a?(Array) until attributes_xml.first.empty? attribute = attributes_xml.first.shift attributes[attribute.localname] = attribute.value end else attribute = attributes_xml.shift attributes[attribute.localname] = attribute.value end end attributes end def extract_link(attributes_xml) response = {} link_attrs = extract_attributes(attributes_xml) response[:type] = link_attrs["type"] response[:rel] = link_attrs["rel"] response[:href] = link_attrs["href"] if response[:type] && response[:rel] short_type = response[:type].scan(/.*\.(.*)\+/).first.first snake_case_short_type = short_type.gsub(/([A-Z])/) { '_' + $1.downcase } response[:method_name] = response[:rel] + '_' + snake_case_short_type end response end end module Fog module Compute class Vcloudng < Fog::Service module Defaults PATH = '/api' PORT = 443 SCHEME = 'https' end module Errors class ServiceError < Fog::Errors::Error; end class Task < ServiceError; end end requires :vcloudng_username, :vcloudng_password, :vcloudng_host secrets :vcloudng_password model_path 'fog/vcloudng/models/compute' model :organization collection :organizations model :catalog collection :catalogs model :catalog_item collection :catalog_items model :vdc collection :vdcs model :vapp collection :vapps model :task collection :tasks model :vm collection :vms model :vm_customization collection :vm_customizations model :network collection :networks model :disk collection :disks model :vm_network collection :vm_networks model :tag # this is called metadata in vcloud collection :tags request_path 'fog/vcloudng/requests/compute' request :get_organizations request :get_organization request :get_catalog request :get_catalog_item request :get_vdc request :get_vapp_template request :get_vapp request :get_vms request :instantiate_vapp_template request :get_task request :get_tasks_list request :get_vm_customization request :put_vm_customization request :get_network request :get_vm_cpu request :put_vm_cpu request :get_vm_memory request :put_vm_memory request :get_vm_disks request :put_vm_disks request :get_vm_network request :put_vm_network request :get_vm_metadata request :post_vm_metadata request :put_vm_metadata_value request :delete_vm_metadata request :post_vm_poweron request :get_request request :get_href class Real include Fog::Compute::Helper attr_reader :end_point def initialize(options={}) @vcloudng_password = options[:vcloudng_password] @vcloudng_username = options[:vcloudng_username] @connection_options = options[:connection_options] || {} @host = options[:vcloudng_host] @path = options[:path] || Fog::Compute::Vcloudng::Defaults::PATH @persistent = options[:persistent] || false @port = options[:port] || Fog::Compute::Vcloudng::Defaults::PORT @scheme = options[:scheme] || Fog::Compute::Vcloudng::Defaults::SCHEME @connection = Fog::Connection.new("#{@scheme}://#{@host}:#{@port}", @persistent, @connection_options) @end_point = "#{@scheme}://#{@host}#{@path}/" end def auth_token response = @connection.request({ :expects => 200, :headers => { 'Authorization' => "Basic #{Base64.encode64("#{@vcloudng_username}:#{@vcloudng_password}").delete("\r\n")}", 'Accept' => 'application/*+xml;version=1.5' }, :host => @host, :method => 'POST', :parser => Fog::ToHashDocument.new, :path => "/api/sessions" # curl http://example.com/api/versions | grep LoginUrl }) response.headers['Set-Cookie'] end def reload @cookie = nil # verify that this makes the connection to be restored, if so use Excon::Errors::Forbidden instead of Excon::Errors::Unauthorized @connection.reset end def request(params) unless @cookie @cookie = auth_token end begin do_request(params) # this is to know if Excon::Errors::Unauthorized really happens #rescue Excon::Errors::Unauthorized # @cookie = auth_token # do_request(params) end end def do_request(params) headers = {} if @cookie headers.merge!('Cookie' => @cookie) end if params[:path] if params[:override_path] == true path = params[:path] else path = "#{@path}/#{params[:path]}" end else path = "#{@path}" end @connection.request({ :body => params[:body], :expects => params[:expects], :headers => headers.merge!(params[:headers] || {}), :host => @host, :method => params[:method], :parser => params[:parser], :path => path }) rescue => @e raise @e unless @e.class.to_s =~ /^Excon::Errors/ puts @e.response.status puts CGI::unescapeHTML(@e.response.body) raise @e end def process_task(request) response = request task = make_task_object(response.body) wait_and_raise_unless_success(task) true end def make_task_object(task_response) task_response[:id] = task_response[:href].split('/').last tasks.new(task_response) end def wait_and_raise_unless_success(task) task.wait_for { non_running? } raise Errors::Task.new "status: #{task.status}, error: #{task.error}" unless task.success? end end class Mock end end end end