source: trunk/packages/invirt-dev/python/invirt/builder.py @ 3037

Last change on this file since 3037 was 3036, checked in by gdb, 14 years ago

Added reporting

File size: 5.3 KB
Line 
1"""Invirt build utilities.
2
3This module contains utility functions used by both the invirtibuilder
4and the remctl submission scripts that insert items into its queue.
5"""
6
7
8import os
9
10from debian_bundle import changelog
11from debian_bundle import deb822
12
13import invirt.common as c
14from invirt.config import structs as config
15
16
17_QUEUE_DIR = '/var/lib/invirt-dev/queue'
18_REPO_DIR = '/srv/git'
19_LOG_DIR = '/var/log/invirt/builds'
20_HOOKS_DIR = '/usr/share/invirt-dev/build-hooks'
21
22
23class InvalidBuild(ValueError):
24    pass
25
26
27def getRepo(package):
28    """Return the path to the git repo for a given package."""
29    return os.path.join(_REPO_DIR, 'invirt/packages', '%s.git' % package)
30
31def ensureValidPackage(package):
32    """Perform some basic sanity checks that the requested repo is in a
33    subdirectory of _REPO_DIR/invirt/packages.  This prevents weirdness
34    such as submitting a package like '../prod/...git'.  Also ensures that
35    the repo exists."""
36    # TODO: this might be easier just to regex
37    repo = os.path.abspath(getRepo(package))
38    parent_dir = os.path.dirname(repo)
39    prefix = os.path.join(_REPO_DIR, 'invirt/packages')
40    if not parent_dir.startswith(prefix):
41        raise InvalidBuild('Invalid package name %s' % package)
42    elif not os.path.exists(repo):
43        raise InvalidBuild('Nonexisting package %s' % package)
44
45def canonicalize_commit(package, commit, shorten=False):
46    if shorten:
47        flags = ['--short']
48    else:
49        flags = []
50    return c.captureOutput(['git', 'rev-parse'] + flags + [commit],
51                           cwd=getRepo(package)).strip()
52
53def pocketToGit(pocket):
54    """Map a pocket in the configuration to a git branch."""
55    return getattr(getattr(config.build.pockets, pocket), 'git', pocket)
56
57
58def pocketToApt(pocket):
59    """Map a pocket in the configuration to an apt repo pocket."""
60    return getattr(getattr(config.build.pockets, pocket), 'apt', pocket)
61
62
63def getGitFile(package, ref, path):
64    """Return the contents of a path from a git ref in a package."""
65    return c.captureOutput(['git', 'cat-file', 'blob', '%s:%s' % (ref, path)],
66                           cwd=getRepo(package))
67
68
69def getChangelog(package, ref):
70    """Get a changelog object for a given ref in a given package.
71
72    This returns a debian_bundle.changelog.Changelog object for a
73    given ref of a given package.
74    """
75    return changelog.Changelog(getGitFile(package, ref, 'debian/changelog'))
76
77def runHook(hook, args=[], stdin_str=None):
78    """Run a named hook."""
79    hook = os.path.join(_HOOKS_DIR, hook)
80    try:
81        c.captureOutput([hook] + args, stdin_str=stdin_str)
82    except OSError:
83        pass
84
85def getVersion(package, ref):
86    """Get the version of a given package at a particular ref."""
87    return getChangelog(package, ref).get_version()
88
89
90def validateBuild(pocket, package, commit):
91    """Given the parameters of a new build, validate that build.
92
93    The checks this function performs vary based on whether or not the
94    pocket is configured with allow_backtracking.
95
96    A build of a pocket without allow_backtracking set must be a
97    fast-forward of the previous revision, and the most recent version
98    in the changelog most be strictly greater than the version
99    currently in the repository.
100
101    In all cases, this revision of the package can only have the same
102    version number as any other revision currently in the apt
103    repository if they have the same commit ID.
104
105    If it's unspecified, it is assumed that pocket do not
106    allow_backtracking.
107
108    If this build request fails validation, this function will raise a
109    InvalidBuild exception, with information about why the validation
110    failed.
111
112    If this build request can be satisfied by copying the package from
113    another pocket, then this function returns that pocket. Otherwise,
114    it returns True.
115    """
116    ensureValidPackage(package)
117    package_repo = getRepo(package)
118    new_version = getVersion(package, commit)
119
120    ret = True
121
122    for p in config.build.pockets:
123        if p == pocket:
124            continue
125
126        b = pocketToGit(p)
127        current_commit = c.captureOutput(['git', 'rev-parse', b],
128                                         cwd=package_repo).strip()
129        current_version = getVersion(package, b)
130
131        if current_version == new_version:
132            if current_commit == commit:
133                ret = p
134            else:
135                raise InvalidBuild('Version %s of %s already available is in '
136                                   'pocket %s from commit %s' %
137                                   (new_version, package, p, current_commit))
138
139    if not config.build.pockets[pocket].get('allow_backtracking', False):
140        branch = pocketToGit(pocket)
141        current_version = getVersion(package, branch)
142        if new_version <= current_version:
143            raise InvalidBuild('New version %s of %s is not newer than '
144                               'version %s currently in pocket %s' %
145                               (new_version, package, current_version, pocket))
146
147        # Almost by definition, A is a fast-forward of B if B..A is
148        # empty
149        if not c.captureOutput(['git', 'rev-list', '%s..%s' % (commit, branch)]):
150            raise InvalidBuild('New commit %s of %s is not a fast-forward of'
151                               'commit currently in pocket %s' %
152                               (commit, package, pocket))
153
154    return ret
Note: See TracBrowser for help on using the repository browser.