# # Author:: Tim Dysinger () # Author:: Benjamin Black () # Author:: Christopher Brown () # Copyright:: Copyright (c) 2009 Opscode, Inc. # License:: Apache License, Version 2.0 # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. provides "ec2" require 'open-uri' require 'socket' require_plugin "hostname" require_plugin "kernel" require_plugin "network" EC2_METADATA_ADDR = "169.254.169.254" unless defined?(EC2_METADATA_ADDR) EC2_METADATA_URL = "http://#{EC2_METADATA_ADDR}/2008-02-01/meta-data" unless defined?(EC2_METADATA_URL) EC2_USERDATA_URL = "http://#{EC2_METADATA_ADDR}/2008-02-01/user-data" unless defined?(EC2_USERDATA_URL) EC2_ARRAY_VALUES = %w(security-groups) def can_metadata_connect?(addr, port, timeout=2) t = Socket.new(Socket::Constants::AF_INET, Socket::Constants::SOCK_STREAM, 0) saddr = Socket.pack_sockaddr_in(port, addr) connected = false begin t.connect_nonblock(saddr) rescue Errno::EINPROGRESS r,w,e = IO::select(nil,[t],nil,timeout) if !w.nil? connected = true else begin t.connect_nonblock(saddr) rescue Errno::EISCONN t.close connected = true rescue SystemCallError end end rescue SystemCallError end connected end def has_ec2_mac? network[:interfaces].values.each do |iface| unless iface[:arp].nil? return true if iface[:arp].value?("fe:ff:ff:ff:ff:ff") end end false end def metadata(id='') OpenURI.open_uri("#{EC2_METADATA_URL}/#{id}").read.split("\n").each do |o| key = "#{id}#{o.gsub(/\=.*$/, '/')}" if key[-1..-1] != '/' ec2[key.gsub(/\-|\//, '_').to_sym] = if EC2_ARRAY_VALUES.include? key OpenURI.open_uri("#{EC2_METADATA_URL}/#{key}").read.split("\n") else OpenURI.open_uri("#{EC2_METADATA_URL}/#{key}").read end else metadata(key) end end end def userdata() ec2[:userdata] = nil # assumes the only expected error is the 404 if there's no user-data begin ec2[:userdata] = OpenURI.open_uri("#{EC2_USERDATA_URL}/").read rescue OpenURI::HTTPError end end def looks_like_ec2? # Try non-blocking connect so we don't "block" if # the Xen environment is *not* EC2 has_ec2_mac? && can_metadata_connect?(EC2_METADATA_ADDR,80) end if looks_like_ec2? ec2 Mash.new self.metadata self.userdata end