require 'json' require 'builder' require 'fileutils' require 'rexml/document' require 'rexml/xpath' module Kytoon module Providers module CloudCue class ServerGroup @@data_dir=File.join(KYTOON_PROJECT, "tmp", "cloudcue") def self.data_dir @@data_dir end def self.data_dir=(dir) @@data_dir=dir end CONFIG_FILE = KYTOON_PROJECT + File::SEPARATOR + "config" + File::SEPARATOR + "server_group.json" attr_accessor :id attr_accessor :name attr_accessor :description attr_accessor :domain_name attr_accessor :owner_name attr_reader :ssh_public_keys def initialize(options={}) @id=options[:id] @name=options[:name] @description=options[:description] @domain_name=options[:domain_name] @owner_name=options[:owner_name] or @owner_name=ENV['USER'] @servers=[] @ssh_public_keys=[] end def server(name) @servers.select {|s| s.name == name}[0] if @servers.size > 0 end def servers @servers end def gateway_ip @servers.select {|s| s.gateway? }[0].external_ip_addr if @servers.size > 0 end def ssh_public_keys @ssh_public_keys end # generate a Server Group XML from server_group.json def self.from_json(json) json_hash=JSON.parse(json) sg=ServerGroup.new( :name => json_hash["name"], :description => json_hash["description"], :domain_name => json_hash["domain_name"] ) json_hash["servers"].each_pair do |server_name, server_config| sg.servers << Server.new( :name => server_name, :description => server_config["description"], :flavor_id => server_config["flavor_id"], :image_id => server_config["image_id"], :image_name => server_config["image_name"], :gateway => server_config["gateway"] ) end # automatically add a key for the current user sg.ssh_public_keys << SshPublicKey.new( :description => "#{ENV['USER']}'s public key", :public_key => Util.load_public_key ) return sg end def to_xml xml = Builder::XmlMarkup.new xml.tag! "server-group" do |sg| sg.id(@id) sg.name(@name) sg.description(@description) sg.tag! "owner-name", @owner_name sg.tag! "domain-name", @domain_name sg.servers("type" => "array") do |xml_servers| self.servers.each do |server| xml_servers.server do |xml_server| xml_server.name(server.name) xml_server.description(server.description) xml_server.tag! "flavor-id", server.flavor_id if server.image_id then xml_server.tag! "image-id", server.image_id end if server.image_name then xml_server.tag! "image-name", server.image_name end if server.admin_password then xml_server.tag! "admin-password", server.admin_password end xml_server.tag! "cloud-server-id-number", server.cloud_server_id_number if server.cloud_server_id_number xml_server.tag! "status", server.status if server.status xml_server.tag! "external-ip-addr", server.external_ip_addr if server.external_ip_addr xml_server.tag! "internal-ip-addr", server.internal_ip_addr if server.internal_ip_addr xml_server.tag! "error-message", server.error_message if server.error_message if server.gateway? xml_server.tag! "gateway", "true", { "type" => "boolean"} end end end end sg.tag! "ssh-public-keys", { "type" => "array"} do |xml_ssh_pub_keys| self.ssh_public_keys.each do |ssh_public_key| xml_ssh_pub_keys.tag! "ssh-public-key" do |xml_ssh_pub_key| xml_ssh_pub_key.description ssh_public_key.description xml_ssh_pub_key.tag! "public-key", ssh_public_key.public_key end end end end xml.target! end def self.from_xml(xml) sg=nil dom = REXML::Document.new(xml) REXML::XPath.each(dom, "/server-group") do |sg_xml| sg=ServerGroup.new( :name => XMLUtil.element_text(sg_xml, "name"), :id => XMLUtil.element_text(sg_xml, "id").to_i, :owner_name => XMLUtil.element_text(sg_xml, "owner-name"), :domain_name => XMLUtil.element_text(sg_xml, "domain-name"), :description => XMLUtil.element_text(sg_xml, "description") ) REXML::XPath.each(dom, "//server") do |server_xml| server=Server.new( :id => XMLUtil.element_text(server_xml, "id").to_i, :name => XMLUtil.element_text(server_xml, "name"), :cloud_server_id_number => XMLUtil.element_text(server_xml, "cloud-server-id-number"), :status => XMLUtil.element_text(server_xml, "status"), :external_ip_addr => XMLUtil.element_text(server_xml, "external-ip-addr"), :internal_ip_addr => XMLUtil.element_text(server_xml, "internal-ip-addr"), :error_message => XMLUtil.element_text(server_xml, "error-message"), :image_id => XMLUtil.element_text(server_xml, "image-id"), :admin_password => XMLUtil.element_text(server_xml, "admin-password"), :flavor_id => XMLUtil.element_text(server_xml, "flavor-id"), :retry_count => XMLUtil.element_text(server_xml, "retry-count"), :gateway => XMLUtil.element_text(server_xml, "gateway") ) sg.servers << server end end sg end def pretty_print puts "Group ID: #{@id}" puts "name: #{@name}" puts "description: #{@description}" puts "domain name: #{@domain_name}" puts "Gateway IP: #{self.gateway_ip}" puts "Servers:" servers.each do |server| puts "\tname: #{server.name} (id: #{server.id})" puts "\tstatus: #{server.status}" if server.gateway? puts "\tGateway server: #{server.gateway?}" end if server.error_message then puts "\tlast error message: #{server.error_message}" end puts "\t--" end end def server_names names=[] servers.each do |server| if block_given? then yield server.name else names << server.name end end names end def cache_to_disk FileUtils.mkdir_p(@@data_dir) File.open(File.join(@@data_dir, "#{@id}.xml"), 'w') do |f| f.chmod(0600) f.write(self.to_xml) end end def delete Connection.delete("/server_groups/#{@id}.xml") out_file=File.join(@@data_dir, "#{@id}.xml") File.delete(out_file) if File.exists?(out_file) end # Poll the server group until it is online. # :timeout - max number of seconds to wait before raising an exception. # Defaults to 1500 def poll_until_online(options={}) timeout=options[:timeout] or timeout = ENV['TIMEOUT'] if timeout.nil? or timeout.empty? then timeout=1500 # defaults to 25 minutes end online = false count=0 until online or (count*20) >= timeout.to_i do count+=1 begin sg=ServerGroup.get(:id => @id, :remote => true) online=true sg.servers.each do |server| if ["Pending", "Rebuilding"].include?(server.status) then online=false end if server.status == "Failed" then raise "Failed to create server group with the following message: #{server.error_message}" end end if not online yield sg if block_given? sleep 20 end rescue EOFError end end if (count*20) >= timeout.to_i then raise "Timeout waiting for server groups to come online." end end def self.create(sg) xml=Connection.post("/server_groups.xml", sg.to_xml) sg=ServerGroup.from_xml(xml) old_group_xml=nil gateway_ip=nil sg.poll_until_online do |server_group| if old_group_xml != server_group.to_xml then old_group_xml = server_group.to_xml gateway_ip = server_group.gateway_ip if server_group.gateway_ip if not gateway_ip.nil? and not gateway_ip.empty? then SshUtil.remove_known_hosts_ip(gateway_ip) end server_group.pretty_print end end sg=ServerGroup.get(:id => sg.id, :remote => true) sg.cache_to_disk puts "Server group online." sg end # Get a server group. The following options are available: # # :id - The ID of the server group to get. Defaults to ENV['GROUP_ID'] # :remote - true or false def self.get(options={}) remote = options[:remote] or false id = options[:id] if id.nil? then group = ServerGroup.most_recent raise "No recent server group files exist." if group.nil? id = group.id end if remote then xml=Connection.get("/server_groups/#{id}.xml") ServerGroup.from_xml(xml) else out_file = File.join(@@data_dir, "#{id}.xml") raise "No server group files exist." if not File.exists?(out_file) ServerGroup.from_xml(IO.read(out_file)) end end # :remote - true or false def self.index(options={}) remote = options[:remote] or false server_groups=[] if remote then xml=Connection.get("/server_groups.xml") dom = REXML::Document.new(xml) REXML::XPath.each(dom, "//server-group") do |group_xml| server_groups << ServerGroup.from_xml(group_xml.to_s) end else Dir[File.join(ServerGroup.data_dir, '*.xml')].each do |file| server_groups << ServerGroup.from_xml(IO.read(file)) end end server_groups end def self.most_recent server_groups=[] Dir[File.join(@@data_dir, "*.xml")].each do |file| server_groups << ServerGroup.from_xml(IO.read(file)) end if server_groups.size > 0 then return server_groups.sort { |a,b| b.id <=> a.id }[0] else nil end end end end end end