2011-08-11 12:53:52 -04:00
|
|
|
require 'fog/compute/models/server'
|
2010-03-16 18:46:21 -04:00
|
|
|
|
2009-09-16 02:11:06 -04:00
|
|
|
module Fog
|
2011-06-16 19:28:54 -04:00
|
|
|
module Compute
|
|
|
|
class AWS
|
2009-09-16 02:11:06 -04:00
|
|
|
|
2011-08-11 12:53:52 -04:00
|
|
|
class Server < Fog::Compute::Server
|
2011-02-23 13:45:50 -05:00
|
|
|
extend Fog::Deprecation
|
2011-02-22 19:36:15 -05:00
|
|
|
deprecate :ip_address, :public_ip_address
|
2009-09-16 02:11:06 -04:00
|
|
|
|
2010-09-07 14:30:02 -04:00
|
|
|
identity :id, :aliases => 'instanceId'
|
2009-10-24 01:23:55 -04:00
|
|
|
|
2010-04-30 17:40:29 -04:00
|
|
|
attr_accessor :architecture
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :ami_launch_index, :aliases => 'amiLaunchIndex'
|
2011-06-24 20:31:44 -04:00
|
|
|
attribute :availability_zone, :aliases => 'availabilityZone'
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :block_device_mapping, :aliases => 'blockDeviceMapping'
|
2010-10-04 19:16:03 -04:00
|
|
|
attribute :client_token, :aliases => 'clientToken'
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :dns_name, :aliases => 'dnsName'
|
2010-01-21 23:11:14 -05:00
|
|
|
attribute :groups
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :flavor_id, :aliases => 'instanceType'
|
|
|
|
attribute :image_id, :aliases => 'imageId'
|
2010-11-15 14:49:16 -05:00
|
|
|
attr_accessor :instance_initiated_shutdown_behavior
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :kernel_id, :aliases => 'kernelId'
|
|
|
|
attribute :key_name, :aliases => 'keyName'
|
|
|
|
attribute :created_at, :aliases => 'launchTime'
|
2010-11-19 16:45:45 -05:00
|
|
|
attribute :monitoring, :squash => 'state'
|
2011-06-24 20:31:44 -04:00
|
|
|
attribute :placement_group, :aliases => 'groupName'
|
2011-07-26 13:11:44 -04:00
|
|
|
attribute :platform, :aliases => 'platform'
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :product_codes, :aliases => 'productCodes'
|
|
|
|
attribute :private_dns_name, :aliases => 'privateDnsName'
|
|
|
|
attribute :private_ip_address, :aliases => 'privateIpAddress'
|
2011-02-22 19:36:15 -05:00
|
|
|
attribute :public_ip_address, :aliases => 'ipAddress'
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :ramdisk_id, :aliases => 'ramdiskId'
|
2009-09-16 02:11:06 -04:00
|
|
|
attribute :reason
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :root_device_name, :aliases => 'rootDeviceName'
|
|
|
|
attribute :root_device_type, :aliases => 'rootDeviceType'
|
2011-12-13 14:02:40 -05:00
|
|
|
attribute :security_group_ids, :aliases => 'securityGroupIds'
|
2010-11-19 16:45:45 -05:00
|
|
|
attribute :state, :aliases => 'instanceState', :squash => 'name'
|
2010-10-04 18:46:12 -04:00
|
|
|
attribute :state_reason, :aliases => 'stateReason'
|
2010-09-07 14:30:02 -04:00
|
|
|
attribute :subnet_id, :aliases => 'subnetId'
|
2011-06-24 20:31:44 -04:00
|
|
|
attribute :tenancy
|
2010-10-12 14:00:59 -04:00
|
|
|
attribute :tags, :aliases => 'tagSet'
|
2009-09-16 02:11:06 -04:00
|
|
|
attribute :user_data
|
2012-01-31 23:30:44 -05:00
|
|
|
attribute :vpc_id, :aliases => 'vpcId'
|
2009-09-16 02:11:06 -04:00
|
|
|
|
2010-10-12 19:22:12 -04:00
|
|
|
attr_accessor :password
|
2010-10-20 17:36:41 -04:00
|
|
|
attr_writer :private_key, :private_key_path, :public_key, :public_key_path, :username
|
2010-09-15 20:31:45 -04:00
|
|
|
|
2010-06-19 20:46:21 -04:00
|
|
|
def initialize(attributes={})
|
2011-12-13 12:49:19 -05:00
|
|
|
self.groups ||= ["default"] unless (attributes[:subnet_id] || attributes[:security_group_ids])
|
2011-04-15 13:43:18 -04:00
|
|
|
self.flavor_id ||= 't1.micro'
|
2011-06-24 17:27:25 -04:00
|
|
|
self.image_id ||= begin
|
|
|
|
self.username = 'ubuntu'
|
2011-04-14 19:31:58 -04:00
|
|
|
case attributes[:connection].instance_variable_get(:@region) # Ubuntu 10.04 LTS 64bit (EBS)
|
|
|
|
when 'ap-northeast-1'
|
|
|
|
'ami-5e0fa45f'
|
|
|
|
when 'ap-southeast-1'
|
|
|
|
'ami-f092eca2'
|
|
|
|
when 'eu-west-1'
|
|
|
|
'ami-3d1f2b49'
|
|
|
|
when 'us-east-1'
|
|
|
|
'ami-3202f25b'
|
|
|
|
when 'us-west-1'
|
|
|
|
'ami-f5bfefb0'
|
2012-04-14 14:30:56 -04:00
|
|
|
when 'us-west-2'
|
|
|
|
'ami-e0ec60d0'
|
2011-04-14 19:31:58 -04:00
|
|
|
end
|
|
|
|
end
|
2010-01-21 23:11:14 -05:00
|
|
|
super
|
|
|
|
end
|
|
|
|
|
2009-10-20 22:39:57 -04:00
|
|
|
def addresses
|
2009-12-05 17:53:42 -05:00
|
|
|
requires :id
|
2009-11-21 16:56:39 -05:00
|
|
|
|
2010-01-08 14:29:07 -05:00
|
|
|
connection.addresses(:server => self)
|
2009-09-18 03:01:10 -04:00
|
|
|
end
|
|
|
|
|
2010-05-01 19:20:34 -04:00
|
|
|
def console_output
|
|
|
|
requires :id
|
|
|
|
|
2010-11-19 16:45:45 -05:00
|
|
|
connection.get_console_output(id)
|
2010-05-01 19:20:34 -04:00
|
|
|
end
|
|
|
|
|
2009-09-20 12:21:03 -04:00
|
|
|
def destroy
|
2009-12-05 17:53:42 -05:00
|
|
|
requires :id
|
2009-11-21 16:56:39 -05:00
|
|
|
|
2010-11-19 16:45:45 -05:00
|
|
|
connection.terminate_instances(id)
|
2009-09-16 02:11:06 -04:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2010-10-12 19:22:12 -04:00
|
|
|
remove_method :flavor_id
|
2010-01-13 00:59:39 -05:00
|
|
|
def flavor_id
|
2010-11-19 16:45:45 -05:00
|
|
|
@flavor && @flavor.id || attributes[:flavor_id]
|
2010-01-10 16:22:18 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def flavor=(new_flavor)
|
2010-01-13 00:59:39 -05:00
|
|
|
@flavor = new_flavor
|
|
|
|
end
|
|
|
|
|
|
|
|
def flavor
|
2010-11-19 16:45:45 -05:00
|
|
|
@flavor ||= connection.flavors.all.detect {|flavor| flavor.id == flavor_id}
|
2010-01-10 16:22:18 -05:00
|
|
|
end
|
|
|
|
|
2009-09-18 11:42:47 -04:00
|
|
|
def key_pair
|
2009-11-21 16:56:39 -05:00
|
|
|
requires :key_name
|
|
|
|
|
2011-03-28 16:47:57 -04:00
|
|
|
connection.key_pairs.all(key_name).first
|
2009-09-18 11:42:47 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def key_pair=(new_keypair)
|
2010-12-01 18:29:35 -05:00
|
|
|
self.key_name = new_keypair && new_keypair.name
|
2009-09-16 02:11:06 -04:00
|
|
|
end
|
|
|
|
|
2010-09-15 20:31:45 -04:00
|
|
|
def private_key_path
|
2010-10-20 17:32:30 -04:00
|
|
|
@private_key_path ||= Fog.credentials[:private_key_path]
|
|
|
|
@private_key_path &&= File.expand_path(@private_key_path)
|
2010-09-23 20:39:25 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def private_key
|
2010-10-20 17:32:30 -04:00
|
|
|
@private_key ||= private_key_path && File.read(private_key_path)
|
2010-09-15 20:31:45 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def public_key_path
|
2010-10-20 17:32:30 -04:00
|
|
|
@public_key_path ||= Fog.credentials[:public_key_path]
|
|
|
|
@public_key_path &&= File.expand_path(@public_key_path)
|
2010-09-15 20:31:45 -04:00
|
|
|
end
|
|
|
|
|
2010-09-23 20:39:25 -04:00
|
|
|
def public_key
|
2010-10-20 17:32:30 -04:00
|
|
|
@public_key ||= public_key_path && File.read(public_key_path)
|
2010-09-23 20:39:25 -04:00
|
|
|
end
|
2010-10-20 17:32:30 -04:00
|
|
|
|
2010-01-13 00:59:39 -05:00
|
|
|
def ready?
|
2010-11-19 16:45:45 -05:00
|
|
|
state == 'running'
|
2010-01-13 00:59:39 -05:00
|
|
|
end
|
|
|
|
|
2009-11-21 13:42:20 -05:00
|
|
|
def reboot
|
2009-12-05 17:53:42 -05:00
|
|
|
requires :id
|
2010-11-19 16:45:45 -05:00
|
|
|
connection.reboot_instances(id)
|
2009-11-21 13:42:20 -05:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2009-09-16 02:11:06 -04:00
|
|
|
def save
|
2010-10-04 19:08:34 -04:00
|
|
|
raise Fog::Errors::Error.new('Resaving an existing object may create a duplicate') if identity
|
2009-11-21 16:56:39 -05:00
|
|
|
requires :image_id
|
|
|
|
|
2010-05-18 14:22:32 -04:00
|
|
|
options = {
|
2010-09-15 20:31:45 -04:00
|
|
|
'BlockDeviceMapping' => block_device_mapping,
|
2010-10-03 21:07:33 -04:00
|
|
|
'ClientToken' => client_token,
|
2010-11-15 14:49:16 -05:00
|
|
|
'InstanceInitiatedShutdownBehavior' => instance_initiated_shutdown_behavior,
|
2010-05-25 19:37:19 -04:00
|
|
|
'InstanceType' => flavor_id,
|
2010-09-15 20:31:45 -04:00
|
|
|
'KernelId' => kernel_id,
|
|
|
|
'KeyName' => key_name,
|
|
|
|
'Monitoring.Enabled' => monitoring,
|
|
|
|
'Placement.AvailabilityZone' => availability_zone,
|
2011-06-24 20:31:44 -04:00
|
|
|
'Placement.GroupName' => placement_group,
|
|
|
|
'Placement.Tenancy' => tenancy,
|
2012-03-10 14:22:35 -05:00
|
|
|
'PrivateIpAddress' => private_ip_address,
|
2010-09-15 20:31:45 -04:00
|
|
|
'RamdiskId' => ramdisk_id,
|
|
|
|
'SecurityGroup' => groups,
|
2011-12-13 12:49:19 -05:00
|
|
|
'SecurityGroupId' => security_group_ids,
|
2010-08-17 06:28:06 -04:00
|
|
|
'SubnetId' => subnet_id,
|
2010-09-15 20:31:45 -04:00
|
|
|
'UserData' => user_data
|
2010-05-18 14:22:32 -04:00
|
|
|
}
|
2010-11-15 14:49:16 -05:00
|
|
|
options.delete_if {|key, value| value.nil?}
|
2010-09-15 20:31:45 -04:00
|
|
|
|
2010-08-17 11:45:18 -04:00
|
|
|
# If subnet is defined we are working on a virtual private cloud.
|
2010-08-17 06:28:06 -04:00
|
|
|
# subnet & security group cannot co-exist. I wish VPC just ignored
|
|
|
|
# the security group parameter instead, it would be much easier!
|
2010-08-18 11:47:51 -04:00
|
|
|
if subnet_id
|
2010-08-17 06:28:06 -04:00
|
|
|
options.delete('SecurityGroup')
|
2010-08-18 11:47:51 -04:00
|
|
|
else
|
|
|
|
options.delete('SubnetId')
|
2010-08-17 06:28:06 -04:00
|
|
|
end
|
2010-05-18 14:22:32 -04:00
|
|
|
|
2010-09-15 20:31:45 -04:00
|
|
|
data = connection.run_instances(image_id, 1, 1, options)
|
2009-09-16 02:11:06 -04:00
|
|
|
merge_attributes(data.body['instancesSet'].first)
|
2011-08-08 13:17:14 -04:00
|
|
|
|
2011-10-22 13:21:35 -04:00
|
|
|
if tags = self.tags
|
2011-10-19 15:13:42 -04:00
|
|
|
# expect eventual consistency
|
|
|
|
Fog.wait_for { self.reload rescue nil }
|
2011-10-22 13:21:35 -04:00
|
|
|
for key, value in (self.tags = tags)
|
2011-08-09 14:39:02 -04:00
|
|
|
connection.tags.create(
|
|
|
|
:key => key,
|
|
|
|
:resource_id => self.identity,
|
|
|
|
:value => value
|
|
|
|
)
|
|
|
|
end
|
2011-08-08 13:17:14 -04:00
|
|
|
end
|
|
|
|
|
2009-09-16 02:11:06 -04:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2010-09-15 20:31:45 -04:00
|
|
|
def setup(credentials = {})
|
2011-08-09 14:40:26 -04:00
|
|
|
requires :public_ip_address, :username
|
2011-08-09 14:39:43 -04:00
|
|
|
require 'net/ssh'
|
2010-12-02 16:37:03 -05:00
|
|
|
|
2010-10-20 17:32:30 -04:00
|
|
|
commands = [
|
2010-09-15 20:31:45 -04:00
|
|
|
%{mkdir .ssh},
|
2011-01-31 20:42:53 -05:00
|
|
|
%{passwd -l #{username}},
|
2012-04-25 10:31:28 -04:00
|
|
|
%{echo "#{Fog::JSON.encode(Fog::JSON.sanitize(attributes))}" >> ~/attributes.json}
|
2010-10-20 17:32:30 -04:00
|
|
|
]
|
|
|
|
if public_key
|
|
|
|
commands << %{echo "#{public_key}" >> ~/.ssh/authorized_keys}
|
|
|
|
end
|
2011-06-23 20:10:38 -04:00
|
|
|
|
|
|
|
# wait for aws to be ready
|
2011-06-24 17:27:25 -04:00
|
|
|
Timeout::timeout(360) do
|
2010-12-02 16:37:03 -05:00
|
|
|
begin
|
2011-06-24 17:27:25 -04:00
|
|
|
Timeout::timeout(8) do
|
|
|
|
Fog::SSH.new(public_ip_address, username, credentials.merge(:timeout => 4)).run('pwd')
|
|
|
|
end
|
2011-06-23 20:10:38 -04:00
|
|
|
rescue Errno::ECONNREFUSED
|
|
|
|
sleep(2)
|
|
|
|
retry
|
2011-06-24 17:27:25 -04:00
|
|
|
rescue Net::SSH::AuthenticationFailed, Timeout::Error
|
2010-12-02 16:37:03 -05:00
|
|
|
retry
|
|
|
|
end
|
|
|
|
end
|
2011-06-23 20:10:38 -04:00
|
|
|
Fog::SSH.new(public_ip_address, username, credentials).run(commands)
|
2010-09-15 20:31:45 -04:00
|
|
|
end
|
|
|
|
|
2010-07-23 00:52:27 -04:00
|
|
|
def start
|
|
|
|
requires :id
|
2010-11-19 16:45:45 -05:00
|
|
|
connection.start_instances(id)
|
2010-07-23 00:52:27 -04:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2011-08-08 03:29:16 -04:00
|
|
|
def stop(force = false)
|
2010-07-23 00:52:27 -04:00
|
|
|
requires :id
|
2011-08-08 03:29:16 -04:00
|
|
|
connection.stop_instances(id, force)
|
2010-07-23 00:52:27 -04:00
|
|
|
true
|
|
|
|
end
|
2010-10-11 16:40:34 -04:00
|
|
|
|
2010-09-15 20:31:45 -04:00
|
|
|
def username
|
|
|
|
@username ||= 'root'
|
|
|
|
end
|
|
|
|
|
2009-09-19 15:31:15 -04:00
|
|
|
def volumes
|
2009-12-05 17:53:42 -05:00
|
|
|
requires :id
|
2010-01-08 14:29:07 -05:00
|
|
|
connection.volumes(:server => self)
|
2009-09-19 15:31:15 -04:00
|
|
|
end
|
|
|
|
|
2011-03-11 16:06:44 -05:00
|
|
|
#I tried to call it monitoring= and be smart with attributes[]
|
|
|
|
#but in #save a merge_attribute is called after run_instance
|
|
|
|
#thus making an un-necessary request. Use this until finding a clever solution
|
2011-03-21 19:45:52 -04:00
|
|
|
def monitor=(new_monitor)
|
|
|
|
if identity
|
|
|
|
case new_monitor
|
2011-03-11 16:06:44 -05:00
|
|
|
when true
|
|
|
|
response = connection.monitor_instances(identity)
|
|
|
|
when false
|
|
|
|
response = connection.unmonitor_instances(identity)
|
|
|
|
else
|
|
|
|
raise ArgumentError.new("only Boolean allowed here")
|
2011-03-21 19:45:52 -04:00
|
|
|
end
|
2011-03-11 16:06:44 -05:00
|
|
|
end
|
2011-03-21 19:45:52 -04:00
|
|
|
self.monitoring = new_monitor
|
2011-03-11 16:06:44 -05:00
|
|
|
end
|
|
|
|
|
2011-06-24 20:31:44 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def placement=(new_placement)
|
|
|
|
if new_placement.is_a?(Hash)
|
|
|
|
merge_attributes(new_placement)
|
|
|
|
else
|
|
|
|
self.attributes[:placement] = new_placement
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2009-09-16 02:11:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|