| # Copyright 2017 The Bazel Authors. All rights reserved. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| import os |
| import re |
| import time |
| from absl.testing import absltest |
| from src.test.py.bazel import test_base |
| |
| |
| class BazelCleanTest(test_base.TestBase): |
| |
| def testBazelClean(self): |
| self.ScratchFile('MODULE.bazel') |
| self.ScratchFile('foo/BUILD', [ |
| 'genrule(', |
| ' name = "x",', |
| ' outs = ["x.out"],', |
| ' cmd = "touch $@",', |
| ')', |
| ]) |
| |
| _, stdout, _ = self.RunBazel(['info', 'bazel-genfiles']) |
| bazel_genfiles = stdout[0] |
| |
| _, stdout, _ = self.RunBazel(['info', 'output_base']) |
| output_base = stdout[0] |
| |
| # Repeat 10 times to ensure flaky error like |
| # https://github.com/bazelbuild/bazel/issues/5907 are caught. |
| for _ in range(0, 10): |
| self.RunBazel(['build', '//foo:x']) |
| self.assertTrue(os.path.exists( |
| os.path.join(bazel_genfiles, 'foo/x.out'))) |
| |
| self.RunBazel(['clean']) |
| self.assertFalse(os.path.exists( |
| os.path.join(bazel_genfiles, 'foo/x.out'))) |
| self.assertTrue(os.path.exists(output_base)) |
| |
| self.RunBazel(['build', '//foo:x']) |
| self.assertTrue(os.path.exists(os.path.join(bazel_genfiles, 'foo/x.out'))) |
| |
| self.RunBazel(['clean', '--expunge']) |
| self.assertFalse(os.path.exists( |
| os.path.join(bazel_genfiles, 'foo/x.out'))) |
| self.assertFalse(os.path.exists(output_base)) |
| |
| @absltest.skipIf(not test_base.TestBase.IsLinux(), |
| 'Async clean only supported on Linux') |
| def testBazelAsyncClean(self): |
| self.ScratchFile('MODULE.bazel') |
| _, _, stderr = self.RunBazel(['clean', '--async']) |
| matcher = self._findMatch(' moved to (.*) for deletion', stderr) |
| self.assertTrue(matcher, stderr) |
| first_temp = matcher.group(1) |
| self.assertTrue(first_temp, stderr) |
| # Now do it again (we need to build to recreate exec root). |
| self.RunBazel(['build']) |
| _, _, stderr = self.RunBazel(['clean', '--async']) |
| matcher = self._findMatch(' moved to (.*) for deletion', stderr) |
| self.assertTrue(matcher, stderr) |
| second_temp = matcher.group(1) |
| self.assertTrue(second_temp, stderr) |
| # Two directories should be different. |
| self.assertNotEqual(second_temp, first_temp, stderr) |
| |
| @absltest.skipIf(not test_base.TestBase.IsLinux(), |
| 'Async clean only supported on Linux') |
| def testBazelAsyncCleanWithReadonlyDirectories(self): |
| self.ScratchFile('MODULE.bazel') |
| self.RunBazel(['build']) |
| _, stdout, _ = self.RunBazel(['info', 'execution_root']) |
| execroot = stdout[0] |
| readonly_dir = os.path.join(execroot, 'readonly') |
| os.mkdir(readonly_dir) |
| open(os.path.join(readonly_dir, 'somefile'), 'wb').close() |
| os.chmod(readonly_dir, 0o555) |
| _, _, stderr = self.RunBazel(['clean', '--async']) |
| matcher = self._findMatch(' moved to (.*) for deletion', stderr) |
| self.assertTrue(matcher, stderr) |
| temp = matcher.group(1) |
| for _ in range(50): |
| if not os.path.isdir(temp): |
| break |
| time.sleep(.1) |
| else: |
| self.fail('temporary directory not removed: {!r}'.format(stderr)) |
| |
| def _findMatch(self, pattern, items): |
| r = re.compile(pattern) |
| for line in items: |
| matcher = r.search(line) |
| if matcher: |
| return matcher |
| return None |
| |
| |
| if __name__ == '__main__': |
| absltest.main() |