1/*
2 * Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.  Oracle designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Oracle in the LICENSE file that accompanied this code.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
23 * questions.
24 */
25
26package java.util.jar;
27
28import java.io.*;
29import java.lang.ref.SoftReference;
30import java.net.URL;
31import java.util.*;
32import java.util.stream.Stream;
33import java.util.stream.StreamSupport;
34import java.util.zip.*;
35import java.security.CodeSigner;
36import java.security.cert.Certificate;
37import java.security.CodeSource;
38import jdk.internal.misc.SharedSecrets;
39import sun.security.action.GetPropertyAction;
40import sun.security.util.ManifestEntryVerifier;
41import sun.security.util.SignatureFileVerifier;
42
43/**
44 * The {@code JarFile} class is used to read the contents of a jar file
45 * from any file that can be opened with {@code java.io.RandomAccessFile}.
46 * It extends the class {@code java.util.zip.ZipFile} with support
47 * for reading an optional {@code Manifest} entry, and support for
48 * processing multi-release jar files.  The {@code Manifest} can be used
49 * to specify meta-information about the jar file and its entries.
50 *
51 * <p><a id="multirelease">A multi-release jar file</a> is a jar file that
52 * contains a manifest with a main attribute named "Multi-Release",
53 * a set of "base" entries, some of which are public classes with public
54 * or protected methods that comprise the public interface of the jar file,
55 * and a set of "versioned" entries contained in subdirectories of the
56 * "META-INF/versions" directory.  The versioned entries are partitioned by the
57 * major version of the Java platform.  A versioned entry, with a version
58 * {@code n}, {@code 8 < n}, in the "META-INF/versions/{n}" directory overrides
59 * the base entry as well as any entry with a version number {@code i} where
60 * {@code 8 < i < n}.
61 *
62 * <p>By default, a {@code JarFile} for a multi-release jar file is configured
63 * to process the multi-release jar file as if it were a plain (unversioned) jar
64 * file, and as such an entry name is associated with at most one base entry.
65 * The {@code JarFile} may be configured to process a multi-release jar file by
66 * creating the {@code JarFile} with the
67 * {@link JarFile#JarFile(File, boolean, int, Runtime.Version)} constructor.  The
68 * {@code Runtime.Version} object sets a maximum version used when searching for
69 * versioned entries.  When so configured, an entry name
70 * can correspond with at most one base entry and zero or more versioned
71 * entries. A search is required to associate the entry name with the latest
72 * versioned entry whose version is less than or equal to the maximum version
73 * (see {@link #getEntry(String)}).
74 *
75 * <p>Class loaders that utilize {@code JarFile} to load classes from the
76 * contents of {@code JarFile} entries should construct the {@code JarFile}
77 * by invoking the {@link JarFile#JarFile(File, boolean, int, Runtime.Version)}
78 * constructor with the value {@code Runtime.version()} assigned to the last
79 * argument.  This assures that classes compatible with the major
80 * version of the running JVM are loaded from multi-release jar files.
81 *
82 * <p>If the verify flag is on when opening a signed jar file, the content of
83 * the file is verified against its signature embedded inside the file. Please
84 * note that the verification process does not include validating the signer's
85 * certificate. A caller should inspect the return value of
86 * {@link JarEntry#getCodeSigners()} to further determine if the signature
87 * can be trusted.
88 *
89 * <p> Unless otherwise noted, passing a {@code null} argument to a constructor
90 * or method in this class will cause a {@link NullPointerException} to be
91 * thrown.
92 *
93 * @implNote
94 * <div class="block">
95 * If the API can not be used to configure a {@code JarFile} (e.g. to override
96 * the configuration of a compiled application or library), two {@code System}
97 * properties are available.
98 * <ul>
99 * <li>
100 * {@code jdk.util.jar.version} can be assigned a value that is the
101 * {@code String} representation of a non-negative integer
102 * {@code <= Runtime.version().major()}.  The value is used to set the effective
103 * runtime version to something other than the default value obtained by
104 * evaluating {@code Runtime.version().major()}. The effective runtime version
105 * is the version that the {@link JarFile#JarFile(File, boolean, int, Runtime.Version)}
106 * constructor uses when the value of the last argument is
107 * {@code JarFile.runtimeVersion()}.
108 * </li>
109 * <li>
110 * {@code jdk.util.jar.enableMultiRelease} can be assigned one of the three
111 * {@code String} values <em>true</em>, <em>false</em>, or <em>force</em>.  The
112 * value <em>true</em>, the default value, enables multi-release jar file
113 * processing.  The value <em>false</em> disables multi-release jar processing,
114 * ignoring the "Multi-Release" manifest attribute, and the versioned
115 * directories in a multi-release jar file if they exist.  Furthermore,
116 * the method {@link JarFile#isMultiRelease()} returns <em>false</em>. The value
117 * <em>force</em> causes the {@code JarFile} to be initialized to runtime
118 * versioning after construction.  It effectively does the same as this code:
119 * {@code (new JarFile(File, boolean, int, JarFile.runtimeVersion())}.
120 * </li>
121 * </ul>
122 * </div>
123 *
124 * @author  David Connelly
125 * @see     Manifest
126 * @see     java.util.zip.ZipFile
127 * @see     java.util.jar.JarEntry
128 * @since   1.2
129 */
130public
131class JarFile extends ZipFile {
132    private final static Runtime.Version BASE_VERSION;
133    private final static int BASE_VERSION_MAJOR;
134    private final static Runtime.Version RUNTIME_VERSION;
135    private final static boolean MULTI_RELEASE_ENABLED;
136    private final static boolean MULTI_RELEASE_FORCED;
137    private SoftReference<Manifest> manRef;
138    private JarEntry manEntry;
139    private JarVerifier jv;
140    private boolean jvInitialized;
141    private boolean verify;
142    private final Runtime.Version version;  // current version
143    private final int versionMajor;         // version.major()
144    private boolean isMultiRelease;         // is jar multi-release?
145
146    // indicates if Class-Path attribute present
147    private boolean hasClassPathAttribute;
148    // true if manifest checked for special attributes
149    private volatile boolean hasCheckedSpecialAttributes;
150
151    static {
152        // Set up JavaUtilJarAccess in SharedSecrets
153        SharedSecrets.setJavaUtilJarAccess(new JavaUtilJarAccessImpl());
154        // multi-release jar file versions >= 9
155        BASE_VERSION = Runtime.Version.parse(Integer.toString(8));
156        BASE_VERSION_MAJOR = BASE_VERSION.major();
157        String jarVersion = GetPropertyAction.privilegedGetProperty("jdk.util.jar.version");
158        int runtimeVersion = Runtime.version().major();
159        if (jarVersion != null) {
160            int jarVer = Integer.parseInt(jarVersion);
161            runtimeVersion = (jarVer > runtimeVersion)
162                    ? runtimeVersion
163                    : Math.max(jarVer, BASE_VERSION_MAJOR);
164        }
165        RUNTIME_VERSION = Runtime.Version.parse(Integer.toString(runtimeVersion));
166        String enableMultiRelease = GetPropertyAction
167                .privilegedGetProperty("jdk.util.jar.enableMultiRelease", "true");
168        switch (enableMultiRelease) {
169            case "true":
170            default:
171                MULTI_RELEASE_ENABLED = true;
172                MULTI_RELEASE_FORCED = false;
173                break;
174            case "false":
175                MULTI_RELEASE_ENABLED = false;
176                MULTI_RELEASE_FORCED = false;
177                break;
178            case "force":
179                MULTI_RELEASE_ENABLED = true;
180                MULTI_RELEASE_FORCED = true;
181                break;
182        }
183    }
184
185    private static final String META_INF = "META-INF/";
186
187    private static final String META_INF_VERSIONS = META_INF + "versions/";
188
189    /**
190     * The JAR manifest file name.
191     */
192    public static final String MANIFEST_NAME = META_INF + "MANIFEST.MF";
193
194    /**
195     * Returns the version that represents the unversioned configuration of a
196     * multi-release jar file.
197     *
198     * @return the version that represents the unversioned configuration
199     *
200     * @since 9
201     */
202    public static Runtime.Version baseVersion() {
203        return BASE_VERSION;
204    }
205
206    /**
207     * Returns the version that represents the effective runtime versioned
208     * configuration of a multi-release jar file.
209     * <p>
210     * By default the major version number of the returned {@code Version} will
211     * be equal to the major version number of {@code Runtime.version()}.
212     * However, if the {@code jdk.util.jar.version} property is set, the
213     * returned {@code Version} is derived from that property and major version
214     * numbers may not be equal.
215     *
216     * @return the version that represents the runtime versioned configuration
217     *
218     * @since 9
219     */
220    public static Runtime.Version runtimeVersion() {
221        return RUNTIME_VERSION;
222    }
223
224    /**
225     * Creates a new {@code JarFile} to read from the specified
226     * file {@code name}. The {@code JarFile} will be verified if
227     * it is signed.
228     * @param name the name of the jar file to be opened for reading
229     * @throws IOException if an I/O error has occurred
230     * @throws SecurityException if access to the file is denied
231     *         by the SecurityManager
232     */
233    public JarFile(String name) throws IOException {
234        this(new File(name), true, ZipFile.OPEN_READ);
235    }
236
237    /**
238     * Creates a new {@code JarFile} to read from the specified
239     * file {@code name}.
240     * @param name the name of the jar file to be opened for reading
241     * @param verify whether or not to verify the jar file if
242     * it is signed.
243     * @throws IOException if an I/O error has occurred
244     * @throws SecurityException if access to the file is denied
245     *         by the SecurityManager
246     */
247    public JarFile(String name, boolean verify) throws IOException {
248        this(new File(name), verify, ZipFile.OPEN_READ);
249    }
250
251    /**
252     * Creates a new {@code JarFile} to read from the specified
253     * {@code File} object. The {@code JarFile} will be verified if
254     * it is signed.
255     * @param file the jar file to be opened for reading
256     * @throws IOException if an I/O error has occurred
257     * @throws SecurityException if access to the file is denied
258     *         by the SecurityManager
259     */
260    public JarFile(File file) throws IOException {
261        this(file, true, ZipFile.OPEN_READ);
262    }
263
264    /**
265     * Creates a new {@code JarFile} to read from the specified
266     * {@code File} object.
267     * @param file the jar file to be opened for reading
268     * @param verify whether or not to verify the jar file if
269     * it is signed.
270     * @throws IOException if an I/O error has occurred
271     * @throws SecurityException if access to the file is denied
272     *         by the SecurityManager.
273     */
274    public JarFile(File file, boolean verify) throws IOException {
275        this(file, verify, ZipFile.OPEN_READ);
276    }
277
278    /**
279     * Creates a new {@code JarFile} to read from the specified
280     * {@code File} object in the specified mode.  The mode argument
281     * must be either {@code OPEN_READ} or {@code OPEN_READ | OPEN_DELETE}.
282     *
283     * @param file the jar file to be opened for reading
284     * @param verify whether or not to verify the jar file if
285     * it is signed.
286     * @param mode the mode in which the file is to be opened
287     * @throws IOException if an I/O error has occurred
288     * @throws IllegalArgumentException
289     *         if the {@code mode} argument is invalid
290     * @throws SecurityException if access to the file is denied
291     *         by the SecurityManager
292     * @since 1.3
293     */
294    public JarFile(File file, boolean verify, int mode) throws IOException {
295        this(file, verify, mode, BASE_VERSION);
296    }
297
298    /**
299     * Creates a new {@code JarFile} to read from the specified
300     * {@code File} object in the specified mode.  The mode argument
301     * must be either {@code OPEN_READ} or {@code OPEN_READ | OPEN_DELETE}.
302     * The version argument, after being converted to a canonical form, is
303     * used to configure the {@code JarFile} for processing
304     * multi-release jar files.
305     * <p>
306     * The canonical form derived from the version parameter is
307     * {@code Runtime.Version.parse(Integer.toString(n))} where {@code n} is
308     * {@code Math.max(version.major(), JarFile.baseVersion().major())}.
309     *
310     * @param file the jar file to be opened for reading
311     * @param verify whether or not to verify the jar file if
312     * it is signed.
313     * @param mode the mode in which the file is to be opened
314     * @param version specifies the release version for a multi-release jar file
315     * @throws IOException if an I/O error has occurred
316     * @throws IllegalArgumentException
317     *         if the {@code mode} argument is invalid
318     * @throws SecurityException if access to the file is denied
319     *         by the SecurityManager
320     * @throws NullPointerException if {@code version} is {@code null}
321     * @since 9
322     */
323    public JarFile(File file, boolean verify, int mode, Runtime.Version version) throws IOException {
324        super(file, mode);
325        this.verify = verify;
326        Objects.requireNonNull(version);
327        if (MULTI_RELEASE_FORCED || version.major() == RUNTIME_VERSION.major()) {
328            // This deals with the common case where the value from JarFile.runtimeVersion() is passed
329            this.version = RUNTIME_VERSION;
330        } else if (version.major() <= BASE_VERSION_MAJOR) {
331            // This also deals with the common case where the value from JarFile.baseVersion() is passed
332            this.version = BASE_VERSION;
333        } else {
334            // Canonicalize
335            this.version = Runtime.Version.parse(Integer.toString(version.major()));
336        }
337        this.versionMajor = this.version.major();
338    }
339
340    /**
341     * Returns the maximum version used when searching for versioned entries.
342     * <p>
343     * If this {@code JarFile} is not a multi-release jar file or is not
344     * configured to be processed as such, then the version returned will be the
345     * same as that returned from {@link #baseVersion()}.
346     *
347     * @return the maximum version
348     * @since 9
349     */
350    public final Runtime.Version getVersion() {
351        return isMultiRelease() ? this.version : BASE_VERSION;
352    }
353
354    /**
355     * Indicates whether or not this jar file is a multi-release jar file.
356     *
357     * @return true if this JarFile is a multi-release jar file
358     * @since 9
359     */
360    public final boolean isMultiRelease() {
361        if (isMultiRelease) {
362            return true;
363        }
364        if (MULTI_RELEASE_ENABLED) {
365            try {
366                checkForSpecialAttributes();
367            } catch (IOException io) {
368                isMultiRelease = false;
369            }
370        }
371        return isMultiRelease;
372    }
373
374    /**
375     * Returns the jar file manifest, or {@code null} if none.
376     *
377     * @return the jar file manifest, or {@code null} if none
378     *
379     * @throws IllegalStateException
380     *         may be thrown if the jar file has been closed
381     * @throws IOException  if an I/O error has occurred
382     */
383    public Manifest getManifest() throws IOException {
384        return getManifestFromReference();
385    }
386
387    private Manifest getManifestFromReference() throws IOException {
388        Manifest man = manRef != null ? manRef.get() : null;
389
390        if (man == null) {
391
392            JarEntry manEntry = getManEntry();
393
394            // If found then load the manifest
395            if (manEntry != null) {
396                if (verify) {
397                    byte[] b = getBytes(manEntry);
398                    man = new Manifest(new ByteArrayInputStream(b));
399                    if (!jvInitialized) {
400                        jv = new JarVerifier(b);
401                    }
402                } else {
403                    man = new Manifest(super.getInputStream(manEntry));
404                }
405                manRef = new SoftReference<>(man);
406            }
407        }
408        return man;
409    }
410
411    private String[] getMetaInfEntryNames() {
412        return jdk.internal.misc.SharedSecrets.getJavaUtilZipFileAccess()
413                                              .getMetaInfEntryNames((ZipFile)this);
414    }
415
416    /**
417     * Returns the {@code JarEntry} for the given base entry name or
418     * {@code null} if not found.
419     *
420     * <p>If this {@code JarFile} is a multi-release jar file and is configured
421     * to be processed as such, then a search is performed to find and return
422     * a {@code JarEntry} that is the latest versioned entry associated with the
423     * given entry name.  The returned {@code JarEntry} is the versioned entry
424     * corresponding to the given base entry name prefixed with the string
425     * {@code "META-INF/versions/{n}/"}, for the largest value of {@code n} for
426     * which an entry exists.  If such a versioned entry does not exist, then
427     * the {@code JarEntry} for the base entry is returned, otherwise
428     * {@code null} is returned if no entries are found.  The initial value for
429     * the version {@code n} is the maximum version as returned by the method
430     * {@link JarFile#getVersion()}.
431     *
432     * @param name the jar file entry name
433     * @return the {@code JarEntry} for the given entry name, or
434     *         the versioned entry name, or {@code null} if not found
435     *
436     * @throws IllegalStateException
437     *         may be thrown if the jar file has been closed
438     *
439     * @see java.util.jar.JarEntry
440     *
441     * @implSpec
442     * <div class="block">
443     * This implementation invokes {@link JarFile#getEntry(String)}.
444     * </div>
445     */
446    public JarEntry getJarEntry(String name) {
447        return (JarEntry)getEntry(name);
448    }
449
450    /**
451     * Returns the {@code ZipEntry} for the given base entry name or
452     * {@code null} if not found.
453     *
454     * <p>If this {@code JarFile} is a multi-release jar file and is configured
455     * to be processed as such, then a search is performed to find and return
456     * a {@code ZipEntry} that is the latest versioned entry associated with the
457     * given entry name.  The returned {@code ZipEntry} is the versioned entry
458     * corresponding to the given base entry name prefixed with the string
459     * {@code "META-INF/versions/{n}/"}, for the largest value of {@code n} for
460     * which an entry exists.  If such a versioned entry does not exist, then
461     * the {@code ZipEntry} for the base entry is returned, otherwise
462     * {@code null} is returned if no entries are found.  The initial value for
463     * the version {@code n} is the maximum version as returned by the method
464     * {@link JarFile#getVersion()}.
465     *
466     * @param name the jar file entry name
467     * @return the {@code ZipEntry} for the given entry name or
468     *         the versioned entry name or {@code null} if not found
469     *
470     * @throws IllegalStateException
471     *         may be thrown if the jar file has been closed
472     *
473     * @see java.util.zip.ZipEntry
474     *
475     * @implSpec
476     * <div class="block">
477     * This implementation may return a versioned entry for the requested name
478     * even if there is not a corresponding base entry.  This can occur
479     * if there is a private or package-private versioned entry that matches.
480     * If a subclass overrides this method, assure that the override method
481     * invokes {@code super.getEntry(name)} to obtain all versioned entries.
482     * </div>
483     */
484    public ZipEntry getEntry(String name) {
485        ZipEntry ze = super.getEntry(name);
486        if (ze != null) {
487            return new JarFileEntry(ze);
488        }
489        // no matching base entry, but maybe there is a versioned entry,
490        // like a new private class
491        if (isMultiRelease()) {
492            ze = new ZipEntry(name);
493            ZipEntry vze = getVersionedEntry(ze);
494            if (ze != vze) {
495                return new JarFileEntry(name, vze);
496            }
497        }
498        return null;
499    }
500
501    private class JarEntryIterator implements Enumeration<JarEntry>,
502            Iterator<JarEntry>
503    {
504        final Enumeration<? extends ZipEntry> e = JarFile.super.entries();
505
506        public boolean hasNext() {
507            return e.hasMoreElements();
508        }
509
510        public JarEntry next() {
511            ZipEntry ze = e.nextElement();
512            return new JarFileEntry(ze.getName(), ze);
513        }
514
515        public boolean hasMoreElements() {
516            return hasNext();
517        }
518
519        public JarEntry nextElement() {
520            return next();
521        }
522
523        public Iterator<JarEntry> asIterator() {
524            return this;
525        }
526    }
527
528    /**
529     * Returns an enumeration of the jar file entries.
530     *
531     * @return an enumeration of the jar file entries
532     * @throws IllegalStateException
533     *         may be thrown if the jar file has been closed
534     */
535    public Enumeration<JarEntry> entries() {
536        return new JarEntryIterator();
537    }
538
539    /**
540     * Returns an ordered {@code Stream} over the jar file entries.
541     * Entries appear in the {@code Stream} in the order they appear in
542     * the central directory of the jar file.
543     *
544     * @return an ordered {@code Stream} of entries in this jar file
545     * @throws IllegalStateException if the jar file has been closed
546     * @since 1.8
547     */
548    public Stream<JarEntry> stream() {
549        return StreamSupport.stream(Spliterators.spliterator(
550                new JarEntryIterator(), size(),
551                Spliterator.ORDERED | Spliterator.DISTINCT |
552                        Spliterator.IMMUTABLE | Spliterator.NONNULL), false);
553    }
554
555    private ZipEntry searchForVersionedEntry(final int version, String name) {
556        ZipEntry vze = null;
557        String sname = "/" + name;
558        int i = version;
559        while (i > BASE_VERSION_MAJOR) {
560            vze = super.getEntry(META_INF_VERSIONS + i + sname);
561            if (vze != null) break;
562            i--;
563        }
564        return vze;
565    }
566
567    private ZipEntry getVersionedEntry(ZipEntry ze) {
568        ZipEntry vze = null;
569        if (BASE_VERSION_MAJOR < versionMajor) {
570            String name = ze.getName();
571            if (!name.startsWith(META_INF)) {
572                vze = searchForVersionedEntry(versionMajor, name);
573            }
574        }
575        return vze == null ? ze : vze;
576    }
577
578    /**
579     * Returns the real name of a {@code JarEntry}.  If this {@code JarFile} is
580     * a multi-release jar file and is configured to be processed as such, the
581     * name returned by this method is the path name of the versioned entry
582     * that the {@code JarEntry} represents, rather than the path name of the
583     * base entry that {@link JarEntry#getName()} returns.  If the
584     * {@code JarEntry} does not represent a versioned entry, or the
585     * jar file is not a multi-release jar file or {@code JarFile} is not
586     * configured for processing a multi-release jar file, this method returns
587     * the same name that {@link JarEntry#getName()} returns.
588     *
589     * @param entry the JarEntry
590     * @return the real name of the JarEntry
591     * @since 9
592     */
593    String getRealName(JarEntry entry) {
594        if (entry instanceof JarFileEntry) {
595            return ((JarFileEntry)entry).realName();
596        }
597        return entry.getName();
598    }
599
600    private class JarFileEntry extends JarEntry {
601        final private String name;
602
603        JarFileEntry(ZipEntry ze) {
604            super(isMultiRelease() ? getVersionedEntry(ze) : ze);
605            this.name = ze.getName();
606        }
607        JarFileEntry(String name, ZipEntry vze) {
608            super(vze);
609            this.name = name;
610        }
611        public Attributes getAttributes() throws IOException {
612            Manifest man = JarFile.this.getManifest();
613            if (man != null) {
614                return man.getAttributes(super.getName());
615            } else {
616                return null;
617            }
618        }
619        public Certificate[] getCertificates() {
620            try {
621                maybeInstantiateVerifier();
622            } catch (IOException e) {
623                throw new RuntimeException(e);
624            }
625            if (certs == null && jv != null) {
626                certs = jv.getCerts(JarFile.this, realEntry());
627            }
628            return certs == null ? null : certs.clone();
629        }
630        public CodeSigner[] getCodeSigners() {
631            try {
632                maybeInstantiateVerifier();
633            } catch (IOException e) {
634                throw new RuntimeException(e);
635            }
636            if (signers == null && jv != null) {
637                signers = jv.getCodeSigners(JarFile.this, realEntry());
638            }
639            return signers == null ? null : signers.clone();
640        }
641        JarFileEntry realEntry() {
642            if (isMultiRelease() && versionMajor != BASE_VERSION_MAJOR) {
643                String entryName = super.getName();
644                return entryName.equals(this.name) ? this : new JarFileEntry(entryName, this);
645            }
646            return this;
647        }
648        String realName() {
649            return super.getName();
650        }
651
652        @Override
653        public String getName() {
654            return name;
655        }
656    }
657
658    /*
659     * Ensures that the JarVerifier has been created if one is
660     * necessary (i.e., the jar appears to be signed.) This is done as
661     * a quick check to avoid processing of the manifest for unsigned
662     * jars.
663     */
664    private void maybeInstantiateVerifier() throws IOException {
665        if (jv != null) {
666            return;
667        }
668
669        if (verify) {
670            String[] names = getMetaInfEntryNames();
671            if (names != null) {
672                for (String nameLower : names) {
673                    String name = nameLower.toUpperCase(Locale.ENGLISH);
674                    if (name.endsWith(".DSA") ||
675                        name.endsWith(".RSA") ||
676                        name.endsWith(".EC") ||
677                        name.endsWith(".SF")) {
678                        // Assume since we found a signature-related file
679                        // that the jar is signed and that we therefore
680                        // need a JarVerifier and Manifest
681                        getManifest();
682                        return;
683                    }
684                }
685            }
686            // No signature-related files; don't instantiate a
687            // verifier
688            verify = false;
689        }
690    }
691
692
693    /*
694     * Initializes the verifier object by reading all the manifest
695     * entries and passing them to the verifier.
696     */
697    private void initializeVerifier() {
698        ManifestEntryVerifier mev = null;
699
700        // Verify "META-INF/" entries...
701        try {
702            String[] names = getMetaInfEntryNames();
703            if (names != null) {
704                for (String name : names) {
705                    String uname = name.toUpperCase(Locale.ENGLISH);
706                    if (MANIFEST_NAME.equals(uname)
707                            || SignatureFileVerifier.isBlockOrSF(uname)) {
708                        JarEntry e = getJarEntry(name);
709                        if (e == null) {
710                            throw new JarException("corrupted jar file");
711                        }
712                        if (mev == null) {
713                            mev = new ManifestEntryVerifier
714                                (getManifestFromReference());
715                        }
716                        byte[] b = getBytes(e);
717                        if (b != null && b.length > 0) {
718                            jv.beginEntry(e, mev);
719                            jv.update(b.length, b, 0, b.length, mev);
720                            jv.update(-1, null, 0, 0, mev);
721                        }
722                    }
723                }
724            }
725        } catch (IOException ex) {
726            // if we had an error parsing any blocks, just
727            // treat the jar file as being unsigned
728            jv = null;
729            verify = false;
730            if (JarVerifier.debug != null) {
731                JarVerifier.debug.println("jarfile parsing error!");
732                ex.printStackTrace();
733            }
734        }
735
736        // if after initializing the verifier we have nothing
737        // signed, we null it out.
738
739        if (jv != null) {
740
741            jv.doneWithMeta();
742            if (JarVerifier.debug != null) {
743                JarVerifier.debug.println("done with meta!");
744            }
745
746            if (jv.nothingToVerify()) {
747                if (JarVerifier.debug != null) {
748                    JarVerifier.debug.println("nothing to verify!");
749                }
750                jv = null;
751                verify = false;
752            }
753        }
754    }
755
756    /*
757     * Reads all the bytes for a given entry. Used to process the
758     * META-INF files.
759     */
760    private byte[] getBytes(ZipEntry ze) throws IOException {
761        try (InputStream is = super.getInputStream(ze)) {
762            int len = (int)ze.getSize();
763            int bytesRead;
764            byte[] b;
765            // trust specified entry sizes when reasonably small
766            if (len != -1 && len <= 65535) {
767                b = new byte[len];
768                bytesRead = is.readNBytes(b, 0, len);
769            } else {
770                b = is.readAllBytes();
771                bytesRead = b.length;
772            }
773            if (len != -1 && len != bytesRead) {
774                throw new EOFException("Expected:" + len + ", read:" + bytesRead);
775            }
776            return b;
777        }
778    }
779
780    /**
781     * Returns an input stream for reading the contents of the specified
782     * zip file entry.
783     * @param ze the zip file entry
784     * @return an input stream for reading the contents of the specified
785     *         zip file entry
786     * @throws ZipException if a zip file format error has occurred
787     * @throws IOException if an I/O error has occurred
788     * @throws SecurityException if any of the jar file entries
789     *         are incorrectly signed.
790     * @throws IllegalStateException
791     *         may be thrown if the jar file has been closed
792     */
793    public synchronized InputStream getInputStream(ZipEntry ze)
794        throws IOException
795    {
796        maybeInstantiateVerifier();
797        if (jv == null) {
798            return super.getInputStream(ze);
799        }
800        if (!jvInitialized) {
801            initializeVerifier();
802            jvInitialized = true;
803            // could be set to null after a call to
804            // initializeVerifier if we have nothing to
805            // verify
806            if (jv == null)
807                return super.getInputStream(ze);
808        }
809
810        // wrap a verifier stream around the real stream
811        return new JarVerifier.VerifierStream(
812            getManifestFromReference(),
813            verifiableEntry(ze),
814            super.getInputStream(ze),
815            jv);
816    }
817
818    private JarEntry verifiableEntry(ZipEntry ze) {
819        if (ze instanceof JarFileEntry) {
820            // assure the name and entry match for verification
821            return ((JarFileEntry)ze).realEntry();
822        }
823        ze = getJarEntry(ze.getName());
824        if (ze instanceof JarFileEntry) {
825            return ((JarFileEntry)ze).realEntry();
826        }
827        return (JarEntry)ze;
828    }
829
830    // Statics for hand-coded Boyer-Moore search
831    private static final byte[] CLASSPATH_CHARS =
832            {'C','L','A','S','S','-','P','A','T','H', ':', ' '};
833
834    // The bad character shift for "class-path: "
835    private static final byte[] CLASSPATH_LASTOCC;
836
837    // The good suffix shift for "class-path: "
838    private static final byte[] CLASSPATH_OPTOSFT;
839
840    private static final byte[] MULTIRELEASE_CHARS =
841            {'M','U','L','T','I','-','R','E','L','E', 'A', 'S', 'E', ':',
842                    ' ', 'T', 'R', 'U', 'E'};
843
844    // The bad character shift for "multi-release: true"
845    private static final byte[] MULTIRELEASE_LASTOCC;
846
847    // The good suffix shift for "multi-release: true"
848    private static final byte[] MULTIRELEASE_OPTOSFT;
849
850    static {
851        CLASSPATH_LASTOCC = new byte[64];
852        CLASSPATH_OPTOSFT = new byte[12];
853        CLASSPATH_LASTOCC[(int)'C' - 32] = 1;
854        CLASSPATH_LASTOCC[(int)'L' - 32] = 2;
855        CLASSPATH_LASTOCC[(int)'S' - 32] = 5;
856        CLASSPATH_LASTOCC[(int)'-' - 32] = 6;
857        CLASSPATH_LASTOCC[(int)'P' - 32] = 7;
858        CLASSPATH_LASTOCC[(int)'A' - 32] = 8;
859        CLASSPATH_LASTOCC[(int)'T' - 32] = 9;
860        CLASSPATH_LASTOCC[(int)'H' - 32] = 10;
861        CLASSPATH_LASTOCC[(int)':' - 32] = 11;
862        CLASSPATH_LASTOCC[(int)' ' - 32] = 12;
863        for (int i = 0; i < 11; i++) {
864            CLASSPATH_OPTOSFT[i] = 12;
865        }
866        CLASSPATH_OPTOSFT[11] = 1;
867
868        MULTIRELEASE_LASTOCC = new byte[64];
869        MULTIRELEASE_OPTOSFT = new byte[19];
870        MULTIRELEASE_LASTOCC[(int)'M' - 32] = 1;
871        MULTIRELEASE_LASTOCC[(int)'I' - 32] = 5;
872        MULTIRELEASE_LASTOCC[(int)'-' - 32] = 6;
873        MULTIRELEASE_LASTOCC[(int)'L' - 32] = 9;
874        MULTIRELEASE_LASTOCC[(int)'A' - 32] = 11;
875        MULTIRELEASE_LASTOCC[(int)'S' - 32] = 12;
876        MULTIRELEASE_LASTOCC[(int)':' - 32] = 14;
877        MULTIRELEASE_LASTOCC[(int)' ' - 32] = 15;
878        MULTIRELEASE_LASTOCC[(int)'T' - 32] = 16;
879        MULTIRELEASE_LASTOCC[(int)'R' - 32] = 17;
880        MULTIRELEASE_LASTOCC[(int)'U' - 32] = 18;
881        MULTIRELEASE_LASTOCC[(int)'E' - 32] = 19;
882        for (int i = 0; i < 17; i++) {
883            MULTIRELEASE_OPTOSFT[i] = 19;
884        }
885        MULTIRELEASE_OPTOSFT[17] = 6;
886        MULTIRELEASE_OPTOSFT[18] = 1;
887    }
888
889    private JarEntry getManEntry() {
890        if (manEntry == null) {
891            // First look up manifest entry using standard name
892            ZipEntry manEntry = super.getEntry(MANIFEST_NAME);
893            if (manEntry == null) {
894                // If not found, then iterate through all the "META-INF/"
895                // entries to find a match.
896                String[] names = getMetaInfEntryNames();
897                if (names != null) {
898                    for (String name : names) {
899                        if (MANIFEST_NAME.equals(name.toUpperCase(Locale.ENGLISH))) {
900                            manEntry = super.getEntry(name);
901                            break;
902                        }
903                    }
904                }
905            }
906            this.manEntry = (manEntry == null)
907                    ? null
908                    : new JarFileEntry(manEntry.getName(), manEntry);
909        }
910        return manEntry;
911    }
912
913   /**
914    * Returns {@code true} iff this JAR file has a manifest with the
915    * Class-Path attribute
916    */
917    boolean hasClassPathAttribute() throws IOException {
918        checkForSpecialAttributes();
919        return hasClassPathAttribute;
920    }
921
922    /**
923     * Returns true if the pattern {@code src} is found in {@code b}.
924     * The {@code lastOcc} array is the precomputed bad character shifts.
925     * Since there are no repeated substring in our search strings,
926     * the good suffix shifts can be replaced with a comparison.
927     */
928    private int match(byte[] src, byte[] b, byte[] lastOcc, byte[] optoSft) {
929        int len = src.length;
930        int last = b.length - len;
931        int i = 0;
932        next:
933        while (i <= last) {
934            for (int j = (len - 1); j >= 0; j--) {
935                byte c = b[i + j];
936                if (c >= ' ' && c <= 'z') {
937                    if (c >= 'a') c -= 32; // Canonicalize
938
939                    if (c != src[j]) {
940                        // no match
941                        int badShift = lastOcc[c - 32];
942                        i += Math.max(j + 1 - badShift, optoSft[j]);
943                        continue next;
944                    }
945                } else {
946                    // no match, character not valid for name
947                    i += len;
948                    continue next;
949                }
950            }
951            return i;
952        }
953        return -1;
954    }
955
956    /**
957     * On first invocation, check if the JAR file has the Class-Path
958     * and the Multi-Release attribute. A no-op on subsequent calls.
959     */
960    private void checkForSpecialAttributes() throws IOException {
961        if (hasCheckedSpecialAttributes) {
962            return;
963        }
964        synchronized (this) {
965            if (hasCheckedSpecialAttributes) {
966                return;
967            }
968            JarEntry manEntry = getManEntry();
969            if (manEntry != null) {
970                byte[] b = getBytes(manEntry);
971                hasClassPathAttribute = match(CLASSPATH_CHARS, b,
972                        CLASSPATH_LASTOCC, CLASSPATH_OPTOSFT) != -1;
973                // is this a multi-release jar file
974                if (MULTI_RELEASE_ENABLED) {
975                    int i = match(MULTIRELEASE_CHARS, b, MULTIRELEASE_LASTOCC,
976                            MULTIRELEASE_OPTOSFT);
977                    if (i != -1) {
978                        i += MULTIRELEASE_CHARS.length;
979                        if (i < b.length) {
980                            byte c = b[i++];
981                            // Check that the value is followed by a newline
982                            // and does not have a continuation
983                            if (c == '\n' &&
984                                    (i == b.length || b[i] != ' ')) {
985                                isMultiRelease = true;
986                            } else if (c == '\r') {
987                                if (i == b.length) {
988                                    isMultiRelease = true;
989                                } else {
990                                    c = b[i++];
991                                    if (c == '\n') {
992                                        if (i == b.length || b[i] != ' ') {
993                                            isMultiRelease = true;
994                                        }
995                                    } else if (c != ' ') {
996                                        isMultiRelease = true;
997                                    }
998                                }
999                            }
1000                        }
1001                    }
1002                }
1003            }
1004            hasCheckedSpecialAttributes = true;
1005        }
1006    }
1007
1008    private synchronized void ensureInitialization() {
1009        try {
1010            maybeInstantiateVerifier();
1011        } catch (IOException e) {
1012            throw new RuntimeException(e);
1013        }
1014        if (jv != null && !jvInitialized) {
1015            initializeVerifier();
1016            jvInitialized = true;
1017        }
1018    }
1019
1020    JarEntry newEntry(ZipEntry ze) {
1021        return new JarFileEntry(ze);
1022    }
1023
1024    Enumeration<String> entryNames(CodeSource[] cs) {
1025        ensureInitialization();
1026        if (jv != null) {
1027            return jv.entryNames(this, cs);
1028        }
1029
1030        /*
1031         * JAR file has no signed content. Is there a non-signing
1032         * code source?
1033         */
1034        boolean includeUnsigned = false;
1035        for (CodeSource c : cs) {
1036            if (c.getCodeSigners() == null) {
1037                includeUnsigned = true;
1038                break;
1039            }
1040        }
1041        if (includeUnsigned) {
1042            return unsignedEntryNames();
1043        } else {
1044            return new Enumeration<>() {
1045
1046                public boolean hasMoreElements() {
1047                    return false;
1048                }
1049
1050                public String nextElement() {
1051                    throw new NoSuchElementException();
1052                }
1053            };
1054        }
1055    }
1056
1057    /**
1058     * Returns an enumeration of the zip file entries
1059     * excluding internal JAR mechanism entries and including
1060     * signed entries missing from the ZIP directory.
1061     */
1062    Enumeration<JarEntry> entries2() {
1063        ensureInitialization();
1064        if (jv != null) {
1065            return jv.entries2(this, super.entries());
1066        }
1067
1068        // screen out entries which are never signed
1069        final Enumeration<? extends ZipEntry> enum_ = super.entries();
1070        return new Enumeration<>() {
1071
1072            ZipEntry entry;
1073
1074            public boolean hasMoreElements() {
1075                if (entry != null) {
1076                    return true;
1077                }
1078                while (enum_.hasMoreElements()) {
1079                    ZipEntry ze = enum_.nextElement();
1080                    if (JarVerifier.isSigningRelated(ze.getName())) {
1081                        continue;
1082                    }
1083                    entry = ze;
1084                    return true;
1085                }
1086                return false;
1087            }
1088
1089            public JarFileEntry nextElement() {
1090                if (hasMoreElements()) {
1091                    ZipEntry ze = entry;
1092                    entry = null;
1093                    return new JarFileEntry(ze);
1094                }
1095                throw new NoSuchElementException();
1096            }
1097        };
1098    }
1099
1100    CodeSource[] getCodeSources(URL url) {
1101        ensureInitialization();
1102        if (jv != null) {
1103            return jv.getCodeSources(this, url);
1104        }
1105
1106        /*
1107         * JAR file has no signed content. Is there a non-signing
1108         * code source?
1109         */
1110        Enumeration<String> unsigned = unsignedEntryNames();
1111        if (unsigned.hasMoreElements()) {
1112            return new CodeSource[]{JarVerifier.getUnsignedCS(url)};
1113        } else {
1114            return null;
1115        }
1116    }
1117
1118    private Enumeration<String> unsignedEntryNames() {
1119        final Enumeration<JarEntry> entries = entries();
1120        return new Enumeration<>() {
1121
1122            String name;
1123
1124            /*
1125             * Grab entries from ZIP directory but screen out
1126             * metadata.
1127             */
1128            public boolean hasMoreElements() {
1129                if (name != null) {
1130                    return true;
1131                }
1132                while (entries.hasMoreElements()) {
1133                    String value;
1134                    ZipEntry e = entries.nextElement();
1135                    value = e.getName();
1136                    if (e.isDirectory() || JarVerifier.isSigningRelated(value)) {
1137                        continue;
1138                    }
1139                    name = value;
1140                    return true;
1141                }
1142                return false;
1143            }
1144
1145            public String nextElement() {
1146                if (hasMoreElements()) {
1147                    String value = name;
1148                    name = null;
1149                    return value;
1150                }
1151                throw new NoSuchElementException();
1152            }
1153        };
1154    }
1155
1156    CodeSource getCodeSource(URL url, String name) {
1157        ensureInitialization();
1158        if (jv != null) {
1159            if (jv.eagerValidation) {
1160                CodeSource cs = null;
1161                JarEntry je = getJarEntry(name);
1162                if (je != null) {
1163                    cs = jv.getCodeSource(url, this, je);
1164                } else {
1165                    cs = jv.getCodeSource(url, name);
1166                }
1167                return cs;
1168            } else {
1169                return jv.getCodeSource(url, name);
1170            }
1171        }
1172
1173        return JarVerifier.getUnsignedCS(url);
1174    }
1175
1176    void setEagerValidation(boolean eager) {
1177        try {
1178            maybeInstantiateVerifier();
1179        } catch (IOException e) {
1180            throw new RuntimeException(e);
1181        }
1182        if (jv != null) {
1183            jv.setEagerValidation(eager);
1184        }
1185    }
1186
1187    List<Object> getManifestDigests() {
1188        ensureInitialization();
1189        if (jv != null) {
1190            return jv.getManifestDigests();
1191        }
1192        return new ArrayList<>();
1193    }
1194}
1195