module Rudy class Machine < Storable include Rudy::MetaData::ObjectBase field :rtype field :awsid field :region field :zone field :environment field :role field :position field :created => Time field :started => Time field :dns_public field :dns_private field :state attr_reader :instance def init #@created = @rtype = 'm' @region = @@global.region @zone = @@global.zone @environment = @@global.environment @role = @@global.role @position = find_next_position || '01' @state = 'no-instance' end def liner_note update #if !dns_public? && @awsid info = !@dns_public.nil? && !@dns_public.empty? ? @dns_public : "#{@awsid}:#{@state}" "%s %s" % [self.name.bright, info] end def to_s(with_title=false) lines = [] lines << liner_note #if self.running? # k, g = @keyname || 'no-keypair', self.groups.join(', ') # lines << @@sformat % %w{zone size ami keyname groups} if with_title # lines << @@sformat % [@zone, @size, @ami, k, g] #end lines.join($/) end def inspect update #if !dns_public? && @awsid lines = [] lines << liner_note field_names.each do |key| next unless self.respond_to?(key) val = self.send(key) lines << sprintf(" %22s: %s", key, (val.is_a?(Array) ? val.join(', ') : val)) end lines.join($/) end def find_next_position list = @sdb.select(self.to_select(nil, [:position])) || [] pos = list.size + 1 pos.to_s.rjust(2, '0') end def name Machine.generate_name(@zone, @environment, @role, @position) end def Machine.generate_name(zon, env, rol, pos) pos = pos.to_s.rjust 2, '0' ["m", zon, env, rol, pos].join(Rudy::DELIM) end def get_instance @ec2inst.get(@awsid) end def update return false unless @awsid @instance = get_instance if @instance.is_a?(Rudy::AWS::EC2::Instance) @dns_public = @instance.dns_public @dns_private = @instance.dns_private @state = @instance.state save elsif @instance.nil? @awsid = @dns_public = @dns_private = nil @state = 'rogue' # Don't save it b/c it's possible the EC2 server is just down. end end def dns_public? !@dns_public.nil? && !@dns_public.empty? end def dns_private? !@dns_private.nil? && !@dns_private.empty? end def start(opts={}) raise "#{name} is already running" if running? opts = { :min => 1, :size => current_machine_size, :ami => current_machine_image, :group => current_group_name, :keypair => root_keypairname, :zone => @@global.zone.to_s, :machine_data => Machine.generate_machine_data.to_yaml }.merge(opts) @ec2inst.create(opts) do |inst| @awsid = inst.awsid @created = @starts = Time.now @state = inst.state # Assign IP address only if we have one for that position if current_machine_address(@position) address = current_machine_address(@position) puts "Associating #{address} to #{inst.awsid}" begin @radd.associate(address, inst.awsid) rescue => ex STDERR.puts "Error while associating address (#{ex.class.to_s})" Rudy::Utils.bug() end end end self.save self end def destroy @ec2inst.destroy(@awsid) if running? super end def restart @ec2inst.restart(@awsid) if running? end def Machine.generate_machine_data data = { # Give the machine an identity :region => @@global.region.to_s, :zone => @@global.zone.to_s, :environment => @@global.environment.to_s, :role => @@global.role.to_s, :position => @@global.position.to_s, :hosts => { # Add hosts to the /etc/hosts file :dbmaster => "127.0.0.1", } } data end def running? return false if @awsid.nil? || @awsid.empty? @ec2inst.running?(@awsid) end end class Machines include Rudy::MetaData def create(&each_mach) raise MachineGroupAlreadyRunning, current_machine_group if running? raise MachineGroupNotDefined, current_machine_group unless known_machine_group? unless (1..MAX_INSTANCES).member?(current_machine_count) raise "Instance count must be more than 0, less than #{MAX_INSTANCES}" end unless @rgrp.exists?(current_group_name) puts "Creating group: #{current_group_name}" @rgrp.create(current_group_name) end unless @rkey.exists?(root_keypairname) kp_file = File.join(Rudy::CONFIG_DIR, root_keypairname) raise PrivateKeyFileExists, kp_file if File.exists?(kp_file) puts "Creating keypair: #{root_keypairname}" kp = @rkey.create(root_keypairname) puts "Saving #{kp_file}" Rudy::Utils.write_to_file(kp_file, kp.private_key, 'w', 0600) else kp_file = root_keypairpath # This means no keypair file can be found raise PrivateKeyNotFound, root_keypairname if kp_file.nil? # This means we found a keypair in the config but we cannot find the private key file. raise PrivateKeyNotFound, kp_file if !File.exists?(kp_file) end machines = [] current_machine_count.times do |i| machine = Rudy::Machine.new #puts "Starting %s" % machine.name machine.start machines << machine end machines.each { |m| each_mach.call(m) } if each_mach machines end def destroy(&each_mach) raise MachineGroupNotDefined, current_machine_group unless known_machine_group? raise MachineGroupNotRunning, current_machine_group unless running? list.each { |m| each_mach.call(m); } if each_mach list do |mach| #puts "Destroying #{mach.name}" mach.destroy end end def restart(&each_mach) raise MachineGroupNotDefined, current_machine_group unless known_machine_group? raise MachineGroupNotRunning, current_machine_group unless running? machines = list machines.each do |mach| each_mach.call(mach) if each_mach puts "Restarting #{mach.name}" mach.restart end machines end def list(more=[], less=[], &each_mach) machines = list_as_hash(more, less, &each_mach) machines &&= machines.values machines end def list_as_hash(more=[], less=[], &each_mach) query = to_select([:rtype, 'm'], less) list = @sdb.select(query) || {} machines = {} list.each_pair do |n,m| machines[n] = Rudy::Machine.from_hash(m) end machines.each_pair { |n,mach| each_mach.call(mach) } if each_mach machines = nil if machines.empty? machines end def get(rname=nil) Rudy::Machine.from_hash(@sdb.get(Rudy::DOMAIN, rname)) # Returns nil if empty end def running? !list.nil? # TODO: add logic that checks whether the instances are running. end end class Machines::Offline def list(more=[], less=[], &each_mach) m = Rudy::Machine.new m.dns_public = 'localhost' each_mach.call(m) if each_mach [m] end end end