summaryrefslogtreecommitdiffstats
path: root/meta/classes-global/sanity.bbclass
diff options
context:
space:
mode:
authorRichard Purdie <richard.purdie@linuxfoundation.org>2022-08-10 14:35:29 +0100
committerRichard Purdie <richard.purdie@linuxfoundation.org>2022-08-12 15:27:17 +0100
commitfd1517e2b51a170f2427122c6b95396db251d827 (patch)
treedabfe3e631339c2fc99a9ee7febb0f9c128e325e /meta/classes-global/sanity.bbclass
parent10317912ee319ccf7f83605d438b5cbf9663f296 (diff)
downloadpoky-fd1517e2b51a170f2427122c6b95396db251d827.tar.gz
classes: Update classes to match new bitbake class scope functionality
Move classes to classes-global or classes-recipe as appropriate to take advantage of new bitbake functionality to check class scope/usage. (From OE-Core rev: f5c128008365e141082c129417eb72d2751e8045) Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
Diffstat (limited to 'meta/classes-global/sanity.bbclass')
-rw-r--r--meta/classes-global/sanity.bbclass1028
1 files changed, 1028 insertions, 0 deletions
diff --git a/meta/classes-global/sanity.bbclass b/meta/classes-global/sanity.bbclass
new file mode 100644
index 0000000000..4104694478
--- /dev/null
+++ b/meta/classes-global/sanity.bbclass
@@ -0,0 +1,1028 @@
1#
2# Copyright OpenEmbedded Contributors
3#
4# SPDX-License-Identifier: MIT
5#
6
7#
8# Sanity check the users setup for common misconfigurations
9#
10
11SANITY_REQUIRED_UTILITIES ?= "patch diffstat git bzip2 tar \
12 gzip gawk chrpath wget cpio perl file which"
13
14def bblayers_conf_file(d):
15 return os.path.join(d.getVar('TOPDIR'), 'conf/bblayers.conf')
16
17def sanity_conf_read(fn):
18 with open(fn, 'r') as f:
19 lines = f.readlines()
20 return lines
21
22def sanity_conf_find_line(pattern, lines):
23 import re
24 return next(((index, line)
25 for index, line in enumerate(lines)
26 if re.search(pattern, line)), (None, None))
27
28def sanity_conf_update(fn, lines, version_var_name, new_version):
29 index, line = sanity_conf_find_line(r"^%s" % version_var_name, lines)
30 lines[index] = '%s = "%d"\n' % (version_var_name, new_version)
31 with open(fn, "w") as f:
32 f.write(''.join(lines))
33
34# Functions added to this variable MUST throw a NotImplementedError exception unless
35# they successfully changed the config version in the config file. Exceptions
36# are used since exec_func doesn't handle return values.
37BBLAYERS_CONF_UPDATE_FUNCS += " \
38 conf/bblayers.conf:LCONF_VERSION:LAYER_CONF_VERSION:oecore_update_bblayers \
39 conf/local.conf:CONF_VERSION:LOCALCONF_VERSION:oecore_update_localconf \
40 conf/site.conf:SCONF_VERSION:SITE_CONF_VERSION:oecore_update_siteconf \
41"
42
43SANITY_DIFF_TOOL ?= "meld"
44
45SANITY_LOCALCONF_SAMPLE ?= "${COREBASE}/meta*/conf/local.conf.sample"
46python oecore_update_localconf() {
47 # Check we are using a valid local.conf
48 current_conf = d.getVar('CONF_VERSION')
49 conf_version = d.getVar('LOCALCONF_VERSION')
50
51 failmsg = """Your version of local.conf was generated from an older/newer version of
52local.conf.sample and there have been updates made to this file. Please compare the two
53files and merge any changes before continuing.
54
55Matching the version numbers will remove this message.
56
57\"${SANITY_DIFF_TOOL} conf/local.conf ${SANITY_LOCALCONF_SAMPLE}\"
58
59is a good way to visualise the changes."""
60 failmsg = d.expand(failmsg)
61
62 raise NotImplementedError(failmsg)
63}
64
65SANITY_SITECONF_SAMPLE ?= "${COREBASE}/meta*/conf/site.conf.sample"
66python oecore_update_siteconf() {
67 # If we have a site.conf, check it's valid
68 current_sconf = d.getVar('SCONF_VERSION')
69 sconf_version = d.getVar('SITE_CONF_VERSION')
70
71 failmsg = """Your version of site.conf was generated from an older version of
72site.conf.sample and there have been updates made to this file. Please compare the two
73files and merge any changes before continuing.
74
75Matching the version numbers will remove this message.
76
77\"${SANITY_DIFF_TOOL} conf/site.conf ${SANITY_SITECONF_SAMPLE}\"
78
79is a good way to visualise the changes."""
80 failmsg = d.expand(failmsg)
81
82 raise NotImplementedError(failmsg)
83}
84
85SANITY_BBLAYERCONF_SAMPLE ?= "${COREBASE}/meta*/conf/bblayers.conf.sample"
86python oecore_update_bblayers() {
87 # bblayers.conf is out of date, so see if we can resolve that
88
89 current_lconf = int(d.getVar('LCONF_VERSION'))
90 lconf_version = int(d.getVar('LAYER_CONF_VERSION'))
91
92 failmsg = """Your version of bblayers.conf has the wrong LCONF_VERSION (has ${LCONF_VERSION}, expecting ${LAYER_CONF_VERSION}).
93Please compare your file against bblayers.conf.sample and merge any changes before continuing.
94"${SANITY_DIFF_TOOL} conf/bblayers.conf ${SANITY_BBLAYERCONF_SAMPLE}"
95
96is a good way to visualise the changes."""
97 failmsg = d.expand(failmsg)
98
99 if not current_lconf:
100 raise NotImplementedError(failmsg)
101
102 lines = []
103
104 if current_lconf < 4:
105 raise NotImplementedError(failmsg)
106
107 bblayers_fn = bblayers_conf_file(d)
108 lines = sanity_conf_read(bblayers_fn)
109
110 if current_lconf == 4 and lconf_version > 4:
111 topdir_var = '$' + '{TOPDIR}'
112 index, bbpath_line = sanity_conf_find_line('BBPATH', lines)
113 if bbpath_line:
114 start = bbpath_line.find('"')
115 if start != -1 and (len(bbpath_line) != (start + 1)):
116 if bbpath_line[start + 1] == '"':
117 lines[index] = (bbpath_line[:start + 1] +
118 topdir_var + bbpath_line[start + 1:])
119 else:
120 if not topdir_var in bbpath_line:
121 lines[index] = (bbpath_line[:start + 1] +
122 topdir_var + ':' + bbpath_line[start + 1:])
123 else:
124 raise NotImplementedError(failmsg)
125 else:
126 index, bbfiles_line = sanity_conf_find_line('BBFILES', lines)
127 if bbfiles_line:
128 lines.insert(index, 'BBPATH = "' + topdir_var + '"\n')
129 else:
130 raise NotImplementedError(failmsg)
131
132 current_lconf += 1
133 sanity_conf_update(bblayers_fn, lines, 'LCONF_VERSION', current_lconf)
134 bb.note("Your conf/bblayers.conf has been automatically updated.")
135 return
136
137 elif current_lconf == 5 and lconf_version > 5:
138 # Null update, to avoid issues with people switching between poky and other distros
139 current_lconf = 6
140 sanity_conf_update(bblayers_fn, lines, 'LCONF_VERSION', current_lconf)
141 bb.note("Your conf/bblayers.conf has been automatically updated.")
142 return
143
144 status.addresult()
145
146 elif current_lconf == 6 and lconf_version > 6:
147 # Handle rename of meta-yocto -> meta-poky
148 # This marks the start of separate version numbers but code is needed in OE-Core
149 # for the migration, one last time.
150 layers = d.getVar('BBLAYERS').split()
151 layers = [ os.path.basename(path) for path in layers ]
152 if 'meta-yocto' in layers:
153 found = False
154 while True:
155 index, meta_yocto_line = sanity_conf_find_line(r'.*meta-yocto[\'"\s\n]', lines)
156 if meta_yocto_line:
157 lines[index] = meta_yocto_line.replace('meta-yocto', 'meta-poky')
158 found = True
159 else:
160 break
161 if not found:
162 raise NotImplementedError(failmsg)
163 index, meta_yocto_line = sanity_conf_find_line('LCONF_VERSION.*\n', lines)
164 if meta_yocto_line:
165 lines[index] = 'POKY_BBLAYERS_CONF_VERSION = "1"\n'
166 else:
167 raise NotImplementedError(failmsg)
168 with open(bblayers_fn, "w") as f:
169 f.write(''.join(lines))
170 bb.note("Your conf/bblayers.conf has been automatically updated.")
171 return
172 current_lconf += 1
173 sanity_conf_update(bblayers_fn, lines, 'LCONF_VERSION', current_lconf)
174 bb.note("Your conf/bblayers.conf has been automatically updated.")
175 return
176
177 raise NotImplementedError(failmsg)
178}
179
180def raise_sanity_error(msg, d, network_error=False):
181 if d.getVar("SANITY_USE_EVENTS") == "1":
182 try:
183 bb.event.fire(bb.event.SanityCheckFailed(msg, network_error), d)
184 except TypeError:
185 bb.event.fire(bb.event.SanityCheckFailed(msg), d)
186 return
187
188 bb.fatal(""" OE-core's config sanity checker detected a potential misconfiguration.
189 Either fix the cause of this error or at your own risk disable the checker (see sanity.conf).
190 Following is the list of potential problems / advisories:
191
192 %s""" % msg)
193
194# Check a single tune for validity.
195def check_toolchain_tune(data, tune, multilib):
196 tune_errors = []
197 if not tune:
198 return "No tuning found for %s multilib." % multilib
199 localdata = bb.data.createCopy(data)
200 if multilib != "default":
201 # Apply the overrides so we can look at the details.
202 overrides = localdata.getVar("OVERRIDES", False) + ":virtclass-multilib-" + multilib
203 localdata.setVar("OVERRIDES", overrides)
204 bb.debug(2, "Sanity-checking tuning '%s' (%s) features:" % (tune, multilib))
205 features = (localdata.getVar("TUNE_FEATURES:tune-%s" % tune) or "").split()
206 if not features:
207 return "Tuning '%s' has no defined features, and cannot be used." % tune
208 valid_tunes = localdata.getVarFlags('TUNEVALID') or {}
209 conflicts = localdata.getVarFlags('TUNECONFLICTS') or {}
210 # [doc] is the documentation for the variable, not a real feature
211 if 'doc' in valid_tunes:
212 del valid_tunes['doc']
213 if 'doc' in conflicts:
214 del conflicts['doc']
215 for feature in features:
216 if feature in conflicts:
217 for conflict in conflicts[feature].split():
218 if conflict in features:
219 tune_errors.append("Feature '%s' conflicts with '%s'." %
220 (feature, conflict))
221 if feature in valid_tunes:
222 bb.debug(2, " %s: %s" % (feature, valid_tunes[feature]))
223 else:
224 tune_errors.append("Feature '%s' is not defined." % feature)
225 if tune_errors:
226 return "Tuning '%s' has the following errors:\n" % tune + '\n'.join(tune_errors)
227
228def check_toolchain(data):
229 tune_error_set = []
230 deftune = data.getVar("DEFAULTTUNE")
231 tune_errors = check_toolchain_tune(data, deftune, 'default')
232 if tune_errors:
233 tune_error_set.append(tune_errors)
234
235 multilibs = (data.getVar("MULTILIB_VARIANTS") or "").split()
236 global_multilibs = (data.getVar("MULTILIB_GLOBAL_VARIANTS") or "").split()
237
238 if multilibs:
239 seen_libs = []
240 seen_tunes = []
241 for lib in multilibs:
242 if lib in seen_libs:
243 tune_error_set.append("The multilib '%s' appears more than once." % lib)
244 else:
245 seen_libs.append(lib)
246 if not lib in global_multilibs:
247 tune_error_set.append("Multilib %s is not present in MULTILIB_GLOBAL_VARIANTS" % lib)
248 tune = data.getVar("DEFAULTTUNE:virtclass-multilib-%s" % lib)
249 if tune in seen_tunes:
250 tune_error_set.append("The tuning '%s' appears in more than one multilib." % tune)
251 else:
252 seen_libs.append(tune)
253 if tune == deftune:
254 tune_error_set.append("Multilib '%s' (%s) is also the default tuning." % (lib, deftune))
255 else:
256 tune_errors = check_toolchain_tune(data, tune, lib)
257 if tune_errors:
258 tune_error_set.append(tune_errors)
259 if tune_error_set:
260 return "Toolchain tunings invalid:\n" + '\n'.join(tune_error_set) + "\n"
261
262 return ""
263
264def check_conf_exists(fn, data):
265 bbpath = []
266 fn = data.expand(fn)
267 vbbpath = data.getVar("BBPATH", False)
268 if vbbpath:
269 bbpath += vbbpath.split(":")
270 for p in bbpath:
271 currname = os.path.join(data.expand(p), fn)
272 if os.access(currname, os.R_OK):
273 return True
274 return False
275
276def check_create_long_filename(filepath, pathname):
277 import string, random
278 testfile = os.path.join(filepath, ''.join(random.choice(string.ascii_letters) for x in range(200)))
279 try:
280 if not os.path.exists(filepath):
281 bb.utils.mkdirhier(filepath)
282 f = open(testfile, "w")
283 f.close()
284 os.remove(testfile)
285 except IOError as e:
286 import errno
287 err, strerror = e.args
288 if err == errno.ENAMETOOLONG:
289 return "Failed to create a file with a long name in %s. Please use a filesystem that does not unreasonably limit filename length.\n" % pathname
290 else:
291 return "Failed to create a file in %s: %s.\n" % (pathname, strerror)
292 except OSError as e:
293 errno, strerror = e.args
294 return "Failed to create %s directory in which to run long name sanity check: %s.\n" % (pathname, strerror)
295 return ""
296
297def check_path_length(filepath, pathname, limit):
298 if len(filepath) > limit:
299 return "The length of %s is longer than %s, this would cause unexpected errors, please use a shorter path.\n" % (pathname, limit)
300 return ""
301
302def get_filesystem_id(path):
303 import subprocess
304 try:
305 return subprocess.check_output(["stat", "-f", "-c", "%t", path]).decode('utf-8').strip()
306 except subprocess.CalledProcessError:
307 bb.warn("Can't get filesystem id of: %s" % path)
308 return None
309
310# Check that the path isn't located on nfs.
311def check_not_nfs(path, name):
312 # The nfs' filesystem id is 6969
313 if get_filesystem_id(path) == "6969":
314 return "The %s: %s can't be located on nfs.\n" % (name, path)
315 return ""
316
317# Check that the path is on a case-sensitive file system
318def check_case_sensitive(path, name):
319 import tempfile
320 with tempfile.NamedTemporaryFile(prefix='TmP', dir=path) as tmp_file:
321 if os.path.exists(tmp_file.name.lower()):
322 return "The %s (%s) can't be on a case-insensitive file system.\n" % (name, path)
323 return ""
324
325# Check that path isn't a broken symlink
326def check_symlink(lnk, data):
327 if os.path.islink(lnk) and not os.path.exists(lnk):
328 raise_sanity_error("%s is a broken symlink." % lnk, data)
329
330def check_connectivity(d):
331 # URI's to check can be set in the CONNECTIVITY_CHECK_URIS variable
332 # using the same syntax as for SRC_URI. If the variable is not set
333 # the check is skipped
334 test_uris = (d.getVar('CONNECTIVITY_CHECK_URIS') or "").split()
335 retval = ""
336
337 bbn = d.getVar('BB_NO_NETWORK')
338 if bbn not in (None, '0', '1'):
339 return 'BB_NO_NETWORK should be "0" or "1", but it is "%s"' % bbn
340
341 # Only check connectivity if network enabled and the
342 # CONNECTIVITY_CHECK_URIS are set
343 network_enabled = not (bbn == '1')
344 check_enabled = len(test_uris)
345 if check_enabled and network_enabled:
346 # Take a copy of the data store and unset MIRRORS and PREMIRRORS
347 data = bb.data.createCopy(d)
348 data.delVar('PREMIRRORS')
349 data.delVar('MIRRORS')
350 try:
351 fetcher = bb.fetch2.Fetch(test_uris, data)
352 fetcher.checkstatus()
353 except Exception as err:
354 # Allow the message to be configured so that users can be
355 # pointed to a support mechanism.
356 msg = data.getVar('CONNECTIVITY_CHECK_MSG') or ""
357 if len(msg) == 0:
358 msg = "%s.\n" % err
359 msg += " Please ensure your host's network is configured correctly.\n"
360 msg += " If your ISP or network is blocking the above URL,\n"
361 msg += " try with another domain name, for example by setting:\n"
362 msg += " CONNECTIVITY_CHECK_URIS = \"https://www.example.com/\""
363 msg += " You could also set BB_NO_NETWORK = \"1\" to disable network\n"
364 msg += " access if all required sources are on local disk.\n"
365 retval = msg
366
367 return retval
368
369def check_supported_distro(sanity_data):
370 from fnmatch import fnmatch
371
372 tested_distros = sanity_data.getVar('SANITY_TESTED_DISTROS')
373 if not tested_distros:
374 return
375
376 try:
377 distro = oe.lsb.distro_identifier()
378 except Exception:
379 distro = None
380
381 if not distro:
382 bb.warn('Host distribution could not be determined; you may possibly experience unexpected failures. It is recommended that you use a tested distribution.')
383
384 for supported in [x.strip() for x in tested_distros.split('\\n')]:
385 if fnmatch(distro, supported):
386 return
387
388 bb.warn('Host distribution "%s" has not been validated with this version of the build system; you may possibly experience unexpected failures. It is recommended that you use a tested distribution.' % distro)
389
390# Checks we should only make if MACHINE is set correctly
391def check_sanity_validmachine(sanity_data):
392 messages = ""
393
394 # Check TUNE_ARCH is set
395 if sanity_data.getVar('TUNE_ARCH') == 'INVALID':
396 messages = messages + 'TUNE_ARCH is unset. Please ensure your MACHINE configuration includes a valid tune configuration file which will set this correctly.\n'
397
398 # Check TARGET_OS is set
399 if sanity_data.getVar('TARGET_OS') == 'INVALID':
400 messages = messages + 'Please set TARGET_OS directly, or choose a MACHINE or DISTRO that does so.\n'
401
402 # Check that we don't have duplicate entries in PACKAGE_ARCHS & that TUNE_PKGARCH is in PACKAGE_ARCHS
403 pkgarchs = sanity_data.getVar('PACKAGE_ARCHS')
404 tunepkg = sanity_data.getVar('TUNE_PKGARCH')
405 defaulttune = sanity_data.getVar('DEFAULTTUNE')
406 tunefound = False
407 seen = {}
408 dups = []
409
410 for pa in pkgarchs.split():
411 if seen.get(pa, 0) == 1:
412 dups.append(pa)
413 else:
414 seen[pa] = 1
415 if pa == tunepkg:
416 tunefound = True
417
418 if len(dups):
419 messages = messages + "Error, the PACKAGE_ARCHS variable contains duplicates. The following archs are listed more than once: %s" % " ".join(dups)
420
421 if tunefound == False:
422 messages = messages + "Error, the PACKAGE_ARCHS variable (%s) for DEFAULTTUNE (%s) does not contain TUNE_PKGARCH (%s)." % (pkgarchs, defaulttune, tunepkg)
423
424 return messages
425
426# Patch before 2.7 can't handle all the features in git-style diffs. Some
427# patches may incorrectly apply, and others won't apply at all.
428def check_patch_version(sanity_data):
429 import re, subprocess
430
431 try:
432 result = subprocess.check_output(["patch", "--version"], stderr=subprocess.STDOUT).decode('utf-8')
433 version = re.search(r"[0-9.]+", result.splitlines()[0]).group()
434 if bb.utils.vercmp_string_op(version, "2.7", "<"):
435 return "Your version of patch is older than 2.7 and has bugs which will break builds. Please install a newer version of patch.\n"
436 else:
437 return None
438 except subprocess.CalledProcessError as e:
439 return "Unable to execute patch --version, exit code %d:\n%s\n" % (e.returncode, e.output)
440
441# Glibc needs make 4.0 or later, we may as well match at this point
442def check_make_version(sanity_data):
443 import subprocess
444
445 try:
446 result = subprocess.check_output(['make', '--version'], stderr=subprocess.STDOUT).decode('utf-8')
447 except subprocess.CalledProcessError as e:
448 return "Unable to execute make --version, exit code %d\n%s\n" % (e.returncode, e.output)
449 version = result.split()[2]
450 if bb.utils.vercmp_string_op(version, "4.0", "<"):
451 return "Please install a make version of 4.0 or later.\n"
452
453 if bb.utils.vercmp_string_op(version, "4.2.1", "=="):
454 distro = oe.lsb.distro_identifier()
455 if "ubuntu" in distro or "debian" in distro or "linuxmint" in distro:
456 return None
457 return "make version 4.2.1 is known to have issues on Centos/OpenSUSE and other non-Ubuntu systems. Please use a buildtools-make-tarball or a newer version of make.\n"
458 return None
459
460
461# Check if we're running on WSL (Windows Subsystem for Linux).
462# WSLv1 is known not to work but WSLv2 should work properly as
463# long as the VHDX file is optimized often, let the user know
464# upfront.
465# More information on installing WSLv2 at:
466# https://docs.microsoft.com/en-us/windows/wsl/wsl2-install
467def check_wsl(d):
468 with open("/proc/version", "r") as f:
469 verdata = f.readlines()
470 for l in verdata:
471 if "Microsoft" in l:
472 return "OpenEmbedded doesn't work under WSLv1, please upgrade to WSLv2 if you want to run builds on Windows"
473 elif "microsoft" in l:
474 bb.warn("You are running bitbake under WSLv2, this works properly but you should optimize your VHDX file eventually to avoid running out of storage space")
475 return None
476
477# Require at least gcc version 7.5.
478#
479# This can be fixed on CentOS-7 with devtoolset-6+
480# https://www.softwarecollections.org/en/scls/rhscl/devtoolset-6/
481#
482# A less invasive fix is with scripts/install-buildtools (or with user
483# built buildtools-extended-tarball)
484#
485def check_gcc_version(sanity_data):
486 import subprocess
487
488 build_cc, version = oe.utils.get_host_compiler_version(sanity_data)
489 if build_cc.strip() == "gcc":
490 if bb.utils.vercmp_string_op(version, "7.5", "<"):
491 return "Your version of gcc is older than 7.5 and will break builds. Please install a newer version of gcc (you could use the project's buildtools-extended-tarball or use scripts/install-buildtools).\n"
492 return None
493
494# Tar version 1.24 and onwards handle overwriting symlinks correctly
495# but earlier versions do not; this needs to work properly for sstate
496# Version 1.28 is needed so opkg-build works correctly when reproducibile builds are enabled
497def check_tar_version(sanity_data):
498 import subprocess
499 try:
500 result = subprocess.check_output(["tar", "--version"], stderr=subprocess.STDOUT).decode('utf-8')
501 except subprocess.CalledProcessError as e:
502 return "Unable to execute tar --version, exit code %d\n%s\n" % (e.returncode, e.output)
503 version = result.split()[3]
504 if bb.utils.vercmp_string_op(version, "1.28", "<"):
505 return "Your version of tar is older than 1.28 and does not have the support needed to enable reproducible builds. Please install a newer version of tar (you could use the project's buildtools-tarball from our last release or use scripts/install-buildtools).\n"
506 return None
507
508# We use git parameters and functionality only found in 1.7.8 or later
509# The kernel tools assume git >= 1.8.3.1 (verified needed > 1.7.9.5) see #6162
510# The git fetcher also had workarounds for git < 1.7.9.2 which we've dropped
511def check_git_version(sanity_data):
512 import subprocess
513 try:
514 result = subprocess.check_output(["git", "--version"], stderr=subprocess.DEVNULL).decode('utf-8')
515 except subprocess.CalledProcessError as e:
516 return "Unable to execute git --version, exit code %d\n%s\n" % (e.returncode, e.output)
517 version = result.split()[2]
518 if bb.utils.vercmp_string_op(version, "1.8.3.1", "<"):
519 return "Your version of git is older than 1.8.3.1 and has bugs which will break builds. Please install a newer version of git.\n"
520 return None
521
522# Check the required perl modules which may not be installed by default
523def check_perl_modules(sanity_data):
524 import subprocess
525 ret = ""
526 modules = ( "Text::ParseWords", "Thread::Queue", "Data::Dumper" )
527 errresult = ''
528 for m in modules:
529 try:
530 subprocess.check_output(["perl", "-e", "use %s" % m])
531 except subprocess.CalledProcessError as e:
532 errresult += bytes.decode(e.output)
533 ret += "%s " % m
534 if ret:
535 return "Required perl module(s) not found: %s\n\n%s\n" % (ret, errresult)
536 return None
537
538def sanity_check_conffiles(d):
539 funcs = d.getVar('BBLAYERS_CONF_UPDATE_FUNCS').split()
540 for func in funcs:
541 conffile, current_version, required_version, func = func.split(":")
542 if check_conf_exists(conffile, d) and d.getVar(current_version) is not None and \
543 d.getVar(current_version) != d.getVar(required_version):
544 try:
545 bb.build.exec_func(func, d)
546 except NotImplementedError as e:
547 bb.fatal(str(e))
548 d.setVar("BB_INVALIDCONF", True)
549
550def drop_v14_cross_builds(d):
551 import glob
552 indexes = glob.glob(d.expand("${SSTATE_MANIFESTS}/index-${BUILD_ARCH}_*"))
553 for i in indexes:
554 with open(i, "r") as f:
555 lines = f.readlines()
556 for l in reversed(lines):
557 try:
558 (stamp, manifest, workdir) = l.split()
559 except ValueError:
560 bb.fatal("Invalid line '%s' in sstate manifest '%s'" % (l, i))
561 for m in glob.glob(manifest + ".*"):
562 if m.endswith(".postrm"):
563 continue
564 sstate_clean_manifest(m, d)
565 bb.utils.remove(stamp + "*")
566 bb.utils.remove(workdir, recurse = True)
567
568def sanity_handle_abichanges(status, d):
569 #
570 # Check the 'ABI' of TMPDIR
571 #
572 import subprocess
573
574 current_abi = d.getVar('OELAYOUT_ABI')
575 abifile = d.getVar('SANITY_ABIFILE')
576 if os.path.exists(abifile):
577 with open(abifile, "r") as f:
578 abi = f.read().strip()
579 if not abi.isdigit():
580 with open(abifile, "w") as f:
581 f.write(current_abi)
582 elif int(abi) <= 11 and current_abi == "12":
583 status.addresult("The layout of TMPDIR changed for Recipe Specific Sysroots.\nConversion doesn't make sense and this change will rebuild everything so please delete TMPDIR (%s).\n" % d.getVar("TMPDIR"))
584 elif int(abi) <= 13 and current_abi == "14":
585 status.addresult("TMPDIR changed to include path filtering from the pseudo database.\nIt is recommended to use a clean TMPDIR with the new pseudo path filtering so TMPDIR (%s) would need to be removed to continue.\n" % d.getVar("TMPDIR"))
586 elif int(abi) == 14 and current_abi == "15":
587 drop_v14_cross_builds(d)
588 with open(abifile, "w") as f:
589 f.write(current_abi)
590 elif (abi != current_abi):
591 # Code to convert from one ABI to another could go here if possible.
592 status.addresult("Error, TMPDIR has changed its layout version number (%s to %s) and you need to either rebuild, revert or adjust it at your own risk.\n" % (abi, current_abi))
593 else:
594 with open(abifile, "w") as f:
595 f.write(current_abi)
596
597def check_sanity_sstate_dir_change(sstate_dir, data):
598 # Sanity checks to be done when the value of SSTATE_DIR changes
599
600 # Check that SSTATE_DIR isn't on a filesystem with limited filename length (eg. eCryptFS)
601 testmsg = ""
602 if sstate_dir != "":
603 testmsg = check_create_long_filename(sstate_dir, "SSTATE_DIR")
604 # If we don't have permissions to SSTATE_DIR, suggest the user set it as an SSTATE_MIRRORS
605 try:
606 err = testmsg.split(': ')[1].strip()
607 if err == "Permission denied.":
608 testmsg = testmsg + "You could try using %s in SSTATE_MIRRORS rather than as an SSTATE_CACHE.\n" % (sstate_dir)
609 except IndexError:
610 pass
611 return testmsg
612
613def check_sanity_version_change(status, d):
614 # Sanity checks to be done when SANITY_VERSION or NATIVELSBSTRING changes
615 # In other words, these tests run once in a given build directory and then
616 # never again until the sanity version or host distrubution id/version changes.
617
618 # Check the python install is complete. Examples that are often removed in
619 # minimal installations: glib-2.0-natives requries # xml.parsers.expat and icu
620 # requires distutils.sysconfig.
621 try:
622 import xml.parsers.expat
623 import distutils.sysconfig
624 except ImportError as e:
625 status.addresult('Your Python 3 is not a full install. Please install the module %s (see the Getting Started guide for further information).\n' % e.name)
626
627 status.addresult(check_gcc_version(d))
628 status.addresult(check_make_version(d))
629 status.addresult(check_patch_version(d))
630 status.addresult(check_tar_version(d))
631 status.addresult(check_git_version(d))
632 status.addresult(check_perl_modules(d))
633 status.addresult(check_wsl(d))
634
635 missing = ""
636
637 if not check_app_exists("${MAKE}", d):
638 missing = missing + "GNU make,"
639
640 if not check_app_exists('${BUILD_CC}', d):
641 missing = missing + "C Compiler (%s)," % d.getVar("BUILD_CC")
642
643 if not check_app_exists('${BUILD_CXX}', d):
644 missing = missing + "C++ Compiler (%s)," % d.getVar("BUILD_CXX")
645
646 required_utilities = d.getVar('SANITY_REQUIRED_UTILITIES')
647
648 for util in required_utilities.split():
649 if not check_app_exists(util, d):
650 missing = missing + "%s," % util
651
652 if missing:
653 missing = missing.rstrip(',')
654 status.addresult("Please install the following missing utilities: %s\n" % missing)
655
656 assume_provided = d.getVar('ASSUME_PROVIDED').split()
657 # Check user doesn't have ASSUME_PROVIDED = instead of += in local.conf
658 if "diffstat-native" not in assume_provided:
659 status.addresult('Please use ASSUME_PROVIDED +=, not ASSUME_PROVIDED = in your local.conf\n')
660
661 # Check that TMPDIR isn't on a filesystem with limited filename length (eg. eCryptFS)
662 import stat
663 tmpdir = d.getVar('TMPDIR')
664 status.addresult(check_create_long_filename(tmpdir, "TMPDIR"))
665 tmpdirmode = os.stat(tmpdir).st_mode
666 if (tmpdirmode & stat.S_ISGID):
667 status.addresult("TMPDIR is setgid, please don't build in a setgid directory")
668 if (tmpdirmode & stat.S_ISUID):
669 status.addresult("TMPDIR is setuid, please don't build in a setuid directory")
670
671 # Check that a user isn't building in a path in PSEUDO_IGNORE_PATHS
672 pseudoignorepaths = d.getVar('PSEUDO_IGNORE_PATHS', expand=True).split(",")
673 workdir = d.getVar('WORKDIR', expand=True)
674 for i in pseudoignorepaths:
675 if i and workdir.startswith(i):
676 status.addresult("You are building in a path included in PSEUDO_IGNORE_PATHS " + str(i) + " please locate the build outside this path.\n")
677
678 # Check if PSEUDO_IGNORE_PATHS and and paths under pseudo control overlap
679 pseudoignorepaths = d.getVar('PSEUDO_IGNORE_PATHS', expand=True).split(",")
680 pseudo_control_dir = "${D},${PKGD},${PKGDEST},${IMAGEROOTFS},${SDK_OUTPUT}"
681 pseudocontroldir = d.expand(pseudo_control_dir).split(",")
682 for i in pseudoignorepaths:
683 for j in pseudocontroldir:
684 if i and j:
685 if j.startswith(i):
686 status.addresult("A path included in PSEUDO_IGNORE_PATHS " + str(i) + " and the path " + str(j) + " overlap and this will break pseudo permission and ownership tracking. Please set the path " + str(j) + " to a different directory which does not overlap with pseudo controlled directories. \n")
687
688 # Some third-party software apparently relies on chmod etc. being suid root (!!)
689 import stat
690 suid_check_bins = "chown chmod mknod".split()
691 for bin_cmd in suid_check_bins:
692 bin_path = bb.utils.which(os.environ["PATH"], bin_cmd)
693 if bin_path:
694 bin_stat = os.stat(bin_path)
695 if bin_stat.st_uid == 0 and bin_stat.st_mode & stat.S_ISUID:
696 status.addresult('%s has the setuid bit set. This interferes with pseudo and may cause other issues that break the build process.\n' % bin_path)
697
698 # Check that we can fetch from various network transports
699 netcheck = check_connectivity(d)
700 status.addresult(netcheck)
701 if netcheck:
702 status.network_error = True
703
704 nolibs = d.getVar('NO32LIBS')
705 if not nolibs:
706 lib32path = '/lib'
707 if os.path.exists('/lib64') and ( os.path.islink('/lib64') or os.path.islink('/lib') ):
708 lib32path = '/lib32'
709
710 if os.path.exists('%s/libc.so.6' % lib32path) and not os.path.exists('/usr/include/gnu/stubs-32.h'):
711 status.addresult("You have a 32-bit libc, but no 32-bit headers. You must install the 32-bit libc headers.\n")
712
713 bbpaths = d.getVar('BBPATH').split(":")
714 if ("." in bbpaths or "./" in bbpaths or "" in bbpaths):
715 status.addresult("BBPATH references the current directory, either through " \
716 "an empty entry, a './' or a '.'.\n\t This is unsafe and means your "\
717 "layer configuration is adding empty elements to BBPATH.\n\t "\
718 "Please check your layer.conf files and other BBPATH " \
719 "settings to remove the current working directory " \
720 "references.\n" \
721 "Parsed BBPATH is" + str(bbpaths));
722
723 oes_bb_conf = d.getVar( 'OES_BITBAKE_CONF')
724 if not oes_bb_conf:
725 status.addresult('You are not using the OpenEmbedded version of conf/bitbake.conf. This means your environment is misconfigured, in particular check BBPATH.\n')
726
727 # The length of TMPDIR can't be longer than 410
728 status.addresult(check_path_length(tmpdir, "TMPDIR", 410))
729
730 # Check that TMPDIR isn't located on nfs
731 status.addresult(check_not_nfs(tmpdir, "TMPDIR"))
732
733 # Check for case-insensitive file systems (such as Linux in Docker on
734 # macOS with default HFS+ file system)
735 status.addresult(check_case_sensitive(tmpdir, "TMPDIR"))
736
737def sanity_check_locale(d):
738 """
739 Currently bitbake switches locale to en_US.UTF-8 so check that this locale actually exists.
740 """
741 import locale
742 try:
743 locale.setlocale(locale.LC_ALL, "en_US.UTF-8")
744 except locale.Error:
745 raise_sanity_error("Your system needs to support the en_US.UTF-8 locale.", d)
746
747def check_sanity_everybuild(status, d):
748 import os, stat
749 # Sanity tests which test the users environment so need to run at each build (or are so cheap
750 # it makes sense to always run them.
751
752 if 0 == os.getuid():
753 raise_sanity_error("Do not use Bitbake as root.", d)
754
755 # Check the Python version, we now have a minimum of Python 3.6
756 import sys
757 if sys.hexversion < 0x030600F0:
758 status.addresult('The system requires at least Python 3.6 to run. Please update your Python interpreter.\n')
759
760 # Check the bitbake version meets minimum requirements
761 minversion = d.getVar('BB_MIN_VERSION')
762 if bb.utils.vercmp_string_op(bb.__version__, minversion, "<"):
763 status.addresult('Bitbake version %s is required and version %s was found\n' % (minversion, bb.__version__))
764
765 sanity_check_locale(d)
766
767 paths = d.getVar('PATH').split(":")
768 if "." in paths or "./" in paths or "" in paths:
769 status.addresult("PATH contains '.', './' or '' (empty element), which will break the build, please remove this.\nParsed PATH is " + str(paths) + "\n")
770
771 #Check if bitbake is present in PATH environment variable
772 bb_check = bb.utils.which(d.getVar('PATH'), 'bitbake')
773 if not bb_check:
774 bb.warn("bitbake binary is not found in PATH, did you source the script?")
775
776 # Check whether 'inherit' directive is found (used for a class to inherit)
777 # in conf file it's supposed to be uppercase INHERIT
778 inherit = d.getVar('inherit')
779 if inherit:
780 status.addresult("Please don't use inherit directive in your local.conf. The directive is supposed to be used in classes and recipes only to inherit of bbclasses. Here INHERIT should be used.\n")
781
782 # Check that the DISTRO is valid, if set
783 # need to take into account DISTRO renaming DISTRO
784 distro = d.getVar('DISTRO')
785 if distro and distro != "nodistro":
786 if not ( check_conf_exists("conf/distro/${DISTRO}.conf", d) or check_conf_exists("conf/distro/include/${DISTRO}.inc", d) ):
787 status.addresult("DISTRO '%s' not found. Please set a valid DISTRO in your local.conf\n" % d.getVar("DISTRO"))
788
789 # Check that these variables don't use tilde-expansion as we don't do that
790 for v in ("TMPDIR", "DL_DIR", "SSTATE_DIR"):
791 if d.getVar(v).startswith("~"):
792 status.addresult("%s uses ~ but Bitbake will not expand this, use an absolute path or variables." % v)
793
794 # Check that DL_DIR is set, exists and is writable. In theory, we should never even hit the check if DL_DIR isn't
795 # set, since so much relies on it being set.
796 dldir = d.getVar('DL_DIR')
797 if not dldir:
798 status.addresult("DL_DIR is not set. Your environment is misconfigured, check that DL_DIR is set, and if the directory exists, that it is writable. \n")
799 if os.path.exists(dldir) and not os.access(dldir, os.W_OK):
800 status.addresult("DL_DIR: %s exists but you do not appear to have write access to it. \n" % dldir)
801 check_symlink(dldir, d)
802
803 # Check that the MACHINE is valid, if it is set
804 machinevalid = True
805 if d.getVar('MACHINE'):
806 if not check_conf_exists("conf/machine/${MACHINE}.conf", d):
807 status.addresult('MACHINE=%s is invalid. Please set a valid MACHINE in your local.conf, environment or other configuration file.\n' % (d.getVar('MACHINE')))
808 machinevalid = False
809 else:
810 status.addresult(check_sanity_validmachine(d))
811 else:
812 status.addresult('Please set a MACHINE in your local.conf or environment\n')
813 machinevalid = False
814 if machinevalid:
815 status.addresult(check_toolchain(d))
816
817 # Check that the SDKMACHINE is valid, if it is set
818 if d.getVar('SDKMACHINE'):
819 if not check_conf_exists("conf/machine-sdk/${SDKMACHINE}.conf", d):
820 status.addresult('Specified SDKMACHINE value is not valid\n')
821 elif d.getVar('SDK_ARCH', False) == "${BUILD_ARCH}":
822 status.addresult('SDKMACHINE is set, but SDK_ARCH has not been changed as a result - SDKMACHINE may have been set too late (e.g. in the distro configuration)\n')
823
824 # If SDK_VENDOR looks like "-my-sdk" then the triples are badly formed so fail early
825 sdkvendor = d.getVar("SDK_VENDOR")
826 if not (sdkvendor.startswith("-") and sdkvendor.count("-") == 1):
827 status.addresult("SDK_VENDOR should be of the form '-foosdk' with a single dash; found '%s'\n" % sdkvendor)
828
829 check_supported_distro(d)
830
831 omask = os.umask(0o022)
832 if omask & 0o755:
833 status.addresult("Please use a umask which allows a+rx and u+rwx\n")
834 os.umask(omask)
835
836 if d.getVar('TARGET_ARCH') == "arm":
837 # This path is no longer user-readable in modern (very recent) Linux
838 try:
839 if os.path.exists("/proc/sys/vm/mmap_min_addr"):
840 f = open("/proc/sys/vm/mmap_min_addr", "r")
841 try:
842 if (int(f.read().strip()) > 65536):
843 status.addresult("/proc/sys/vm/mmap_min_addr is not <= 65536. This will cause problems with qemu so please fix the value (as root).\n\nTo fix this in later reboots, set vm.mmap_min_addr = 65536 in /etc/sysctl.conf.\n")
844 finally:
845 f.close()
846 except:
847 pass
848
849 for checkdir in ['COREBASE', 'TMPDIR']:
850 val = d.getVar(checkdir)
851 if val.find('..') != -1:
852 status.addresult("Error, you have '..' in your %s directory path. Please ensure the variable contains an absolute path as this can break some recipe builds in obtuse ways." % checkdir)
853 if val.find('+') != -1:
854 status.addresult("Error, you have an invalid character (+) in your %s directory path. Please move the installation to a directory which doesn't include any + characters." % checkdir)
855 if val.find('@') != -1:
856 status.addresult("Error, you have an invalid character (@) in your %s directory path. Please move the installation to a directory which doesn't include any @ characters." % checkdir)
857 if val.find(' ') != -1:
858 status.addresult("Error, you have a space in your %s directory path. Please move the installation to a directory which doesn't include a space since autotools doesn't support this." % checkdir)
859 if val.find('%') != -1:
860 status.addresult("Error, you have an invalid character (%) in your %s directory path which causes problems with python string formatting. Please move the installation to a directory which doesn't include any % characters." % checkdir)
861
862 # Check the format of MIRRORS, PREMIRRORS and SSTATE_MIRRORS
863 import re
864 mirror_vars = ['MIRRORS', 'PREMIRRORS', 'SSTATE_MIRRORS']
865 protocols = ['http', 'ftp', 'file', 'https', \
866 'git', 'gitsm', 'hg', 'osc', 'p4', 'svn', \
867 'bzr', 'cvs', 'npm', 'sftp', 'ssh', 's3', 'az', 'ftps']
868 for mirror_var in mirror_vars:
869 mirrors = (d.getVar(mirror_var) or '').replace('\\n', ' ').split()
870
871 # Split into pairs
872 if len(mirrors) % 2 != 0:
873 bb.warn('Invalid mirror variable value for %s: %s, should contain paired members.' % (mirror_var, str(mirrors)))
874 continue
875 mirrors = list(zip(*[iter(mirrors)]*2))
876
877 for mirror_entry in mirrors:
878 pattern, mirror = mirror_entry
879
880 decoded = bb.fetch2.decodeurl(pattern)
881 try:
882 pattern_scheme = re.compile(decoded[0])
883 except re.error as exc:
884 bb.warn('Invalid scheme regex (%s) in %s; %s' % (pattern, mirror_var, mirror_entry))
885 continue
886
887 if not any(pattern_scheme.match(protocol) for protocol in protocols):
888 bb.warn('Invalid protocol (%s) in %s: %s' % (decoded[0], mirror_var, mirror_entry))
889 continue
890
891 if not any(mirror.startswith(protocol + '://') for protocol in protocols):
892 bb.warn('Invalid protocol in %s: %s' % (mirror_var, mirror_entry))
893 continue
894
895 if mirror.startswith('file://'):
896 import urllib
897 check_symlink(urllib.parse.urlparse(mirror).path, d)
898 # SSTATE_MIRROR ends with a /PATH string
899 if mirror.endswith('/PATH'):
900 # remove /PATH$ from SSTATE_MIRROR to get a working
901 # base directory path
902 mirror_base = urllib.parse.urlparse(mirror[:-1*len('/PATH')]).path
903 check_symlink(mirror_base, d)
904
905 # Check sstate mirrors aren't being used with a local hash server and no remote
906 hashserv = d.getVar("BB_HASHSERVE")
907 if d.getVar("SSTATE_MIRRORS") and hashserv and hashserv.startswith("unix://") and not d.getVar("BB_HASHSERVE_UPSTREAM"):
908 bb.warn("You are using a local hash equivalence server but have configured an sstate mirror. This will likely mean no sstate will match from the mirror. You may wish to disable the hash equivalence use (BB_HASHSERVE), or use a hash equivalence server alongside the sstate mirror.")
909
910 # Check that TMPDIR hasn't changed location since the last time we were run
911 tmpdir = d.getVar('TMPDIR')
912 checkfile = os.path.join(tmpdir, "saved_tmpdir")
913 if os.path.exists(checkfile):
914 with open(checkfile, "r") as f:
915 saved_tmpdir = f.read().strip()
916 if (saved_tmpdir != tmpdir):
917 status.addresult("Error, TMPDIR has changed location. You need to either move it back to %s or delete it and rebuild\n" % saved_tmpdir)
918 else:
919 bb.utils.mkdirhier(tmpdir)
920 # Remove setuid, setgid and sticky bits from TMPDIR
921 try:
922 os.chmod(tmpdir, os.stat(tmpdir).st_mode & ~ stat.S_ISUID)
923 os.chmod(tmpdir, os.stat(tmpdir).st_mode & ~ stat.S_ISGID)
924 os.chmod(tmpdir, os.stat(tmpdir).st_mode & ~ stat.S_ISVTX)
925 except OSError as exc:
926 bb.warn("Unable to chmod TMPDIR: %s" % exc)
927 with open(checkfile, "w") as f:
928 f.write(tmpdir)
929
930 # If /bin/sh is a symlink, check that it points to dash or bash
931 if os.path.islink('/bin/sh'):
932 real_sh = os.path.realpath('/bin/sh')
933 # Due to update-alternatives, the shell name may take various
934 # forms, such as /bin/dash, bin/bash, /bin/bash.bash ...
935 if '/dash' not in real_sh and '/bash' not in real_sh:
936 status.addresult("Error, /bin/sh links to %s, must be dash or bash\n" % real_sh)
937
938def check_sanity(sanity_data):
939 class SanityStatus(object):
940 def __init__(self):
941 self.messages = ""
942 self.network_error = False
943
944 def addresult(self, message):
945 if message:
946 self.messages = self.messages + message
947
948 status = SanityStatus()
949
950 tmpdir = sanity_data.getVar('TMPDIR')
951 sstate_dir = sanity_data.getVar('SSTATE_DIR')
952
953 check_symlink(sstate_dir, sanity_data)
954
955 # Check saved sanity info
956 last_sanity_version = 0
957 last_tmpdir = ""
958 last_sstate_dir = ""
959 last_nativelsbstr = ""
960 sanityverfile = sanity_data.expand("${TOPDIR}/cache/sanity_info")
961 if os.path.exists(sanityverfile):
962 with open(sanityverfile, 'r') as f:
963 for line in f:
964 if line.startswith('SANITY_VERSION'):
965 last_sanity_version = int(line.split()[1])
966 if line.startswith('TMPDIR'):
967 last_tmpdir = line.split()[1]
968 if line.startswith('SSTATE_DIR'):
969 last_sstate_dir = line.split()[1]
970 if line.startswith('NATIVELSBSTRING'):
971 last_nativelsbstr = line.split()[1]
972
973 check_sanity_everybuild(status, sanity_data)
974
975 sanity_version = int(sanity_data.getVar('SANITY_VERSION') or 1)
976 network_error = False
977 # NATIVELSBSTRING var may have been overridden with "universal", so
978 # get actual host distribution id and version
979 nativelsbstr = lsb_distro_identifier(sanity_data)
980 if last_sanity_version < sanity_version or last_nativelsbstr != nativelsbstr:
981 check_sanity_version_change(status, sanity_data)
982 status.addresult(check_sanity_sstate_dir_change(sstate_dir, sanity_data))
983 else:
984 if last_sstate_dir != sstate_dir:
985 status.addresult(check_sanity_sstate_dir_change(sstate_dir, sanity_data))
986
987 if os.path.exists(os.path.dirname(sanityverfile)) and not status.messages:
988 with open(sanityverfile, 'w') as f:
989 f.write("SANITY_VERSION %s\n" % sanity_version)
990 f.write("TMPDIR %s\n" % tmpdir)
991 f.write("SSTATE_DIR %s\n" % sstate_dir)
992 f.write("NATIVELSBSTRING %s\n" % nativelsbstr)
993
994 sanity_handle_abichanges(status, sanity_data)
995
996 if status.messages != "":
997 raise_sanity_error(sanity_data.expand(status.messages), sanity_data, status.network_error)
998
999# Create a copy of the datastore and finalise it to ensure appends and
1000# overrides are set - the datastore has yet to be finalised at ConfigParsed
1001def copy_data(e):
1002 sanity_data = bb.data.createCopy(e.data)
1003 sanity_data.finalize()
1004 return sanity_data
1005
1006addhandler config_reparse_eventhandler
1007config_reparse_eventhandler[eventmask] = "bb.event.ConfigParsed"
1008python config_reparse_eventhandler() {
1009 sanity_check_conffiles(e.data)
1010}
1011
1012addhandler check_sanity_eventhandler
1013check_sanity_eventhandler[eventmask] = "bb.event.SanityCheck bb.event.NetworkTest"
1014python check_sanity_eventhandler() {
1015 if bb.event.getName(e) == "SanityCheck":
1016 sanity_data = copy_data(e)
1017 check_sanity(sanity_data)
1018 if e.generateevents:
1019 sanity_data.setVar("SANITY_USE_EVENTS", "1")
1020 bb.event.fire(bb.event.SanityCheckPassed(), e.data)
1021 elif bb.event.getName(e) == "NetworkTest":
1022 sanity_data = copy_data(e)
1023 if e.generateevents:
1024 sanity_data.setVar("SANITY_USE_EVENTS", "1")
1025 bb.event.fire(bb.event.NetworkTestFailed() if check_connectivity(sanity_data) else bb.event.NetworkTestPassed(), e.data)
1026
1027 return
1028}