summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorScott Moser <smoser@ubuntu.com>2011-08-02 17:09:43 -0400
committerScott Moser <smoser@ubuntu.com>2011-08-02 17:09:43 -0400
commit9d4a9dc17696d0465ae4f8b56db9c0948fcac28c (patch)
tree77b6c6c2d98016524f4191786dd6e89f2e28090b
parentfb0150dd71bf54bd46e30efab62dbb2f6eca2ba2 (diff)
downloadvyos-cloud-init-9d4a9dc17696d0465ae4f8b56db9c0948fcac28c.tar.gz
vyos-cloud-init-9d4a9dc17696d0465ae4f8b56db9c0948fcac28c.zip
This fixes LP: #819507, to make consume_userdata run 'always'
consume_userdata should really run always, rather than once per instance. The documentation says that boothooks were on their own for per-instance but since this routine was only being called once, they would only get called once. This modifies the behavior to be: user_script: per_always cloud_config : per_always upstart_job : per_instance cloud_boothook: per_always In order to not break part handlers that are existing, and expect to only be called once per instance, this adds a 'handler_version' item in a handler that can indicate the version (currently 1 or 2). If it is 2, then the hander will be passed the frequency (per-instance or per-always) that this is being run. That way the handler can differenciate between them. This also makes 'bootcmd' run every boot. That should be changable in cloud-config though, so users who dont like the behavior can modify it. LP: #819507
-rwxr-xr-xcloud-init.py6
-rw-r--r--cloudinit/CloudConfig/__init__.py6
-rw-r--r--cloudinit/CloudConfig/cc_bootcmd.py2
-rw-r--r--cloudinit/UserDataHandler.py10
-rw-r--r--cloudinit/__init__.py184
5 files changed, 139 insertions, 69 deletions
diff --git a/cloud-init.py b/cloud-init.py
index 09c537f1..0ba00699 100755
--- a/cloud-init.py
+++ b/cloud-init.py
@@ -140,8 +140,10 @@ def main():
# parse the user data (ec2-run-userdata.py)
try:
- cloud.sem_and_run("consume_userdata", "once-per-instance",
- cloud.consume_userdata,[],False)
+ ran = cloud.sem_and_run("consume_userdata", cloudinit.per_instance,
+ cloud.consume_userdata,[cloudinit.per_instance],False)
+ if not ran:
+ cloud.consume_userdata(cloudinit.per_always)
except:
warn("consuming user data failed!\n")
raise
diff --git a/cloudinit/CloudConfig/__init__.py b/cloudinit/CloudConfig/__init__.py
index 91853dfd..82f422fc 100644
--- a/cloudinit/CloudConfig/__init__.py
+++ b/cloudinit/CloudConfig/__init__.py
@@ -25,9 +25,9 @@ import os
import subprocess
import time
-per_instance="once-per-instance"
-per_always="always"
-per_once="once"
+per_instance= cloudinit.per_instance
+per_always = cloudinit.per_always
+per_once = cloudinit.per_once
class CloudConfig():
cfgfile = None
diff --git a/cloudinit/CloudConfig/cc_bootcmd.py b/cloudinit/CloudConfig/cc_bootcmd.py
index 9eccfd78..11e9938c 100644
--- a/cloudinit/CloudConfig/cc_bootcmd.py
+++ b/cloudinit/CloudConfig/cc_bootcmd.py
@@ -18,6 +18,8 @@
import cloudinit.util as util
import subprocess
import tempfile
+from cloudinit.CloudConfig import per_always
+frequency = per_always
def handle(name,cfg,cloud,log,args):
if not cfg.has_key("bootcmd"):
diff --git a/cloudinit/UserDataHandler.py b/cloudinit/UserDataHandler.py
index 9670c0cb..afd5dd99 100644
--- a/cloudinit/UserDataHandler.py
+++ b/cloudinit/UserDataHandler.py
@@ -190,11 +190,10 @@ def preprocess_userdata(data):
process_includes(email.message_from_string(decomp_str(data)),parts)
return(parts2mime(parts))
-# callbacks is a dictionary with:
-# { 'content-type': handler(data,content_type,filename,payload) }
-def walk_userdata(str, callbacks, data = None):
+# callback is a function that will be called with (data, content_type, filename, payload)
+def walk_userdata(istr, callback, data = None):
partnum = 0
- for part in email.message_from_string(str).walk():
+ for part in email.message_from_string(istr).walk():
# multipart/* are just containers
if part.get_content_maintype() == 'multipart':
continue
@@ -207,8 +206,7 @@ def walk_userdata(str, callbacks, data = None):
if not filename:
filename = 'part-%03d' % partnum
- if callbacks.has_key(ctype):
- callbacks[ctype](data,ctype,filename,part.get_payload())
+ callback(data, ctype, filename, part.get_payload())
partnum = partnum+1
diff --git a/cloudinit/__init__.py b/cloudinit/__init__.py
index f471f363..d01d443d 100644
--- a/cloudinit/__init__.py
+++ b/cloudinit/__init__.py
@@ -46,6 +46,10 @@ pathmap = {
None : "",
}
+per_instance="once-per-instance"
+per_always="always"
+per_once="once"
+
parsed_cfgs = { }
import os
@@ -63,6 +67,7 @@ import logging
import logging.config
import StringIO
import glob
+import traceback
class NullHandler(logging.Handler):
def emit(self,record): pass
@@ -111,14 +116,16 @@ class CloudInit:
ds_deps = [ DataSource.DEP_FILESYSTEM, DataSource.DEP_NETWORK ]
datasource = None
+ builtin_handlers = [ ]
+
def __init__(self, ds_deps = None, sysconfig=system_config):
- self.part_handlers = {
- 'text/x-shellscript' : self.handle_user_script,
- 'text/cloud-config' : self.handle_cloud_config,
- 'text/upstart-job' : self.handle_upstart_job,
- 'text/part-handler' : self.handle_handler,
- 'text/cloud-boothook' : self.handle_cloud_boothook
- }
+ self.builtin_handlers = [
+ [ 'text/x-shellscript', self.handle_user_script, per_always ],
+ [ 'text/cloud-config', self.handle_cloud_config, per_always ],
+ [ 'text/upstart-job', self.handle_upstart_job, per_instance ],
+ [ 'text/cloud-boothook', self.handle_cloud_boothook, per_always ],
+ ]
+
if ds_deps != None:
self.ds_deps = ds_deps
self.sysconfig=sysconfig
@@ -249,7 +256,7 @@ class CloudInit:
return("%s/%s.%s" % (get_cpath("sem"), name, freq))
def sem_has_run(self,name,freq):
- if freq == "always": return False
+ if freq == per_always: return False
semfile = self.sem_getpath(name,freq)
if os.path.exists(semfile):
return True
@@ -265,7 +272,7 @@ class CloudInit:
if e.errno != errno.EEXIST:
raise e
- if os.path.exists(semfile) and freq != "always":
+ if os.path.exists(semfile) and freq != per_always:
return False
# race condition
@@ -294,7 +301,7 @@ class CloudInit:
def sem_and_run(self,semname,freq,func,args=[],clear_on_fail=False):
if self.sem_has_run(semname,freq):
log.debug("%s already ran %s", semname, freq)
- return
+ return False
try:
if not self.sem_acquire(semname,freq):
raise Exception("Failed to acquire lock on %s" % semname)
@@ -305,13 +312,15 @@ class CloudInit:
self.sem_clear(semname,freq)
raise
+ return True
+
# get_ipath : get the instance path for a name in pathmap
# (/var/lib/cloud/instances/<instance>/name)<name>)
def get_ipath(self, name=None):
return("%s/instances/%s%s"
% (varlibdir,self.get_instance_id(), pathmap[name]))
- def consume_userdata(self):
+ def consume_userdata(self, frequency=per_instance):
self.get_userdata()
data = self
@@ -323,63 +332,40 @@ class CloudInit:
sys.path.insert(0,cdir)
sys.path.insert(0,idir)
+ part_handlers = { }
# add handlers in cdir
for fname in glob.glob("%s/*.py" % cdir):
if not os.path.isfile(fname): continue
modname = os.path.basename(fname)[0:-3]
try:
mod = __import__(modname)
- lister = getattr(mod, "list_types")
- handler = getattr(mod, "handle_part")
- mtypes = lister()
- for mtype in mtypes:
- self.part_handlers[mtype]=handler
- log.debug("added handler for [%s] from %s" % (mtypes,fname))
+ handler_register(mod, part_handlers, data, frequency)
+ log.debug("added handler for [%s] from %s" % (mod.list_types(), fname))
except:
log.warn("failed to initialize handler in %s" % fname)
util.logexc(log)
-
- # give callbacks opportunity to initialize
- for ctype, func in self.part_handlers.items():
- func(data, "__begin__",None,None)
- UserDataHandler.walk_userdata(self.get_userdata(),
- self.part_handlers, data)
- # give callbacks opportunity to finalize
- for ctype, func in self.part_handlers.items():
- func(data,"__end__",None,None)
+ # add the internal handers if their type hasn't been already claimed
+ for (btype, bhand, bfreq) in self.builtin_handlers:
+ if btype in part_handlers:
+ continue
+ handler_register(InternalPartHandler(bhand, [btype], bfreq),
+ part_handlers, data, frequency)
- def handle_handler(self,data,ctype,filename,payload):
- if ctype == "__end__": return
- if ctype == "__begin__" :
- self.handlercount = 0
- return
-
- self.handlercount=self.handlercount+1
-
- # write content to instance's handlerdir
- handlerdir = self.get_ipath("handlers")
- modname = 'part-handler-%03d' % self.handlercount
- modfname = modname + ".py"
- util.write_file("%s/%s" % (handlerdir,modfname), payload, 0600)
-
- try:
- mod = __import__(modname)
- lister = getattr(mod, "list_types")
- handler = getattr(mod, "handle_part")
- except:
- import traceback
- traceback.print_exc(file=sys.stderr)
- return
-
- # - call it with '__begin__'
- handler(data, "__begin__", None, None)
+ # walk the data
+ pdata = { 'handlers': part_handlers, 'handlerdir': idir,
+ 'data' : data, 'frequency': frequency }
+ UserDataHandler.walk_userdata(self.get_userdata(),
+ partwalker_callback, data = pdata)
- # - add it self.part_handlers
- for mtype in lister():
- self.part_handlers[mtype]=handler
+ # give callbacks opportunity to finalize
+ called = [ ]
+ for (mtype, mod) in part_handlers.iteritems():
+ if mod in called:
+ continue
+ handler_call_end(mod, data, frequency)
- def handle_user_script(self,data,ctype,filename,payload):
+ def handle_user_script(self,data,ctype,filename,payload, frequency):
if ctype == "__end__": return
if ctype == "__begin__":
# maybe delete existing things here
@@ -390,7 +376,11 @@ class CloudInit:
util.write_file("%s/%s" %
(scriptsdir,filename), util.dos2unix(payload), 0700)
- def handle_upstart_job(self,data,ctype,filename,payload):
+ def handle_upstart_job(self,data,ctype,filename,payload, frequency):
+ # upstart jobs are only written on the first boot
+ if frequency != per_instance:
+ return
+
if ctype == "__end__" or ctype == "__begin__": return
if not filename.endswith(".conf"):
filename=filename+".conf"
@@ -398,7 +388,7 @@ class CloudInit:
util.write_file("%s/%s" % ("/etc/init",filename),
util.dos2unix(payload), 0644)
- def handle_cloud_config(self,data,ctype,filename,payload):
+ def handle_cloud_config(self,data,ctype,filename,payload, frequency):
if ctype == "__begin__":
self.cloud_config_str=""
return
@@ -418,7 +408,7 @@ class CloudInit:
self.cloud_config_str+="\n#%s\n%s" % (filename,payload)
- def handle_cloud_boothook(self,data,ctype,filename,payload):
+ def handle_cloud_boothook(self,data,ctype,filename,payload, frequency):
if ctype == "__end__": return
if ctype == "__begin__": return
@@ -520,3 +510,81 @@ class DataSourceNotFoundException(Exception):
def list_sources(cfg_list, depends):
return(DataSource.list_sources(cfg_list,depends, ["cloudinit", "" ]))
+
+def handler_register(mod, part_handlers, data, frequency=per_instance):
+ if not hasattr(mod, "handler_version"):
+ setattr(mod, "handler_version", 1)
+
+ for mtype in mod.list_types():
+ part_handlers[mtype] = mod
+
+ handler_call_begin(mod, data, frequency)
+ return(mod)
+
+def handler_call_begin(mod, data, frequency):
+ handler_handle_part(mod, data, "__begin__", None, None, frequency)
+
+def handler_call_end(mod, data, frequency):
+ handler_handle_part(mod, data, "__end__", None, None, frequency)
+
+def handler_handle_part(mod, data, ctype, filename, payload, frequency):
+ # only add the handler if the module should run
+ modfreq = getattr(mod, "frequency", per_instance)
+ if not ( modfreq == per_always or
+ ( frequency == per_instance and modfreq == per_instance)):
+ return
+ if mod.handler_version == 1:
+ mod.handle_part(data, ctype, filename, payload)
+ else:
+ mod.handle_part(data, ctype, filename, payload, frequency)
+
+def partwalker_handle_handler(pdata, ctype, filename, payload):
+
+ curcount = pdata['handlercount']
+ modname = 'part-handler-%03d' % curcount
+ frequency = pdata['frequency']
+
+ modfname = modname + ".py"
+ util.write_file("%s/%s" % (pdata['handlerdir'], modfname), payload, 0600)
+
+ pdata['handlercount'] = curcount + 1
+
+ try:
+ mod = __import__(modname)
+ handler_register(mod, pdata['handlers'], pdata['data'], frequency)
+ except:
+ util.logexc(log)
+ traceback.print_exc(file=sys.stderr)
+ return
+
+def partwalker_callback(pdata, ctype, filename, payload):
+ # data here is the part_handlers array and then the data to pass through
+ if ctype == "text/part-handler":
+ if 'handlercount' not in pdata:
+ pdata['handlercount'] = 0
+ partwalker_handle_handler(pdata, ctype, filename, payload)
+ return
+ if ctype not in pdata['handlers']:
+ return
+ handler_handle_part(pdata['handlers'][ctype], pdata['data'],
+ ctype, filename, payload, pdata['frequency'])
+
+class InternalPartHandler:
+ freq = per_instance
+ mtypes = [ ]
+ handler_version = 1
+ handler = None
+ def __init__(self, handler, mtypes, frequency, version = 2):
+ self.handler = handler
+ self.mtypes = mtypes
+ self.frequency = frequency
+ self.handler_version = version
+
+ def __repr__():
+ return("InternalPartHandler: [%s]" % self.mtypes)
+
+ def list_types(self):
+ return(self.mtypes)
+
+ def handle_part(self, data, ctype, filename, payload, frequency):
+ return(self.handler(data, ctype, filename, payload, frequency))