#!/usr/bin/python # vi: ts=4 expandtab # # Copyright (C) 2009-2010 Canonical Ltd. # # Author: Scott Moser # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License version 3, as # published by the Free Software Foundation. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . import subprocess import sys import cloudinit import cloudinit.util as util import time import logging import errno def warn(str): sys.stderr.write(str) def main(): now = time.strftime("%a, %d %b %Y %H:%M:%S %z") try: uptimef=open("/proc/uptime") uptime=uptimef.read().split(" ")[0] uptimef.close() except IOError as e: warn("unable to open /proc/uptime\n") uptime = "na" msg = "cloud-init running: %s. up %s seconds" % (now, uptime) sys.stderr.write(msg + "\n") sys.stderr.flush() cloudinit.logging_set_from_cfg_file() log = logging.getLogger() log.info(msg) # cache is not instance specific, so it has to be purged cloudinit.purge_cache() cloud = cloudinit.CloudInit() try: cloud.get_data_source() except Exception as e: print e sys.stderr.write("Failed to get instance data\n") sys.exit(1) # store the metadata cloud.update_cache() # parse the user data (ec2-run-userdata.py) try: cloud.sem_and_run("consume_userdata", "once-per-instance", cloud.consume_userdata,[],False) except: warn("consuming user data failed!\n") raise try: if util.get_cfg_option_bool(cloud.cfg,"preserve_hostname",False): log.debug("preserve_hostname is set. not managing hostname") else: hostname = cloud.get_hostname() cloud.sem_and_run("set_hostname", "once-per-instance", set_hostname, [ hostname, log ], False) cloud.sem_and_run("update_hostname", "always", update_hostname, [ hostname, log ], False) except: warn("failed to set hostname\n") #print "user data is:" + cloud.get_user_data() # set the defaults (like what ec2-set-defaults.py did) try: cloud.sem_and_run("set_defaults", "once-per-instance", set_defaults,[ cloud ],False) except: warn("failed to set defaults\n") # finish, send the cloud-config event cloud.initctl_emit() sys.exit(0) def set_defaults(cloud): apply_locale(cloud.get_locale()) def apply_locale(locale): subprocess.Popen(['locale-gen', locale]).communicate() subprocess.Popen(['update-locale', locale]).communicate() util.render_to_file('default-locale', '/etc/default/locale', \ { 'locale' : locale }) # read hostname from a 'hostname' file # allow for comments and stripping line endings. # if file doesn't exist, or no contents, return default def read_hostname(filename, default=None): try: fp = open(filename,"r") lines = fp.readlines() fp.close() for line in lines: hpos = line.find("#") if hpos != -1: line = line[0:hpos] line = line.rstrip() if line: return line except IOError, e: if e.errno == errno.ENOENT: pass return default def set_hostname(hostname, log): try: subprocess.Popen(['hostname', hostname]).communicate() util.write_file("/etc/hostname","%s\n" % hostname, 0644) log.debug("populated /etc/hostname with %s on first boot", hostname) except: log.error("failed to set_hostname") def update_hostname(hostname, log): prev_file="%s/%s" % (cloudinit.datadir,"previous-hostname") etc_file = "/etc/hostname" hostname_prev = None hostname_in_etc = None try: hostname_prev = read_hostname(prev_file) except: log.warn("Failed to open %s" % prev_file) try: hostname_in_etc = read_hostname(etc_file) except: log.warn("Failed to open %s" % etc_file) update_files = [] if not hostname_prev or hostname_prev != hostname: update_files.append(prev_file) if (not hostname_in_etc or (hostname_in_etc == hostname_prev and hostname_in_etc != hostname)): update_files.append(etc_file) try: for fname in update_files: util.write_file(fname,"%s\n" % hostname, 0644) log.debug("wrote %s to %s" % (hostname,fname)) except: log.warn("failed to write hostname to %s" % fname) if hostname_in_etc and hostname_prev and hostname_in_etc != hostname_prev: log.debug("%s differs from %s. assuming user maintained" % (prev_file,etc_file)) if etc_file in update_files: log.debug("setting hostname to %s" % hostname) subprocess.Popen(['hostname', hostname]).communicate() if __name__ == '__main__': main()