diff options
Diffstat (limited to 'meta')
-rw-r--r-- | meta/classes-global/retain.bbclass | 182 | ||||
-rw-r--r-- | meta/lib/oeqa/selftest/cases/retain.py | 241 |
2 files changed, 423 insertions, 0 deletions
diff --git a/meta/classes-global/retain.bbclass b/meta/classes-global/retain.bbclass new file mode 100644 index 0000000000..46e8c256cf --- /dev/null +++ b/meta/classes-global/retain.bbclass | |||
@@ -0,0 +1,182 @@ | |||
1 | # Creates a tarball of the work directory for a recipe when one of its | ||
2 | # tasks fails, or any other nominated directories. | ||
3 | # Useful in cases where the environment in which builds are run is | ||
4 | # ephemeral or otherwise inaccessible for examination during | ||
5 | # debugging. | ||
6 | # | ||
7 | # To enable, simply add the following to your configuration: | ||
8 | # | ||
9 | # INHERIT += "retain" | ||
10 | # | ||
11 | # You can specify the recipe-specific directories to save upon failure | ||
12 | # or always (space-separated) e.g.: | ||
13 | # | ||
14 | # RETAIN_DIRS_FAILURE = "${WORKDIR};prefix=workdir" # default | ||
15 | # RETAIN_DIRS_ALWAYS = "${T}" | ||
16 | # | ||
17 | # Naturally you can use overrides to limit it to a specific recipe: | ||
18 | # RETAIN_DIRS_ALWAYS:pn-somerecipe = "${T}" | ||
19 | # | ||
20 | # You can also specify global (non-recipe-specific) directories to save: | ||
21 | # | ||
22 | # RETAIN_DIRS_GLOBAL_FAILURE = "${LOG_DIR}" | ||
23 | # RETAIN_DIRS_GLOBAL_ALWAYS = "${BUILDSTATS_BASE}" | ||
24 | # | ||
25 | # If you wish to use a different tarball name prefix than the default of | ||
26 | # the directory name, you can do so by specifying a ;prefix= followed by | ||
27 | # the desired prefix (no spaces) in any of the RETAIN_DIRS_* variables. | ||
28 | # e.g. to always save the log files with a "recipelogs" as the prefix for | ||
29 | # the tarball of ${T} you would do this: | ||
30 | # | ||
31 | # RETAIN_DIRS_ALWAYS = "${T};prefix=recipelogs" | ||
32 | # | ||
33 | # Notes: | ||
34 | # * For this to be useful you also need corresponding logic in your build | ||
35 | # orchestration tool to pick up any files written out to RETAIN_OUTDIR | ||
36 | # (with the other assumption being that no files are present there at | ||
37 | # the start of the build, since there is no logic to purge old files). | ||
38 | # * Work directories can be quite large, so saving them can take some time | ||
39 | # and of course space. | ||
40 | # * Tarball creation is deferred to the end of the build, thus you will | ||
41 | # get the state at the end, not immediately upon failure. | ||
42 | # * Extra directories must naturally be populated at the time the retain | ||
43 | # class goes to save them (build completion); to try ensure this for | ||
44 | # things that are also saved on build completion (e.g. buildstats), put | ||
45 | # the INHERIT += "retain" after the INHERIT += lines for the class that | ||
46 | # is writing out the data that you wish to save. | ||
47 | # * The tarballs have the tarball name as a top-level directory so that | ||
48 | # multiple tarballs can be extracted side-by-side easily. | ||
49 | # | ||
50 | # Copyright (c) 2020, 2024 Microsoft Corporation | ||
51 | # | ||
52 | # SPDX-License-Identifier: GPL-2.0-only | ||
53 | # | ||
54 | |||
55 | RETAIN_OUTDIR ?= "${TMPDIR}/retained" | ||
56 | RETAIN_DIRS_FAILURE ?= "${WORKDIR};prefix=workdir" | ||
57 | RETAIN_DIRS_ALWAYS ?= "" | ||
58 | RETAIN_DIRS_GLOBAL_FAILURE ?= "" | ||
59 | RETAIN_DIRS_GLOBAL_ALWAYS ?= "" | ||
60 | RETAIN_TARBALL_SUFFIX ?= "${DATETIME}.tar.gz" | ||
61 | RETAIN_ENABLED ?= "1" | ||
62 | |||
63 | |||
64 | def retain_retain_dir(desc, tarprefix, path, tarbasepath, d): | ||
65 | import datetime | ||
66 | |||
67 | outdir = d.getVar('RETAIN_OUTDIR') | ||
68 | bb.utils.mkdirhier(outdir) | ||
69 | suffix = d.getVar('RETAIN_TARBALL_SUFFIX') | ||
70 | tarname = '%s_%s' % (tarprefix, suffix) | ||
71 | tarfp = os.path.join(outdir, '%s' % tarname) | ||
72 | tardir = os.path.relpath(path, tarbasepath) | ||
73 | cmdargs = ['tar', 'cfa', tarfp] | ||
74 | # Prefix paths within the tarball with the tarball name so that | ||
75 | # multiple tarballs can be extracted side-by-side | ||
76 | tarname_noext = os.path.splitext(tarname)[0] | ||
77 | if tarname_noext.endswith('.tar'): | ||
78 | tarname_noext = tarname_noext[:-4] | ||
79 | cmdargs += ['--transform', 's:^:%s/:' % tarname_noext] | ||
80 | cmdargs += [tardir] | ||
81 | try: | ||
82 | bb.process.run(cmdargs, cwd=tarbasepath) | ||
83 | except bb.process.ExecutionError as e: | ||
84 | # It is possible for other tasks to be writing to the workdir | ||
85 | # while we are tarring it up, in which case tar will return 1, | ||
86 | # but we don't care in this situation (tar returns 2 for other | ||
87 | # errors so we we will see those) | ||
88 | if e.exitcode != 1: | ||
89 | bb.warn('retain: error saving %s: %s' % (desc, str(e))) | ||
90 | |||
91 | |||
92 | addhandler retain_task_handler | ||
93 | retain_task_handler[eventmask] = "bb.build.TaskFailed bb.build.TaskSucceeded" | ||
94 | |||
95 | addhandler retain_build_handler | ||
96 | retain_build_handler[eventmask] = "bb.event.BuildStarted bb.event.BuildCompleted" | ||
97 | |||
98 | python retain_task_handler() { | ||
99 | if d.getVar('RETAIN_ENABLED') != '1': | ||
100 | return | ||
101 | |||
102 | dirs = d.getVar('RETAIN_DIRS_ALWAYS') | ||
103 | if isinstance(e, bb.build.TaskFailed): | ||
104 | dirs += ' ' + d.getVar('RETAIN_DIRS_FAILURE') | ||
105 | |||
106 | dirs = dirs.strip().split() | ||
107 | if dirs: | ||
108 | outdir = d.getVar('RETAIN_OUTDIR') | ||
109 | bb.utils.mkdirhier(outdir) | ||
110 | dirlist_file = os.path.join(outdir, 'retain_dirs.list') | ||
111 | pn = d.getVar('PN') | ||
112 | taskname = d.getVar('BB_CURRENTTASK') | ||
113 | with open(dirlist_file, 'a') as f: | ||
114 | for entry in dirs: | ||
115 | f.write('%s %s %s\n' % (pn, taskname, entry)) | ||
116 | } | ||
117 | |||
118 | python retain_build_handler() { | ||
119 | outdir = d.getVar('RETAIN_OUTDIR') | ||
120 | dirlist_file = os.path.join(outdir, 'retain_dirs.list') | ||
121 | |||
122 | if isinstance(e, bb.event.BuildStarted): | ||
123 | if os.path.exists(dirlist_file): | ||
124 | os.remove(dirlist_file) | ||
125 | return | ||
126 | |||
127 | if d.getVar('RETAIN_ENABLED') != '1': | ||
128 | return | ||
129 | |||
130 | savedirs = {} | ||
131 | try: | ||
132 | with open(dirlist_file, 'r') as f: | ||
133 | for line in f: | ||
134 | pn, _, path = line.rstrip().split() | ||
135 | if not path in savedirs: | ||
136 | savedirs[path] = pn | ||
137 | os.remove(dirlist_file) | ||
138 | except FileNotFoundError: | ||
139 | pass | ||
140 | |||
141 | if e.getFailures(): | ||
142 | for path in (d.getVar('RETAIN_DIRS_GLOBAL_FAILURE') or '').strip().split(): | ||
143 | savedirs[path] = '' | ||
144 | |||
145 | for path in (d.getVar('RETAIN_DIRS_GLOBAL_ALWAYS') or '').strip().split(): | ||
146 | savedirs[path] = '' | ||
147 | |||
148 | if savedirs: | ||
149 | bb.plain('NOTE: retain: retaining build output...') | ||
150 | count = 0 | ||
151 | for path, pn in savedirs.items(): | ||
152 | prefix = None | ||
153 | if ';' in path: | ||
154 | pathsplit = path.split(';') | ||
155 | path = pathsplit[0] | ||
156 | for param in pathsplit[1:]: | ||
157 | if '=' in param: | ||
158 | name, value = param.split('=', 1) | ||
159 | if name == 'prefix': | ||
160 | prefix = value | ||
161 | else: | ||
162 | bb.error('retain: invalid parameter "%s" in RETAIN_* variable value' % param) | ||
163 | return | ||
164 | else: | ||
165 | bb.error('retain: parameter "%s" missing value in RETAIN_* variable value' % param) | ||
166 | return | ||
167 | if prefix: | ||
168 | itemname = prefix | ||
169 | else: | ||
170 | itemname = os.path.basename(path) | ||
171 | if pn: | ||
172 | # Always add the recipe name in front | ||
173 | itemname = pn + '_' + itemname | ||
174 | if os.path.exists(path): | ||
175 | retain_retain_dir(itemname, itemname, path, os.path.dirname(path), d) | ||
176 | count += 1 | ||
177 | else: | ||
178 | bb.warn('retain: path %s does not currently exist' % path) | ||
179 | if count: | ||
180 | item = 'archive' if count == 1 else 'archives' | ||
181 | bb.plain('NOTE: retain: saved %d %s to %s' % (count, item, outdir)) | ||
182 | } | ||
diff --git a/meta/lib/oeqa/selftest/cases/retain.py b/meta/lib/oeqa/selftest/cases/retain.py new file mode 100644 index 0000000000..892be45857 --- /dev/null +++ b/meta/lib/oeqa/selftest/cases/retain.py | |||
@@ -0,0 +1,241 @@ | |||
1 | # Tests for retain.bbclass | ||
2 | # | ||
3 | # Copyright OpenEmbedded Contributors | ||
4 | # | ||
5 | # SPDX-License-Identifier: MIT | ||
6 | # | ||
7 | |||
8 | import os | ||
9 | import glob | ||
10 | import fnmatch | ||
11 | import oe.path | ||
12 | import shutil | ||
13 | import tarfile | ||
14 | from oeqa.utils.commands import bitbake, get_bb_vars | ||
15 | from oeqa.selftest.case import OESelftestTestCase | ||
16 | |||
17 | class Retain(OESelftestTestCase): | ||
18 | |||
19 | def test_retain_always(self): | ||
20 | """ | ||
21 | Summary: Test retain class with RETAIN_DIRS_ALWAYS | ||
22 | Expected: Archive written to RETAIN_OUTDIR when build of test recipe completes | ||
23 | Product: oe-core | ||
24 | Author: Paul Eggleton <paul.eggleton@microsoft.com> | ||
25 | """ | ||
26 | |||
27 | test_recipe = 'quilt-native' | ||
28 | |||
29 | features = 'INHERIT += "retain"\n' | ||
30 | features += 'RETAIN_DIRS_ALWAYS = "${T}"\n' | ||
31 | self.write_config(features) | ||
32 | |||
33 | bitbake('-c clean %s' % test_recipe) | ||
34 | |||
35 | bb_vars = get_bb_vars(['RETAIN_OUTDIR', 'TMPDIR']) | ||
36 | retain_outdir = bb_vars['RETAIN_OUTDIR'] or '' | ||
37 | tmpdir = bb_vars['TMPDIR'] | ||
38 | if len(retain_outdir) < 5: | ||
39 | self.fail('RETAIN_OUTDIR value "%s" is invalid' % retain_outdir) | ||
40 | if not oe.path.is_path_parent(tmpdir, retain_outdir): | ||
41 | self.fail('RETAIN_OUTDIR (%s) is not underneath TMPDIR (%s)' % (retain_outdir, tmpdir)) | ||
42 | try: | ||
43 | shutil.rmtree(retain_outdir) | ||
44 | except FileNotFoundError: | ||
45 | pass | ||
46 | |||
47 | bitbake(test_recipe) | ||
48 | if not glob.glob(os.path.join(retain_outdir, '%s_temp_*.tar.gz' % test_recipe)): | ||
49 | self.fail('No output archive for %s created' % test_recipe) | ||
50 | |||
51 | |||
52 | def test_retain_failure(self): | ||
53 | """ | ||
54 | Summary: Test retain class default behaviour | ||
55 | Expected: Archive written to RETAIN_OUTDIR only when build of test | ||
56 | recipe fails, and archive contents are as expected | ||
57 | Product: oe-core | ||
58 | Author: Paul Eggleton <paul.eggleton@microsoft.com> | ||
59 | """ | ||
60 | |||
61 | test_recipe_fail = 'error' | ||
62 | |||
63 | features = 'INHERIT += "retain"\n' | ||
64 | self.write_config(features) | ||
65 | |||
66 | bb_vars = get_bb_vars(['RETAIN_OUTDIR', 'TMPDIR', 'RETAIN_DIRS_ALWAYS', 'RETAIN_DIRS_GLOBAL_ALWAYS']) | ||
67 | if bb_vars['RETAIN_DIRS_ALWAYS']: | ||
68 | self.fail('RETAIN_DIRS_ALWAYS is set, this interferes with the test') | ||
69 | if bb_vars['RETAIN_DIRS_GLOBAL_ALWAYS']: | ||
70 | self.fail('RETAIN_DIRS_GLOBAL_ALWAYS is set, this interferes with the test') | ||
71 | retain_outdir = bb_vars['RETAIN_OUTDIR'] or '' | ||
72 | tmpdir = bb_vars['TMPDIR'] | ||
73 | if len(retain_outdir) < 5: | ||
74 | self.fail('RETAIN_OUTDIR value "%s" is invalid' % retain_outdir) | ||
75 | if not oe.path.is_path_parent(tmpdir, retain_outdir): | ||
76 | self.fail('RETAIN_OUTDIR (%s) is not underneath TMPDIR (%s)' % (retain_outdir, tmpdir)) | ||
77 | |||
78 | try: | ||
79 | shutil.rmtree(retain_outdir) | ||
80 | except FileNotFoundError: | ||
81 | pass | ||
82 | |||
83 | bitbake('-c clean %s' % test_recipe_fail) | ||
84 | |||
85 | if os.path.exists(retain_outdir): | ||
86 | retain_dirlist = os.listdir(retain_outdir) | ||
87 | if retain_dirlist: | ||
88 | self.fail('RETAIN_OUTDIR should be empty without failure, contents:\n%s' % '\n'.join(retain_dirlist)) | ||
89 | |||
90 | result = bitbake('-c compile %s' % test_recipe_fail, ignore_status=True) | ||
91 | if result.status == 0: | ||
92 | self.fail('Build of %s did not fail as expected' % test_recipe_fail) | ||
93 | |||
94 | archives = glob.glob(os.path.join(retain_outdir, '%s_*.tar.gz' % test_recipe_fail)) | ||
95 | if not archives: | ||
96 | self.fail('No output archive for %s created' % test_recipe_fail) | ||
97 | if len(archives) > 1: | ||
98 | self.fail('More than one archive for %s created' % test_recipe_fail) | ||
99 | for archive in archives: | ||
100 | found = False | ||
101 | archive_prefix = os.path.basename(archive).split('.tar')[0] | ||
102 | expected_prefix_start = '%s_workdir' % test_recipe_fail | ||
103 | if not archive_prefix.startswith(expected_prefix_start): | ||
104 | self.fail('Archive %s name does not start with expected prefix "%s"' % (os.path.basename(archive), expected_prefix_start)) | ||
105 | with tarfile.open(archive) as tf: | ||
106 | for ti in tf: | ||
107 | if not fnmatch.fnmatch(ti.name, '%s/*' % archive_prefix): | ||
108 | self.fail('File without tarball-named subdirectory within tarball %s: %s' % (os.path.basename(archive), ti.name)) | ||
109 | if ti.name.endswith('/temp/log.do_compile'): | ||
110 | found = True | ||
111 | if not found: | ||
112 | self.fail('Did not find log.do_compile in output archive %s' % os.path.basename(archive)) | ||
113 | |||
114 | |||
115 | def test_retain_global(self): | ||
116 | """ | ||
117 | Summary: Test retain class RETAIN_DIRS_GLOBAL_* behaviour | ||
118 | Expected: Ensure RETAIN_DIRS_GLOBAL_ALWAYS always causes an | ||
119 | archive to be created, and RETAIN_DIRS_GLOBAL_FAILURE | ||
120 | only causes an archive to be created on failure. | ||
121 | Also test archive naming (with : character) as an | ||
122 | added bonus. | ||
123 | Product: oe-core | ||
124 | Author: Paul Eggleton <paul.eggleton@microsoft.com> | ||
125 | """ | ||
126 | |||
127 | test_recipe = 'quilt-native' | ||
128 | test_recipe_fail = 'error' | ||
129 | |||
130 | features = 'INHERIT += "retain"\n' | ||
131 | features += 'RETAIN_DIRS_GLOBAL_ALWAYS = "${LOG_DIR};prefix=buildlogs"\n' | ||
132 | features += 'RETAIN_DIRS_GLOBAL_FAILURE = "${STAMPS_DIR}"\n' | ||
133 | self.write_config(features) | ||
134 | |||
135 | bitbake('-c clean %s' % test_recipe) | ||
136 | |||
137 | bb_vars = get_bb_vars(['RETAIN_OUTDIR', 'TMPDIR', 'STAMPS_DIR']) | ||
138 | retain_outdir = bb_vars['RETAIN_OUTDIR'] or '' | ||
139 | tmpdir = bb_vars['TMPDIR'] | ||
140 | if len(retain_outdir) < 5: | ||
141 | self.fail('RETAIN_OUTDIR value "%s" is invalid' % retain_outdir) | ||
142 | if not oe.path.is_path_parent(tmpdir, retain_outdir): | ||
143 | self.fail('RETAIN_OUTDIR (%s) is not underneath TMPDIR (%s)' % (retain_outdir, tmpdir)) | ||
144 | try: | ||
145 | shutil.rmtree(retain_outdir) | ||
146 | except FileNotFoundError: | ||
147 | pass | ||
148 | |||
149 | # Test success case | ||
150 | bitbake(test_recipe) | ||
151 | if not glob.glob(os.path.join(retain_outdir, 'buildlogs_*.tar.gz')): | ||
152 | self.fail('No output archive for LOG_DIR created') | ||
153 | stamps_dir = bb_vars['STAMPS_DIR'] | ||
154 | if glob.glob(os.path.join(retain_outdir, '%s_*.tar.gz' % os.path.basename(stamps_dir))): | ||
155 | self.fail('Output archive for STAMPS_DIR created when it should not have been') | ||
156 | |||
157 | # Test failure case | ||
158 | result = bitbake('-c compile %s' % test_recipe_fail, ignore_status=True) | ||
159 | if result.status == 0: | ||
160 | self.fail('Build of %s did not fail as expected' % test_recipe_fail) | ||
161 | if not glob.glob(os.path.join(retain_outdir, '%s_*.tar.gz' % os.path.basename(stamps_dir))): | ||
162 | self.fail('Output archive for STAMPS_DIR not created') | ||
163 | if len(glob.glob(os.path.join(retain_outdir, 'buildlogs_*.tar.gz'))) != 2: | ||
164 | self.fail('Should be exactly two buildlogs archives in output dir') | ||
165 | |||
166 | |||
167 | def test_retain_misc(self): | ||
168 | """ | ||
169 | Summary: Test retain class with RETAIN_ENABLED and RETAIN_TARBALL_SUFFIX | ||
170 | Expected: Archive written to RETAIN_OUTDIR only when RETAIN_ENABLED is set | ||
171 | and archive contents are as expected. Also test archive naming | ||
172 | (with : character) as an added bonus. | ||
173 | Product: oe-core | ||
174 | Author: Paul Eggleton <paul.eggleton@microsoft.com> | ||
175 | """ | ||
176 | |||
177 | test_recipe_fail = 'error' | ||
178 | |||
179 | features = 'INHERIT += "retain"\n' | ||
180 | features += 'RETAIN_DIRS_ALWAYS = "${T}"\n' | ||
181 | features += 'RETAIN_ENABLED = "0"\n' | ||
182 | self.write_config(features) | ||
183 | |||
184 | bb_vars = get_bb_vars(['RETAIN_OUTDIR', 'TMPDIR']) | ||
185 | retain_outdir = bb_vars['RETAIN_OUTDIR'] or '' | ||
186 | tmpdir = bb_vars['TMPDIR'] | ||
187 | if len(retain_outdir) < 5: | ||
188 | self.fail('RETAIN_OUTDIR value "%s" is invalid' % retain_outdir) | ||
189 | if not oe.path.is_path_parent(tmpdir, retain_outdir): | ||
190 | self.fail('RETAIN_OUTDIR (%s) is not underneath TMPDIR (%s)' % (retain_outdir, tmpdir)) | ||
191 | |||
192 | try: | ||
193 | shutil.rmtree(retain_outdir) | ||
194 | except FileNotFoundError: | ||
195 | pass | ||
196 | |||
197 | bitbake('-c clean %s' % test_recipe_fail) | ||
198 | result = bitbake('-c compile %s' % test_recipe_fail, ignore_status=True) | ||
199 | if result.status == 0: | ||
200 | self.fail('Build of %s did not fail as expected' % test_recipe_fail) | ||
201 | |||
202 | if os.path.exists(retain_outdir) and os.listdir(retain_outdir): | ||
203 | self.fail('RETAIN_OUTDIR should be empty with RETAIN_ENABLED = "0"') | ||
204 | |||
205 | features = 'INHERIT += "retain"\n' | ||
206 | features += 'RETAIN_DIRS_ALWAYS = "${T};prefix=recipelogs"\n' | ||
207 | features += 'RETAIN_TARBALL_SUFFIX = "${DATETIME}-testsuffix.tar.bz2"\n' | ||
208 | features += 'RETAIN_ENABLED = "1"\n' | ||
209 | self.write_config(features) | ||
210 | |||
211 | result = bitbake('-c compile %s' % test_recipe_fail, ignore_status=True) | ||
212 | if result.status == 0: | ||
213 | self.fail('Build of %s did not fail as expected' % test_recipe_fail) | ||
214 | |||
215 | archives = glob.glob(os.path.join(retain_outdir, '%s_*-testsuffix.tar.bz2' % test_recipe_fail)) | ||
216 | if not archives: | ||
217 | self.fail('No output archive for %s created' % test_recipe_fail) | ||
218 | if len(archives) != 2: | ||
219 | self.fail('Two archives for %s expected, but %d exist' % (test_recipe_fail, len(archives))) | ||
220 | recipelogs_found = False | ||
221 | workdir_found = False | ||
222 | for archive in archives: | ||
223 | contents_found = False | ||
224 | archive_prefix = os.path.basename(archive).split('.tar')[0] | ||
225 | if archive_prefix.startswith('%s_recipelogs' % test_recipe_fail): | ||
226 | recipelogs_found = True | ||
227 | if archive_prefix.startswith('%s_workdir' % test_recipe_fail): | ||
228 | workdir_found = True | ||
229 | with tarfile.open(archive, 'r:bz2') as tf: | ||
230 | for ti in tf: | ||
231 | if not fnmatch.fnmatch(ti.name, '%s/*' % (archive_prefix)): | ||
232 | self.fail('File without tarball-named subdirectory within tarball %s: %s' % (os.path.basename(archive), ti.name)) | ||
233 | if ti.name.endswith('/log.do_compile'): | ||
234 | contents_found = True | ||
235 | if not contents_found: | ||
236 | # Both archives should contain this file | ||
237 | self.fail('Did not find log.do_compile in output archive %s' % os.path.basename(archive)) | ||
238 | if not recipelogs_found: | ||
239 | self.fail('No archive with expected "recipelogs" prefix found') | ||
240 | if not workdir_found: | ||
241 | self.fail('No archive with expected "workdir" prefix found') | ||