Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | # |
| 3 | # Copyright (C) 2018 The Android Open Source Project |
| 4 | # |
| 5 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 6 | # you may not use this file except in compliance with the License. |
| 7 | # You may obtain a copy of the License at |
| 8 | # |
| 9 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | # |
| 11 | # Unless required by applicable law or agreed to in writing, software |
| 12 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 13 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 14 | # See the License for the specific language governing permissions and |
| 15 | # limitations under the License. |
| 16 | # |
| 17 | """Unit tests for manifest_fixer.py.""" |
| 18 | |
| 19 | import sys |
| 20 | import unittest |
| 21 | from xml.dom import minidom |
| 22 | |
| 23 | import manifest_check |
| 24 | |
| 25 | sys.dont_write_bytecode = True |
| 26 | |
| 27 | |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 28 | def uses_library_xml(name, attr=''): |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 29 | return '<uses-library android:name="%s"%s />' % (name, attr) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 30 | |
| 31 | |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 32 | def required_xml(value): |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 33 | return ' android:required="%s"' % ('true' if value else 'false') |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 34 | |
| 35 | |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 36 | def uses_library_apk(name, sfx=''): |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 37 | return "uses-library%s:'%s'" % (sfx, name) |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 38 | |
| 39 | |
| 40 | def required_apk(value): |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 41 | return '' if value else '-not-required' |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 42 | |
| 43 | |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 44 | class EnforceUsesLibrariesTest(unittest.TestCase): |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 45 | """Unit tests for add_extract_native_libs function.""" |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 46 | |
Jiakai Zhang | f98da19 | 2024-04-15 11:15:41 +0000 | [diff] [blame] | 47 | def run_test(self, xml, apk, uses_libraries=[], optional_uses_libraries=[], |
| 48 | missing_optional_uses_libraries=[]): #pylint: disable=dangerous-default-value |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 49 | doc = minidom.parseString(xml) |
| 50 | try: |
| 51 | relax = False |
| 52 | manifest_check.enforce_uses_libraries( |
Jiakai Zhang | f98da19 | 2024-04-15 11:15:41 +0000 | [diff] [blame] | 53 | doc, uses_libraries, optional_uses_libraries, missing_optional_uses_libraries, |
| 54 | relax, False, 'path/to/X/AndroidManifest.xml') |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 55 | manifest_check.enforce_uses_libraries(apk, uses_libraries, |
| 56 | optional_uses_libraries, |
Jiakai Zhang | f98da19 | 2024-04-15 11:15:41 +0000 | [diff] [blame] | 57 | missing_optional_uses_libraries, |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 58 | relax, True, |
| 59 | 'path/to/X/X.apk') |
| 60 | return True |
| 61 | except manifest_check.ManifestMismatchError: |
| 62 | return False |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 63 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 64 | xml_tmpl = ( |
| 65 | '<?xml version="1.0" encoding="utf-8"?>\n<manifest ' |
| 66 | 'xmlns:android="http://schemas.android.com/apk/res/android">\n ' |
| 67 | '<application>\n %s\n </application>\n</manifest>\n') |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 68 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 69 | apk_tmpl = ( |
| 70 | "package: name='com.google.android.something' versionCode='100'\n" |
| 71 | "sdkVersion:'29'\n" |
| 72 | "targetSdkVersion:'29'\n" |
| 73 | "uses-permission: name='android.permission.ACCESS_NETWORK_STATE'\n" |
| 74 | '%s\n' |
| 75 | "densities: '160' '240' '320' '480' '640' '65534") |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 76 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 77 | def test_uses_library(self): |
| 78 | xml = self.xml_tmpl % (uses_library_xml('foo')) |
| 79 | apk = self.apk_tmpl % (uses_library_apk('foo')) |
| 80 | matches = self.run_test(xml, apk, uses_libraries=['foo']) |
| 81 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 82 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 83 | def test_uses_library_required(self): |
| 84 | xml = self.xml_tmpl % (uses_library_xml('foo', required_xml(True))) |
| 85 | apk = self.apk_tmpl % (uses_library_apk('foo', required_apk(True))) |
| 86 | matches = self.run_test(xml, apk, uses_libraries=['foo']) |
| 87 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 88 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 89 | def test_optional_uses_library(self): |
| 90 | xml = self.xml_tmpl % (uses_library_xml('foo', required_xml(False))) |
| 91 | apk = self.apk_tmpl % (uses_library_apk('foo', required_apk(False))) |
| 92 | matches = self.run_test(xml, apk, optional_uses_libraries=['foo']) |
| 93 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 94 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 95 | def test_expected_uses_library(self): |
| 96 | xml = self.xml_tmpl % (uses_library_xml('foo', required_xml(False))) |
| 97 | apk = self.apk_tmpl % (uses_library_apk('foo', required_apk(False))) |
| 98 | matches = self.run_test(xml, apk, uses_libraries=['foo']) |
| 99 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 100 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 101 | def test_expected_optional_uses_library(self): |
| 102 | xml = self.xml_tmpl % (uses_library_xml('foo')) |
| 103 | apk = self.apk_tmpl % (uses_library_apk('foo')) |
| 104 | matches = self.run_test(xml, apk, optional_uses_libraries=['foo']) |
| 105 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 106 | |
Jiakai Zhang | f98da19 | 2024-04-15 11:15:41 +0000 | [diff] [blame] | 107 | def test_expected_missing_optional_uses_library(self): |
| 108 | xml = self.xml_tmpl % ( |
| 109 | uses_library_xml('foo') + uses_library_xml('missing') + uses_library_xml('bar')) |
| 110 | apk = self.apk_tmpl % ( |
| 111 | uses_library_apk('foo') + uses_library_apk('missing') + uses_library_apk('bar')) |
| 112 | matches = self.run_test(xml, apk, optional_uses_libraries=['foo', 'bar'], |
| 113 | missing_optional_uses_libraries=['missing']) |
| 114 | self.assertFalse(matches) |
| 115 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 116 | def test_missing_uses_library(self): |
| 117 | xml = self.xml_tmpl % ('') |
| 118 | apk = self.apk_tmpl % ('') |
| 119 | matches = self.run_test(xml, apk, uses_libraries=['foo']) |
| 120 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 121 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 122 | def test_missing_optional_uses_library(self): |
| 123 | xml = self.xml_tmpl % ('') |
| 124 | apk = self.apk_tmpl % ('') |
| 125 | matches = self.run_test(xml, apk, optional_uses_libraries=['foo']) |
| 126 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 127 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 128 | def test_extra_uses_library(self): |
| 129 | xml = self.xml_tmpl % (uses_library_xml('foo')) |
| 130 | apk = self.apk_tmpl % (uses_library_xml('foo')) |
| 131 | matches = self.run_test(xml, apk) |
| 132 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 133 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 134 | def test_extra_optional_uses_library(self): |
| 135 | xml = self.xml_tmpl % (uses_library_xml('foo', required_xml(False))) |
| 136 | apk = self.apk_tmpl % (uses_library_apk('foo', required_apk(False))) |
| 137 | matches = self.run_test(xml, apk) |
| 138 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 139 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 140 | def test_multiple_uses_library(self): |
| 141 | xml = self.xml_tmpl % ('\n'.join( |
| 142 | [uses_library_xml('foo'), |
| 143 | uses_library_xml('bar')])) |
| 144 | apk = self.apk_tmpl % ('\n'.join( |
| 145 | [uses_library_apk('foo'), |
| 146 | uses_library_apk('bar')])) |
| 147 | matches = self.run_test(xml, apk, uses_libraries=['foo', 'bar']) |
| 148 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 149 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 150 | def test_multiple_optional_uses_library(self): |
| 151 | xml = self.xml_tmpl % ('\n'.join([ |
| 152 | uses_library_xml('foo', required_xml(False)), |
| 153 | uses_library_xml('bar', required_xml(False)) |
| 154 | ])) |
| 155 | apk = self.apk_tmpl % ('\n'.join([ |
| 156 | uses_library_apk('foo', required_apk(False)), |
| 157 | uses_library_apk('bar', required_apk(False)) |
| 158 | ])) |
| 159 | matches = self.run_test( |
| 160 | xml, apk, optional_uses_libraries=['foo', 'bar']) |
| 161 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 162 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 163 | def test_order_uses_library(self): |
| 164 | xml = self.xml_tmpl % ('\n'.join( |
| 165 | [uses_library_xml('foo'), |
| 166 | uses_library_xml('bar')])) |
| 167 | apk = self.apk_tmpl % ('\n'.join( |
| 168 | [uses_library_apk('foo'), |
| 169 | uses_library_apk('bar')])) |
| 170 | matches = self.run_test(xml, apk, uses_libraries=['bar', 'foo']) |
| 171 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 172 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 173 | def test_order_optional_uses_library(self): |
| 174 | xml = self.xml_tmpl % ('\n'.join([ |
| 175 | uses_library_xml('foo', required_xml(False)), |
| 176 | uses_library_xml('bar', required_xml(False)) |
| 177 | ])) |
| 178 | apk = self.apk_tmpl % ('\n'.join([ |
| 179 | uses_library_apk('foo', required_apk(False)), |
| 180 | uses_library_apk('bar', required_apk(False)) |
| 181 | ])) |
| 182 | matches = self.run_test( |
| 183 | xml, apk, optional_uses_libraries=['bar', 'foo']) |
| 184 | self.assertFalse(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 185 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 186 | def test_duplicate_uses_library(self): |
| 187 | xml = self.xml_tmpl % ('\n'.join( |
| 188 | [uses_library_xml('foo'), |
| 189 | uses_library_xml('foo')])) |
| 190 | apk = self.apk_tmpl % ('\n'.join( |
| 191 | [uses_library_apk('foo'), |
| 192 | uses_library_apk('foo')])) |
| 193 | matches = self.run_test(xml, apk, uses_libraries=['foo']) |
| 194 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 195 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 196 | def test_duplicate_optional_uses_library(self): |
| 197 | xml = self.xml_tmpl % ('\n'.join([ |
| 198 | uses_library_xml('foo', required_xml(False)), |
| 199 | uses_library_xml('foo', required_xml(False)) |
| 200 | ])) |
| 201 | apk = self.apk_tmpl % ('\n'.join([ |
| 202 | uses_library_apk('foo', required_apk(False)), |
| 203 | uses_library_apk('foo', required_apk(False)) |
| 204 | ])) |
| 205 | matches = self.run_test(xml, apk, optional_uses_libraries=['foo']) |
| 206 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 207 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 208 | def test_mixed(self): |
| 209 | xml = self.xml_tmpl % ('\n'.join([ |
| 210 | uses_library_xml('foo'), |
| 211 | uses_library_xml('bar', required_xml(False)) |
| 212 | ])) |
| 213 | apk = self.apk_tmpl % ('\n'.join([ |
| 214 | uses_library_apk('foo'), |
| 215 | uses_library_apk('bar', required_apk(False)) |
| 216 | ])) |
| 217 | matches = self.run_test( |
| 218 | xml, apk, uses_libraries=['foo'], optional_uses_libraries=['bar']) |
| 219 | self.assertTrue(matches) |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 220 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 221 | def test_mixed_with_namespace(self): |
| 222 | xml = self.xml_tmpl % ('\n'.join([ |
| 223 | uses_library_xml('foo'), |
| 224 | uses_library_xml('bar', required_xml(False)) |
| 225 | ])) |
| 226 | apk = self.apk_tmpl % ('\n'.join([ |
| 227 | uses_library_apk('foo'), |
| 228 | uses_library_apk('bar', required_apk(False)) |
| 229 | ])) |
| 230 | matches = self.run_test( |
| 231 | xml, |
| 232 | apk, |
| 233 | uses_libraries=['//x/y/z:foo'], |
| 234 | optional_uses_libraries=['//x/y/z:bar']) |
| 235 | self.assertTrue(matches) |
Ulya Trafimovich | 1b51345 | 2021-07-20 14:27:32 +0100 | [diff] [blame] | 236 | |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 237 | |
| 238 | class ExtractTargetSdkVersionTest(unittest.TestCase): |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 239 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 240 | def run_test(self, xml, apk, version): |
| 241 | doc = minidom.parseString(xml) |
| 242 | v = manifest_check.extract_target_sdk_version(doc, is_apk=False) |
| 243 | self.assertEqual(v, version) |
| 244 | v = manifest_check.extract_target_sdk_version(apk, is_apk=True) |
| 245 | self.assertEqual(v, version) |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 246 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 247 | xml_tmpl = ( |
| 248 | '<?xml version="1.0" encoding="utf-8"?>\n<manifest ' |
| 249 | 'xmlns:android="http://schemas.android.com/apk/res/android">\n ' |
| 250 | '<uses-sdk android:minSdkVersion="28" android:targetSdkVersion="%s" ' |
| 251 | '/>\n</manifest>\n') |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 252 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 253 | apk_tmpl = ( |
| 254 | "package: name='com.google.android.something' versionCode='100'\n" |
| 255 | "sdkVersion:'28'\n" |
| 256 | "targetSdkVersion:'%s'\n" |
| 257 | "uses-permission: name='android.permission.ACCESS_NETWORK_STATE'\n") |
Ulya Trafimovich | 0aba252 | 2021-03-03 16:38:37 +0000 | [diff] [blame] | 258 | |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 259 | def test_targert_sdk_version_28(self): |
| 260 | xml = self.xml_tmpl % '28' |
| 261 | apk = self.apk_tmpl % '28' |
| 262 | self.run_test(xml, apk, '28') |
| 263 | |
| 264 | def test_targert_sdk_version_29(self): |
| 265 | xml = self.xml_tmpl % '29' |
| 266 | apk = self.apk_tmpl % '29' |
| 267 | self.run_test(xml, apk, '29') |
| 268 | |
Colin Cross | 7211910 | 2019-05-20 13:14:18 -0700 | [diff] [blame] | 269 | |
| 270 | if __name__ == '__main__': |
Spandan Das | f880742 | 2021-08-25 20:01:17 +0000 | [diff] [blame] | 271 | unittest.main(verbosity=2) |