diff options
Diffstat (limited to 'scripts/oe-git-archive')
-rwxr-xr-x | scripts/oe-git-archive | 244 |
1 files changed, 244 insertions, 0 deletions
diff --git a/scripts/oe-git-archive b/scripts/oe-git-archive new file mode 100755 index 0000000000..419332ded1 --- /dev/null +++ b/scripts/oe-git-archive | |||
@@ -0,0 +1,244 @@ | |||
1 | #!/usr/bin/python3 | ||
2 | # | ||
3 | # Helper script for committing data to git and pushing upstream | ||
4 | # | ||
5 | # Copyright (c) 2017, Intel Corporation. | ||
6 | # | ||
7 | # This program is free software; you can redistribute it and/or modify it | ||
8 | # under the terms and conditions of the GNU General Public License, | ||
9 | # version 2, as published by the Free Software Foundation. | ||
10 | # | ||
11 | # This program is distributed in the hope it will be useful, but WITHOUT | ||
12 | # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
13 | # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
14 | # more details. | ||
15 | # | ||
16 | import argparse | ||
17 | import glob | ||
18 | import json | ||
19 | import logging | ||
20 | import math | ||
21 | import os | ||
22 | import re | ||
23 | import sys | ||
24 | from collections import namedtuple, OrderedDict | ||
25 | from datetime import datetime, timedelta, tzinfo | ||
26 | from operator import attrgetter | ||
27 | |||
28 | # Import oe and bitbake libs | ||
29 | scripts_path = os.path.dirname(os.path.realpath(__file__)) | ||
30 | sys.path.append(os.path.join(scripts_path, 'lib')) | ||
31 | import scriptpath | ||
32 | scriptpath.add_bitbake_lib_path() | ||
33 | scriptpath.add_oe_lib_path() | ||
34 | |||
35 | from oeqa.utils.git import GitRepo, GitError | ||
36 | from oeqa.utils.metadata import metadata_from_bb | ||
37 | |||
38 | |||
39 | # Setup logging | ||
40 | logging.basicConfig(level=logging.INFO, format="%(levelname)s: %(message)s") | ||
41 | log = logging.getLogger() | ||
42 | |||
43 | |||
44 | class ArchiveError(Exception): | ||
45 | """Internal error handling of this script""" | ||
46 | |||
47 | |||
48 | def format_str(string, fields): | ||
49 | """Format string using the given fields (dict)""" | ||
50 | try: | ||
51 | return string.format(**fields) | ||
52 | except KeyError as err: | ||
53 | raise ArchiveError("Unable to expand string '{}': unknown field {} " | ||
54 | "(valid fields are: {})".format( | ||
55 | string, err, ', '.join(sorted(fields.keys())))) | ||
56 | |||
57 | |||
58 | def init_git_repo(path, no_create): | ||
59 | """Initialize local Git repository""" | ||
60 | path = os.path.abspath(path) | ||
61 | if os.path.isfile(path): | ||
62 | raise ArchiveError("Invalid Git repo at {}: path exists but is not a " | ||
63 | "directory".format(path)) | ||
64 | if not os.path.isdir(path) or not os.listdir(path): | ||
65 | if no_create: | ||
66 | raise ArchiveError("No git repo at {}, refusing to create " | ||
67 | "one".format(path)) | ||
68 | if not os.path.isdir(path): | ||
69 | try: | ||
70 | os.mkdir(path) | ||
71 | except (FileNotFoundError, PermissionError) as err: | ||
72 | raise ArchiveError("Failed to mkdir {}: {}".format(path, err)) | ||
73 | if not os.listdir(path): | ||
74 | log.info("Initializing a new Git repo at %s", path) | ||
75 | repo = GitRepo.init(path) | ||
76 | try: | ||
77 | repo = GitRepo(path, is_topdir=True) | ||
78 | except GitError: | ||
79 | raise ArchiveError("Non-empty directory that is not a Git repository " | ||
80 | "at {}\nPlease specify an existing Git repository, " | ||
81 | "an empty directory or a non-existing directory " | ||
82 | "path.".format(path)) | ||
83 | return repo | ||
84 | |||
85 | |||
86 | def git_commit_data(repo, data_dir, branch, message): | ||
87 | """Commit data into a Git repository""" | ||
88 | log.info("Committing data into to branch %s", branch) | ||
89 | tmp_index = os.path.join(repo.git_dir, 'index.oe-git-archive') | ||
90 | try: | ||
91 | # Create new tree object from the data | ||
92 | env_update = {'GIT_INDEX_FILE': tmp_index, | ||
93 | 'GIT_WORK_TREE': os.path.abspath(data_dir)} | ||
94 | repo.run_cmd('add .', env_update) | ||
95 | tree = repo.run_cmd('write-tree', env_update) | ||
96 | |||
97 | # Create new commit object from the tree | ||
98 | parent = repo.rev_parse(branch) | ||
99 | git_cmd = ['commit-tree', tree, '-m', message] | ||
100 | if parent: | ||
101 | git_cmd += ['-p', parent] | ||
102 | commit = repo.run_cmd(git_cmd, env_update) | ||
103 | |||
104 | # Update branch head | ||
105 | git_cmd = ['update-ref', 'refs/heads/' + branch, commit] | ||
106 | if parent: | ||
107 | git_cmd.append(parent) | ||
108 | repo.run_cmd(git_cmd) | ||
109 | |||
110 | # Update current HEAD, if we're on branch 'branch' | ||
111 | if repo.get_current_branch() == branch: | ||
112 | log.info("Updating %s HEAD to latest commit", repo.top_dir) | ||
113 | repo.run_cmd('reset --hard') | ||
114 | |||
115 | return commit | ||
116 | finally: | ||
117 | if os.path.exists(tmp_index): | ||
118 | os.unlink(tmp_index) | ||
119 | |||
120 | |||
121 | def expand_tag_strings(repo, name_pattern, msg_subj_pattern, msg_body_pattern, | ||
122 | keywords): | ||
123 | """Generate tag name and message, with support for running id number""" | ||
124 | keyws = keywords.copy() | ||
125 | # Tag number is handled specially: if not defined, we autoincrement it | ||
126 | if 'tag_number' not in keyws: | ||
127 | # Fill in all other fields than 'tag_number' | ||
128 | keyws['tag_number'] = '{tag_number}' | ||
129 | tag_re = format_str(name_pattern, keyws) | ||
130 | # Replace parentheses for proper regex matching | ||
131 | tag_re = tag_re.replace('(', '\(').replace(')', '\)') + '$' | ||
132 | # Inject regex group pattern for 'tag_number' | ||
133 | tag_re = tag_re.format(tag_number='(?P<tag_number>[0-9]{1,5})') | ||
134 | |||
135 | keyws['tag_number'] = 0 | ||
136 | for existing_tag in repo.run_cmd('tag').splitlines(): | ||
137 | match = re.match(tag_re, existing_tag) | ||
138 | |||
139 | if match and int(match.group('tag_number')) >= keyws['tag_number']: | ||
140 | keyws['tag_number'] = int(match.group('tag_number')) + 1 | ||
141 | |||
142 | tag_name = format_str(name_pattern, keyws) | ||
143 | msg_subj= format_str(msg_subj_pattern.strip(), keyws) | ||
144 | msg_body = format_str(msg_body_pattern, keyws) | ||
145 | return tag_name, msg_subj + '\n\n' + msg_body | ||
146 | |||
147 | |||
148 | def parse_args(argv): | ||
149 | """Parse command line arguments""" | ||
150 | parser = argparse.ArgumentParser( | ||
151 | description="Commit data to git and push upstream", | ||
152 | formatter_class=argparse.ArgumentDefaultsHelpFormatter) | ||
153 | |||
154 | parser.add_argument('--debug', '-D', action='store_true', | ||
155 | help="Verbose logging") | ||
156 | parser.add_argument('--git-dir', '-g', required=True, | ||
157 | help="Local git directory to use") | ||
158 | parser.add_argument('--no-create', action='store_true', | ||
159 | help="If GIT_DIR is not a valid Git repository, do not " | ||
160 | "try to create one") | ||
161 | parser.add_argument('--push', '-p', nargs='?', default=False, const=True, | ||
162 | help="Push to remote") | ||
163 | parser.add_argument('--branch-name', '-b', | ||
164 | default='{hostname}/{branch}/{machine}', | ||
165 | help="Git branch name (pattern) to use") | ||
166 | parser.add_argument('--no-tag', action='store_true', | ||
167 | help="Do not create Git tag") | ||
168 | parser.add_argument('--tag-name', '-t', | ||
169 | default='{hostname}/{branch}/{machine}/{commit_count}-g{commit}/{tag_number}', | ||
170 | help="Tag name (pattern) to use") | ||
171 | parser.add_argument('--commit-msg-subject', | ||
172 | default='Results of {branch}:{commit} on {hostname}', | ||
173 | help="Subject line (pattern) to use in the commit message") | ||
174 | parser.add_argument('--commit-msg-body', | ||
175 | default='branch: {branch}\ncommit: {commit}\nhostname: {hostname}', | ||
176 | help="Commit message body (pattern)") | ||
177 | parser.add_argument('--tag-msg-subject', | ||
178 | default='Test run #{tag_number} of {branch}:{commit} on {hostname}', | ||
179 | help="Subject line (pattern) of the tag message") | ||
180 | parser.add_argument('--tag-msg-body', | ||
181 | default='', | ||
182 | help="Tag message body (pattern)") | ||
183 | parser.add_argument('data_dir', metavar='DATA_DIR', | ||
184 | help="Data to commit") | ||
185 | return parser.parse_args(argv) | ||
186 | |||
187 | |||
188 | def main(argv=None): | ||
189 | """Script entry point""" | ||
190 | args = parse_args(argv) | ||
191 | if args.debug: | ||
192 | log.setLevel(logging.DEBUG) | ||
193 | |||
194 | try: | ||
195 | if not os.path.isdir(args.data_dir): | ||
196 | raise ArchiveError("Not a directory: {}".format(args.data_dir)) | ||
197 | |||
198 | data_repo = init_git_repo(args.git_dir, args.no_create) | ||
199 | |||
200 | # Get keywords to be used in tag and branch names and messages | ||
201 | metadata = metadata_from_bb() | ||
202 | keywords = {'hostname': metadata['hostname'], | ||
203 | 'branch': metadata['layers']['meta']['branch'], | ||
204 | 'commit': metadata['layers']['meta']['commit'], | ||
205 | 'commit_count': metadata['layers']['meta']['commit_count'], | ||
206 | 'machine': metadata['config']['MACHINE']} | ||
207 | |||
208 | # Expand strings early in order to avoid getting into inconsistent | ||
209 | # state (e.g. no tag even if data was committed) | ||
210 | commit_msg = format_str(args.commit_msg_subject.strip(), keywords) | ||
211 | commit_msg += '\n\n' + format_str(args.commit_msg_body, keywords) | ||
212 | branch_name = format_str(args.branch_name, keywords) | ||
213 | tag_name = None | ||
214 | if not args.no_tag and args.tag_name: | ||
215 | tag_name, tag_msg = expand_tag_strings(data_repo, args.tag_name, | ||
216 | args.tag_msg_subject, | ||
217 | args.tag_msg_body, keywords) | ||
218 | |||
219 | # Commit data | ||
220 | commit = git_commit_data(data_repo, args.data_dir, branch_name, | ||
221 | commit_msg) | ||
222 | |||
223 | # Create tag | ||
224 | if tag_name: | ||
225 | log.info("Creating tag %s", tag_name) | ||
226 | data_repo.run_cmd(['tag', '-a', '-m', tag_msg, tag_name, commit]) | ||
227 | |||
228 | # Push data to remote | ||
229 | if args.push: | ||
230 | cmd = ['push', '--tags'] | ||
231 | if args.push is not True: | ||
232 | cmd.extend(['--repo', args.push]) | ||
233 | cmd.append(branch_name) | ||
234 | log.info("Pushing data to remote") | ||
235 | data_repo.run_cmd(cmd) | ||
236 | |||
237 | except ArchiveError as err: | ||
238 | log.error(str(err)) | ||
239 | return 1 | ||
240 | |||
241 | return 0 | ||
242 | |||
243 | if __name__ == "__main__": | ||
244 | sys.exit(main()) | ||