#!/usr/bin/env python3 # Copyright 2012 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. """Tests for jni_zero.py. This test suite contains various tests for the JNI generator. It exercises the low-level parser all the way up to the code generator and ensures the output matches a golden file. """ import collections import copy import difflib import glob import logging import os import pathlib import re import shlex import subprocess import sys import tempfile import unittest import zipfile _SCRIPT_DIR = os.path.normpath(os.path.dirname(__file__)) _GOLDENS_DIR = os.path.join(_SCRIPT_DIR, 'golden') _EXTRA_INCLUDES = 'third_party/jni_zero/jni_zero_helper.h' _JAVA_SRC_DIR = os.path.join(_SCRIPT_DIR, 'java', 'src', 'org', 'jni_zero') # Set this environment variable in order to regenerate the golden text # files. _REBASELINE = os.environ.get('REBASELINE', '0') != '0' _accessed_goldens = set() class CliOptions: def __init__(self, is_final=False, is_javap=False, **kwargs): if is_final: self.action = 'generate-final' elif is_javap: self.action = 'from-jar' else: self.action = 'from-source' self.input_files = [] self.jar_file = None self.output_dir = None self.output_files = None if is_final else [] self.header_path = None self.enable_jni_multiplexing = False self.package_prefix = None self.package_prefix_filter = None self.use_proxy_hash = False self.extra_include = None if is_final else _EXTRA_INCLUDES self.module_name = None self.add_stubs_for_missing_native = False self.include_test_only = False self.manual_jni_registration = False self.remove_uncalled_methods = False self.__dict__.update(kwargs) def to_args(self): ret = [os.path.join(_SCRIPT_DIR, os.pardir, 'jni_zero.py'), self.action] if self.enable_jni_multiplexing: ret.append('--enable-jni-multiplexing') if self.package_prefix: ret += ['--package-prefix', self.package_prefix] if self.package_prefix_filter: ret += ['--package-prefix-filter', self.package_prefix_filter] if self.use_proxy_hash: ret.append('--use-proxy-hash') if self.output_dir: ret += ['--output-dir', self.output_dir] if self.input_files: for f in self.input_files: ret += ['--input-file', f] if self.output_files: for f in self.output_files: ret += ['--output-name', f] if self.jar_file: ret += ['--jar-file', self.jar_file] if self.extra_include: ret += ['--extra-include', self.extra_include] if self.add_stubs_for_missing_native: ret.append('--add-stubs-for-missing-native') if self.header_path: ret += ['--header-path', self.header_path] if self.include_test_only: ret.append('--include-test-only') if self.manual_jni_registration: ret.append('--manual-jni-registration') if self.module_name: ret += ['--module-name', self.module_name] if self.remove_uncalled_methods: ret.append('--remove-uncalled-methods') return ret def _MakePrefixes(options): package_prefix = '' if options.package_prefix: package_prefix = options.package_prefix.replace('.', '/') + '/' module_prefix = '' if options.module_name: module_prefix = f'{options.module_name}_' return package_prefix, module_prefix class BaseTest(unittest.TestCase): def _CheckSrcjarGoldens(self, srcjar_path, name_to_goldens): with zipfile.ZipFile(srcjar_path, 'r') as srcjar: self.assertEqual(set(srcjar.namelist()), set(name_to_goldens)) for name in srcjar.namelist(): self.assertTrue( name in name_to_goldens, f'Found {name} output, but not present in name_to_goldens map.') contents = srcjar.read(name).decode('utf-8') self.AssertGoldenTextEquals(contents, name_to_goldens[name]) def _CheckPlaceholderSrcjarGolden(self, srcjar_path, golden_path): expected_contents = [ 'This is the concatenated contents of all files ' 'inside the placeholder srcjar.\n\n' ] with zipfile.ZipFile(srcjar_path, 'r') as srcjar: for name in srcjar.namelist(): file_contents = srcjar.read(name).decode('utf-8') expected_contents += [f'## Contents of {name}:', file_contents, '\n'] self.AssertGoldenTextEquals('\n'.join(expected_contents), golden_path) def _TestEndToEndGeneration(self, input_files, *, srcjar=False, generate_placeholders=False, enable_jni_multiplexing=False, per_file_natives=False, **kwargs): is_javap = input_files[0].endswith('.class') golden_name = self._testMethodName options = CliOptions(is_javap=is_javap, **kwargs) name_to_goldens = {} if srcjar: dir_prefix, file_prefix = _MakePrefixes(options) # GEN_JNI ends up in placeholder srcjar instead if passed. if not per_file_natives: name_to_goldens.update({ f'{dir_prefix}org/jni_zero/{file_prefix}GEN_JNI.java': f'{golden_name}-Placeholder-GEN_JNI.java.golden', }) with tempfile.TemporaryDirectory() as tdir: for i in input_files: basename_and_folder = os.path.splitext(i)[0] basename = os.path.basename(basename_and_folder) options.output_files.append(f'{basename}_jni.h') if srcjar: name_to_goldens.update({ f'org/jni_zero/{basename_and_folder}Jni.java': f'{golden_name}-{basename}Jni.java.golden', }) relative_input_file = os.path.join(_JAVA_SRC_DIR, i) if is_javap: jar_path = os.path.join(tdir, 'input.jar') with zipfile.ZipFile(jar_path, 'w') as z: z.write(relative_input_file, i) options.jar_file = jar_path options.input_files.append(i) else: options.input_files.append(relative_input_file) options.output_dir = tdir cmd = options.to_args() if srcjar: srcjar_path = os.path.join(tdir, 'srcjar.jar') cmd += ['--srcjar-path', srcjar_path] if generate_placeholders: placeholder_srcjar_path = os.path.join(tdir, 'placeholders.srcjar') cmd += ['--placeholder-srcjar-path', placeholder_srcjar_path] if enable_jni_multiplexing: cmd += ['--enable-jni-multiplexing'] if per_file_natives: cmd += ['--per-file-natives'] logging.info('Running: %s', shlex.join(cmd)) subprocess.check_call(cmd) for o in options.output_files: output_path = os.path.join(tdir, o) with open(output_path, 'r') as f: contents = f.read() basename = os.path.splitext(o)[0] header_golden = f'{golden_name}-{basename}.h.golden' self.AssertGoldenTextEquals(contents, header_golden) if srcjar: self._CheckSrcjarGoldens(srcjar_path, name_to_goldens) if generate_placeholders: placeholder_srcjar_golden = f'{golden_name}-placeholder.srcjar.golden' self._CheckPlaceholderSrcjarGolden(placeholder_srcjar_path, placeholder_srcjar_golden) def _TestEndToEndRegistration(self, input_files, golden_name=None, src_files_for_asserts_and_stubs=None, priority_java_files=None, inspection_func=None, **kwargs): golden_name = golden_name or self._testMethodName options = CliOptions(is_final=True, **kwargs) dir_prefix, file_prefix = _MakePrefixes(options) name_to_goldens = { f'{dir_prefix}org/jni_zero/{file_prefix}GEN_JNI.java': f'{golden_name}-Final-GEN_JNI.java.golden', } if options.use_proxy_hash or options.enable_jni_multiplexing: name_to_goldens[f'{dir_prefix}J/{file_prefix}N.java'] = ( f'{golden_name}-Final-N.java.golden') header_golden = None if options.use_proxy_hash or options.manual_jni_registration or options.enable_jni_multiplexing: header_golden = f'{golden_name}-Registration.h.golden' with tempfile.TemporaryDirectory() as tdir: native_sources = [os.path.join(_JAVA_SRC_DIR, f) for f in input_files] if src_files_for_asserts_and_stubs: java_sources = [ os.path.join(_JAVA_SRC_DIR, f) for f in src_files_for_asserts_and_stubs ] else: java_sources = native_sources cmd = options.to_args() java_sources_file = pathlib.Path(tdir) / 'java_sources.txt' java_sources_file.write_text('\n'.join(java_sources)) cmd += ['--java-sources-file', str(java_sources_file)] if native_sources: native_sources_file = pathlib.Path(tdir) / 'native_sources.txt' native_sources_file.write_text('\n'.join(native_sources)) cmd += ['--native-sources-file', str(native_sources_file)] if priority_java_files: priority_java_sources = [ os.path.join(_JAVA_SRC_DIR, f) for f in priority_java_files ] priority_java_file = pathlib.Path(tdir) / 'java_priority_sources.txt' priority_java_file.write_text('\n'.join(priority_java_sources)) cmd += ['--priority-java-sources-file', str(priority_java_file)] if priority_java_files is not None: cmd += ['--never-omit-switch-num'] srcjar_path = os.path.join(tdir, 'srcjar.jar') cmd += ['--srcjar-path', srcjar_path] if header_golden: header_path = os.path.join(tdir, 'header.h') cmd += ['--header-path', header_path] logging.info('Running: %s', shlex.join(cmd)) subprocess.check_call(cmd) self._CheckSrcjarGoldens(srcjar_path, name_to_goldens) if header_golden: with open(header_path, 'r') as f: # Temp directory will cause some diffs each time we run if we don't # normalize. contents = f.read().replace( tdir.replace('/', '_').upper(), 'TEMP_DIR') self.AssertGoldenTextEquals(contents, header_golden) if inspection_func: inspection_func(tdir) def _TestParseError(self, error_snippet, input_data): with tempfile.TemporaryDirectory() as tdir: input_file = os.path.join(tdir, 'MyFile.java') pathlib.Path(input_file).write_text(input_data) options = CliOptions() options.input_files = [input_file] options.output_files = [f'{input_file}_jni.h'] options.output_dir = tdir cmd = options.to_args() logging.info('Running: %s', shlex.join(cmd)) result = subprocess.run(cmd, capture_output=True, check=False, text=True) self.assertIn('MyFile.java', result.stderr) self.assertIn(error_snippet, result.stderr) self.assertEqual(result.returncode, 1) return result.stderr def _ReadGoldenFile(self, path): _accessed_goldens.add(path) if not os.path.exists(path): return None with open(path, 'r') as f: return f.read() def AssertTextEquals(self, golden_text, generated_text): if not self.CompareText(golden_text, generated_text): self.fail('Golden text mismatch.') def CompareText(self, golden_text, generated_text): def FilterText(text): return [ l.strip() for l in text.split('\n') if not l.startswith('// Copyright') ] stripped_golden = FilterText(golden_text) stripped_generated = FilterText(generated_text) if stripped_golden == stripped_generated: return True print(self.id()) for line in difflib.context_diff(stripped_golden, stripped_generated): print(line) print('\n\nGenerated') print('=' * 80) print(generated_text) print('=' * 80) print('Run with:') print('REBASELINE=1', sys.argv[0]) print('to regenerate the data files.') def AssertGoldenTextEquals(self, generated_text, golden_file): """Compares generated text with the corresponding golden_file It will instead compare the generated text with script_dir/golden/golden_file.""" golden_path = os.path.join(_GOLDENS_DIR, golden_file) golden_text = self._ReadGoldenFile(golden_path) if _REBASELINE: if golden_text != generated_text: print('Updated', golden_path) with open(golden_path, 'w') as f: f.write(generated_text) return # golden_text is None if no file is found. Better to fail than in # AssertTextEquals so we can give a clearer message. if golden_text is None: self.fail('Golden file does not exist: ' + golden_path) self.AssertTextEquals(golden_text, generated_text) @unittest.skipIf(os.name == 'nt', 'Not intended to work on Windows') class Tests(BaseTest): def testNonProxy(self): self._TestEndToEndGeneration(['SampleNonProxy.java']) def testBirectionalNonProxy(self): self._TestEndToEndGeneration(['SampleBidirectionalNonProxy.java']) def testBidirectionalClass(self): self._TestEndToEndGeneration(['SampleForTests.java'], srcjar=True) self._TestEndToEndRegistration(['SampleForTests.java']) def testFromClassFile(self): self._TestEndToEndGeneration(['JavapClass.class']) def testUniqueAnnotations(self): self._TestEndToEndGeneration(['SampleUniqueAnnotations.java'], srcjar=True) def testPerFileNatives(self): self._TestEndToEndGeneration(['SampleForAnnotationProcessor.java'], srcjar=True, per_file_natives=True) def testEndToEndProxyHashed(self): self._TestEndToEndGeneration(['SampleForAnnotationProcessor.java'], srcjar=True, generate_placeholders=True) self._TestEndToEndRegistration(['SampleForAnnotationProcessor.java'], use_proxy_hash=True) def testEndToEndManualRegistration(self): self._TestEndToEndRegistration(['SampleForAnnotationProcessor.java'], manual_jni_registration=True) def testEndToEndManualRegistration_NonProxy(self): self._TestEndToEndRegistration(['SampleNonProxy.java'], manual_jni_registration=True) def testEndToEndProxyJniWithModules(self): self._TestEndToEndGeneration(['SampleModule.java'], srcjar=True, use_proxy_hash=True, module_name='module') self._TestEndToEndRegistration( ['SampleForAnnotationProcessor.java', 'SampleModule.java'], use_proxy_hash=True, manual_jni_registration=True, module_name='module') def testModulesWithMultiplexing(self): self._TestEndToEndRegistration( ['SampleForAnnotationProcessor.java', 'SampleModule.java'], enable_jni_multiplexing=True, manual_jni_registration=True, module_name='module') def testStubRegistration(self): input_java_files = ['SampleForAnnotationProcessor.java'] stubs_java_files = input_java_files + [ 'TinySample.java', 'SampleProxyEdgeCases.java' ] extra_input_java_files = ['TinySample2.java'] self._TestEndToEndRegistration( input_java_files + extra_input_java_files, src_files_for_asserts_and_stubs=stubs_java_files, add_stubs_for_missing_native=True, remove_uncalled_methods=True) def testPriorityRegistration(self): input_java_files = [ 'TinySample2.java', 'TinySample.java', 'SampleProxyEdgeCases.java' ] # Add an entry not in input_java_files to simulate one that is in native # sources but not java source (e.g. contains only @CalledByNative) priority_java_files = ['TinySample2.java', 'SampleModule.java'] hash_holder = [] def inspection_func(tdir): header_path = os.path.join(tdir, 'header.h') header_text = pathlib.Path(header_path).read_text() whole = re.findall(r'HashWhole.*?= (.*?);', header_text)[0] priority = re.findall(r'HashPriority.*?= (.*?);', header_text)[0] hash_holder.append((whole, priority)) self._TestEndToEndRegistration(input_java_files, priority_java_files=priority_java_files, inspection_func=inspection_func, enable_jni_multiplexing=True) self._TestEndToEndRegistration(priority_java_files, golden_name='testPriorityRegistrationPart2', priority_java_files=[], inspection_func=inspection_func, enable_jni_multiplexing=True) self.assertEqual(hash_holder[0][1], hash_holder[1][0]) def testFullStubs(self): self._TestEndToEndRegistration( [], src_files_for_asserts_and_stubs=['TinySample.java'], add_stubs_for_missing_native=True) def testForTestingKeptHash(self): input_java_file = 'SampleProxyEdgeCases.java' self._TestEndToEndGeneration([input_java_file], srcjar=True) self._TestEndToEndRegistration([input_java_file], use_proxy_hash=True, include_test_only=True) def testForTestingRemovedHash(self): self._TestEndToEndRegistration(['SampleProxyEdgeCases.java'], use_proxy_hash=True, include_test_only=False) def testForTestingKeptMultiplexing(self): input_java_file = 'SampleProxyEdgeCases.java' self._TestEndToEndGeneration([input_java_file], enable_jni_multiplexing=True, srcjar=True) self._TestEndToEndRegistration([input_java_file], enable_jni_multiplexing=True, include_test_only=True) def testForTestingRemovedMultiplexing(self): self._TestEndToEndRegistration(['SampleProxyEdgeCases.java'], enable_jni_multiplexing=True, include_test_only=False) def testPackagePrefixGenerator(self): self._TestEndToEndGeneration(['SampleForTests.java'], srcjar=True, package_prefix='this.is.a.package.prefix') def testPackagePrefixWithFilter(self): self._TestEndToEndGeneration(['SampleForTests.java'], srcjar=True, package_prefix='this.is.a.package.prefix', package_prefix_filter='org.jni_zero') def testPackagePrefixWithManualRegistration(self): self._TestEndToEndRegistration(['SampleForAnnotationProcessor.java'], package_prefix='this.is.a.package.prefix', manual_jni_registration=True) def testPackagePrefixWithMultiplexing(self): self._TestEndToEndRegistration(['SampleForAnnotationProcessor.java'], package_prefix='this.is.a.package.prefix', enable_jni_multiplexing=True) def testPackagePrefixWithManualRegistrationWithMultiplexing(self): self._TestEndToEndRegistration(['SampleForAnnotationProcessor.java'], package_prefix='this.is.a.package.prefix', enable_jni_multiplexing=True, manual_jni_registration=True) def testPlaceholdersOverlapping(self): self._TestEndToEndGeneration([ 'TinySample.java', 'extrapackage/ImportsTinySample.java', ], srcjar=True, generate_placeholders=True) def testMultiplexing(self): self._TestEndToEndRegistration(['SampleForAnnotationProcessor.java'], enable_jni_multiplexing=True, manual_jni_registration=True) def testParseError_noPackage(self): data = """ class MyFile {} """ self._TestParseError('Unable to find "package" line', data) def testParseError_noClass(self): data = """ package foo; """ self._TestParseError('No classes found', data) def testParseError_wrongClass(self): data = """ package foo; class YourFile {} """ self._TestParseError('Found class "YourFile" but expected "MyFile"', data) def testParseError_noMethods(self): data = """ package foo; class MyFile { void foo() {} } """ self._TestParseError('No native methods found', data) def testParseError_noInterfaceMethods(self): data = """ package foo; class MyFile { @NativeMethods interface A {} } """ self._TestParseError('Found no methods within', data) def testParseError_twoInterfaces(self): data = """ package foo; class MyFile { @NativeMethods interface A { void a(); } @NativeMethods interface B { void b(); } } """ self._TestParseError('Multiple @NativeMethod interfaces', data) def testParseError_twoNamespaces(self): data = """ package foo; @JNINamespace("one") @JNINamespace("two") class MyFile { @NativeMethods interface A { void a(); } } """ self._TestParseError('Found multiple @JNINamespace', data) def main(): try: unittest.main() finally: if _REBASELINE and not any(not x.startswith('-') for x in sys.argv[1:]): for path in glob.glob(os.path.join(_GOLDENS_DIR, '*.golden')): if path not in _accessed_goldens: print('Removing obsolete golden:', path) os.unlink(path) if __name__ == '__main__': main()