Metalava has two different flags surrounding api-levels: - one for generating api-versions.xml to a file - one for applying api-versions.xml from a file Previously, soong always applied both of these arguments at the same time, such that framework-doc-stubs both generated and applied api-versions.xml. Add support for using api-versions.xml from another module name as well. Bug: 187398174 Test: droidstubs_test.go Change-Id: I8288fe4788336d5d5c60d09d48b00ca111449fba
		
			
				
	
	
		
			262 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			262 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2021 Google Inc. All rights reserved.
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package java
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"reflect"
 | |
| 	"regexp"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| )
 | |
| 
 | |
| func TestDroidstubs(t *testing.T) {
 | |
| 	ctx, _ := testJavaWithFS(t, `
 | |
| 		droiddoc_exported_dir {
 | |
| 			name: "droiddoc-templates-sdk",
 | |
| 			path: ".",
 | |
| 		}
 | |
| 
 | |
| 		droidstubs {
 | |
| 			name: "bar-stubs",
 | |
| 			srcs: ["bar-doc/a.java"],
 | |
| 			api_levels_annotations_dirs: ["droiddoc-templates-sdk"],
 | |
| 			api_levels_annotations_enabled: true,
 | |
| 		}
 | |
| 
 | |
| 		droidstubs {
 | |
| 			name: "bar-stubs-other",
 | |
| 			srcs: ["bar-doc/a.java"],
 | |
| 			high_mem: true,
 | |
| 			api_levels_annotations_dirs: ["droiddoc-templates-sdk"],
 | |
| 			api_levels_annotations_enabled: true,
 | |
| 			api_levels_jar_filename: "android.other.jar",
 | |
| 		}
 | |
| 
 | |
| 		droidstubs {
 | |
| 			name: "stubs-applying-api-versions",
 | |
| 			srcs: ["bar-doc/a.java"],
 | |
| 			api_levels_module: "bar-stubs-other",
 | |
| 		}
 | |
| 		`,
 | |
| 		map[string][]byte{
 | |
| 			"bar-doc/a.java": nil,
 | |
| 		})
 | |
| 	testcases := []struct {
 | |
| 		moduleName          string
 | |
| 		expectedJarFilename string
 | |
| 		generate_xml        bool
 | |
| 		high_mem            bool
 | |
| 	}{
 | |
| 		{
 | |
| 			moduleName:          "bar-stubs",
 | |
| 			generate_xml:        true,
 | |
| 			expectedJarFilename: "android.jar",
 | |
| 			high_mem:            false,
 | |
| 		},
 | |
| 		{
 | |
| 			moduleName:          "bar-stubs-other",
 | |
| 			generate_xml:        true,
 | |
| 			expectedJarFilename: "android.other.jar",
 | |
| 			high_mem:            true,
 | |
| 		},
 | |
| 		{
 | |
| 			moduleName:   "stubs-applying-api-versions",
 | |
| 			generate_xml: false,
 | |
| 		},
 | |
| 	}
 | |
| 	for _, c := range testcases {
 | |
| 		m := ctx.ModuleForTests(c.moduleName, "android_common")
 | |
| 		manifest := m.Output("metalava.sbox.textproto")
 | |
| 		sboxProto := android.RuleBuilderSboxProtoForTests(t, manifest)
 | |
| 		cmdline := String(sboxProto.Commands[0].Command)
 | |
| 		android.AssertStringContainsEquals(t, "api-versions generation flag", cmdline, "--generate-api-levels", c.generate_xml)
 | |
| 		if c.expectedJarFilename != "" {
 | |
| 			expected := "--android-jar-pattern ./%/public/" + c.expectedJarFilename
 | |
| 			if !strings.Contains(cmdline, expected) {
 | |
| 				t.Errorf("For %q, expected metalava argument %q, but was not found %q", c.moduleName, expected, cmdline)
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		metalava := m.Rule("metalava")
 | |
| 		rp := metalava.RuleParams
 | |
| 		if actual := rp.Pool != nil && strings.Contains(rp.Pool.String(), "highmem"); actual != c.high_mem {
 | |
| 			t.Errorf("Expected %q high_mem to be %v, was %v", c.moduleName, c.high_mem, actual)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // runs a test for droidstubs with a customizable sdkType argument and returns
 | |
| // the list of jar patterns that is passed as `--android-jar-pattern`
 | |
| func getAndroidJarPatternsForDroidstubs(t *testing.T, sdkType string) []string {
 | |
| 	ctx, _ := testJavaWithFS(t, fmt.Sprintf(`
 | |
| 		droiddoc_exported_dir {
 | |
| 			name: "some-exported-dir",
 | |
| 			path: "somedir",
 | |
| 		}
 | |
| 
 | |
| 		droiddoc_exported_dir {
 | |
| 			name: "some-other-exported-dir",
 | |
| 			path: "someotherdir",
 | |
| 		}
 | |
| 
 | |
| 		droidstubs {
 | |
| 			name: "foo-stubs",
 | |
| 			srcs: ["foo-doc/a.java"],
 | |
| 			api_levels_annotations_dirs: [
 | |
| 				"some-exported-dir",
 | |
| 				"some-other-exported-dir",
 | |
| 			],
 | |
| 			api_levels_annotations_enabled: true,
 | |
|       api_levels_sdk_type: "%s",
 | |
| 		}
 | |
| 		`, sdkType),
 | |
| 		map[string][]byte{
 | |
| 			"foo-doc/a.java": nil,
 | |
| 		})
 | |
| 
 | |
| 	m := ctx.ModuleForTests("foo-stubs", "android_common")
 | |
| 	manifest := m.Output("metalava.sbox.textproto")
 | |
| 	cmd := String(android.RuleBuilderSboxProtoForTests(t, manifest).Commands[0].Command)
 | |
| 	r := regexp.MustCompile(`--android-jar-pattern [^ ]+/android.jar`)
 | |
| 	return r.FindAllString(cmd, -1)
 | |
| }
 | |
| 
 | |
| func TestPublicDroidstubs(t *testing.T) {
 | |
| 	patterns := getAndroidJarPatternsForDroidstubs(t, "public")
 | |
| 
 | |
| 	android.AssertArrayString(t, "order of patterns", []string{
 | |
| 		"--android-jar-pattern somedir/%/public/android.jar",
 | |
| 		"--android-jar-pattern someotherdir/%/public/android.jar",
 | |
| 	}, patterns)
 | |
| }
 | |
| 
 | |
| func TestSystemDroidstubs(t *testing.T) {
 | |
| 	patterns := getAndroidJarPatternsForDroidstubs(t, "system")
 | |
| 
 | |
| 	android.AssertArrayString(t, "order of patterns", []string{
 | |
| 		"--android-jar-pattern somedir/%/system/android.jar",
 | |
| 		"--android-jar-pattern someotherdir/%/system/android.jar",
 | |
| 		"--android-jar-pattern somedir/%/public/android.jar",
 | |
| 		"--android-jar-pattern someotherdir/%/public/android.jar",
 | |
| 	}, patterns)
 | |
| }
 | |
| 
 | |
| func TestModuleLibDroidstubs(t *testing.T) {
 | |
| 	patterns := getAndroidJarPatternsForDroidstubs(t, "module-lib")
 | |
| 
 | |
| 	android.AssertArrayString(t, "order of patterns", []string{
 | |
| 		"--android-jar-pattern somedir/%/module-lib/android.jar",
 | |
| 		"--android-jar-pattern someotherdir/%/module-lib/android.jar",
 | |
| 		"--android-jar-pattern somedir/%/system/android.jar",
 | |
| 		"--android-jar-pattern someotherdir/%/system/android.jar",
 | |
| 		"--android-jar-pattern somedir/%/public/android.jar",
 | |
| 		"--android-jar-pattern someotherdir/%/public/android.jar",
 | |
| 	}, patterns)
 | |
| }
 | |
| 
 | |
| func TestDroidstubsSandbox(t *testing.T) {
 | |
| 	ctx, _ := testJavaWithFS(t, `
 | |
| 		genrule {
 | |
| 			name: "foo",
 | |
| 			out: ["foo.txt"],
 | |
| 			cmd: "touch $(out)",
 | |
| 		}
 | |
| 
 | |
| 		droidstubs {
 | |
| 			name: "bar-stubs",
 | |
| 			srcs: ["bar-doc/a.java"],
 | |
| 
 | |
| 			args: "--reference $(location :foo)",
 | |
| 			arg_files: [":foo"],
 | |
| 		}
 | |
| 		`,
 | |
| 		map[string][]byte{
 | |
| 			"bar-doc/a.java": nil,
 | |
| 		})
 | |
| 
 | |
| 	m := ctx.ModuleForTests("bar-stubs", "android_common")
 | |
| 	metalava := m.Rule("metalava")
 | |
| 	if g, w := metalava.Inputs.Strings(), []string{"bar-doc/a.java"}; !reflect.DeepEqual(w, g) {
 | |
| 		t.Errorf("Expected inputs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	manifest := android.RuleBuilderSboxProtoForTests(t, m.Output("metalava.sbox.textproto"))
 | |
| 	if g, w := manifest.Commands[0].GetCommand(), "reference __SBOX_SANDBOX_DIR__/out/.intermediates/foo/gen/foo.txt"; !strings.Contains(g, w) {
 | |
| 		t.Errorf("Expected command to contain %q, got %q", w, g)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDroidstubsWithSystemModules(t *testing.T) {
 | |
| 	ctx, _ := testJava(t, `
 | |
| 		droidstubs {
 | |
| 		    name: "stubs-source-system-modules",
 | |
| 		    srcs: [
 | |
| 		        "bar-doc/a.java",
 | |
| 		    ],
 | |
| 				sdk_version: "none",
 | |
| 				system_modules: "source-system-modules",
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 				name: "source-jar",
 | |
| 		    srcs: [
 | |
| 		        "a.java",
 | |
| 		    ],
 | |
| 		}
 | |
| 
 | |
| 		java_system_modules {
 | |
| 				name: "source-system-modules",
 | |
| 				libs: ["source-jar"],
 | |
| 		}
 | |
| 
 | |
| 		droidstubs {
 | |
| 		    name: "stubs-prebuilt-system-modules",
 | |
| 		    srcs: [
 | |
| 		        "bar-doc/a.java",
 | |
| 		    ],
 | |
| 				sdk_version: "none",
 | |
| 				system_modules: "prebuilt-system-modules",
 | |
| 		}
 | |
| 
 | |
| 		java_import {
 | |
| 				name: "prebuilt-jar",
 | |
| 				jars: ["a.jar"],
 | |
| 		}
 | |
| 
 | |
| 		java_system_modules_import {
 | |
| 				name: "prebuilt-system-modules",
 | |
| 				libs: ["prebuilt-jar"],
 | |
| 		}
 | |
| 		`)
 | |
| 
 | |
| 	checkSystemModulesUseByDroidstubs(t, ctx, "stubs-source-system-modules", "source-jar.jar")
 | |
| 
 | |
| 	checkSystemModulesUseByDroidstubs(t, ctx, "stubs-prebuilt-system-modules", "prebuilt-jar.jar")
 | |
| }
 | |
| 
 | |
| func checkSystemModulesUseByDroidstubs(t *testing.T, ctx *android.TestContext, moduleName string, systemJar string) {
 | |
| 	metalavaRule := ctx.ModuleForTests(moduleName, "android_common").Rule("metalava")
 | |
| 	var systemJars []string
 | |
| 	for _, i := range metalavaRule.Implicits {
 | |
| 		systemJars = append(systemJars, i.Base())
 | |
| 	}
 | |
| 	if len(systemJars) < 1 || systemJars[0] != systemJar {
 | |
| 		t.Errorf("inputs of %q must be []string{%q}, but was %#v.", moduleName, systemJar, systemJars)
 | |
| 	}
 | |
| }
 |