Class: VsphereHelper

Inherits:
Object
  • Object
show all
Defined in:
lib/beaker/hypervisor/vsphere_helper.rb

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(vInfo) ⇒ VsphereHelper

Returns a new instance of VsphereHelper.



6
7
8
9
10
11
12
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 6

def initialize vInfo
  @logger = vInfo[:logger] || Beaker::Logger.new
  @connection = RbVmomi::VIM.connect :host     => vInfo[:server],
                                     :user     => vInfo[:user],
                                     :password => vInfo[:pass],
                                     :insecure => true
end

Class Method Details

.load_config(dot_fog = '.fog') ⇒ Object



14
15
16
17
18
19
20
21
22
23
24
25
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 14

def self.load_config(dot_fog = '.fog')
  # support Fog/Cloud Provisioner layout
  # (ie, someplace besides my made up conf)
  vsphere_credentials = nil
  if File.exists?( dot_fog )
    vsphere_credentials = load_fog_credentials(dot_fog)
  else
    raise ArgumentError, ".fog file '#{dot_fog}' does not exist"
  end

  return vsphere_credentials
end

.load_fog_credentials(dot_fog = '.fog') ⇒ Object



27
28
29
30
31
32
33
34
35
36
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 27

def self.load_fog_credentials(dot_fog = '.fog')
  vInfo = YAML.load_file( dot_fog )

  vsphere_credentials = {}
  vsphere_credentials[:server] = vInfo[:default][:vsphere_server]
  vsphere_credentials[:user]   = vInfo[:default][:vsphere_username]
  vsphere_credentials[:pass]   = vInfo[:default][:vsphere_password]

  return vsphere_credentials
end

Instance Method Details

#closeObject



200
201
202
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 200

def close
  @connection.close
end

#find_customization(name) ⇒ Object



58
59
60
61
62
63
64
65
66
67
68
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 58

def find_customization name
  csm = @connection.serviceContent.customizationSpecManager

  begin
    customizationSpec = csm.GetCustomizationSpec({:name => name}).spec
  rescue
    customizationSpec = nil
  end

  return customizationSpec
end

#find_datastore(dc, datastorename) ⇒ Object



121
122
123
124
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 121

def find_datastore(dc,datastorename)
  datacenter = @connection.serviceInstance.find_datacenter(dc)
  datacenter.find_datastore(datastorename)
end

#find_folder(dc, foldername) ⇒ Object



126
127
128
129
130
131
132
133
134
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 126

def find_folder(dc,foldername)
  datacenter = @connection.serviceInstance.find_datacenter(dc)
  base = datacenter.vmFolder.traverse(foldername)
  if base != nil
    base
  else
    abort "Failed to find folder #{foldername}"
  end
end

#find_pool(dc, poolname) ⇒ Object



136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 136

def find_pool(dc,poolname)
  datacenter = @connection.serviceInstance.find_datacenter(dc)
  base = datacenter.hostFolder
  pools = poolname.split('/')
  pools.each do |pool|
    case base
      when RbVmomi::VIM::Folder
        base = base.childEntity.find { |f| f.name == pool }
      when RbVmomi::VIM::ClusterComputeResource
        base = base.resourcePool.resourcePool.find { |f| f.name == pool }
      when RbVmomi::VIM::ResourcePool
        base = base.resourcePool.find { |f| f.name == pool }
      else
        abort "Unexpected object type encountered (#{base.class}) while finding resource pool"
    end
  end

  base = base.resourcePool unless base.is_a?(RbVmomi::VIM::ResourcePool) and base.respond_to?(:resourcePool)
  base
end

#find_snapshot(vm, snapname) ⇒ Object



38
39
40
41
42
43
44
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 38

def find_snapshot vm, snapname
  if vm.snapshot
    search_child_snaps vm.snapshot.rootSnapshotList, snapname
  else
    raise "vm #{vm.name} has no snapshots to revert to"
  end
end

#find_vms(names, connection = @connection) ⇒ Object

an easier wrapper around the horrid PropertyCollector interface, necessary for searching VMs in all Datacenters that may be nested within folders of arbitrary depth returns a hash array of <name> => <VirtualMachine ManagedObjects>



74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
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
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 74

def find_vms names, connection = @connection
  names = names.is_a?(Array) ? names : [ names ]
  containerView = get_base_vm_container_from connection
  propertyCollector = connection.propertyCollector

  objectSet = [{
    :obj => containerView,
    :skip => true,
    :selectSet => [ RbVmomi::VIM::TraversalSpec.new({
        :name => 'gettingTheVMs',
        :path => 'view',
        :skip => false,
        :type => 'ContainerView'
    }) ]
  }]

  propSet = [{
    :pathSet => [ 'name' ],
    :type => 'VirtualMachine'
  }]

  results = propertyCollector.RetrievePropertiesEx({
    :specSet => [{
      :objectSet => objectSet,
      :propSet   => propSet
    }],
    :options => { :maxObjects => nil }
  })

  vms = {}
  results.objects.each do |result|
    name = result.propSet.first.val
    next unless names.include? name
    vms[name] = result.obj
  end

  while results.token do
    results = propertyCollector.ContinueRetrievePropertiesEx({:token => results.token})
    results.objects.each do |result|
      name = result.propSet.first.val
      next unless names.include? name
      vms[name] = result.obj
    end
  end
  vms
end

#get_base_vm_container_from(connection) ⇒ Object



157
158
159
160
161
162
163
164
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 157

def get_base_vm_container_from connection
  viewManager = connection.serviceContent.viewManager
  viewManager.CreateContainerView({
    :container => connection.serviceContent.rootFolder,
    :recursive => true,
    :type      => [ 'VirtualMachine' ]
  })
end

#search_child_snaps(tree, snapname) ⇒ Object



46
47
48
49
50
51
52
53
54
55
56
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 46

def search_child_snaps tree, snapname
  snapshot = nil
  tree.each do |child|
    if child.name == snapname
      snapshot ||= child.snapshot
    else
      snapshot ||= search_child_snaps child.childSnapshotList, snapname
    end
  end
  snapshot
end

#wait_for_tasks(tasks, try, attempts) ⇒ Object



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
# File 'lib/beaker/hypervisor/vsphere_helper.rb', line 166

def wait_for_tasks tasks, try, attempts
  obj_set = tasks.map { |task| { :obj => task } }
  filter = @connection.propertyCollector.CreateFilter(
    :spec => {
      :propSet => [{ :type => 'Task',
                     :all  => false,
                     :pathSet => ['info.state']}],
      :objectSet => obj_set
    },
    :partialUpdates => false
  )
  ver = ''
  while true
    result = @connection.propertyCollector.WaitForUpdates(:version => ver)
    ver = result.version
    complete = 0
    tasks.each do |task|
      if ['success', 'error'].member? task.info.state
        complete += 1
      end
    end
    break if (complete == tasks.length)
    if try <= attempts
      sleep 5
      try += 1
    else
      raise "unable to complete Vsphere tasks before timeout"
    end
  end

  filter.DestroyPropertyFilter
  tasks
end