OLD | NEW |
| (Empty) |
1 #!/usr/bin/env python | |
2 # Copyright 2014 the V8 project authors. All rights reserved. | |
3 # Use of this source code is governed by a BSD-style license that can be | |
4 # found in the LICENSE file. | |
5 | |
6 import argparse | |
7 import sys | |
8 | |
9 from common_includes import * | |
10 | |
11 | |
12 class Preparation(Step): | |
13 MESSAGE = "Preparation." | |
14 | |
15 def RunStep(self): | |
16 # TODO(machenbach): Remove after the git switch. | |
17 if self.Config("PERSISTFILE_BASENAME") == "/tmp/v8-auto-tag-tempfile": | |
18 print "This script is disabled until after the v8 git migration." | |
19 return True | |
20 | |
21 self.CommonPrepare() | |
22 self.PrepareBranch() | |
23 self.GitCheckout("master") | |
24 self.vc.Pull() | |
25 | |
26 | |
27 class GetTags(Step): | |
28 MESSAGE = "Get all V8 tags." | |
29 | |
30 def RunStep(self): | |
31 self.GitCreateBranch(self._config["BRANCHNAME"]) | |
32 self["tags"] = self.vc.GetTags() | |
33 | |
34 | |
35 class GetOldestUntaggedVersion(Step): | |
36 MESSAGE = "Check if there's a version on bleeding edge without a tag." | |
37 | |
38 def RunStep(self): | |
39 tags = set(self["tags"]) | |
40 self["candidate"] = None | |
41 self["candidate_version"] = None | |
42 self["next"] = None | |
43 self["next_version"] = None | |
44 | |
45 # Iterate backwards through all automatic version updates. | |
46 for git_hash in self.GitLog( | |
47 format="%H", grep="\\[Auto\\-roll\\] Bump up version to").splitlines(): | |
48 | |
49 # Get the version. | |
50 if not self.GitCheckoutFileSafe(VERSION_FILE, git_hash): | |
51 continue | |
52 | |
53 self.ReadAndPersistVersion() | |
54 version = self.ArrayToVersion("") | |
55 | |
56 # Strip off trailing patch level (tags don't include tag level 0). | |
57 if version.endswith(".0"): | |
58 version = version[:-2] | |
59 | |
60 # Clean up checked-out version file. | |
61 self.GitCheckoutFileSafe(VERSION_FILE, "HEAD") | |
62 | |
63 if version in tags: | |
64 if self["candidate"]: | |
65 # Revision "git_hash" is tagged already and "candidate" was the next | |
66 # newer revision without a tag. | |
67 break | |
68 else: | |
69 print("Stop as %s is the latest version and it has been tagged." % | |
70 version) | |
71 self.CommonCleanup() | |
72 return True | |
73 else: | |
74 # This is the second oldest version without a tag. | |
75 self["next"] = self["candidate"] | |
76 self["next_version"] = self["candidate_version"] | |
77 | |
78 # This is the oldest version without a tag. | |
79 self["candidate"] = git_hash | |
80 self["candidate_version"] = version | |
81 | |
82 if not self["candidate"] or not self["candidate_version"]: | |
83 print "Nothing found to tag." | |
84 self.CommonCleanup() | |
85 return True | |
86 | |
87 print("Candidate for tagging is %s with version %s" % | |
88 (self["candidate"], self["candidate_version"])) | |
89 | |
90 | |
91 class GetLKGRs(Step): | |
92 MESSAGE = "Get the last lkgrs." | |
93 | |
94 def RunStep(self): | |
95 revision_url = "https://v8-status.appspot.com/revisions?format=json" | |
96 status_json = self.ReadURL(revision_url, wait_plan=[5, 20]) | |
97 self["lkgrs"] = [entry["revision"] | |
98 for entry in json.loads(status_json) if entry["status"]] | |
99 | |
100 | |
101 class CalculateTagRevision(Step): | |
102 MESSAGE = "Calculate the revision to tag." | |
103 | |
104 def LastLKGR(self, min_rev, max_rev): | |
105 """Finds the newest lkgr between min_rev (inclusive) and max_rev | |
106 (exclusive). | |
107 """ | |
108 for lkgr in self["lkgrs"]: | |
109 # LKGRs are reverse sorted. | |
110 if int(min_rev) <= int(lkgr) and int(lkgr) < int(max_rev): | |
111 return lkgr | |
112 return None | |
113 | |
114 def RunStep(self): | |
115 # Get the lkgr after the tag candidate and before the next tag candidate. | |
116 candidate_svn = self.vc.GitSvn(self["candidate"]) | |
117 if self["next"]: | |
118 next_svn = self.vc.GitSvn(self["next"]) | |
119 else: | |
120 # Don't include the version change commit itself if there is no upper | |
121 # limit yet. | |
122 candidate_svn = str(int(candidate_svn) + 1) | |
123 next_svn = sys.maxint | |
124 lkgr_svn = self.LastLKGR(candidate_svn, next_svn) | |
125 | |
126 if not lkgr_svn: | |
127 print "There is no lkgr since the candidate version yet." | |
128 self.CommonCleanup() | |
129 return True | |
130 | |
131 # Let's check if the lkgr is at least three hours old. | |
132 self["lkgr"] = self.vc.SvnGit(lkgr_svn) | |
133 if not self["lkgr"]: | |
134 print "Couldn't find git hash for lkgr %s" % lkgr_svn | |
135 self.CommonCleanup() | |
136 return True | |
137 | |
138 lkgr_utc_time = int(self.GitLog(n=1, format="%at", git_hash=self["lkgr"])) | |
139 current_utc_time = self._side_effect_handler.GetUTCStamp() | |
140 | |
141 if current_utc_time < lkgr_utc_time + 10800: | |
142 print "Candidate lkgr %s is too recent for tagging." % lkgr_svn | |
143 self.CommonCleanup() | |
144 return True | |
145 | |
146 print "Tagging revision %s with %s" % (lkgr_svn, self["candidate_version"]) | |
147 | |
148 | |
149 class MakeTag(Step): | |
150 MESSAGE = "Tag the version." | |
151 | |
152 def RunStep(self): | |
153 if not self._options.dry_run: | |
154 self.GitReset(self["lkgr"]) | |
155 # FIXME(machenbach): Make this work with the git repo. | |
156 self.vc.Tag(self["candidate_version"], | |
157 "svn/bleeding_edge", | |
158 "This won't work!") | |
159 | |
160 | |
161 class CleanUp(Step): | |
162 MESSAGE = "Clean up." | |
163 | |
164 def RunStep(self): | |
165 self.CommonCleanup() | |
166 | |
167 | |
168 class AutoTag(ScriptsBase): | |
169 def _PrepareOptions(self, parser): | |
170 parser.add_argument("--dry_run", help="Don't tag the new version.", | |
171 default=False, action="store_true") | |
172 | |
173 def _ProcessOptions(self, options): # pragma: no cover | |
174 if not options.dry_run and not options.author: | |
175 print "Specify your chromium.org email with -a" | |
176 return False | |
177 options.wait_for_lgtm = False | |
178 options.force_readline_defaults = True | |
179 options.force_upload = True | |
180 return True | |
181 | |
182 def _Config(self): | |
183 return { | |
184 "BRANCHNAME": "auto-tag-v8", | |
185 "PERSISTFILE_BASENAME": "/tmp/v8-auto-tag-tempfile", | |
186 } | |
187 | |
188 def _Steps(self): | |
189 return [ | |
190 Preparation, | |
191 GetTags, | |
192 GetOldestUntaggedVersion, | |
193 GetLKGRs, | |
194 CalculateTagRevision, | |
195 MakeTag, | |
196 CleanUp, | |
197 ] | |
198 | |
199 | |
200 if __name__ == "__main__": # pragma: no cover | |
201 sys.exit(AutoTag().Run()) | |
OLD | NEW |