| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 1 | // Copyright 2017 Google Inc. All rights reserved. | 
|  | 2 | // | 
|  | 3 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 4 | // you may not use this file except in compliance with the License. | 
|  | 5 | // You may obtain a copy of the License at | 
|  | 6 | // | 
|  | 7 | //     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 8 | // | 
|  | 9 | // Unless required by applicable law or agreed to in writing, software | 
|  | 10 | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 12 | // See the License for the specific language governing permissions and | 
|  | 13 | // limitations under the License. | 
|  | 14 |  | 
|  | 15 | package java | 
|  | 16 |  | 
|  | 17 | // Rules for instrumenting classes using jacoco | 
|  | 18 |  | 
|  | 19 | import ( | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 20 | "fmt" | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 21 | "path/filepath" | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 22 | "strings" | 
|  | 23 |  | 
|  | 24 | "github.com/google/blueprint" | 
| Nan Zhang | ffe2c1c | 2018-01-25 13:53:22 -0800 | [diff] [blame] | 25 | "github.com/google/blueprint/proptools" | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 26 |  | 
|  | 27 | "android/soong/android" | 
| Chris Gross | 2f74869 | 2020-06-24 20:36:59 +0000 | [diff] [blame] | 28 | "android/soong/java/config" | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 29 | ) | 
|  | 30 |  | 
|  | 31 | var ( | 
|  | 32 | jacoco = pctx.AndroidStaticRule("jacoco", blueprint.RuleParams{ | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 33 | Command: `rm -rf $tmpDir && mkdir -p $tmpDir && ` + | 
|  | 34 | `${config.Zip2ZipCmd} -i $in -o $strippedJar $stripSpec && ` + | 
| Sasha Smundak | 26c6d9e | 2019-06-11 13:30:13 -0700 | [diff] [blame] | 35 | `${config.JavaCmd} ${config.JavaVmFlags} -jar ${config.JacocoCLIJar} ` + | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 36 | `  instrument --quiet --dest $tmpDir $strippedJar && ` + | 
|  | 37 | `${config.Ziptime} $tmpJar && ` + | 
|  | 38 | `${config.MergeZipsCmd} --ignore-duplicates -j $out $tmpJar $in`, | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 39 | CommandDeps: []string{ | 
|  | 40 | "${config.Zip2ZipCmd}", | 
|  | 41 | "${config.JavaCmd}", | 
|  | 42 | "${config.JacocoCLIJar}", | 
|  | 43 | "${config.Ziptime}", | 
|  | 44 | "${config.MergeZipsCmd}", | 
|  | 45 | }, | 
|  | 46 | }, | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 47 | "strippedJar", "stripSpec", "tmpDir", "tmpJar") | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 48 | ) | 
|  | 49 |  | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 50 | // Instruments a jar using the Jacoco command line interface.  Uses stripSpec to extract a subset | 
|  | 51 | // of the classes in inputJar into strippedJar, instruments strippedJar into tmpJar, and then | 
|  | 52 | // combines the classes in tmpJar with inputJar (preferring the instrumented classes in tmpJar) | 
|  | 53 | // to produce instrumentedJar. | 
|  | 54 | func jacocoInstrumentJar(ctx android.ModuleContext, instrumentedJar, strippedJar android.WritablePath, | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 55 | inputJar android.Path, stripSpec string) { | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 56 |  | 
|  | 57 | // The basename of tmpJar has to be the same as the basename of strippedJar | 
|  | 58 | tmpJar := android.PathForModuleOut(ctx, "jacoco", "tmp", strippedJar.Base()) | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 59 |  | 
|  | 60 | ctx.Build(pctx, android.BuildParams{ | 
|  | 61 | Rule:           jacoco, | 
|  | 62 | Description:    "jacoco", | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 63 | Output:         instrumentedJar, | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 64 | ImplicitOutput: strippedJar, | 
|  | 65 | Input:          inputJar, | 
|  | 66 | Args: map[string]string{ | 
| Colin Cross | 84c3882 | 2018-01-03 15:59:46 -0800 | [diff] [blame] | 67 | "strippedJar": strippedJar.String(), | 
|  | 68 | "stripSpec":   stripSpec, | 
|  | 69 | "tmpDir":      filepath.Dir(tmpJar.String()), | 
|  | 70 | "tmpJar":      tmpJar.String(), | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 71 | }, | 
|  | 72 | }) | 
|  | 73 | } | 
|  | 74 |  | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 75 | func (j *Module) jacocoModuleToZipCommand(ctx android.ModuleContext) string { | 
|  | 76 | includes, err := jacocoFiltersToSpecs(j.properties.Jacoco.Include_filter) | 
|  | 77 | if err != nil { | 
|  | 78 | ctx.PropertyErrorf("jacoco.include_filter", "%s", err.Error()) | 
|  | 79 | } | 
| Chris Gross | 2f74869 | 2020-06-24 20:36:59 +0000 | [diff] [blame] | 80 | // Also include the default list of classes to exclude from instrumentation. | 
|  | 81 | excludes, err := jacocoFiltersToSpecs(append(j.properties.Jacoco.Exclude_filter, config.DefaultJacocoExcludeFilter...)) | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 82 | if err != nil { | 
|  | 83 | ctx.PropertyErrorf("jacoco.exclude_filter", "%s", err.Error()) | 
|  | 84 | } | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 85 |  | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 86 | return jacocoFiltersToZipCommand(includes, excludes) | 
|  | 87 | } | 
|  | 88 |  | 
|  | 89 | func jacocoFiltersToZipCommand(includes, excludes []string) string { | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 90 | specs := "" | 
|  | 91 | if len(excludes) > 0 { | 
| Colin Cross | d7deceb | 2017-12-19 13:59:44 -0800 | [diff] [blame] | 92 | specs += android.JoinWithPrefix(excludes, "-x ") + " " | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 93 | } | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 94 | if len(includes) > 0 { | 
|  | 95 | specs += strings.Join(includes, " ") | 
|  | 96 | } else { | 
|  | 97 | specs += "**/*.class" | 
|  | 98 | } | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 99 | return specs | 
|  | 100 | } | 
|  | 101 |  | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 102 | func jacocoFiltersToSpecs(filters []string) ([]string, error) { | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 103 | specs := make([]string, len(filters)) | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 104 | var err error | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 105 | for i, f := range filters { | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 106 | specs[i], err = jacocoFilterToSpec(f) | 
|  | 107 | if err != nil { | 
|  | 108 | return nil, err | 
|  | 109 | } | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 110 | } | 
| Colin Cross | 0b9f31f | 2019-02-28 11:00:01 -0800 | [diff] [blame] | 111 | return proptools.NinjaAndShellEscapeList(specs), nil | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 112 | } | 
|  | 113 |  | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 114 | func jacocoFilterToSpec(filter string) (string, error) { | 
| Nan Zhang | ffe2c1c | 2018-01-25 13:53:22 -0800 | [diff] [blame] | 115 | recursiveWildcard := strings.HasSuffix(filter, "**") | 
|  | 116 | nonRecursiveWildcard := false | 
|  | 117 | if !recursiveWildcard { | 
|  | 118 | nonRecursiveWildcard = strings.HasSuffix(filter, "*") | 
|  | 119 | filter = strings.TrimSuffix(filter, "*") | 
|  | 120 | } else { | 
|  | 121 | filter = strings.TrimSuffix(filter, "**") | 
|  | 122 | } | 
|  | 123 |  | 
|  | 124 | if recursiveWildcard && !(strings.HasSuffix(filter, ".") || filter == "") { | 
|  | 125 | return "", fmt.Errorf("only '**' or '.**' is supported as recursive wildcard in a filter") | 
|  | 126 | } | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 127 |  | 
|  | 128 | if strings.ContainsRune(filter, '*') { | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 129 | return "", fmt.Errorf("'*' is only supported as the last character in a filter") | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 130 | } | 
|  | 131 |  | 
|  | 132 | spec := strings.Replace(filter, ".", "/", -1) | 
|  | 133 |  | 
|  | 134 | if recursiveWildcard { | 
|  | 135 | spec += "**/*.class" | 
| Nan Zhang | ffe2c1c | 2018-01-25 13:53:22 -0800 | [diff] [blame] | 136 | } else if nonRecursiveWildcard { | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 137 | spec += "*.class" | 
| Colin Cross | d7deceb | 2017-12-19 13:59:44 -0800 | [diff] [blame] | 138 | } else { | 
|  | 139 | spec += ".class" | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 140 | } | 
|  | 141 |  | 
| Colin Cross | 7a3139e | 2017-12-19 13:57:50 -0800 | [diff] [blame] | 142 | return spec, nil | 
| Colin Cross | cb93359 | 2017-11-22 13:49:43 -0800 | [diff] [blame] | 143 | } |