blob: 55ccb7bbe26549f2ac4b7928661194a78c407d51 [file] [log] [blame]
Dan Willemsen2902fa72017-04-27 21:16:35 -07001// 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
15package main
16
17import (
Colin Cross9f5633d2018-04-19 16:52:21 -070018 "bufio"
19 "bytes"
Dan Willemsen2902fa72017-04-27 21:16:35 -070020 "encoding/xml"
21 "flag"
22 "fmt"
Dan Willemsen2902fa72017-04-27 21:16:35 -070023 "io/ioutil"
24 "os"
Colin Cross9f5633d2018-04-19 16:52:21 -070025 "os/exec"
Dan Willemsen2902fa72017-04-27 21:16:35 -070026 "path/filepath"
27 "regexp"
28 "sort"
29 "strings"
30 "text/template"
31
32 "github.com/google/blueprint/proptools"
33)
34
35type RewriteNames []RewriteName
36type RewriteName struct {
37 regexp *regexp.Regexp
38 repl string
39}
40
41func (r *RewriteNames) String() string {
42 return ""
43}
44
45func (r *RewriteNames) Set(v string) error {
46 split := strings.SplitN(v, "=", 2)
47 if len(split) != 2 {
48 return fmt.Errorf("Must be in the form of <regex>=<replace>")
49 }
50 regex, err := regexp.Compile(split[0])
51 if err != nil {
52 return nil
53 }
54 *r = append(*r, RewriteName{
55 regexp: regex,
56 repl: split[1],
57 })
58 return nil
59}
60
Alan Viverette75b95f82017-12-04 16:24:07 -050061func (r *RewriteNames) MavenToMk(groupId string, artifactId string) string {
Dan Willemsen2902fa72017-04-27 21:16:35 -070062 for _, r := range *r {
Alan Viverette75b95f82017-12-04 16:24:07 -050063 if r.regexp.MatchString(groupId + ":" + artifactId) {
64 return r.regexp.ReplaceAllString(groupId+":"+artifactId, r.repl)
65 } else if r.regexp.MatchString(artifactId) {
66 return r.regexp.ReplaceAllString(artifactId, r.repl)
Dan Willemsen2902fa72017-04-27 21:16:35 -070067 }
68 }
Alan Viverette75b95f82017-12-04 16:24:07 -050069 return artifactId
Dan Willemsen2902fa72017-04-27 21:16:35 -070070}
71
72var rewriteNames = RewriteNames{}
73
74type ExtraDeps map[string][]string
75
76func (d ExtraDeps) String() string {
77 return ""
78}
79
80func (d ExtraDeps) Set(v string) error {
81 split := strings.SplitN(v, "=", 2)
82 if len(split) != 2 {
83 return fmt.Errorf("Must be in the form of <module>=<module>[,<module>]")
84 }
85 d[split[0]] = strings.Split(split[1], ",")
86 return nil
87}
88
89var extraDeps = make(ExtraDeps)
90
Dan Willemsen15a8e792017-11-10 14:02:44 -080091var sdkVersion string
Dan Willemsen47e44a42017-10-05 13:28:16 -070092var useVersion string
Alan Viverette1593d3d2017-12-11 17:14:26 -050093var staticDeps bool
Dan Willemsen47e44a42017-10-05 13:28:16 -070094
Jeff Gaston8f084822018-03-29 14:58:50 -040095func InList(s string, list []string) bool {
96 for _, l := range list {
97 if l == s {
98 return true
99 }
100 }
101
102 return false
103}
104
Dan Willemsen2902fa72017-04-27 21:16:35 -0700105type Dependency struct {
106 XMLName xml.Name `xml:"dependency"`
107
Alan Viverette1593d3d2017-12-11 17:14:26 -0500108 MakeTarget string `xml:"-"`
109
Dan Willemsen2902fa72017-04-27 21:16:35 -0700110 GroupId string `xml:"groupId"`
111 ArtifactId string `xml:"artifactId"`
112 Version string `xml:"version"`
113 Type string `xml:"type"`
Alan Viverette6bd35eb2018-02-13 13:25:24 -0500114 Scope string `xml:"scope"`
Dan Willemsen2902fa72017-04-27 21:16:35 -0700115}
116
Alan Viverette1593d3d2017-12-11 17:14:26 -0500117func (d Dependency) MkName() string {
118 if d.MakeTarget == "" {
119 d.MakeTarget = rewriteNames.MavenToMk(d.GroupId, d.ArtifactId)
120 }
121 return d.MakeTarget
122}
123
Dan Willemsen2902fa72017-04-27 21:16:35 -0700124type Pom struct {
125 XMLName xml.Name `xml:"http://maven.apache.org/POM/4.0.0 project"`
126
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700127 PomFile string `xml:"-"`
Dan Willemsen2902fa72017-04-27 21:16:35 -0700128 ArtifactFile string `xml:"-"`
Alan Viverette75b95f82017-12-04 16:24:07 -0500129 MakeTarget string `xml:"-"`
Dan Willemsen2902fa72017-04-27 21:16:35 -0700130
131 GroupId string `xml:"groupId"`
132 ArtifactId string `xml:"artifactId"`
133 Version string `xml:"version"`
134 Packaging string `xml:"packaging"`
135
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700136 Dependencies []*Dependency `xml:"dependencies>dependency"`
Dan Willemsen2902fa72017-04-27 21:16:35 -0700137}
138
Alan Viverette1593d3d2017-12-11 17:14:26 -0500139func (p Pom) IsAar() bool {
140 return p.Packaging == "aar"
141}
142
143func (p Pom) IsJar() bool {
144 return p.Packaging == "jar"
145}
146
Dan Willemsen2902fa72017-04-27 21:16:35 -0700147func (p Pom) MkName() string {
Alan Viverette75b95f82017-12-04 16:24:07 -0500148 if p.MakeTarget == "" {
149 p.MakeTarget = rewriteNames.MavenToMk(p.GroupId, p.ArtifactId)
150 }
151 return p.MakeTarget
Dan Willemsen2902fa72017-04-27 21:16:35 -0700152}
153
Alan Viverette1593d3d2017-12-11 17:14:26 -0500154func (p Pom) MkJarDeps() []string {
Jeff Gaston8f084822018-03-29 14:58:50 -0400155 return p.MkDeps("jar", []string{"compile", "runtime"})
Alan Viverette1593d3d2017-12-11 17:14:26 -0500156}
157
158func (p Pom) MkAarDeps() []string {
Jeff Gaston8f084822018-03-29 14:58:50 -0400159 return p.MkDeps("aar", []string{"compile", "runtime"})
Alan Viverette1593d3d2017-12-11 17:14:26 -0500160}
161
Alan Viverette6bd35eb2018-02-13 13:25:24 -0500162// MkDeps obtains dependencies filtered by type and scope. The results of this
163// method are formatted as Make targets, e.g. run through MavenToMk rules.
Jeff Gaston8f084822018-03-29 14:58:50 -0400164func (p Pom) MkDeps(typeExt string, scopes []string) []string {
Dan Willemsen2902fa72017-04-27 21:16:35 -0700165 var ret []string
Jeff Gaston358f25e2018-04-23 15:32:19 -0400166 if typeExt == "jar" {
167 // all top-level extra deps are assumed to be of type "jar" until we add syntax to specify other types
168 ret = append(ret, extraDeps[p.MkName()]...)
169 }
Dan Willemsen2902fa72017-04-27 21:16:35 -0700170 for _, d := range p.Dependencies {
Jeff Gaston8f084822018-03-29 14:58:50 -0400171 if d.Type != typeExt || !InList(d.Scope, scopes) {
Dan Willemsen2902fa72017-04-27 21:16:35 -0700172 continue
173 }
Alan Viverette75b95f82017-12-04 16:24:07 -0500174 name := rewriteNames.MavenToMk(d.GroupId, d.ArtifactId)
Dan Willemsen2902fa72017-04-27 21:16:35 -0700175 ret = append(ret, name)
176 ret = append(ret, extraDeps[name]...)
177 }
178 return ret
179}
180
Dan Willemsen15a8e792017-11-10 14:02:44 -0800181func (p Pom) SdkVersion() string {
182 return sdkVersion
183}
184
Alan Viverette6bd35eb2018-02-13 13:25:24 -0500185func (p *Pom) FixDeps(modules map[string]*Pom) {
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700186 for _, d := range p.Dependencies {
Alan Viverette6bd35eb2018-02-13 13:25:24 -0500187 if d.Type == "" {
188 if depPom, ok := modules[d.MkName()]; ok {
189 // We've seen the POM for this dependency, use its packaging
190 // as the dependency type rather than Maven spec default.
191 d.Type = depPom.Packaging
192 } else {
193 // Dependency type was not specified and we don't have the POM
194 // for this artifact, use the default from Maven spec.
195 d.Type = "jar"
196 }
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700197 }
Alan Viverette6bd35eb2018-02-13 13:25:24 -0500198 if d.Scope == "" {
199 // Scope was not specified, use the default from Maven spec.
200 d.Scope = "compile"
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700201 }
202 }
203}
204
Dan Willemsen2902fa72017-04-27 21:16:35 -0700205var mkTemplate = template.Must(template.New("mk").Parse(`
206include $(CLEAR_VARS)
207LOCAL_MODULE := {{.MkName}}
208LOCAL_MODULE_CLASS := JAVA_LIBRARIES
209LOCAL_UNINSTALLABLE_MODULE := true
210LOCAL_SRC_FILES := {{.ArtifactFile}}
211LOCAL_BUILT_MODULE_STEM := javalib.jar
212LOCAL_MODULE_SUFFIX := .{{.Packaging}}
213LOCAL_USE_AAPT2 := true
Dan Willemsen15a8e792017-11-10 14:02:44 -0800214LOCAL_SDK_VERSION := {{.SdkVersion}}
Alan Viverette2b53a0c2018-03-28 10:32:10 -0400215LOCAL_STATIC_JAVA_LIBRARIES :={{range .MkJarDeps}} \
216 {{.}}{{end}}
217LOCAL_STATIC_ANDROID_LIBRARIES :={{range .MkAarDeps}} \
218 {{.}}{{end}}
Dan Willemsen2902fa72017-04-27 21:16:35 -0700219include $(BUILD_PREBUILT)
220`))
221
Alan Viverette1593d3d2017-12-11 17:14:26 -0500222var mkDepsTemplate = template.Must(template.New("mk").Parse(`
223include $(CLEAR_VARS)
224LOCAL_MODULE := {{.MkName}}-nodeps
225LOCAL_MODULE_CLASS := JAVA_LIBRARIES
226LOCAL_UNINSTALLABLE_MODULE := true
227LOCAL_SRC_FILES := {{.ArtifactFile}}
228LOCAL_BUILT_MODULE_STEM := javalib.jar
229LOCAL_MODULE_SUFFIX := .{{.Packaging}}
230LOCAL_USE_AAPT2 := true
231LOCAL_SDK_VERSION := {{.SdkVersion}}
232LOCAL_STATIC_ANDROID_LIBRARIES :={{range .MkAarDeps}} \
233 {{.}}{{end}}
234include $(BUILD_PREBUILT)
235include $(CLEAR_VARS)
236LOCAL_MODULE := {{.MkName}}
237LOCAL_SDK_VERSION := {{.SdkVersion}}{{if .IsAar}}
238LOCAL_MANIFEST_FILE := manifests/{{.MkName}}/AndroidManifest.xml{{end}}
239LOCAL_STATIC_JAVA_LIBRARIES :={{if .IsJar}} \
240 {{.MkName}}-nodeps{{end}}{{range .MkJarDeps}} \
241 {{.}}{{end}}
242LOCAL_STATIC_ANDROID_LIBRARIES :={{if .IsAar}} \
243 {{.MkName}}-nodeps{{end}}{{range .MkAarDeps}} \
244 {{.}}{{end}}
245LOCAL_JAR_EXCLUDE_FILES := none
246LOCAL_JAVA_LANGUAGE_VERSION := 1.7
247LOCAL_USE_AAPT2 := true
248include $(BUILD_STATIC_JAVA_LIBRARY)
249`))
250
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700251func parse(filename string) (*Pom, error) {
Dan Willemsen2902fa72017-04-27 21:16:35 -0700252 data, err := ioutil.ReadFile(filename)
253 if err != nil {
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700254 return nil, err
Dan Willemsen2902fa72017-04-27 21:16:35 -0700255 }
256
257 var pom Pom
258 err = xml.Unmarshal(data, &pom)
259 if err != nil {
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700260 return nil, err
Dan Willemsen2902fa72017-04-27 21:16:35 -0700261 }
262
Dan Willemsen47e44a42017-10-05 13:28:16 -0700263 if useVersion != "" && pom.Version != useVersion {
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700264 return nil, nil
Dan Willemsen47e44a42017-10-05 13:28:16 -0700265 }
266
Dan Willemsen2902fa72017-04-27 21:16:35 -0700267 if pom.Packaging == "" {
268 pom.Packaging = "jar"
269 }
270
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700271 pom.PomFile = filename
Dan Willemsen2902fa72017-04-27 21:16:35 -0700272 pom.ArtifactFile = strings.TrimSuffix(filename, ".pom") + "." + pom.Packaging
273
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700274 return &pom, nil
Dan Willemsen2902fa72017-04-27 21:16:35 -0700275}
276
Colin Cross9f5633d2018-04-19 16:52:21 -0700277func rerunForRegen(filename string) error {
278 buf, err := ioutil.ReadFile(filename)
279 if err != nil {
280 return err
281 }
282
283 scanner := bufio.NewScanner(bytes.NewBuffer(buf))
284
285 // Skip the first line in the file
286 for i := 0; i < 2; i++ {
287 if !scanner.Scan() {
288 if scanner.Err() != nil {
289 return scanner.Err()
290 } else {
291 return fmt.Errorf("unexpected EOF")
292 }
293 }
294 }
295
296 // Extract the old args from the file
297 line := scanner.Text()
298 if strings.HasPrefix(line, "# pom2mk ") {
299 line = strings.TrimPrefix(line, "# pom2mk ")
300 } else {
301 return fmt.Errorf("unexpected second line: %q", line)
302 }
303 args := strings.Split(line, " ")
304 lastArg := args[len(args)-1]
305 args = args[:len(args)-1]
306
307 // Append all current command line args except -regen <file> to the ones from the file
308 for i := 1; i < len(os.Args); i++ {
309 if os.Args[i] == "-regen" {
310 i++
311 } else {
312 args = append(args, os.Args[i])
313 }
314 }
315 args = append(args, lastArg)
316
317 cmd := os.Args[0] + " " + strings.Join(args, " ")
318 // Re-exec pom2mk with the new arguments
319 output, err := exec.Command("/bin/sh", "-c", cmd).Output()
320 if exitErr, _ := err.(*exec.ExitError); exitErr != nil {
321 return fmt.Errorf("failed to run %s\n%s", cmd, string(exitErr.Stderr))
322 } else if err != nil {
323 return err
324 }
325
326 return ioutil.WriteFile(filename, output, 0666)
327}
328
Dan Willemsen2902fa72017-04-27 21:16:35 -0700329func main() {
330 flag.Usage = func() {
331 fmt.Fprintf(os.Stderr, `pom2mk, a tool to create Android.mk files from maven repos
332
333The tool will extract the necessary information from *.pom files to create an Android.mk whose
334aar libraries can be linked against when using AAPT2.
335
Colin Cross9f5633d2018-04-19 16:52:21 -0700336Usage: %s [--rewrite <regex>=<replace>] [--extra-deps <module>=<module>[,<module>]] [<dir>] [-regen <file>]
Dan Willemsen2902fa72017-04-27 21:16:35 -0700337
338 -rewrite <regex>=<replace>
Alan Viverette75b95f82017-12-04 16:24:07 -0500339 rewrite can be used to specify mappings between Maven projects and Make modules. The -rewrite
340 option can be specified multiple times. When determining the Make module for a given Maven
341 project, mappings are searched in the order they were specified. The first <regex> matching
342 either the Maven project's <groupId>:<artifactId> or <artifactId> will be used to generate
343 the Make module name using <replace>. If no matches are found, <artifactId> is used.
Dan Willemsen2902fa72017-04-27 21:16:35 -0700344 -extra-deps <module>=<module>[,<module>]
345 Some Android.mk modules have transitive dependencies that must be specified when they are
346 depended upon (like android-support-v7-mediarouter requires android-support-v7-appcompat).
347 This may be specified multiple times to declare these dependencies.
Dan Willemsen15a8e792017-11-10 14:02:44 -0800348 -sdk-version <version>
349 Sets LOCAL_SDK_VERSION := <version> for all modules.
Dan Willemsen47e44a42017-10-05 13:28:16 -0700350 -use-version <version>
351 If the maven directory contains multiple versions of artifacts and their pom files,
352 -use-version can be used to only write makefiles for a specific version of those artifacts.
Alan Viverette1593d3d2017-12-11 17:14:26 -0500353 -static-deps
354 Whether to statically include direct dependencies.
Dan Willemsen2902fa72017-04-27 21:16:35 -0700355 <dir>
356 The directory to search for *.pom files under.
Colin Cross9f5633d2018-04-19 16:52:21 -0700357 The makefile is written to stdout, to be put in the current directory (often as Android.mk)
358 -regen <file>
359 Read arguments from <file> and overwrite it.
Dan Willemsen2902fa72017-04-27 21:16:35 -0700360`, os.Args[0])
361 }
362
Colin Cross9f5633d2018-04-19 16:52:21 -0700363 var regen string
364
Dan Willemsen2902fa72017-04-27 21:16:35 -0700365 flag.Var(&extraDeps, "extra-deps", "Extra dependencies needed when depending on a module")
366 flag.Var(&rewriteNames, "rewrite", "Regex(es) to rewrite artifact names")
Dan Willemsen15a8e792017-11-10 14:02:44 -0800367 flag.StringVar(&sdkVersion, "sdk-version", "", "What to write to LOCAL_SDK_VERSION")
Dan Willemsen47e44a42017-10-05 13:28:16 -0700368 flag.StringVar(&useVersion, "use-version", "", "Only read artifacts of a specific version")
Alan Viverette1593d3d2017-12-11 17:14:26 -0500369 flag.BoolVar(&staticDeps, "static-deps", false, "Statically include direct dependencies")
Colin Cross9f5633d2018-04-19 16:52:21 -0700370 flag.StringVar(&regen, "regen", "", "Rewrite specified file")
Dan Willemsen2902fa72017-04-27 21:16:35 -0700371 flag.Parse()
372
Colin Cross9f5633d2018-04-19 16:52:21 -0700373 if regen != "" {
374 err := rerunForRegen(regen)
375 if err != nil {
376 fmt.Fprintln(os.Stderr, err)
377 os.Exit(1)
378 }
379 os.Exit(0)
380 }
381
382 if flag.NArg() == 0 {
383 fmt.Fprintln(os.Stderr, "Directory argument is required")
384 os.Exit(1)
385 } else if flag.NArg() > 1 {
386 fmt.Fprintln(os.Stderr, "Multiple directories provided:", strings.Join(flag.Args(), " "))
Dan Willemsen2902fa72017-04-27 21:16:35 -0700387 os.Exit(1)
388 }
389
390 dir := flag.Arg(0)
391 absDir, err := filepath.Abs(dir)
392 if err != nil {
Colin Crossf46e37f2018-03-21 16:25:58 -0700393 fmt.Fprintln(os.Stderr, "Failed to get absolute directory:", err)
Dan Willemsen2902fa72017-04-27 21:16:35 -0700394 os.Exit(1)
395 }
396
397 var filenames []string
398 err = filepath.Walk(absDir, func(path string, info os.FileInfo, err error) error {
399 if err != nil {
400 return err
401 }
402
403 name := info.Name()
404 if info.IsDir() {
405 if strings.HasPrefix(name, ".") {
406 return filepath.SkipDir
407 }
408 return nil
409 }
410
411 if strings.HasPrefix(name, ".") {
412 return nil
413 }
414
415 if strings.HasSuffix(name, ".pom") {
416 path, err = filepath.Rel(absDir, path)
417 if err != nil {
418 return err
419 }
420 filenames = append(filenames, filepath.Join(dir, path))
421 }
422 return nil
423 })
424 if err != nil {
425 fmt.Fprintln(os.Stderr, "Error walking files:", err)
426 os.Exit(1)
427 }
428
429 if len(filenames) == 0 {
430 fmt.Fprintln(os.Stderr, "Error: no *.pom files found under", dir)
431 os.Exit(1)
432 }
433
434 sort.Strings(filenames)
435
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700436 poms := []*Pom{}
437 modules := make(map[string]*Pom)
Jeff Gastonaf2191e2018-03-29 14:59:01 -0400438 duplicate := false
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700439 for _, filename := range filenames {
440 pom, err := parse(filename)
441 if err != nil {
442 fmt.Fprintln(os.Stderr, "Error converting", filename, err)
443 os.Exit(1)
444 }
445
446 if pom != nil {
447 poms = append(poms, pom)
Alan Viverette75b95f82017-12-04 16:24:07 -0500448 key := pom.MkName()
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700449
Alan Viverette75b95f82017-12-04 16:24:07 -0500450 if old, ok := modules[key]; ok {
451 fmt.Fprintln(os.Stderr, "Module", key, "defined twice:", old.PomFile, pom.PomFile)
Jeff Gastonaf2191e2018-03-29 14:59:01 -0400452 duplicate = true
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700453 }
454
Alan Viverette75b95f82017-12-04 16:24:07 -0500455 modules[key] = pom
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700456 }
457 }
Jeff Gastonaf2191e2018-03-29 14:59:01 -0400458 if duplicate {
459 os.Exit(1)
460 }
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700461
462 for _, pom := range poms {
Alan Viverette6bd35eb2018-02-13 13:25:24 -0500463 pom.FixDeps(modules)
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700464 }
465
Dan Willemsen2902fa72017-04-27 21:16:35 -0700466 fmt.Println("# Automatically generated with:")
467 fmt.Println("# pom2mk", strings.Join(proptools.ShellEscape(os.Args[1:]), " "))
468 fmt.Println("LOCAL_PATH := $(call my-dir)")
469
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700470 for _, pom := range poms {
Alan Viverette1593d3d2017-12-11 17:14:26 -0500471 var err error
472 if staticDeps {
473 err = mkDepsTemplate.Execute(os.Stdout, pom)
474 } else {
475 err = mkTemplate.Execute(os.Stdout, pom)
476 }
Dan Willemsen2902fa72017-04-27 21:16:35 -0700477 if err != nil {
Dan Willemsen5f9d8a62017-10-05 14:01:31 -0700478 fmt.Fprintln(os.Stderr, "Error writing", pom.PomFile, pom.MkName(), err)
Dan Willemsen2902fa72017-04-27 21:16:35 -0700479 os.Exit(1)
480 }
481 }
482}