|  | package android | 
|  |  | 
|  | import ( | 
|  | "testing" | 
|  |  | 
|  | "github.com/google/blueprint" | 
|  | ) | 
|  |  | 
|  | var licensesTests = []struct { | 
|  | name                       string | 
|  | fs                         MockFS | 
|  | expectedErrors             []string | 
|  | effectiveLicenses          map[string][]string | 
|  | effectiveInheritedLicenses map[string][]string | 
|  | effectivePackage           map[string]string | 
|  | effectiveNotices           map[string][]string | 
|  | effectiveKinds             map[string][]string | 
|  | effectiveConditions        map[string][]string | 
|  | }{ | 
|  | { | 
|  | name: "invalid module type without licenses property", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | mock_bad_module { | 
|  | name: "libexample", | 
|  | }`), | 
|  | }, | 
|  | expectedErrors: []string{`module type "mock_bad_module" must have an applicable licenses property`}, | 
|  | }, | 
|  | { | 
|  | name: "license must exist", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "libexample", | 
|  | licenses: ["notice"], | 
|  | }`), | 
|  | }, | 
|  | expectedErrors: []string{`"libexample" depends on undefined module "notice"`}, | 
|  | }, | 
|  | { | 
|  | name: "all good", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | license_kind { | 
|  | name: "notice", | 
|  | conditions: ["shownotice"], | 
|  | } | 
|  |  | 
|  | license { | 
|  | name: "top_Apache2", | 
|  | license_kinds: ["notice"], | 
|  | package_name: "topDog", | 
|  | license_text: ["LICENSE", "NOTICE"], | 
|  | } | 
|  |  | 
|  | mock_library { | 
|  | name: "libexample1", | 
|  | licenses: ["top_Apache2"], | 
|  | }`), | 
|  | "top/nested/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "libnested", | 
|  | licenses: ["top_Apache2"], | 
|  | }`), | 
|  | "other/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "libother", | 
|  | licenses: ["top_Apache2"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "libexample1": []string{"top_Apache2"}, | 
|  | "libnested":   []string{"top_Apache2"}, | 
|  | "libother":    []string{"top_Apache2"}, | 
|  | }, | 
|  | effectiveKinds: map[string][]string{ | 
|  | "libexample1": []string{"notice"}, | 
|  | "libnested":   []string{"notice"}, | 
|  | "libother":    []string{"notice"}, | 
|  | }, | 
|  | effectivePackage: map[string]string{ | 
|  | "libexample1": "topDog", | 
|  | "libnested":   "topDog", | 
|  | "libother":    "topDog", | 
|  | }, | 
|  | effectiveConditions: map[string][]string{ | 
|  | "libexample1": []string{"shownotice"}, | 
|  | "libnested":   []string{"shownotice"}, | 
|  | "libother":    []string{"shownotice"}, | 
|  | }, | 
|  | effectiveNotices: map[string][]string{ | 
|  | "libexample1": []string{"top/LICENSE:topDog", "top/NOTICE:topDog"}, | 
|  | "libnested":   []string{"top/LICENSE:topDog", "top/NOTICE:topDog"}, | 
|  | "libother":    []string{"top/LICENSE:topDog", "top/NOTICE:topDog"}, | 
|  | }, | 
|  | }, | 
|  |  | 
|  | // Defaults propagation tests | 
|  | { | 
|  | // Check that licenses is the union of the defaults modules. | 
|  | name: "defaults union, basic", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | license_kind { | 
|  | name: "top_notice", | 
|  | conditions: ["notice"], | 
|  | } | 
|  |  | 
|  | license { | 
|  | name: "top_other", | 
|  | license_kinds: ["top_notice"], | 
|  | } | 
|  |  | 
|  | mock_defaults { | 
|  | name: "libexample_defaults", | 
|  | licenses: ["top_other"], | 
|  | } | 
|  | mock_library { | 
|  | name: "libexample", | 
|  | licenses: ["nested_other"], | 
|  | defaults: ["libexample_defaults"], | 
|  | } | 
|  | mock_library { | 
|  | name: "libsamepackage", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | "top/nested/Android.bp": []byte(` | 
|  | license_kind { | 
|  | name: "nested_notice", | 
|  | conditions: ["notice"], | 
|  | } | 
|  |  | 
|  | license { | 
|  | name: "nested_other", | 
|  | license_kinds: ["nested_notice"], | 
|  | } | 
|  |  | 
|  | mock_library { | 
|  | name: "libnested", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | "other/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "libother", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "libexample":     []string{"nested_other", "top_other"}, | 
|  | "libsamepackage": []string{}, | 
|  | "libnested":      []string{}, | 
|  | "libother":       []string{}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "libexample":     []string{"nested_other", "top_other"}, | 
|  | "libsamepackage": []string{"nested_other", "top_other"}, | 
|  | "libnested":      []string{"nested_other", "top_other"}, | 
|  | "libother":       []string{"nested_other", "top_other"}, | 
|  | }, | 
|  | effectiveKinds: map[string][]string{ | 
|  | "libexample":     []string{"nested_notice", "top_notice"}, | 
|  | "libsamepackage": []string{}, | 
|  | "libnested":      []string{}, | 
|  | "libother":       []string{}, | 
|  | }, | 
|  | effectiveConditions: map[string][]string{ | 
|  | "libexample":     []string{"notice"}, | 
|  | "libsamepackage": []string{}, | 
|  | "libnested":      []string{}, | 
|  | "libother":       []string{}, | 
|  | }, | 
|  | }, | 
|  | { | 
|  | name: "defaults union, multiple defaults", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | license { | 
|  | name: "top", | 
|  | } | 
|  | mock_defaults { | 
|  | name: "libexample_defaults_1", | 
|  | licenses: ["other"], | 
|  | } | 
|  | mock_defaults { | 
|  | name: "libexample_defaults_2", | 
|  | licenses: ["top_nested"], | 
|  | } | 
|  | mock_library { | 
|  | name: "libexample", | 
|  | defaults: ["libexample_defaults_1", "libexample_defaults_2"], | 
|  | } | 
|  | mock_library { | 
|  | name: "libsamepackage", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | "top/nested/Android.bp": []byte(` | 
|  | license { | 
|  | name: "top_nested", | 
|  | license_text: ["LICENSE.txt"], | 
|  | } | 
|  | mock_library { | 
|  | name: "libnested", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | "other/Android.bp": []byte(` | 
|  | license { | 
|  | name: "other", | 
|  | } | 
|  | mock_library { | 
|  | name: "libother", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | "outsider/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "liboutsider", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "libexample":     []string{"other", "top_nested"}, | 
|  | "libsamepackage": []string{}, | 
|  | "libnested":      []string{}, | 
|  | "libother":       []string{}, | 
|  | "liboutsider":    []string{}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "libexample":     []string{"other", "top_nested"}, | 
|  | "libsamepackage": []string{"other", "top_nested"}, | 
|  | "libnested":      []string{"other", "top_nested"}, | 
|  | "libother":       []string{"other", "top_nested"}, | 
|  | "liboutsider":    []string{"other", "top_nested"}, | 
|  | }, | 
|  | effectiveKinds: map[string][]string{ | 
|  | "libexample":     []string{}, | 
|  | "libsamepackage": []string{}, | 
|  | "libnested":      []string{}, | 
|  | "libother":       []string{}, | 
|  | "liboutsider":    []string{}, | 
|  | }, | 
|  | effectiveNotices: map[string][]string{ | 
|  | "libexample":     []string{"top/nested/LICENSE.txt"}, | 
|  | "libsamepackage": []string{}, | 
|  | "libnested":      []string{}, | 
|  | "libother":       []string{}, | 
|  | "liboutsider":    []string{}, | 
|  | }, | 
|  | }, | 
|  |  | 
|  | // Defaults module's defaults_licenses tests | 
|  | { | 
|  | name: "defaults_licenses invalid", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | mock_defaults { | 
|  | name: "top_defaults", | 
|  | licenses: ["notice"], | 
|  | }`), | 
|  | }, | 
|  | expectedErrors: []string{`"top_defaults" depends on undefined module "notice"`}, | 
|  | }, | 
|  | { | 
|  | name: "defaults_licenses overrides package default", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | package { | 
|  | default_applicable_licenses: ["by_exception_only"], | 
|  | } | 
|  | license { | 
|  | name: "by_exception_only", | 
|  | } | 
|  | license { | 
|  | name: "notice", | 
|  | } | 
|  | mock_defaults { | 
|  | name: "top_defaults", | 
|  | licenses: ["notice"], | 
|  | } | 
|  | mock_library { | 
|  | name: "libexample", | 
|  | } | 
|  | mock_library { | 
|  | name: "libdefaults", | 
|  | defaults: ["top_defaults"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "libexample":  []string{"by_exception_only"}, | 
|  | "libdefaults": []string{"notice"}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "libexample":  []string{"by_exception_only"}, | 
|  | "libdefaults": []string{"notice"}, | 
|  | }, | 
|  | }, | 
|  |  | 
|  | // Package default_applicable_licenses tests | 
|  | { | 
|  | name: "package default_applicable_licenses must exist", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | package { | 
|  | default_applicable_licenses: ["notice"], | 
|  | }`), | 
|  | }, | 
|  | expectedErrors: []string{`"//top" depends on undefined module "notice"`}, | 
|  | }, | 
|  | { | 
|  | // This test relies on the default licenses being legacy_public. | 
|  | name: "package default_applicable_licenses property used when no licenses specified", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | package { | 
|  | default_applicable_licenses: ["top_notice"], | 
|  | } | 
|  |  | 
|  | license { | 
|  | name: "top_notice", | 
|  | } | 
|  | mock_library { | 
|  | name: "libexample", | 
|  | }`), | 
|  | "outsider/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "liboutsider", | 
|  | deps: ["libexample"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "libexample":  []string{"top_notice"}, | 
|  | "liboutsider": []string{}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "libexample":  []string{"top_notice"}, | 
|  | "liboutsider": []string{"top_notice"}, | 
|  | }, | 
|  | }, | 
|  | { | 
|  | name: "package default_applicable_licenses not inherited to subpackages", | 
|  | fs: map[string][]byte{ | 
|  | "top/Android.bp": []byte(` | 
|  | package { | 
|  | default_applicable_licenses: ["top_notice"], | 
|  | } | 
|  | license { | 
|  | name: "top_notice", | 
|  | } | 
|  | mock_library { | 
|  | name: "libexample", | 
|  | }`), | 
|  | "top/nested/Android.bp": []byte(` | 
|  | package { | 
|  | default_applicable_licenses: ["outsider"], | 
|  | } | 
|  |  | 
|  | mock_library { | 
|  | name: "libnested", | 
|  | }`), | 
|  | "top/other/Android.bp": []byte(` | 
|  | mock_library { | 
|  | name: "libother", | 
|  | }`), | 
|  | "outsider/Android.bp": []byte(` | 
|  | license { | 
|  | name: "outsider", | 
|  | } | 
|  | mock_library { | 
|  | name: "liboutsider", | 
|  | deps: ["libexample", "libother", "libnested"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "libexample":  []string{"top_notice"}, | 
|  | "libnested":   []string{"outsider"}, | 
|  | "libother":    []string{}, | 
|  | "liboutsider": []string{}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "libexample":  []string{"top_notice"}, | 
|  | "libnested":   []string{"outsider"}, | 
|  | "libother":    []string{}, | 
|  | "liboutsider": []string{"top_notice", "outsider"}, | 
|  | }, | 
|  | }, | 
|  | { | 
|  | name: "verify that prebuilt dependencies are included", | 
|  | fs: map[string][]byte{ | 
|  | "prebuilts/Android.bp": []byte(` | 
|  | license { | 
|  | name: "prebuilt" | 
|  | } | 
|  | prebuilt { | 
|  | name: "module", | 
|  | licenses: ["prebuilt"], | 
|  | }`), | 
|  | "top/sources/source_file": nil, | 
|  | "top/sources/Android.bp": []byte(` | 
|  | license { | 
|  | name: "top_sources" | 
|  | } | 
|  | source { | 
|  | name: "module", | 
|  | licenses: ["top_sources"], | 
|  | }`), | 
|  | "top/other/source_file": nil, | 
|  | "top/other/Android.bp": []byte(` | 
|  | source { | 
|  | name: "other", | 
|  | deps: [":module"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "other": []string{}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "other": []string{"prebuilt", "top_sources"}, | 
|  | }, | 
|  | }, | 
|  | { | 
|  | name: "verify that prebuilt dependencies are ignored for licenses reasons (preferred)", | 
|  | fs: map[string][]byte{ | 
|  | "prebuilts/Android.bp": []byte(` | 
|  | license { | 
|  | name: "prebuilt" | 
|  | } | 
|  | prebuilt { | 
|  | name: "module", | 
|  | licenses: ["prebuilt"], | 
|  | prefer: true, | 
|  | }`), | 
|  | "top/sources/source_file": nil, | 
|  | "top/sources/Android.bp": []byte(` | 
|  | license { | 
|  | name: "top_sources" | 
|  | } | 
|  | source { | 
|  | name: "module", | 
|  | licenses: ["top_sources"], | 
|  | }`), | 
|  | "top/other/source_file": nil, | 
|  | "top/other/Android.bp": []byte(` | 
|  | source { | 
|  | name: "other", | 
|  | deps: [":module"], | 
|  | }`), | 
|  | }, | 
|  | effectiveLicenses: map[string][]string{ | 
|  | "other": []string{}, | 
|  | }, | 
|  | effectiveInheritedLicenses: map[string][]string{ | 
|  | "module": []string{"prebuilt", "top_sources"}, | 
|  | "other":  []string{"prebuilt", "top_sources"}, | 
|  | }, | 
|  | }, | 
|  | } | 
|  |  | 
|  | func TestLicenses(t *testing.T) { | 
|  | for _, test := range licensesTests { | 
|  | t.Run(test.name, func(t *testing.T) { | 
|  | // Customize the common license text fixture factory. | 
|  | result := GroupFixturePreparers( | 
|  | prepareForLicenseTest, | 
|  | FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
|  | ctx.RegisterModuleType("mock_bad_module", newMockLicensesBadModule) | 
|  | ctx.RegisterModuleType("mock_library", newMockLicensesLibraryModule) | 
|  | ctx.RegisterModuleType("mock_defaults", defaultsLicensesFactory) | 
|  | }), | 
|  | test.fs.AddToFixture(), | 
|  | ). | 
|  | ExtendWithErrorHandler(FixtureExpectsAllErrorsToMatchAPattern(test.expectedErrors)). | 
|  | RunTest(t) | 
|  |  | 
|  | if test.effectiveLicenses != nil { | 
|  | checkEffectiveLicenses(t, result, test.effectiveLicenses) | 
|  | } | 
|  |  | 
|  | if test.effectivePackage != nil { | 
|  | checkEffectivePackage(t, result, test.effectivePackage) | 
|  | } | 
|  |  | 
|  | if test.effectiveNotices != nil { | 
|  | checkEffectiveNotices(t, result, test.effectiveNotices) | 
|  | } | 
|  |  | 
|  | if test.effectiveKinds != nil { | 
|  | checkEffectiveKinds(t, result, test.effectiveKinds) | 
|  | } | 
|  |  | 
|  | if test.effectiveConditions != nil { | 
|  | checkEffectiveConditions(t, result, test.effectiveConditions) | 
|  | } | 
|  |  | 
|  | if test.effectiveInheritedLicenses != nil { | 
|  | checkEffectiveInheritedLicenses(t, result, test.effectiveInheritedLicenses) | 
|  | } | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func checkEffectiveLicenses(t *testing.T, result *TestResult, effectiveLicenses map[string][]string) { | 
|  | actualLicenses := make(map[string][]string) | 
|  | result.Context.Context.VisitAllModules(func(m blueprint.Module) { | 
|  | if _, ok := m.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | module, ok := m.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", m.Name()) | 
|  | return | 
|  | } | 
|  | base := module.base() | 
|  | if base == nil { | 
|  | return | 
|  | } | 
|  | actualLicenses[m.Name()] = base.commonProperties.Effective_licenses | 
|  | }) | 
|  |  | 
|  | for moduleName, expectedLicenses := range effectiveLicenses { | 
|  | licenses, ok := actualLicenses[moduleName] | 
|  | if !ok { | 
|  | licenses = []string{} | 
|  | } | 
|  | if !compareUnorderedStringArrays(expectedLicenses, licenses) { | 
|  | t.Errorf("effective licenses mismatch for module %q: expected %q, found %q", moduleName, expectedLicenses, licenses) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func checkEffectiveInheritedLicenses(t *testing.T, result *TestResult, effectiveInheritedLicenses map[string][]string) { | 
|  | actualLicenses := make(map[string][]string) | 
|  | result.Context.Context.VisitAllModules(func(m blueprint.Module) { | 
|  | if _, ok := m.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | module, ok := m.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", m.Name()) | 
|  | return | 
|  | } | 
|  | base := module.base() | 
|  | if base == nil { | 
|  | return | 
|  | } | 
|  | inherited := make(map[string]bool) | 
|  | for _, l := range base.commonProperties.Effective_licenses { | 
|  | inherited[l] = true | 
|  | } | 
|  | result.Context.Context.VisitDepsDepthFirst(m, func(c blueprint.Module) { | 
|  | if _, ok := c.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := c.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := c.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | cmodule, ok := c.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", c.Name()) | 
|  | return | 
|  | } | 
|  | cbase := cmodule.base() | 
|  | if cbase == nil { | 
|  | return | 
|  | } | 
|  | for _, l := range cbase.commonProperties.Effective_licenses { | 
|  | inherited[l] = true | 
|  | } | 
|  | }) | 
|  | actualLicenses[m.Name()] = []string{} | 
|  | for l := range inherited { | 
|  | actualLicenses[m.Name()] = append(actualLicenses[m.Name()], l) | 
|  | } | 
|  | }) | 
|  |  | 
|  | for moduleName, expectedInheritedLicenses := range effectiveInheritedLicenses { | 
|  | licenses, ok := actualLicenses[moduleName] | 
|  | if !ok { | 
|  | licenses = []string{} | 
|  | } | 
|  | if !compareUnorderedStringArrays(expectedInheritedLicenses, licenses) { | 
|  | t.Errorf("effective inherited licenses mismatch for module %q: expected %q, found %q", moduleName, expectedInheritedLicenses, licenses) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func checkEffectivePackage(t *testing.T, result *TestResult, effectivePackage map[string]string) { | 
|  | actualPackage := make(map[string]string) | 
|  | result.Context.Context.VisitAllModules(func(m blueprint.Module) { | 
|  | if _, ok := m.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | module, ok := m.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", m.Name()) | 
|  | return | 
|  | } | 
|  | base := module.base() | 
|  | if base == nil { | 
|  | return | 
|  | } | 
|  |  | 
|  | if base.commonProperties.Effective_package_name == nil { | 
|  | actualPackage[m.Name()] = "" | 
|  | } else { | 
|  | actualPackage[m.Name()] = *base.commonProperties.Effective_package_name | 
|  | } | 
|  | }) | 
|  |  | 
|  | for moduleName, expectedPackage := range effectivePackage { | 
|  | packageName, ok := actualPackage[moduleName] | 
|  | if !ok { | 
|  | packageName = "" | 
|  | } | 
|  | if expectedPackage != packageName { | 
|  | t.Errorf("effective package mismatch for module %q: expected %q, found %q", moduleName, expectedPackage, packageName) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func checkEffectiveNotices(t *testing.T, result *TestResult, effectiveNotices map[string][]string) { | 
|  | actualNotices := make(map[string][]string) | 
|  | result.Context.Context.VisitAllModules(func(m blueprint.Module) { | 
|  | if _, ok := m.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | module, ok := m.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", m.Name()) | 
|  | return | 
|  | } | 
|  | base := module.base() | 
|  | if base == nil { | 
|  | return | 
|  | } | 
|  | actualNotices[m.Name()] = base.commonProperties.Effective_license_text.Strings() | 
|  | }) | 
|  |  | 
|  | for moduleName, expectedNotices := range effectiveNotices { | 
|  | notices, ok := actualNotices[moduleName] | 
|  | if !ok { | 
|  | notices = []string{} | 
|  | } | 
|  | if !compareUnorderedStringArrays(expectedNotices, notices) { | 
|  | t.Errorf("effective notice files mismatch for module %q: expected %q, found %q", moduleName, expectedNotices, notices) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func checkEffectiveKinds(t *testing.T, result *TestResult, effectiveKinds map[string][]string) { | 
|  | actualKinds := make(map[string][]string) | 
|  | result.Context.Context.VisitAllModules(func(m blueprint.Module) { | 
|  | if _, ok := m.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | module, ok := m.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", m.Name()) | 
|  | return | 
|  | } | 
|  | base := module.base() | 
|  | if base == nil { | 
|  | return | 
|  | } | 
|  | actualKinds[m.Name()] = base.commonProperties.Effective_license_kinds | 
|  | }) | 
|  |  | 
|  | for moduleName, expectedKinds := range effectiveKinds { | 
|  | kinds, ok := actualKinds[moduleName] | 
|  | if !ok { | 
|  | kinds = []string{} | 
|  | } | 
|  | if !compareUnorderedStringArrays(expectedKinds, kinds) { | 
|  | t.Errorf("effective license kinds mismatch for module %q: expected %q, found %q", moduleName, expectedKinds, kinds) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func checkEffectiveConditions(t *testing.T, result *TestResult, effectiveConditions map[string][]string) { | 
|  | actualConditions := make(map[string][]string) | 
|  | result.Context.Context.VisitAllModules(func(m blueprint.Module) { | 
|  | if _, ok := m.(*licenseModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*licenseKindModule); ok { | 
|  | return | 
|  | } | 
|  | if _, ok := m.(*packageModule); ok { | 
|  | return | 
|  | } | 
|  | module, ok := m.(Module) | 
|  | if !ok { | 
|  | t.Errorf("%q not a module", m.Name()) | 
|  | return | 
|  | } | 
|  | base := module.base() | 
|  | if base == nil { | 
|  | return | 
|  | } | 
|  | actualConditions[m.Name()] = base.commonProperties.Effective_license_conditions | 
|  | }) | 
|  |  | 
|  | for moduleName, expectedConditions := range effectiveConditions { | 
|  | conditions, ok := actualConditions[moduleName] | 
|  | if !ok { | 
|  | conditions = []string{} | 
|  | } | 
|  | if !compareUnorderedStringArrays(expectedConditions, conditions) { | 
|  | t.Errorf("effective license conditions mismatch for module %q: expected %q, found %q", moduleName, expectedConditions, conditions) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func compareUnorderedStringArrays(expected, actual []string) bool { | 
|  | if len(expected) != len(actual) { | 
|  | return false | 
|  | } | 
|  | s := make(map[string]int) | 
|  | for _, v := range expected { | 
|  | s[v] += 1 | 
|  | } | 
|  | for _, v := range actual { | 
|  | c, ok := s[v] | 
|  | if !ok { | 
|  | return false | 
|  | } | 
|  | if c < 1 { | 
|  | return false | 
|  | } | 
|  | s[v] -= 1 | 
|  | } | 
|  | return true | 
|  | } | 
|  |  | 
|  | type mockLicensesBadProperties struct { | 
|  | Visibility []string | 
|  | } | 
|  |  | 
|  | type mockLicensesBadModule struct { | 
|  | ModuleBase | 
|  | DefaultableModuleBase | 
|  | properties mockLicensesBadProperties | 
|  | } | 
|  |  | 
|  | func newMockLicensesBadModule() Module { | 
|  | m := &mockLicensesBadModule{} | 
|  |  | 
|  | base := m.base() | 
|  | m.AddProperties(&base.nameProperties, &m.properties) | 
|  |  | 
|  | // The default_visibility property needs to be checked and parsed by the visibility module during | 
|  | // its checking and parsing phases so make it the primary visibility property. | 
|  | setPrimaryVisibilityProperty(m, "visibility", &m.properties.Visibility) | 
|  |  | 
|  | initAndroidModuleBase(m) | 
|  | InitDefaultableModule(m) | 
|  |  | 
|  | return m | 
|  | } | 
|  |  | 
|  | func (m *mockLicensesBadModule) GenerateAndroidBuildActions(ModuleContext) { | 
|  | } | 
|  |  | 
|  | type mockLicensesLibraryProperties struct { | 
|  | Deps []string | 
|  | } | 
|  |  | 
|  | type mockLicensesLibraryModule struct { | 
|  | ModuleBase | 
|  | DefaultableModuleBase | 
|  | properties mockLicensesLibraryProperties | 
|  | } | 
|  |  | 
|  | func newMockLicensesLibraryModule() Module { | 
|  | m := &mockLicensesLibraryModule{} | 
|  | m.AddProperties(&m.properties) | 
|  | InitAndroidArchModule(m, HostAndDeviceSupported, MultilibCommon) | 
|  | InitDefaultableModule(m) | 
|  | return m | 
|  | } | 
|  |  | 
|  | type dependencyLicensesTag struct { | 
|  | blueprint.BaseDependencyTag | 
|  | name string | 
|  | } | 
|  |  | 
|  | func (j *mockLicensesLibraryModule) DepsMutator(ctx BottomUpMutatorContext) { | 
|  | ctx.AddVariationDependencies(nil, dependencyLicensesTag{name: "mockdeps"}, j.properties.Deps...) | 
|  | } | 
|  |  | 
|  | func (p *mockLicensesLibraryModule) GenerateAndroidBuildActions(ModuleContext) { | 
|  | } | 
|  |  | 
|  | type mockLicensesDefaults struct { | 
|  | ModuleBase | 
|  | DefaultsModuleBase | 
|  | } | 
|  |  | 
|  | func defaultsLicensesFactory() Module { | 
|  | m := &mockLicensesDefaults{} | 
|  | InitDefaultsModule(m) | 
|  | return m | 
|  | } |