xref: /aosp_15_r20/build/soong/jar/jar.go (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
1*333d2b36SAndroid Build Coastguard Worker// Copyright 2017 Google Inc. All rights reserved.
2*333d2b36SAndroid Build Coastguard Worker//
3*333d2b36SAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License");
4*333d2b36SAndroid Build Coastguard Worker// you may not use this file except in compliance with the License.
5*333d2b36SAndroid Build Coastguard Worker// You may obtain a copy of the License at
6*333d2b36SAndroid Build Coastguard Worker//
7*333d2b36SAndroid Build Coastguard Worker//     http://www.apache.org/licenses/LICENSE-2.0
8*333d2b36SAndroid Build Coastguard Worker//
9*333d2b36SAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software
10*333d2b36SAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS,
11*333d2b36SAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*333d2b36SAndroid Build Coastguard Worker// See the License for the specific language governing permissions and
13*333d2b36SAndroid Build Coastguard Worker// limitations under the License.
14*333d2b36SAndroid Build Coastguard Worker
15*333d2b36SAndroid Build Coastguard Workerpackage jar
16*333d2b36SAndroid Build Coastguard Worker
17*333d2b36SAndroid Build Coastguard Workerimport (
18*333d2b36SAndroid Build Coastguard Worker	"bytes"
19*333d2b36SAndroid Build Coastguard Worker	"fmt"
20*333d2b36SAndroid Build Coastguard Worker	"io"
21*333d2b36SAndroid Build Coastguard Worker	"os"
22*333d2b36SAndroid Build Coastguard Worker	"strings"
23*333d2b36SAndroid Build Coastguard Worker	"text/scanner"
24*333d2b36SAndroid Build Coastguard Worker	"time"
25*333d2b36SAndroid Build Coastguard Worker	"unicode"
26*333d2b36SAndroid Build Coastguard Worker
27*333d2b36SAndroid Build Coastguard Worker	"android/soong/third_party/zip"
28*333d2b36SAndroid Build Coastguard Worker)
29*333d2b36SAndroid Build Coastguard Worker
30*333d2b36SAndroid Build Coastguard Workerconst (
31*333d2b36SAndroid Build Coastguard Worker	MetaDir         = "META-INF/"
32*333d2b36SAndroid Build Coastguard Worker	ManifestFile    = MetaDir + "MANIFEST.MF"
33*333d2b36SAndroid Build Coastguard Worker	ModuleInfoClass = "module-info.class"
34*333d2b36SAndroid Build Coastguard Worker)
35*333d2b36SAndroid Build Coastguard Worker
36*333d2b36SAndroid Build Coastguard Workervar DefaultTime = time.Date(2008, 1, 1, 0, 0, 0, 0, time.UTC)
37*333d2b36SAndroid Build Coastguard Worker
38*333d2b36SAndroid Build Coastguard Workervar MetaDirExtra = [2]byte{0xca, 0xfe}
39*333d2b36SAndroid Build Coastguard Worker
40*333d2b36SAndroid Build Coastguard Worker// EntryNamesLess tells whether <filepathA> should precede <filepathB> in
41*333d2b36SAndroid Build Coastguard Worker// the order of files with a .jar
42*333d2b36SAndroid Build Coastguard Workerfunc EntryNamesLess(filepathA string, filepathB string) (less bool) {
43*333d2b36SAndroid Build Coastguard Worker	diff := index(filepathA) - index(filepathB)
44*333d2b36SAndroid Build Coastguard Worker	if diff == 0 {
45*333d2b36SAndroid Build Coastguard Worker		return filepathA < filepathB
46*333d2b36SAndroid Build Coastguard Worker	}
47*333d2b36SAndroid Build Coastguard Worker	return diff < 0
48*333d2b36SAndroid Build Coastguard Worker}
49*333d2b36SAndroid Build Coastguard Worker
50*333d2b36SAndroid Build Coastguard Worker// Treats trailing * as a prefix match
51*333d2b36SAndroid Build Coastguard Workerfunc patternMatch(pattern, name string) bool {
52*333d2b36SAndroid Build Coastguard Worker	if strings.HasSuffix(pattern, "*") {
53*333d2b36SAndroid Build Coastguard Worker		return strings.HasPrefix(name, strings.TrimSuffix(pattern, "*"))
54*333d2b36SAndroid Build Coastguard Worker	} else {
55*333d2b36SAndroid Build Coastguard Worker		return name == pattern
56*333d2b36SAndroid Build Coastguard Worker	}
57*333d2b36SAndroid Build Coastguard Worker}
58*333d2b36SAndroid Build Coastguard Worker
59*333d2b36SAndroid Build Coastguard Workervar jarOrder = []string{
60*333d2b36SAndroid Build Coastguard Worker	MetaDir,
61*333d2b36SAndroid Build Coastguard Worker	ManifestFile,
62*333d2b36SAndroid Build Coastguard Worker	MetaDir + "*",
63*333d2b36SAndroid Build Coastguard Worker	"*",
64*333d2b36SAndroid Build Coastguard Worker}
65*333d2b36SAndroid Build Coastguard Worker
66*333d2b36SAndroid Build Coastguard Workerfunc index(name string) int {
67*333d2b36SAndroid Build Coastguard Worker	for i, pattern := range jarOrder {
68*333d2b36SAndroid Build Coastguard Worker		if patternMatch(pattern, name) {
69*333d2b36SAndroid Build Coastguard Worker			return i
70*333d2b36SAndroid Build Coastguard Worker		}
71*333d2b36SAndroid Build Coastguard Worker	}
72*333d2b36SAndroid Build Coastguard Worker	panic(fmt.Errorf("file %q did not match any pattern", name))
73*333d2b36SAndroid Build Coastguard Worker}
74*333d2b36SAndroid Build Coastguard Worker
75*333d2b36SAndroid Build Coastguard Workerfunc MetaDirFileHeader() *zip.FileHeader {
76*333d2b36SAndroid Build Coastguard Worker	dirHeader := &zip.FileHeader{
77*333d2b36SAndroid Build Coastguard Worker		Name:  MetaDir,
78*333d2b36SAndroid Build Coastguard Worker		Extra: []byte{MetaDirExtra[1], MetaDirExtra[0], 0, 0},
79*333d2b36SAndroid Build Coastguard Worker	}
80*333d2b36SAndroid Build Coastguard Worker	dirHeader.SetMode(0755 | os.ModeDir)
81*333d2b36SAndroid Build Coastguard Worker	dirHeader.SetModTime(DefaultTime)
82*333d2b36SAndroid Build Coastguard Worker
83*333d2b36SAndroid Build Coastguard Worker	return dirHeader
84*333d2b36SAndroid Build Coastguard Worker}
85*333d2b36SAndroid Build Coastguard Worker
86*333d2b36SAndroid Build Coastguard Worker// Create a manifest zip header and contents using the provided contents if any.
87*333d2b36SAndroid Build Coastguard Workerfunc ManifestFileContents(contents []byte) (*zip.FileHeader, []byte, error) {
88*333d2b36SAndroid Build Coastguard Worker	b, err := manifestContents(contents)
89*333d2b36SAndroid Build Coastguard Worker	if err != nil {
90*333d2b36SAndroid Build Coastguard Worker		return nil, nil, err
91*333d2b36SAndroid Build Coastguard Worker	}
92*333d2b36SAndroid Build Coastguard Worker
93*333d2b36SAndroid Build Coastguard Worker	fh := &zip.FileHeader{
94*333d2b36SAndroid Build Coastguard Worker		Name:               ManifestFile,
95*333d2b36SAndroid Build Coastguard Worker		Method:             zip.Store,
96*333d2b36SAndroid Build Coastguard Worker		UncompressedSize64: uint64(len(b)),
97*333d2b36SAndroid Build Coastguard Worker	}
98*333d2b36SAndroid Build Coastguard Worker	fh.SetMode(0644)
99*333d2b36SAndroid Build Coastguard Worker	fh.SetModTime(DefaultTime)
100*333d2b36SAndroid Build Coastguard Worker
101*333d2b36SAndroid Build Coastguard Worker	return fh, b, nil
102*333d2b36SAndroid Build Coastguard Worker}
103*333d2b36SAndroid Build Coastguard Worker
104*333d2b36SAndroid Build Coastguard Worker// Create manifest contents, using the provided contents if any.
105*333d2b36SAndroid Build Coastguard Workerfunc manifestContents(contents []byte) ([]byte, error) {
106*333d2b36SAndroid Build Coastguard Worker	manifestMarker := []byte("Manifest-Version:")
107*333d2b36SAndroid Build Coastguard Worker	header := append(manifestMarker, []byte(" 1.0\nCreated-By: soong_zip\n")...)
108*333d2b36SAndroid Build Coastguard Worker
109*333d2b36SAndroid Build Coastguard Worker	var finalBytes []byte
110*333d2b36SAndroid Build Coastguard Worker	if !bytes.Contains(contents, manifestMarker) {
111*333d2b36SAndroid Build Coastguard Worker		finalBytes = append(append(header, contents...), byte('\n'))
112*333d2b36SAndroid Build Coastguard Worker	} else {
113*333d2b36SAndroid Build Coastguard Worker		finalBytes = contents
114*333d2b36SAndroid Build Coastguard Worker	}
115*333d2b36SAndroid Build Coastguard Worker
116*333d2b36SAndroid Build Coastguard Worker	return finalBytes, nil
117*333d2b36SAndroid Build Coastguard Worker}
118*333d2b36SAndroid Build Coastguard Worker
119*333d2b36SAndroid Build Coastguard Workervar javaIgnorableIdentifier = &unicode.RangeTable{
120*333d2b36SAndroid Build Coastguard Worker	R16: []unicode.Range16{
121*333d2b36SAndroid Build Coastguard Worker		{0x00, 0x08, 1},
122*333d2b36SAndroid Build Coastguard Worker		{0x0e, 0x1b, 1},
123*333d2b36SAndroid Build Coastguard Worker		{0x7f, 0x9f, 1},
124*333d2b36SAndroid Build Coastguard Worker	},
125*333d2b36SAndroid Build Coastguard Worker	LatinOffset: 3,
126*333d2b36SAndroid Build Coastguard Worker}
127*333d2b36SAndroid Build Coastguard Worker
128*333d2b36SAndroid Build Coastguard Workerfunc javaIdentRune(ch rune, i int) bool {
129*333d2b36SAndroid Build Coastguard Worker	if unicode.IsLetter(ch) {
130*333d2b36SAndroid Build Coastguard Worker		return true
131*333d2b36SAndroid Build Coastguard Worker	}
132*333d2b36SAndroid Build Coastguard Worker	if unicode.IsDigit(ch) && i > 0 {
133*333d2b36SAndroid Build Coastguard Worker		return true
134*333d2b36SAndroid Build Coastguard Worker	}
135*333d2b36SAndroid Build Coastguard Worker
136*333d2b36SAndroid Build Coastguard Worker	if unicode.In(ch,
137*333d2b36SAndroid Build Coastguard Worker		unicode.Nl, // letter number
138*333d2b36SAndroid Build Coastguard Worker		unicode.Sc, // currency symbol
139*333d2b36SAndroid Build Coastguard Worker		unicode.Pc, // connecting punctuation
140*333d2b36SAndroid Build Coastguard Worker	) {
141*333d2b36SAndroid Build Coastguard Worker		return true
142*333d2b36SAndroid Build Coastguard Worker	}
143*333d2b36SAndroid Build Coastguard Worker
144*333d2b36SAndroid Build Coastguard Worker	if unicode.In(ch,
145*333d2b36SAndroid Build Coastguard Worker		unicode.Cf, // format
146*333d2b36SAndroid Build Coastguard Worker		unicode.Mc, // combining mark
147*333d2b36SAndroid Build Coastguard Worker		unicode.Mn, // non-spacing mark
148*333d2b36SAndroid Build Coastguard Worker		javaIgnorableIdentifier,
149*333d2b36SAndroid Build Coastguard Worker	) && i > 0 {
150*333d2b36SAndroid Build Coastguard Worker		return true
151*333d2b36SAndroid Build Coastguard Worker	}
152*333d2b36SAndroid Build Coastguard Worker
153*333d2b36SAndroid Build Coastguard Worker	return false
154*333d2b36SAndroid Build Coastguard Worker}
155*333d2b36SAndroid Build Coastguard Worker
156*333d2b36SAndroid Build Coastguard Worker// JavaPackage parses the package out of a java source file by looking for the package statement, or the first valid
157*333d2b36SAndroid Build Coastguard Worker// non-package statement, in which case it returns an empty string for the package.
158*333d2b36SAndroid Build Coastguard Workerfunc JavaPackage(r io.Reader, src string) (string, error) {
159*333d2b36SAndroid Build Coastguard Worker	var s scanner.Scanner
160*333d2b36SAndroid Build Coastguard Worker	var sErr error
161*333d2b36SAndroid Build Coastguard Worker
162*333d2b36SAndroid Build Coastguard Worker	s.Init(r)
163*333d2b36SAndroid Build Coastguard Worker	s.Filename = src
164*333d2b36SAndroid Build Coastguard Worker	s.Error = func(s *scanner.Scanner, msg string) {
165*333d2b36SAndroid Build Coastguard Worker		sErr = fmt.Errorf("error parsing %q: %s", src, msg)
166*333d2b36SAndroid Build Coastguard Worker	}
167*333d2b36SAndroid Build Coastguard Worker	s.IsIdentRune = javaIdentRune
168*333d2b36SAndroid Build Coastguard Worker
169*333d2b36SAndroid Build Coastguard Worker	var tok rune
170*333d2b36SAndroid Build Coastguard Worker	for {
171*333d2b36SAndroid Build Coastguard Worker		tok = s.Scan()
172*333d2b36SAndroid Build Coastguard Worker		if sErr != nil {
173*333d2b36SAndroid Build Coastguard Worker			return "", sErr
174*333d2b36SAndroid Build Coastguard Worker		}
175*333d2b36SAndroid Build Coastguard Worker		// If the first token is an annotation, it could be annotating a package declaration, so consume them.
176*333d2b36SAndroid Build Coastguard Worker		// Note that this does not support "complex" annotations with attributes, e.g. @Foo(x=y).
177*333d2b36SAndroid Build Coastguard Worker		if tok != '@' {
178*333d2b36SAndroid Build Coastguard Worker			break
179*333d2b36SAndroid Build Coastguard Worker		}
180*333d2b36SAndroid Build Coastguard Worker		tok = s.Scan()
181*333d2b36SAndroid Build Coastguard Worker		if tok != scanner.Ident || sErr != nil {
182*333d2b36SAndroid Build Coastguard Worker			return "", fmt.Errorf("expected annotation identifier, got @%v", tok)
183*333d2b36SAndroid Build Coastguard Worker		}
184*333d2b36SAndroid Build Coastguard Worker	}
185*333d2b36SAndroid Build Coastguard Worker
186*333d2b36SAndroid Build Coastguard Worker	if tok == scanner.Ident {
187*333d2b36SAndroid Build Coastguard Worker		switch s.TokenText() {
188*333d2b36SAndroid Build Coastguard Worker		case "package":
189*333d2b36SAndroid Build Coastguard Worker		// Nothing
190*333d2b36SAndroid Build Coastguard Worker		case "import":
191*333d2b36SAndroid Build Coastguard Worker			// File has no package statement, first keyword is an import
192*333d2b36SAndroid Build Coastguard Worker			return "", nil
193*333d2b36SAndroid Build Coastguard Worker		case "class", "enum", "interface":
194*333d2b36SAndroid Build Coastguard Worker			// File has no package statement, first keyword is a type declaration
195*333d2b36SAndroid Build Coastguard Worker			return "", nil
196*333d2b36SAndroid Build Coastguard Worker		case "public", "protected", "private", "abstract", "static", "final", "strictfp":
197*333d2b36SAndroid Build Coastguard Worker			// File has no package statement, first keyword is a modifier
198*333d2b36SAndroid Build Coastguard Worker			return "", nil
199*333d2b36SAndroid Build Coastguard Worker		case "module", "open":
200*333d2b36SAndroid Build Coastguard Worker			// File has no package statement, first keyword is a module declaration
201*333d2b36SAndroid Build Coastguard Worker			return "", nil
202*333d2b36SAndroid Build Coastguard Worker		default:
203*333d2b36SAndroid Build Coastguard Worker			return "", fmt.Errorf(`expected first token of java file to be "package", got %q`, s.TokenText())
204*333d2b36SAndroid Build Coastguard Worker		}
205*333d2b36SAndroid Build Coastguard Worker	} else if tok == scanner.EOF {
206*333d2b36SAndroid Build Coastguard Worker		// File no package statement, it has no non-whitespace non-comment tokens
207*333d2b36SAndroid Build Coastguard Worker		return "", nil
208*333d2b36SAndroid Build Coastguard Worker	} else {
209*333d2b36SAndroid Build Coastguard Worker		return "", fmt.Errorf(`expected first token of java file to be "package", got %q`, s.TokenText())
210*333d2b36SAndroid Build Coastguard Worker	}
211*333d2b36SAndroid Build Coastguard Worker
212*333d2b36SAndroid Build Coastguard Worker	var pkg string
213*333d2b36SAndroid Build Coastguard Worker	for {
214*333d2b36SAndroid Build Coastguard Worker		tok = s.Scan()
215*333d2b36SAndroid Build Coastguard Worker		if sErr != nil {
216*333d2b36SAndroid Build Coastguard Worker			return "", sErr
217*333d2b36SAndroid Build Coastguard Worker		}
218*333d2b36SAndroid Build Coastguard Worker		if tok != scanner.Ident {
219*333d2b36SAndroid Build Coastguard Worker			return "", fmt.Errorf(`expected "package <package>;", got "package %s%s"`, pkg, s.TokenText())
220*333d2b36SAndroid Build Coastguard Worker		}
221*333d2b36SAndroid Build Coastguard Worker		pkg += s.TokenText()
222*333d2b36SAndroid Build Coastguard Worker
223*333d2b36SAndroid Build Coastguard Worker		tok = s.Scan()
224*333d2b36SAndroid Build Coastguard Worker		if sErr != nil {
225*333d2b36SAndroid Build Coastguard Worker			return "", sErr
226*333d2b36SAndroid Build Coastguard Worker		}
227*333d2b36SAndroid Build Coastguard Worker		if tok == ';' {
228*333d2b36SAndroid Build Coastguard Worker			return pkg, nil
229*333d2b36SAndroid Build Coastguard Worker		} else if tok == '.' {
230*333d2b36SAndroid Build Coastguard Worker			pkg += "."
231*333d2b36SAndroid Build Coastguard Worker		} else {
232*333d2b36SAndroid Build Coastguard Worker			return "", fmt.Errorf(`expected "package <package>;", got "package %s%s"`, pkg, s.TokenText())
233*333d2b36SAndroid Build Coastguard Worker		}
234*333d2b36SAndroid Build Coastguard Worker	}
235*333d2b36SAndroid Build Coastguard Worker}
236