Mike Frysinger | f1ba7ad | 2022-09-12 05:42:57 -0400 | [diff] [blame] | 1 | # Copyright 2018 The ChromiumOS Authors |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 2 | # Use of this source code is governed by a BSD-style license that can be |
| 3 | # found in the LICENSE file. |
| 4 | |
Robert Kolchmeyer | d14321e | 2023-07-11 17:36:25 -0700 | [diff] [blame] | 5 | """Test gconv_strip. |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 6 | |
Robert Kolchmeyer | d14321e | 2023-07-11 17:36:25 -0700 | [diff] [blame] | 7 | To run these tests, do the following inside the chroot: |
| 8 | $ pytest -c /dev/null scripts/gconv_strip_unittest.py |
| 9 | |
| 10 | Explicitly using an empty pytest config is necessary because chromite's |
| 11 | pytest.ini assumes pytest-xdist is installed, which it is not inside the chroot. |
| 12 | """ |
| 13 | |
| 14 | import glob |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 15 | import os |
| 16 | |
| 17 | from chromite.lib import cros_test_lib |
| 18 | from chromite.lib import osutils |
| 19 | from chromite.scripts import gconv_strip |
| 20 | |
Mike Frysinger | 807d828 | 2022-04-28 22:45:17 -0400 | [diff] [blame] | 21 | |
Greg Edelston | a4c9b3b | 2020-01-07 17:51:13 -0700 | [diff] [blame] | 22 | pytestmark = cros_test_lib.pytestmark_inside_only |
| 23 | |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 24 | |
Robert Kolchmeyer | d14321e | 2023-07-11 17:36:25 -0700 | [diff] [blame] | 25 | class GconvStripTest(cros_test_lib.MockTempDirTestCase): |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 26 | """Tests for gconv_strip script.""" |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 27 | |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 28 | def testMultipleStringMatch(self): |
| 29 | self.assertEqual( |
| 30 | gconv_strip.MultipleStringMatch( |
| 31 | [b"hell", b"a", b"z", b"k", b"spec"], |
| 32 | b"hello_from a very special place", |
| 33 | ), |
| 34 | [True, True, False, False, True], |
| 35 | ) |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 36 | |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 37 | def testModuleRewrite(self): |
| 38 | tmp_gconv_module = os.path.join(self.tempdir, "gconv-modules") |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 39 | |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 40 | data = """ |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 41 | # from to module cost |
| 42 | alias FOO charset_foo |
| 43 | alias BAR charset_bar |
| 44 | module charset_foo charset_bar UNUSED_MODULE |
| 45 | |
| 46 | # from to module cost |
| 47 | alias CHAR_A charset_A |
| 48 | alias EUROPE charset_B |
| 49 | module charset_A charset_B USED_MODULE |
| 50 | module charset_foo charset_A USED_MODULE |
| 51 | """ |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 52 | osutils.WriteFile(tmp_gconv_module, data) |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 53 | |
Robert Kolchmeyer | d14321e | 2023-07-11 17:36:25 -0700 | [diff] [blame] | 54 | gmods = gconv_strip.GconvModules( |
| 55 | tmp_gconv_module, os.path.dirname(tmp_gconv_module) |
| 56 | ) |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 57 | self.assertEqual( |
| 58 | gmods.Load(), |
| 59 | [ |
| 60 | "BAR", |
| 61 | "CHAR_A", |
| 62 | "EUROPE", |
| 63 | "FOO", |
| 64 | "charset_A", |
| 65 | "charset_B", |
| 66 | "charset_bar", |
| 67 | "charset_foo", |
| 68 | ], |
| 69 | ) |
| 70 | self.PatchObject(gconv_strip.lddtree, "ParseELF", return_value={}) |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 71 | |
Alex Klein | 074f94f | 2023-06-22 10:32:06 -0600 | [diff] [blame] | 72 | class _StubStat: |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 73 | """Fake for lstat.""" |
| 74 | |
| 75 | st_size = 0 |
| 76 | |
| 77 | self.PatchObject(gconv_strip.os, "lstat", return_value=_StubStat) |
| 78 | self.PatchObject(gconv_strip.os, "unlink") |
Mike Frysinger | 61b792c | 2023-02-02 09:02:27 -0500 | [diff] [blame] | 79 | gmods.Rewrite(["charset_A", "charset_B"], dryrun=False) |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 80 | |
| 81 | expected = """ |
Ned Nguyen | bf081d0 | 2018-12-18 16:41:31 -0700 | [diff] [blame] | 82 | # from to module cost |
| 83 | alias FOO charset_foo |
| 84 | |
| 85 | # from to module cost |
| 86 | alias CHAR_A charset_A |
| 87 | alias EUROPE charset_B |
| 88 | module charset_A charset_B USED_MODULE |
| 89 | module charset_foo charset_A USED_MODULE |
| 90 | """ |
| 91 | |
Alex Klein | 1699fab | 2022-09-08 08:46:06 -0600 | [diff] [blame] | 92 | content = osutils.ReadFile(tmp_gconv_module) |
| 93 | self.assertEqual(content, expected) |
Robert Kolchmeyer | d14321e | 2023-07-11 17:36:25 -0700 | [diff] [blame] | 94 | |
| 95 | def testGconvStrip(self): |
| 96 | """Tests GconvStrip end-to-end. |
| 97 | |
| 98 | Creates a fake root directory with fake gconv modules, and expects the |
| 99 | non-sticky modules to be deleted. |
| 100 | """ |
| 101 | modules_dir = os.path.join(self.tempdir, "usr", "lib64", "gconv") |
| 102 | extras_dir = os.path.join(modules_dir, "gconv-modules.d") |
| 103 | os.makedirs(extras_dir) |
| 104 | tmp_gconv_modules = os.path.join(modules_dir, "gconv-modules") |
| 105 | tmp_gconv_extras = os.path.join(extras_dir, "gconv-modules-extras.conf") |
| 106 | |
| 107 | gconv_data = """ |
| 108 | # from to module cost |
| 109 | alias UTF32// UTF-32// |
| 110 | module UTF-32// INTERNAL UTF-32 1 |
| 111 | module INTERNAL UTF-32// UTF-32 1 |
| 112 | |
| 113 | # from to module cost |
| 114 | alias UTF7// UTF-7// |
| 115 | module UTF-7// INTERNAL UTF-7 1 |
| 116 | module INTERNAL UTF-7// UTF-7 1 |
| 117 | """ |
| 118 | gconv_extras_data = """ |
| 119 | # from to module cost |
| 120 | alias UTF16// UTF-16// |
| 121 | module UTF-16// INTERNAL UTF-16 1 |
| 122 | module INTERNAL UTF-16// UTF-16 1 |
| 123 | |
| 124 | # from to module cost |
| 125 | alias EUCTW// EUC-TW// |
| 126 | alias OSF0005000a// EUC-TW// |
| 127 | module EUC-TW// INTERNAL EUC-TW 1 |
| 128 | module INTERNAL EUC-TW// EUC-TW 1 |
| 129 | """ |
| 130 | osutils.WriteFile(tmp_gconv_modules, gconv_data) |
| 131 | osutils.WriteFile(tmp_gconv_extras, gconv_extras_data) |
| 132 | for module in ["UTF-32.so", "UTF-7.so", "UTF-16.so", "EUC-TW.so"]: |
| 133 | osutils.Touch(os.path.join(modules_dir, module)) |
| 134 | |
| 135 | self.PatchObject(gconv_strip.lddtree, "ParseELF", return_value={}) |
| 136 | |
| 137 | class _StubOpts: |
| 138 | """Stub for GconvStrip args.""" |
| 139 | |
| 140 | def __init__(self, root): |
| 141 | self.root = root |
| 142 | self.dryrun = False |
| 143 | |
| 144 | gconv_strip.GconvStrip(_StubOpts(self.tempdir)) |
| 145 | |
| 146 | expected_gconv_data = """ |
| 147 | # from to module cost |
| 148 | alias UTF32// UTF-32// |
| 149 | module UTF-32// INTERNAL UTF-32 1 |
| 150 | module INTERNAL UTF-32// UTF-32 1 |
| 151 | |
| 152 | # from to module cost |
| 153 | """ |
| 154 | expected_gconv_extras_data = """ |
| 155 | # from to module cost |
| 156 | alias UTF16// UTF-16// |
| 157 | module UTF-16// INTERNAL UTF-16 1 |
| 158 | module INTERNAL UTF-16// UTF-16 1 |
| 159 | |
| 160 | # from to module cost |
| 161 | """ |
| 162 | expected_modules = ["UTF-16.so", "UTF-32.so"] |
| 163 | actual_gconv_data = osutils.ReadFile(tmp_gconv_modules) |
| 164 | actual_gconv_extras_data = osutils.ReadFile(tmp_gconv_extras) |
| 165 | actual_modules = glob.glob(os.path.join(modules_dir, "*.so")) |
| 166 | actual_modules_names = sorted( |
| 167 | os.path.basename(x) for x in actual_modules |
| 168 | ) |
| 169 | self.assertEqual(actual_gconv_data, expected_gconv_data) |
| 170 | self.assertEqual(actual_gconv_extras_data, expected_gconv_extras_data) |
| 171 | self.assertEqual(actual_modules_names, expected_modules) |