|
module Foreman::Model
|
|
class GCE < ComputeResource
|
|
has_one :key_pair, :foreign_key => :compute_resource_id, :dependent => :destroy
|
|
before_create :setup_key_pair
|
|
validate :check_google_key_path
|
|
validates :key_path, :project, :email, :presence => true
|
|
|
|
delegate :flavors, :to => :client
|
|
|
|
def self.available?
|
|
Fog::Compute.providers.include?(:google)
|
|
end
|
|
|
|
def to_label
|
|
"#{name} (#{zone}-#{provider_friendly_name})"
|
|
end
|
|
|
|
def capabilities
|
|
[:image, :new_volume]
|
|
end
|
|
|
|
def project
|
|
attrs[:project]
|
|
end
|
|
|
|
def project=(name)
|
|
attrs[:project] = name
|
|
end
|
|
|
|
def key_path
|
|
attrs[:key_path]
|
|
end
|
|
|
|
def key_path=(name)
|
|
attrs[:key_path] = name
|
|
end
|
|
|
|
def email
|
|
attrs[:email]
|
|
end
|
|
|
|
def email=(email)
|
|
attrs[:email] = email
|
|
end
|
|
|
|
def provided_attributes
|
|
super.merge({ :ip => :vm_ip_address })
|
|
end
|
|
|
|
def zones
|
|
client.list_zones.body['items'].map { |zone| zone['name'] }
|
|
end
|
|
|
|
def networks
|
|
client.list_networks.body['items'].map { |n| n['name'] }
|
|
end
|
|
|
|
def disks
|
|
client.list_disks(zone).body['items'].map { |disk| disk['name'] }
|
|
end
|
|
|
|
def zone
|
|
url
|
|
end
|
|
|
|
def zone=(zone)
|
|
self.url = zone
|
|
end
|
|
|
|
def new_vm(args = {})
|
|
# convert rails nested_attributes into a plain hash
|
|
[:volumes].each do |collection|
|
|
nested_attrs = args.delete("#{collection}_attributes".to_sym)
|
|
args[collection] = nested_attributes_for(collection, nested_attrs) if nested_attrs
|
|
end
|
|
|
|
# Dots are not allowed in names
|
|
args[:name] = args[:name].parameterize if args[:name].present?
|
|
args[:external_ip] = args[:external_ip] == '1'
|
|
# GCE network interfaces cannot be defined though Foreman yet
|
|
args[:network_interfaces] = nil
|
|
|
|
if args[:volumes].present?
|
|
if args[:image_id].present?
|
|
args[:volumes].first[:source_image] = client.images.find { |i| i.id == args[:image_id] }.name
|
|
end
|
|
args[:disks] = []
|
|
args[:volumes].each_with_index do |vol_args, i|
|
|
args[:disks] << new_volume(vol_args.merge(:name => "#{args[:name]}-disk#{i + 1}"))
|
|
end
|
|
end
|
|
|
|
super(args)
|
|
end
|
|
|
|
def create_vm(args = {})
|
|
new_vm(args)
|
|
create_volumes(args)
|
|
|
|
username = images.find_by(:uuid => args[:image_name]).try(:username)
|
|
ssh = { :username => username, :public_key => key_pair.public }
|
|
vm = super(args.merge(ssh))
|
|
vm.disks.each { |disk| vm.set_disk_auto_delete(true, disk['deviceName']) }
|
|
vm
|
|
rescue Fog::Errors::Error => e
|
|
args[:disks].find_all(&:status).map(&:destroy) if args[:disks].present?
|
|
Foreman::Logging.exception("Unhandled GCE error", e)
|
|
raise e
|
|
end
|
|
|
|
def create_volumes(args)
|
|
args[:disks].map(&:save)
|
|
args[:disks].each { |disk| disk.wait_for { disk.ready? } }
|
|
end
|
|
|
|
def available_images
|
|
client.images
|
|
end
|
|
|
|
def self.model_name
|
|
ComputeResource.model_name
|
|
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 test_connection(options = {})
|
|
super
|
|
errors[:user].empty? && errors[:password].empty? && zones
|
|
rescue => e
|
|
errors[:base] << e.message
|
|
end
|
|
|
|
def self.provider_friendly_name
|
|
"Google"
|
|
end
|
|
|
|
def interfaces_attrs_name
|
|
:network_interfaces
|
|
end
|
|
|
|
def new_volume(attrs = { })
|
|
args = {
|
|
:size_gb => (attrs[:size_gb] || 10).to_i,
|
|
:zone_name => zone
|
|
}.merge(attrs)
|
|
client.disks.new(args)
|
|
end
|
|
|
|
def normalize_vm_attrs(vm_attrs)
|
|
normalized = slice_vm_attributes(vm_attrs, ['image_id', 'machine_type', 'network'])
|
|
|
|
normalized['external_ip'] = to_bool(vm_attrs['external_ip'])
|
|
normalized['image_name'] = self.images.find_by(:uuid => vm_attrs['image_id']).try(:name)
|
|
|
|
volume_attrs = vm_attrs['volumes_attributes'] || {}
|
|
normalized['volumes_attributes'] = volume_attrs.each_with_object({}) do |(key, vol), volumes|
|
|
volumes[key] = {
|
|
'size' => memory_gb_to_bytes(vol['size_gb']).to_s
|
|
}
|
|
end
|
|
|
|
normalized
|
|
end
|
|
|
|
private
|
|
|
|
def client
|
|
@client ||= ::Fog::Compute.new(:provider => 'google', :google_project => project, :google_client_email => email, :google_key_location => key_path)
|
|
end
|
|
|
|
def check_google_key_path
|
|
return if key_path.blank?
|
|
unless File.exist?(key_path)
|
|
errors.add(:key_path, _('Unable to access key'))
|
|
end
|
|
rescue => e
|
|
Foreman::Logging.exception("Failed to access gce key path", e)
|
|
errors.add(:key_path, e.message.to_s)
|
|
end
|
|
|
|
def vm_instance_defaults
|
|
super.merge(
|
|
:zone => zone,
|
|
:name => "foreman-#{Time.now.to_i}",
|
|
:disks => [new_volume]
|
|
)
|
|
end
|
|
end
|
|
end
|