diff options
| -rw-r--r-- | bitbake/lib/bb/tests/utils.py | 50 | ||||
| -rw-r--r-- | bitbake/lib/bb/utils.py | 21 |
2 files changed, 71 insertions, 0 deletions
diff --git a/bitbake/lib/bb/tests/utils.py b/bitbake/lib/bb/tests/utils.py index 7c50b1d786..677b3872ba 100644 --- a/bitbake/lib/bb/tests/utils.py +++ b/bitbake/lib/bb/tests/utils.py | |||
| @@ -21,6 +21,7 @@ | |||
| 21 | 21 | ||
| 22 | import unittest | 22 | import unittest |
| 23 | import bb | 23 | import bb |
| 24 | import os | ||
| 24 | 25 | ||
| 25 | class VerCmpString(unittest.TestCase): | 26 | class VerCmpString(unittest.TestCase): |
| 26 | 27 | ||
| @@ -51,3 +52,52 @@ class VerCmpString(unittest.TestCase): | |||
| 51 | result = bb.utils.explode_dep_versions2("foo ( =1.10 )") | 52 | result = bb.utils.explode_dep_versions2("foo ( =1.10 )") |
| 52 | self.assertEqual(result, correctresult) | 53 | self.assertEqual(result, correctresult) |
| 53 | 54 | ||
| 55 | def test_vercmp_string_op(self): | ||
| 56 | compareops = [('1', '1', '=', True), | ||
| 57 | ('1', '1', '==', True), | ||
| 58 | ('1', '1', '!=', False), | ||
| 59 | ('1', '1', '>', False), | ||
| 60 | ('1', '1', '<', False), | ||
| 61 | ('1', '1', '>=', True), | ||
| 62 | ('1', '1', '<=', True), | ||
| 63 | ('1', '0', '=', False), | ||
| 64 | ('1', '0', '==', False), | ||
| 65 | ('1', '0', '!=', True), | ||
| 66 | ('1', '0', '>', True), | ||
| 67 | ('1', '0', '<', False), | ||
| 68 | ('1', '0', '>>', True), | ||
| 69 | ('1', '0', '<<', False), | ||
| 70 | ('1', '0', '>=', True), | ||
| 71 | ('1', '0', '<=', False), | ||
| 72 | ('0', '1', '=', False), | ||
| 73 | ('0', '1', '==', False), | ||
| 74 | ('0', '1', '!=', True), | ||
| 75 | ('0', '1', '>', False), | ||
| 76 | ('0', '1', '<', True), | ||
| 77 | ('0', '1', '>>', False), | ||
| 78 | ('0', '1', '<<', True), | ||
| 79 | ('0', '1', '>=', False), | ||
| 80 | ('0', '1', '<=', True)] | ||
| 81 | |||
| 82 | for arg1, arg2, op, correctresult in compareops: | ||
| 83 | result = bb.utils.vercmp_string_op(arg1, arg2, op) | ||
| 84 | self.assertEqual(result, correctresult, 'vercmp_string_op("%s", "%s", "%s") != %s' % (arg1, arg2, op, correctresult)) | ||
| 85 | |||
| 86 | # Check that clearly invalid operator raises an exception | ||
| 87 | self.assertRaises(bb.utils.VersionStringException, bb.utils.vercmp_string_op, '0', '0', '$') | ||
| 88 | |||
| 89 | |||
| 90 | class Path(unittest.TestCase): | ||
| 91 | def test_unsafe_delete_path(self): | ||
| 92 | checkitems = [('/', True), | ||
| 93 | ('//', True), | ||
| 94 | ('///', True), | ||
| 95 | (os.getcwd().count(os.sep) * ('..' + os.sep), True), | ||
| 96 | (os.environ.get('HOME', '/home/test'), True), | ||
| 97 | ('/home/someone', True), | ||
| 98 | ('/home/other/', True), | ||
| 99 | ('/home/other/subdir', False), | ||
| 100 | ('', False)] | ||
| 101 | for arg1, correctresult in checkitems: | ||
| 102 | result = bb.utils._check_unsafe_delete_path(arg1) | ||
| 103 | self.assertEqual(result, correctresult, '_check_unsafe_delete_path("%s") != %s' % (arg1, correctresult)) | ||
diff --git a/bitbake/lib/bb/utils.py b/bitbake/lib/bb/utils.py index 670e592fe0..2562db8e47 100644 --- a/bitbake/lib/bb/utils.py +++ b/bitbake/lib/bb/utils.py | |||
| @@ -575,11 +575,30 @@ def build_environment(d): | |||
| 575 | if export: | 575 | if export: |
| 576 | os.environ[var] = d.getVar(var, True) or "" | 576 | os.environ[var] = d.getVar(var, True) or "" |
| 577 | 577 | ||
| 578 | def _check_unsafe_delete_path(path): | ||
| 579 | """ | ||
| 580 | Basic safeguard against recursively deleting something we shouldn't. If it returns True, | ||
| 581 | the caller should raise an exception with an appropriate message. | ||
| 582 | NOTE: This is NOT meant to be a security mechanism - just a guard against silly mistakes | ||
| 583 | with potentially disastrous results. | ||
| 584 | """ | ||
| 585 | extra = '' | ||
| 586 | # HOME might not be /home/something, so in case we can get it, check against it | ||
| 587 | homedir = os.environ.get('HOME', '') | ||
| 588 | if homedir: | ||
| 589 | extra = '|%s' % homedir | ||
| 590 | if re.match('(/|//|/home|/home/[^/]*%s)$' % extra, os.path.abspath(path)): | ||
| 591 | return True | ||
| 592 | return False | ||
| 593 | |||
| 578 | def remove(path, recurse=False): | 594 | def remove(path, recurse=False): |
| 579 | """Equivalent to rm -f or rm -rf""" | 595 | """Equivalent to rm -f or rm -rf""" |
| 580 | if not path: | 596 | if not path: |
| 581 | return | 597 | return |
| 582 | if recurse: | 598 | if recurse: |
| 599 | for name in glob.glob(path): | ||
| 600 | if _check_unsafe_delete_path(path): | ||
| 601 | raise Exception('bb.utils.remove: called with dangerous path "%s" and recurse=True, refusing to delete!' % path) | ||
| 583 | # shutil.rmtree(name) would be ideal but its too slow | 602 | # shutil.rmtree(name) would be ideal but its too slow |
| 584 | subprocess.call(['rm', '-rf'] + glob.glob(path)) | 603 | subprocess.call(['rm', '-rf'] + glob.glob(path)) |
| 585 | return | 604 | return |
| @@ -593,6 +612,8 @@ def remove(path, recurse=False): | |||
| 593 | def prunedir(topdir): | 612 | def prunedir(topdir): |
| 594 | # Delete everything reachable from the directory named in 'topdir'. | 613 | # Delete everything reachable from the directory named in 'topdir'. |
| 595 | # CAUTION: This is dangerous! | 614 | # CAUTION: This is dangerous! |
| 615 | if _check_unsafe_delete_path(topdir): | ||
| 616 | raise Exception('bb.utils.prunedir: called with dangerous path "%s", refusing to delete!' % topdir) | ||
| 596 | for root, dirs, files in os.walk(topdir, topdown = False): | 617 | for root, dirs, files in os.walk(topdir, topdown = False): |
| 597 | for name in files: | 618 | for name in files: |
| 598 | os.remove(os.path.join(root, name)) | 619 | os.remove(os.path.join(root, name)) |
