~landscape/landscape-bundles/trunk

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
#!/usr/bin/python3

import argparse
import fileinput
from glob import glob
import json
import re
import subprocess
import sys


class Charm(object):

    def __init__(self, charm_name, series):
        """Create a charm object.
        @param charm_name: name of charm
        """
        reference = "cs:{}/{}".format(series, charm_name)
        json_info = subprocess.check_output(
            ["charm", "show", "--format=json", reference, "id"])
        charm_data = json.loads(json_info)
        self.revision = charm_data.get("id", {}).get("Revision")
        if self.revision is None:
            raise IOError("{}: {}".format(charm_name, charm_data))
        self.name = charm_name
        self.store_url = charm_data.get("id", {}).get("Id")


def _get_charm(charm_name, series):
    """
    Small wrapper to try to get the series charm you specify.  Falls back
    to precise if you specified something else
    """
    try:
        return Charm(charm_name, series)
    except IOError as e:
        if series != "trusty":
            try:
                return Charm(charm_name, series="trusty")
            except IOError as trusty_e:
                raise IOError(
                    "%s: not found? (even tried trusty): {}, {}".format(
                        charm_name, e, trusty_e))
        raise


def replace_in_config(filename, charms, series):
    """
    Iterate over each charm, replacing all occurences of old charm store
    urls with new versions.  Intent is to leave file in a state that can
    be tested and checked in if successful.

    Look for lines like:
      charm: cs:juju-gui-83

    Or:
      series: trusty
    """
    for line in fileinput.input(filename, inplace=1):
        pattern = "^(\s*)series:\s.*$"
        line = re.sub(pattern, r"\1series: %s" % series, line)
        for charm in charms:
            pattern = "cs:{}-[0-9]+".format(charm.name)
            line = re.sub(pattern, charm.store_url, line)
        sys.stdout.write(line)

    print("# Config written, use `bzr diff %s` to see any changes" % filename)


def get_options():
    """Parse and return command line options."""
    description = "Put new charmstore descriptors into a deployer file."
    parser = argparse.ArgumentParser(description=description)
    parser.add_argument("charms", help="List charms to update", nargs='*')
    parser.add_argument("--series", help="Charm series", default="bionic")
    return parser.parse_args()


def main():
    options = get_options()
    charms = []
    if len(options.charms):
        print("# Latest upstream versions of charms ({}):".format(
            options.series))
    for charm_name in options.charms:
        charm = _get_charm(charm_name, series=options.series)
        print("#   {}: {}".format(charm_name, charm.revision))
        charms.append(charm)

    for filename in glob("*.jinja2"):
        replace_in_config(filename, charms, series=options.series)


if __name__ == "__main__":
    sys.exit(main())