~launchpad-pqm/launchpad/devel

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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
# Copyright 2009 Canonical Ltd.  This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).

"""Implementation of the Launchpad script to list modified branches."""

__metaclass__ = type
__all__ = ['ModifiedBranchesScript']


from datetime import (
    datetime,
    timedelta,
    )
import os
from time import strptime

import pytz
from zope.component import getUtility

from canonical.config import config
from lp.code.enums import BranchType
from lp.code.interfaces.branchcollection import IAllBranches
from lp.codehosting.vfs import branch_id_to_path
from lp.services.scripts.base import (
    LaunchpadScript,
    LaunchpadScriptFailure,
    )


class ModifiedBranchesScript(LaunchpadScript):
    """List branches modified since the specified time.

    Only branches that have been modified since the specified time will be
    returned.  It is possible that the branch will have been modified only in
    the web UI and not actually received any more revisions, and will be a
    false positive.
    """

    description = (
        "List branch paths for branches modified since the specified time.")

    def __init__(self, name, dbuser=None, test_args=None):
        LaunchpadScript.__init__(self, name, dbuser, test_args)
        # Cache this on object creation so it can be used in tests.
        self.now_timestamp = datetime.utcnow()
        self.locations = set()

    def add_my_options(self):
        self.parser.set_defaults(
            strip_prefix='/srv/',
            append_suffix='/**')
        self.parser.add_option(
            "-s", "--since", metavar="DATE",
            help="A date in the format YYYY-MM-DD.  Branches that "
            "have been modified since this date will be returned.")
        self.parser.add_option(
            "-l", "--last-hours", metavar="HOURS", type="int",
            help="Return the branches that have been modified in "
            "the last HOURS number of hours.")
        self.parser.add_option(
            "--strip-prefix", metavar="PREFIX",
            help="The prefix to remove from the branch locations.  "
            "Defaults to '/srv/'.")
        self.parser.add_option(
            "--append-suffix", metavar="SUFFIX",
            help="A suffix to append to the end of the branch locations.  "
            "Defaults to '/**'.")

    def get_last_modified_epoch(self):
        """Return the timezone aware datetime for the last modified epoch. """
        if (self.options.last_hours is not None and
            self.options.since is not None):
            raise LaunchpadScriptFailure(
                "Only one of --since or --last-hours can be specified.")
        last_modified = None
        if self.options.last_hours is not None:
            last_modified = (
                self.now_timestamp - timedelta(hours=self.options.last_hours))
        elif self.options.since is not None:
            try:
                parsed_time = strptime(self.options.since, '%Y-%m-%d')
                last_modified = datetime(*(parsed_time[:3]))
            except ValueError, e:
                raise LaunchpadScriptFailure(str(e))
        else:
            raise LaunchpadScriptFailure(
                "One of --since or --last-hours needs to be specified.")

        # Make the datetime timezone aware.
        return last_modified.replace(tzinfo=pytz.UTC)

    def branch_location(self, branch):
        """Return the  branch path for the given branch."""
        path = branch_id_to_path(branch.id)
        return os.path.join(config.codehosting.mirrored_branches_root, path)

    def process_location(self, location):
        """Strip the defined prefix, and append the suffix as configured."""
        if location.startswith(self.options.strip_prefix):
            location = location[len(self.options.strip_prefix):]
        return location + self.options.append_suffix

    def update_locations(self, location):
        """Add the location, and all the possible parent directories."""
        paths = location.split('/')
        curr = []
        for segment in paths:
            curr.append(segment)
            # Don't add an empty string.
            if curr != ['']:
                self.locations.add('/'.join(curr))

    def main(self):
        last_modified = self.get_last_modified_epoch()
        self.logger.info(
            "Looking for branches modified since %s", last_modified)
        collection = getUtility(IAllBranches)
        collection = collection.withBranchType(
            BranchType.HOSTED, BranchType.MIRRORED, BranchType.IMPORTED)
        collection = collection.scannedSince(last_modified)
        for branch in collection.getBranches():
            self.logger.info(branch.unique_name)
            location = self.branch_location(branch)
            self.update_locations(self.process_location(location))

        for location in sorted(self.locations):
            print location

        self.logger.info("Done.")