|  | // Copyright 2015 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 android | 
|  |  | 
|  | import ( | 
|  | "errors" | 
|  | "fmt" | 
|  | "reflect" | 
|  | "strings" | 
|  | "testing" | 
|  |  | 
|  | "github.com/google/blueprint/pathtools" | 
|  | ) | 
|  |  | 
|  | type strsTestCase struct { | 
|  | in  []string | 
|  | out string | 
|  | err []error | 
|  | } | 
|  |  | 
|  | var commonValidatePathTestCases = []strsTestCase{ | 
|  | { | 
|  | in:  []string{""}, | 
|  | out: "", | 
|  | }, | 
|  | { | 
|  | in:  []string{"a/b"}, | 
|  | out: "a/b", | 
|  | }, | 
|  | { | 
|  | in:  []string{"a/b", "c"}, | 
|  | out: "a/b/c", | 
|  | }, | 
|  | { | 
|  | in:  []string{"a/.."}, | 
|  | out: ".", | 
|  | }, | 
|  | { | 
|  | in:  []string{"."}, | 
|  | out: ".", | 
|  | }, | 
|  | { | 
|  | in:  []string{".."}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: ..")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"../a"}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: ../a")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"b/../../a"}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: ../a")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"/a"}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: /a")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"a", "../b"}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: ../b")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"a", "b/../../c"}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: ../c")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"a", "./.."}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path is outside directory: ..")}, | 
|  | }, | 
|  | } | 
|  |  | 
|  | var validateSafePathTestCases = append(commonValidatePathTestCases, []strsTestCase{ | 
|  | { | 
|  | in:  []string{"$host/../$a"}, | 
|  | out: "$a", | 
|  | }, | 
|  | }...) | 
|  |  | 
|  | var validatePathTestCases = append(commonValidatePathTestCases, []strsTestCase{ | 
|  | { | 
|  | in:  []string{"$host/../$a"}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path contains invalid character($): $host/../$a")}, | 
|  | }, | 
|  | { | 
|  | in:  []string{"$host/.."}, | 
|  | out: "", | 
|  | err: []error{errors.New("Path contains invalid character($): $host/..")}, | 
|  | }, | 
|  | }...) | 
|  |  | 
|  | func TestValidateSafePath(t *testing.T) { | 
|  | for _, testCase := range validateSafePathTestCases { | 
|  | t.Run(strings.Join(testCase.in, ","), func(t *testing.T) { | 
|  | ctx := &configErrorWrapper{} | 
|  | out, err := validateSafePath(testCase.in...) | 
|  | if err != nil { | 
|  | reportPathError(ctx, err) | 
|  | } | 
|  | check(t, "validateSafePath", p(testCase.in), out, ctx.errors, testCase.out, testCase.err) | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestValidatePath(t *testing.T) { | 
|  | for _, testCase := range validatePathTestCases { | 
|  | t.Run(strings.Join(testCase.in, ","), func(t *testing.T) { | 
|  | ctx := &configErrorWrapper{} | 
|  | out, err := validatePath(testCase.in...) | 
|  | if err != nil { | 
|  | reportPathError(ctx, err) | 
|  | } | 
|  | check(t, "validatePath", p(testCase.in), out, ctx.errors, testCase.out, testCase.err) | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestOptionalPath(t *testing.T) { | 
|  | var path OptionalPath | 
|  | checkInvalidOptionalPath(t, path) | 
|  |  | 
|  | path = OptionalPathForPath(nil) | 
|  | checkInvalidOptionalPath(t, path) | 
|  | } | 
|  |  | 
|  | func checkInvalidOptionalPath(t *testing.T, path OptionalPath) { | 
|  | t.Helper() | 
|  | if path.Valid() { | 
|  | t.Errorf("Uninitialized OptionalPath should not be valid") | 
|  | } | 
|  | if path.String() != "" { | 
|  | t.Errorf("Uninitialized OptionalPath String() should return \"\", not %q", path.String()) | 
|  | } | 
|  | defer func() { | 
|  | if r := recover(); r == nil { | 
|  | t.Errorf("Expected a panic when calling Path() on an uninitialized OptionalPath") | 
|  | } | 
|  | }() | 
|  | path.Path() | 
|  | } | 
|  |  | 
|  | func check(t *testing.T, testType, testString string, | 
|  | got interface{}, err []error, | 
|  | expected interface{}, expectedErr []error) { | 
|  | t.Helper() | 
|  |  | 
|  | printedTestCase := false | 
|  | e := func(s string, expected, got interface{}) { | 
|  | t.Helper() | 
|  | if !printedTestCase { | 
|  | t.Errorf("test case %s: %s", testType, testString) | 
|  | printedTestCase = true | 
|  | } | 
|  | t.Errorf("incorrect %s", s) | 
|  | t.Errorf("  expected: %s", p(expected)) | 
|  | t.Errorf("       got: %s", p(got)) | 
|  | } | 
|  |  | 
|  | if !reflect.DeepEqual(expectedErr, err) { | 
|  | e("errors:", expectedErr, err) | 
|  | } | 
|  |  | 
|  | if !reflect.DeepEqual(expected, got) { | 
|  | e("output:", expected, got) | 
|  | } | 
|  | } | 
|  |  | 
|  | func p(in interface{}) string { | 
|  | if v, ok := in.([]interface{}); ok { | 
|  | s := make([]string, len(v)) | 
|  | for i := range v { | 
|  | s[i] = fmt.Sprintf("%#v", v[i]) | 
|  | } | 
|  | return "[" + strings.Join(s, ", ") + "]" | 
|  | } else { | 
|  | return fmt.Sprintf("%#v", in) | 
|  | } | 
|  | } | 
|  |  | 
|  | type moduleInstallPathContextImpl struct { | 
|  | androidBaseContextImpl | 
|  |  | 
|  | inData         bool | 
|  | inSanitizerDir bool | 
|  | inRecovery     bool | 
|  | } | 
|  |  | 
|  | func (moduleInstallPathContextImpl) Fs() pathtools.FileSystem { | 
|  | return pathtools.MockFs(nil) | 
|  | } | 
|  |  | 
|  | func (m moduleInstallPathContextImpl) Config() Config { | 
|  | return m.androidBaseContextImpl.config | 
|  | } | 
|  |  | 
|  | func (moduleInstallPathContextImpl) AddNinjaFileDeps(deps ...string) {} | 
|  |  | 
|  | func (m moduleInstallPathContextImpl) InstallInData() bool { | 
|  | return m.inData | 
|  | } | 
|  |  | 
|  | func (m moduleInstallPathContextImpl) InstallInSanitizerDir() bool { | 
|  | return m.inSanitizerDir | 
|  | } | 
|  |  | 
|  | func (m moduleInstallPathContextImpl) InstallInRecovery() bool { | 
|  | return m.inRecovery | 
|  | } | 
|  |  | 
|  | func TestPathForModuleInstall(t *testing.T) { | 
|  | testConfig := TestConfig("", nil) | 
|  |  | 
|  | hostTarget := Target{Os: Linux} | 
|  | deviceTarget := Target{Os: Android} | 
|  |  | 
|  | testCases := []struct { | 
|  | name string | 
|  | ctx  *moduleInstallPathContextImpl | 
|  | in   []string | 
|  | out  string | 
|  | }{ | 
|  | { | 
|  | name: "host binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: hostTarget, | 
|  | }, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "host/linux-x86/bin/my_test", | 
|  | }, | 
|  |  | 
|  | { | 
|  | name: "system binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | }, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/system/bin/my_test", | 
|  | }, | 
|  | { | 
|  | name: "vendor binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   socSpecificModule, | 
|  | }, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/vendor/bin/my_test", | 
|  | }, | 
|  | { | 
|  | name: "odm binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   deviceSpecificModule, | 
|  | }, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/odm/bin/my_test", | 
|  | }, | 
|  | { | 
|  | name: "product binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   productSpecificModule, | 
|  | }, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/product/bin/my_test", | 
|  | }, | 
|  |  | 
|  | { | 
|  | name: "system native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | }, | 
|  | inData: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/nativetest/my_test", | 
|  | }, | 
|  | { | 
|  | name: "vendor native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   socSpecificModule, | 
|  | }, | 
|  | inData: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/nativetest/my_test", | 
|  | }, | 
|  | { | 
|  | name: "odm native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   deviceSpecificModule, | 
|  | }, | 
|  | inData: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/nativetest/my_test", | 
|  | }, | 
|  | { | 
|  | name: "product native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   productSpecificModule, | 
|  | }, | 
|  | inData: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/nativetest/my_test", | 
|  | }, | 
|  |  | 
|  | { | 
|  | name: "sanitized system binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | }, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/system/bin/my_test", | 
|  | }, | 
|  | { | 
|  | name: "sanitized vendor binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   socSpecificModule, | 
|  | }, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/vendor/bin/my_test", | 
|  | }, | 
|  | { | 
|  | name: "sanitized odm binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   deviceSpecificModule, | 
|  | }, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/odm/bin/my_test", | 
|  | }, | 
|  | { | 
|  | name: "sanitized product binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   productSpecificModule, | 
|  | }, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"bin", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/product/bin/my_test", | 
|  | }, | 
|  |  | 
|  | { | 
|  | name: "sanitized system native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | }, | 
|  | inData:         true, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/data/nativetest/my_test", | 
|  | }, | 
|  | { | 
|  | name: "sanitized vendor native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   socSpecificModule, | 
|  | }, | 
|  | inData:         true, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/data/nativetest/my_test", | 
|  | }, | 
|  | { | 
|  | name: "sanitized odm native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   deviceSpecificModule, | 
|  | }, | 
|  | inData:         true, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/data/nativetest/my_test", | 
|  | }, | 
|  | { | 
|  | name: "sanitized product native test binary", | 
|  | ctx: &moduleInstallPathContextImpl{ | 
|  | androidBaseContextImpl: androidBaseContextImpl{ | 
|  | target: deviceTarget, | 
|  | kind:   productSpecificModule, | 
|  | }, | 
|  | inData:         true, | 
|  | inSanitizerDir: true, | 
|  | }, | 
|  | in:  []string{"nativetest", "my_test"}, | 
|  | out: "target/product/test_device/data/asan/data/nativetest/my_test", | 
|  | }, | 
|  | } | 
|  |  | 
|  | for _, tc := range testCases { | 
|  | t.Run(tc.name, func(t *testing.T) { | 
|  | tc.ctx.androidBaseContextImpl.config = testConfig | 
|  | output := PathForModuleInstall(tc.ctx, tc.in...) | 
|  | if output.basePath.path != tc.out { | 
|  | t.Errorf("unexpected path:\n got: %q\nwant: %q\n", | 
|  | output.basePath.path, | 
|  | tc.out) | 
|  | } | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestDirectorySortedPaths(t *testing.T) { | 
|  | makePaths := func() Paths { | 
|  | return Paths{ | 
|  | PathForTesting("a.txt"), | 
|  | PathForTesting("a/txt"), | 
|  | PathForTesting("a/b/c"), | 
|  | PathForTesting("a/b/d"), | 
|  | PathForTesting("b"), | 
|  | PathForTesting("b/b.txt"), | 
|  | PathForTesting("a/a.txt"), | 
|  | } | 
|  | } | 
|  |  | 
|  | expected := []string{ | 
|  | "a.txt", | 
|  | "a/a.txt", | 
|  | "a/b/c", | 
|  | "a/b/d", | 
|  | "a/txt", | 
|  | "b", | 
|  | "b/b.txt", | 
|  | } | 
|  |  | 
|  | paths := makePaths() | 
|  | reversePaths := ReversePaths(paths) | 
|  |  | 
|  | sortedPaths := PathsToDirectorySortedPaths(paths) | 
|  | reverseSortedPaths := PathsToDirectorySortedPaths(reversePaths) | 
|  |  | 
|  | if !reflect.DeepEqual(Paths(sortedPaths).Strings(), expected) { | 
|  | t.Fatalf("sorted paths:\n %#v\n != \n %#v", paths.Strings(), expected) | 
|  | } | 
|  |  | 
|  | if !reflect.DeepEqual(Paths(reverseSortedPaths).Strings(), expected) { | 
|  | t.Fatalf("sorted reversed paths:\n %#v\n !=\n %#v", reversePaths.Strings(), expected) | 
|  | } | 
|  |  | 
|  | expectedA := []string{ | 
|  | "a/a.txt", | 
|  | "a/b/c", | 
|  | "a/b/d", | 
|  | "a/txt", | 
|  | } | 
|  |  | 
|  | inA := sortedPaths.PathsInDirectory("a") | 
|  | if !reflect.DeepEqual(inA.Strings(), expectedA) { | 
|  | t.Errorf("FilesInDirectory(a):\n %#v\n != \n %#v", inA.Strings(), expectedA) | 
|  | } | 
|  |  | 
|  | expectedA_B := []string{ | 
|  | "a/b/c", | 
|  | "a/b/d", | 
|  | } | 
|  |  | 
|  | inA_B := sortedPaths.PathsInDirectory("a/b") | 
|  | if !reflect.DeepEqual(inA_B.Strings(), expectedA_B) { | 
|  | t.Errorf("FilesInDirectory(a/b):\n %#v\n != \n %#v", inA_B.Strings(), expectedA_B) | 
|  | } | 
|  |  | 
|  | expectedB := []string{ | 
|  | "b/b.txt", | 
|  | } | 
|  |  | 
|  | inB := sortedPaths.PathsInDirectory("b") | 
|  | if !reflect.DeepEqual(inB.Strings(), expectedB) { | 
|  | t.Errorf("FilesInDirectory(b):\n %#v\n != \n %#v", inA.Strings(), expectedA) | 
|  | } | 
|  | } |