|  | #!/usr/bin/env python3 | 
|  |  | 
|  | # Copyright 2020 The Chromium Authors. All rights reserved. | 
|  | # Use of this source code is governed by a BSD-style license that can be | 
|  | # found in the LICENSE file. | 
|  | """Tests for java_cpp_features.py. | 
|  |  | 
|  | This test suite contains various tests for the C++ -> Java base::Feature | 
|  | generator. | 
|  | """ | 
|  |  | 
|  | import unittest | 
|  |  | 
|  | import java_cpp_features | 
|  | from util import java_cpp_utils | 
|  |  | 
|  |  | 
|  | class _TestFeaturesParser(unittest.TestCase): | 
|  | def testParseComments(self): | 
|  | test_data = """ | 
|  | /** | 
|  | * This should be ignored as well. | 
|  | */ | 
|  |  | 
|  | // Comment followed by a blank line. | 
|  |  | 
|  | // Comment followed by unrelated code. | 
|  | int foo() { return 3; } | 
|  |  | 
|  | // Real comment. | 
|  | const base::Feature kSomeFeature{"SomeFeature", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  |  | 
|  | // Real comment that spans | 
|  | // multiple lines. | 
|  | const base::Feature kSomeOtherFeature{"SomeOtherFeature", | 
|  | base::FEATURE_ENABLED_BY_DEFAULT}; | 
|  |  | 
|  | // Comment followed by nothing. | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual(2, len(features)) | 
|  | self.assertEqual('SOME_FEATURE', features[0].name) | 
|  | self.assertEqual('"SomeFeature"', features[0].value) | 
|  | self.assertEqual(1, len(features[0].comments.split('\n'))) | 
|  | self.assertEqual('SOME_OTHER_FEATURE', features[1].name) | 
|  | self.assertEqual('"SomeOtherFeature"', features[1].value) | 
|  | self.assertEqual(2, len(features[1].comments.split('\n'))) | 
|  |  | 
|  | def testWhitespace(self): | 
|  | test_data = """ | 
|  | // 1 line | 
|  | const base::Feature kShort{"Short", base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  |  | 
|  | // 2 lines | 
|  | const base::Feature kTwoLineFeatureA{"TwoLineFeatureA", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | const base::Feature kTwoLineFeatureB{ | 
|  | "TwoLineFeatureB", base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  |  | 
|  | // 3 lines | 
|  | const base::Feature kFeatureWithAVeryLongNameThatWillHaveToWrap{ | 
|  | "FeatureWithAVeryLongNameThatWillHaveToWrap", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual(4, len(features)) | 
|  | self.assertEqual('SHORT', features[0].name) | 
|  | self.assertEqual('"Short"', features[0].value) | 
|  | self.assertEqual('TWO_LINE_FEATURE_A', features[1].name) | 
|  | self.assertEqual('"TwoLineFeatureA"', features[1].value) | 
|  | self.assertEqual('TWO_LINE_FEATURE_B', features[2].name) | 
|  | self.assertEqual('"TwoLineFeatureB"', features[2].value) | 
|  | self.assertEqual('FEATURE_WITH_A_VERY_LONG_NAME_THAT_WILL_HAVE_TO_WRAP', | 
|  | features[3].name) | 
|  | self.assertEqual('"FeatureWithAVeryLongNameThatWillHaveToWrap"', | 
|  | features[3].value) | 
|  |  | 
|  | def testCppSyntax(self): | 
|  | test_data = """ | 
|  | // Mismatched name | 
|  | const base::Feature kMismatchedFeature{"MismatchedName", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  |  | 
|  | namespace myfeature { | 
|  | // In a namespace | 
|  | const base::Feature kSomeFeature{"SomeFeature", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | } | 
|  |  | 
|  | // Defined with equals sign | 
|  | const base::Feature kFoo = {"Foo", base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  |  | 
|  | // Build config-specific base::Feature | 
|  | #if defined(OS_ANDROID) | 
|  | const base::Feature kAndroidOnlyFeature{"AndroidOnlyFeature", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | #endif | 
|  |  | 
|  | // Value depends on build config | 
|  | const base::Feature kMaybeEnabled{"MaybeEnabled", | 
|  | #if defined(OS_ANDROID) | 
|  | base::FEATURE_DISABLED_BY_DEFAULT | 
|  | #else | 
|  | base::FEATURE_ENABLED_BY_DEFAULT | 
|  | #endif | 
|  | }; | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual(5, len(features)) | 
|  | self.assertEqual('MISMATCHED_FEATURE', features[0].name) | 
|  | self.assertEqual('"MismatchedName"', features[0].value) | 
|  | self.assertEqual('SOME_FEATURE', features[1].name) | 
|  | self.assertEqual('"SomeFeature"', features[1].value) | 
|  | self.assertEqual('FOO', features[2].name) | 
|  | self.assertEqual('"Foo"', features[2].value) | 
|  | self.assertEqual('ANDROID_ONLY_FEATURE', features[3].name) | 
|  | self.assertEqual('"AndroidOnlyFeature"', features[3].value) | 
|  | self.assertEqual('MAYBE_ENABLED', features[4].name) | 
|  | self.assertEqual('"MaybeEnabled"', features[4].value) | 
|  |  | 
|  | def testNotYetSupported(self): | 
|  | # Negative test for cases we don't yet support, to ensure we don't misparse | 
|  | # these until we intentionally add proper support. | 
|  | test_data = """ | 
|  | // Not currently supported: name depends on C++ directive | 
|  | const base::Feature kNameDependsOnOs{ | 
|  | #if defined(OS_ANDROID) | 
|  | "MaybeName1", | 
|  | #else | 
|  | "MaybeName2", | 
|  | #endif | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  |  | 
|  | // Not currently supported: feature named with a constant instead of literal | 
|  | const base::Feature kNamedAfterConstant{kNamedStringConstant, | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual(0, len(features)) | 
|  |  | 
|  | def testTreatWebViewLikeOneWord(self): | 
|  | test_data = """ | 
|  | const base::Feature kSomeWebViewFeature{"SomeWebViewFeature", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | const base::Feature kWebViewOtherFeature{"WebViewOtherFeature", | 
|  | base::FEATURE_ENABLED_BY_DEFAULT}; | 
|  | const base::Feature kFeatureWithPluralWebViews{ | 
|  | "FeatureWithPluralWebViews", | 
|  | base::FEATURE_ENABLED_BY_DEFAULT}; | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual('SOME_WEBVIEW_FEATURE', features[0].name) | 
|  | self.assertEqual('"SomeWebViewFeature"', features[0].value) | 
|  | self.assertEqual('WEBVIEW_OTHER_FEATURE', features[1].name) | 
|  | self.assertEqual('"WebViewOtherFeature"', features[1].value) | 
|  | self.assertEqual('FEATURE_WITH_PLURAL_WEBVIEWS', features[2].name) | 
|  | self.assertEqual('"FeatureWithPluralWebViews"', features[2].value) | 
|  |  | 
|  | def testSpecialCharacters(self): | 
|  | test_data = r""" | 
|  | const base::Feature kFeatureWithEscapes{"Weird\tfeature\"name\n", | 
|  | base::FEATURE_DISABLED_BY_DEFAULT}; | 
|  | const base::Feature kFeatureWithEscapes2{ | 
|  | "Weird\tfeature\"name\n", | 
|  | base::FEATURE_ENABLED_BY_DEFAULT}; | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual('FEATURE_WITH_ESCAPES', features[0].name) | 
|  | self.assertEqual(r'"Weird\tfeature\"name\n"', features[0].value) | 
|  | self.assertEqual('FEATURE_WITH_ESCAPES2', features[1].name) | 
|  | self.assertEqual(r'"Weird\tfeature\"name\n"', features[1].value) | 
|  |  | 
|  | def testNoBaseNamespacePrefix(self): | 
|  | test_data = """ | 
|  | const Feature kSomeFeature{"SomeFeature", FEATURE_DISABLED_BY_DEFAULT}; | 
|  | """.split('\n') | 
|  | feature_file_parser = java_cpp_utils.CppConstantParser( | 
|  | java_cpp_features.FeatureParserDelegate(), test_data) | 
|  | features = feature_file_parser.Parse() | 
|  | self.assertEqual('SOME_FEATURE', features[0].name) | 
|  | self.assertEqual('"SomeFeature"', features[0].value) | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | unittest.main() |