source: trunk/packages/sipb-xen-base/files/usr/share/python-support/sipb-xen-base/invirt/config.py @ 793

Last change on this file since 793 was 793, checked in by y_z, 16 years ago

added shared/exclusive locking; added shared locking of initial JSON cache read

File size: 2.2 KB
RevLine 
[784]1import json
[778]2from invirt.common import *
[781]3from os.path import getmtime
[726]4
[771]5default_src_path   = '/etc/invirt/master.yaml'
[781]6default_cache_path = '/var/lib/invirt/cache.json'
[793]7lock_file          = '/var/lib/invirt/cache.lock'
[726]8
[771]9def load(src_path = default_src_path,
10         cache_path = default_cache_path,
11         force_refresh = False):
12    """
13    Try loading the configuration from the faster-to-load JSON cache at
14    cache_path.  If it doesn't exist or is outdated, load the configuration
15    instead from the original YAML file at src_path and regenerate the cache.
16    I assume I have the permissions to write to the cache directory.
17    """
[793]18    # Namespace container for various state variables, so that they can be
19    # updated by closures.
20    ns = struct()
21
[771]22    if force_refresh:
[793]23        ns.do_refresh = True
[771]24    else:
25        src_mtime = getmtime(src_path)
[793]26        try:            cache_mtime   = getmtime(cache_path)
27        except OSError: ns.do_refresh = True
28        else:           ns.do_refresh = src_mtime > cache_mtime
[771]29
[793]30    if not ns.do_refresh:
31        # Try reading from the cache first.  This must be transactionally
32        # isolated from concurrent writes to prevent reading an incomplete
33        # (changing) version of the data (but the transaction can share the
34        # lock with other concurrent reads).
35        @with_lock_file(lock_file, False)
36        def read_cache():
37            try: ns.cfg = with_closing(file(cache_path))(lambda f: json.read(f.read()))
38            except: ns.do_refresh = True
[778]39
[793]40    if ns.do_refresh:
[781]41        # Atomically reload the source and regenerate the cache.  The read and
42        # write must be a single transaction, or a stale version may be
43        # written.
[793]44        @with_lock_file(lock_file)
45        def refresh_cache():
[784]46            import yaml
[792]47            try:    default_loader = yaml.CSafeLoader
48            except: default_loader = yaml.SafeLoader
[793]49            ns.cfg = with_closing(file(src_path))(lambda f: yaml.load(f, default_loader))
50            try: with_closing(file(cache_path, 'w'))(lambda f: f.write(json.write(ns.cfg)))
[781]51            except: pass # silent failure
[793]52    return ns.cfg
[771]53
[778]54dicts = load()
55structs = dicts2struct(dicts)
56
[726]57# vim:et:sw=4:ts=4
Note: See TracBrowser for help on using the repository browser.