src/jdk.jartool/share/classes/sun/tools/jar/Main.java
author weijun
Thu, 18 Jul 2019 08:53:06 +0800
changeset 57488 94691d8e746f
parent 52902 e3398b2e1ab0
permissions -rw-r--r--
8217375: jarsigner breaks old signature with long lines in manifest Reviewed-by: jjiang, weijun Contributed-by: Philipp Kunz <philipp.kunz@paratix.ch>
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
52723
f24ae8376e71 8210454: jar tool does not allow setting the module version without also setting the main class
lancea
parents: 51188
diff changeset
     2
 * Copyright (c) 1996, 2018, Oracle and/or its affiliates. All rights reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 3288
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 3288
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 3288
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 3288
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 3288
diff changeset
    23
 * questions.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
package sun.tools.jar;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import java.io.*;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
    29
import java.lang.module.Configuration;
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
    30
import java.lang.module.FindException;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    31
import java.lang.module.InvalidModuleDescriptorException;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    32
import java.lang.module.ModuleDescriptor;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    33
import java.lang.module.ModuleDescriptor.Exports;
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    34
import java.lang.module.ModuleDescriptor.Opens;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    35
import java.lang.module.ModuleDescriptor.Provides;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    36
import java.lang.module.ModuleDescriptor.Version;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    37
import java.lang.module.ModuleFinder;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
    38
import java.lang.module.ModuleReader;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    39
import java.lang.module.ModuleReference;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
    40
import java.lang.module.ResolvedModule;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    41
import java.net.URI;
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
    42
import java.nio.ByteBuffer;
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    43
import java.nio.file.Files;
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
    44
import java.nio.file.Path;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    45
import java.nio.file.Paths;
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
    46
import java.nio.file.StandardCopyOption;
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    47
import java.text.MessageFormat;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
import java.util.*;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    49
import java.util.function.Consumer;
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    50
import java.util.jar.Attributes;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    51
import java.util.jar.JarFile;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    52
import java.util.jar.JarOutputStream;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    53
import java.util.jar.Manifest;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    54
import java.util.regex.Pattern;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    55
import java.util.stream.Collectors;
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    56
import java.util.stream.Stream;
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    57
import java.util.zip.CRC32;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    58
import java.util.zip.ZipEntry;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    59
import java.util.zip.ZipFile;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    60
import java.util.zip.ZipInputStream;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    61
import java.util.zip.ZipOutputStream;
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
    62
import jdk.internal.module.Checks;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
    63
import jdk.internal.module.ModuleHashes;
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
    64
import jdk.internal.module.ModuleHashesBuilder;
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
    65
import jdk.internal.module.ModuleInfo;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    66
import jdk.internal.module.ModuleInfoExtender;
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
    67
import jdk.internal.module.ModuleResolution;
44359
c6761862ca0b 8174823: Module system implementation refresh (3/2017)
alanb
parents: 44033
diff changeset
    68
import jdk.internal.module.ModuleTarget;
36745
51effd3e92d0 8152190: Move sun.misc.JarIndex and InvalidJarIndexException to an internal package
chegar
parents: 36511
diff changeset
    69
import jdk.internal.util.jar.JarIndex;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
    70
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    71
import static java.nio.file.StandardCopyOption.REPLACE_EXISTING;
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
    72
import static java.util.jar.JarFile.MANIFEST_NAME;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
    73
import static java.util.stream.Collectors.joining;
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
    74
import static jdk.internal.util.jar.JarIndex.INDEX_NAME;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
 * This class implements a simple utility for creating files in the JAR
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
 * (Java Archive) file format. The JAR format is based on the ZIP file
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
 * format, with optional meta-information stored in a MANIFEST entry.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
 */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    81
public class Main {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
    String program;
41484
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
    83
    PrintWriter out, err;
19409
d7c7b9d56631 8022921: Remove experimental Profile attribute
alanb
parents: 15682
diff changeset
    84
    String fname, mname, ename;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    85
    String zname = "";
90ce3da70b43 Initial load
duke
parents:
diff changeset
    86
    String rootjar = null;
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
    87
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    88
    private static final int BASE_VERSION = 0;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    89
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    90
    private static class Entry {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    91
        final String name;
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    92
        final File file;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    93
        final boolean isDir;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    94
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    95
        Entry(File file, String name, boolean isDir) {
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
    96
            this.file = file;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    97
            this.isDir = isDir;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
    98
            this.name = name;
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
    99
        }
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   100
41221
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   101
        @Override
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   102
        public boolean equals(Object o) {
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   103
            if (this == o) return true;
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   104
            if (!(o instanceof Entry)) return false;
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   105
            return this.file.equals(((Entry)o).file);
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   106
        }
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   107
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   108
        @Override
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   109
        public int hashCode() {
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   110
            return file.hashCode();
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   111
        }
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   112
    }
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   113
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   114
    // An entryName(path)->Entry map generated during "expand", it helps to
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   115
    // decide whether or not an existing entry in a jar file needs to be
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   116
    // replaced, during the "update" operation.
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   117
    Map<String, Entry> entryMap = new HashMap<>();
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   118
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   119
    // All entries need to be added/updated.
41221
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   120
    Set<Entry> entries = new LinkedHashSet<>();
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   121
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   122
    // module-info.class entries need to be added/updated.
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   123
    Map<String,byte[]> moduleInfos = new HashMap<>();
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   124
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   125
    // A paths Set for each version, where each Set contains directories
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   126
    // specified by the "-C" operation.
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   127
    Map<Integer,Set<String>> pathsMap = new HashMap<>();
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   128
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   129
    // There's also a files array per version
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   130
    Map<Integer,String[]> filesMap = new HashMap<>();
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   131
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   132
    // Do we think this is a multi-release jar?  Set to true
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   133
    // if --release option found followed by at least file
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   134
    boolean isMultiRelease;
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   135
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   136
    // The last parsed --release value, if any. Used in conjunction with
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   137
    // "-d,--describe-module" to select the operative module descriptor.
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   138
    int releaseValue = -1;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   139
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
     * cflag: create
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
     * uflag: update
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
     * xflag: xtract
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
     * tflag: table
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
     * vflag: verbose
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
     * flag0: no zip compression (store only)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
     * Mflag: DO NOT generate a manifest file (just ZIP)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
     * iflag: generate jar index
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   149
     * nflag: Perform jar normalization at the end
29914
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
   150
     * pflag: preserve/don't strip leading slash and .. component from file name
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   151
     * dflag: print module descriptor
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
     */
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   153
    boolean cflag, uflag, xflag, tflag, vflag, flag0, Mflag, iflag, nflag, pflag, dflag;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   155
    boolean suppressDeprecateMsg = false;
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   156
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   157
    /* To support additional GNU Style informational options */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   158
    Consumer<PrintWriter> info;
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   159
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   160
    /* Modular jar related options */
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   161
    Version moduleVersion;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
   162
    Pattern modulesToHash;
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
   163
    ModuleResolution moduleResolution = ModuleResolution.empty();
38457
3d019217e322 8152650: ModuleFinder.compose should accept varargs
alanb
parents: 37779
diff changeset
   164
    ModuleFinder moduleFinder = ModuleFinder.of();
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   165
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   166
    static final String MODULE_INFO = "module-info.class";
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
    static final String MANIFEST_DIR = "META-INF/";
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   168
    static final String VERSIONS_DIR = MANIFEST_DIR + "versions/";
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
    static final String VERSION = "1.0";
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
   170
    static final int VERSIONS_DIR_LENGTH = VERSIONS_DIR.length();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
    private static ResourceBundle rsrc;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
     * If true, maintain compatibility with JDK releases prior to 6.0 by
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
     * timestamping extracted files with the time at which they are extracted.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
     * Default is to use the time given in the archive.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
    private static final boolean useExtractionTime =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
        Boolean.getBoolean("sun.tools.jar.useExtractionTime");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
     * Initialize ResourceBundle
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
    static {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
            rsrc = ResourceBundle.getBundle("sun.tools.jar.resources.jar");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
        } catch (MissingResourceException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
            throw new Error("Fatal: Resource for jar is missing");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   192
    static String getMsg(String key) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
            return (rsrc.getString(key));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
        } catch (MissingResourceException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
            throw new Error("Error in message file");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   200
    static String formatMsg(String key, String arg) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   201
        String msg = getMsg(key);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
        String[] args = new String[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
        args[0] = arg;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
        return MessageFormat.format(msg, (Object[]) args);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   207
    static String formatMsg2(String key, String arg, String arg1) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
        String msg = getMsg(key);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
        String[] args = new String[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
        args[0] = arg;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
        args[1] = arg1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
        return MessageFormat.format(msg, (Object[]) args);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   213
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
    public Main(PrintStream out, PrintStream err, String program) {
41484
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
   216
        this.out = new PrintWriter(out, true);
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
   217
        this.err = new PrintWriter(err, true);
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
   218
        this.program = program;
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
   219
    }
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
   220
834b7539ada3 8164689: Retrofit jar, jlink, jmod as a ToolProvider
mchung
parents: 41221
diff changeset
   221
    public Main(PrintWriter out, PrintWriter err, String program) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
        this.out = out;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
        this.err = err;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
        this.program = program;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   227
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   228
     * Creates a new empty temporary file in the same directory as the
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   229
     * specified file.  A variant of File.createTempFile.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   230
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   231
    private static File createTempFileInSameDirectoryAs(File file)
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   232
        throws IOException {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   233
        File dir = file.getParentFile();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   234
        if (dir == null)
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   235
            dir = new File(".");
13154
9e8a04d28ded 7175845: jar uf changes file permissions unexpectedly
sherman
parents: 13041
diff changeset
   236
        return File.createTempFile("jartmp", null, dir);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   237
    }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   238
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
    private boolean ok;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   241
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
     * Starts main program with the specified arguments.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
     */
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   244
    @SuppressWarnings({"removal"})
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
    public synchronized boolean run(String args[]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
        ok = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
        if (!parseArgs(args)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
            return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
        }
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   250
        File tmpFile = null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   252
            if (cflag || uflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   253
                if (fname != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
                    // The name of the zip file as it would appear as its own
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
                    // zip file entry. We use this to make sure that we don't
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
                    // add the zip file to itself.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
                    zname = fname.replace(File.separatorChar, '/');
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
                    if (zname.startsWith("./")) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
                        zname = zname.substring(2);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
            if (cflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
                Manifest manifest = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
                if (!Mflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
                    if (mname != null) {
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   267
                        try (InputStream in = new FileInputStream(mname)) {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   268
                            manifest = new Manifest(new BufferedInputStream(in));
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   269
                        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
                    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
                        manifest = new Manifest();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
                    addVersion(manifest);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
                    addCreatedBy(manifest);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   275
                    if (isAmbiguousMainClass(manifest)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
                        return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
                    if (ename != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
                        addMainClass(manifest, ename);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
                    }
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   281
                    if (isMultiRelease) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   282
                        addMultiRelease(manifest);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   283
                    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
                }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   285
                expand();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   286
                if (!moduleInfos.isEmpty()) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   287
                    // All actual file entries (excl manifest and module-info.class)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   288
                    Set<String> jentries = new HashSet<>();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   289
                    // all packages if it's a class or resource
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   290
                    Set<String> packages = new HashSet<>();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   291
                    entries.stream()
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   292
                           .filter(e -> !e.isDir)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   293
                           .forEach( e -> {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   294
                               addPackageIfNamed(packages, e.name);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   295
                               jentries.add(e.name);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   296
                    });
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   297
                    addExtendedModuleAttributes(moduleInfos, packages);
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   298
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   299
                    // Basic consistency checks for modular jars.
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   300
                    if (!checkModuleInfo(moduleInfos.get(MODULE_INFO), jentries))
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   301
                        return false;
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   302
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
   303
                } else if (moduleVersion != null || modulesToHash != null) {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   304
                    error(getMsg("error.module.options.without.info"));
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   305
                    return false;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   306
                }
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   307
                if (vflag && fname == null) {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   308
                    // Disable verbose output so that it does not appear
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   309
                    // on stdout along with file data
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   310
                    // error("Warning: -v option ignored");
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   311
                    vflag = false;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
                }
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   313
                final String tmpbase = (fname == null)
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   314
                        ? "tmpjar"
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   315
                        : fname.substring(fname.indexOf(File.separatorChar) + 1);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   316
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   317
                tmpFile = createTemporaryFile(tmpbase, ".jar");
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   318
                try (OutputStream out = new FileOutputStream(tmpFile)) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   319
                    create(new BufferedOutputStream(out, 4096), manifest);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
                }
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   321
                if (nflag) {
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   322
                    if (!suppressDeprecateMsg) {
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   323
                        warn(formatMsg("warn.flag.is.deprecated", "-n"));
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   324
                    }
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   325
                    File packFile = createTemporaryFile(tmpbase, ".pack");
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   326
                    try {
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   327
                        java.util.jar.Pack200.Packer packer = java.util.jar.Pack200.newPacker();
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   328
                        Map<String, String> p = packer.properties();
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   329
                        p.put(java.util.jar.Pack200.Packer.EFFORT, "1"); // Minimal effort to conserve CPU
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   330
                        try (JarFile jarFile = new JarFile(tmpFile.getCanonicalPath());
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   331
                             OutputStream pack = new FileOutputStream(packFile))
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   332
                        {
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   333
                            packer.pack(jarFile, pack);
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   334
                        }
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   335
                        if (tmpFile.exists()) {
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   336
                            tmpFile.delete();
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   337
                        }
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   338
                        tmpFile = createTemporaryFile(tmpbase, ".jar");
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   339
                        try (OutputStream out = new FileOutputStream(tmpFile);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   340
                             JarOutputStream jos = new JarOutputStream(out))
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   341
                        {
50603
95c0644a1c47 8199871: Deprecate pack200 and unpack200 tools
henryjen
parents: 50130
diff changeset
   342
                            java.util.jar.Pack200.Unpacker unpacker = java.util.jar.Pack200.newUnpacker();
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   343
                            unpacker.unpack(packFile, jos);
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   344
                        }
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   345
                    } finally {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   346
                        Files.deleteIfExists(packFile.toPath());
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   347
                    }
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
   348
                }
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   349
                validateAndClose(tmpFile);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
            } else if (uflag) {
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   351
                File inputFile = null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
                if (fname != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   353
                    inputFile = new File(fname);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   354
                    tmpFile = createTempFileInSameDirectoryAs(inputFile);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
                    vflag = false;
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   357
                    tmpFile = createTemporaryFile("tmpjar", ".jar");
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
                }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   359
                expand();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   360
                try (FileInputStream in = (fname != null) ? new FileInputStream(inputFile)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   361
                        : new FileInputStream(FileDescriptor.in);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   362
                     FileOutputStream out = new FileOutputStream(tmpFile);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   363
                     InputStream manifest = (!Mflag && (mname != null)) ?
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   364
                            (new FileInputStream(mname)) : null;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   365
                ) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   366
                    boolean updateOk = update(in, new BufferedOutputStream(out),
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   367
                        manifest, moduleInfos, null);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   368
                    if (ok) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   369
                        ok = updateOk;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   370
                    }
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   371
                }
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   372
                validateAndClose(tmpFile);
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   373
            } else if (tflag) {
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   374
                replaceFSC(filesMap);
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   375
                // For the "list table contents" action, access using the
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   376
                // ZipFile class is always most efficient since only a
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   377
                // "one-finger" scan through the central directory is required.
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   378
                String[] files = filesMapToFiles(filesMap);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
                if (fname != null) {
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   380
                    list(fname, files);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
                } else {
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   382
                    InputStream in = new FileInputStream(FileDescriptor.in);
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   383
                    try {
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   384
                        list(new BufferedInputStream(in), files);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   385
                    } finally {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   386
                        in.close();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   387
                    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   388
                }
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   389
            } else if (xflag) {
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   390
                replaceFSC(filesMap);
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   391
                // For the extract action, when extracting all the entries,
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   392
                // access using the ZipInputStream class is most efficient,
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   393
                // since only a single sequential scan through the zip file is
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   394
                // required.  When using the ZipFile class, a "two-finger" scan
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   395
                // is required, but this is likely to be more efficient when a
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   396
                // partial extract is requested.  In case the zip file has
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   397
                // "leading garbage", we fall back from the ZipInputStream
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   398
                // implementation to the ZipFile implementation, since only the
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   399
                // latter can handle it.
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   400
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   401
                String[] files = filesMapToFiles(filesMap);
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   402
                if (fname != null && files != null) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   403
                    extract(fname, files);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
                } else {
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   405
                    InputStream in = (fname == null)
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   406
                        ? new FileInputStream(FileDescriptor.in)
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   407
                        : new FileInputStream(fname);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   408
                    try {
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   409
                        if (!extract(new BufferedInputStream(in), files) && fname != null) {
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   410
                            extract(fname, files);
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
   411
                        }
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   412
                    } finally {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   413
                        in.close();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
   414
                    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
            } else if (iflag) {
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   417
                String[] files = filesMap.get(BASE_VERSION);  // base entries only, can be null
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
                genIndex(rootjar, files);
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   419
            } else if (dflag) {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   420
                boolean found;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   421
                if (fname != null) {
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   422
                    try (ZipFile zf = new ZipFile(fname)) {
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
   423
                        found = describeModule(zf);
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   424
                    }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   425
                } else {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   426
                    try (FileInputStream fin = new FileInputStream(FileDescriptor.in)) {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   427
                        found = describeModuleFromStream(fin);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   428
                    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   429
                }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   430
                if (!found)
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   431
                    error(getMsg("error.module.descriptor.not.found"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
        } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
            fatalError(e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
            ok = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
        } catch (Error ee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
            ee.printStackTrace();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
            ok = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
        } catch (Throwable t) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
            t.printStackTrace();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
            ok = false;
44033
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   442
        } finally {
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   443
            if (tmpFile != null && tmpFile.exists())
ab43ed5eab17 8174996: jar leaves temporary file when exception occur in creating jar
sherman
parents: 43731
diff changeset
   444
                tmpFile.delete();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
        err.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
        return ok;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   450
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   451
    private void validateAndClose(File tmpfile) throws IOException {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   452
        if (ok && isMultiRelease) {
47958
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
   453
            try (ZipFile zf = new ZipFile(tmpfile)) {
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
   454
                ok = Validator.validate(this, zf);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   455
                if (!ok) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   456
                    error(formatMsg("error.validator.jarfile.invalid", fname));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   457
                }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   458
            } catch (IOException e) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   459
                error(formatMsg2("error.validator.jarfile.exception", fname, e.getMessage()));
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   460
            }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   461
        }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   462
        Path path = tmpfile.toPath();
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   463
        try {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   464
            if (ok) {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   465
                if (fname != null) {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   466
                    Files.move(path, Paths.get(fname), StandardCopyOption.REPLACE_EXISTING);
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   467
                } else {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   468
                    Files.copy(path, new FileOutputStream(FileDescriptor.out));
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   469
                }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   470
            }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   471
        } finally {
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   472
            Files.deleteIfExists(path);
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   473
        }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   474
    }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   475
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   476
    private String[] filesMapToFiles(Map<Integer,String[]> filesMap) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   477
        if (filesMap.isEmpty()) return null;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   478
        return filesMap.entrySet()
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   479
                .stream()
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   480
                .flatMap(this::filesToEntryNames)
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   481
                .toArray(String[]::new);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   482
    }
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   483
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   484
    Stream<String> filesToEntryNames(Map.Entry<Integer,String[]> fileEntries) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   485
        int version = fileEntries.getKey();
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   486
        Set<String> cpaths = pathsMap.get(version);
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   487
        return Stream.of(fileEntries.getValue())
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   488
            .map(f -> toVersionedName(toEntryName(f, cpaths, false), version));
40251
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   489
    }
481b890e50a3 8158295: Add a multi-release jar validation mechanism to jar tool
sdrach
parents: 39313
diff changeset
   490
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   491
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   492
     * Parses command line arguments.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
    boolean parseArgs(String args[]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
        /* Preprocess and expand @file arguments */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
            args = CommandLine.parse(args);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
        } catch (FileNotFoundException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
            fatalError(formatMsg("error.cant.open", e.getMessage()));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
            return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
        } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
            fatalError(e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   503
            return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
        /* parse flags */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
        int count = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
            String flags = args[0];
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   509
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   510
            // Note: flags.length == 2 can be treated as the short version of
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   511
            // the GNU option since the there cannot be any other options,
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   512
            // excluding -C, as per the old way.
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   513
            if (flags.startsWith("--") ||
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   514
                (flags.startsWith("-") && flags.length() == 2)) {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   515
                try {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   516
                    count = GNUStyleOptions.parseOptions(this, args);
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   517
                } catch (GNUStyleOptions.BadArgs x) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   518
                    if (info == null) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   519
                        if (x.showUsage) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   520
                            usageError(x.getMessage());
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   521
                        } else {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   522
                            error(x.getMessage());
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   523
                        }
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   524
                        return false;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
                    }
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   526
                }
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   527
                if (info != null) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   528
                    info.accept(out);
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   529
                    return true;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   530
                }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   531
            } else {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   532
                // Legacy/compatibility options
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   533
                if (flags.startsWith("-")) {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   534
                    flags = flags.substring(1);
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   535
                }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   536
                for (int i = 0; i < flags.length(); i++) {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   537
                    switch (flags.charAt(i)) {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   538
                        case 'c':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   539
                            if (xflag || tflag || uflag || iflag) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   540
                                usageError(getMsg("error.multiple.main.operations"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   541
                                return false;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   542
                            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   543
                            cflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   544
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   545
                        case 'u':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   546
                            if (cflag || xflag || tflag || iflag) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   547
                                usageError(getMsg("error.multiple.main.operations"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   548
                                return false;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   549
                            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   550
                            uflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   551
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   552
                        case 'x':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   553
                            if (cflag || uflag || tflag || iflag) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   554
                                usageError(getMsg("error.multiple.main.operations"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   555
                                return false;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   556
                            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   557
                            xflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   558
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   559
                        case 't':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   560
                            if (cflag || uflag || xflag || iflag) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   561
                                usageError(getMsg("error.multiple.main.operations"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   562
                                return false;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   563
                            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   564
                            tflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   565
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   566
                        case 'M':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   567
                            Mflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   568
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   569
                        case 'v':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   570
                            vflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   571
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   572
                        case 'f':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   573
                            fname = args[count++];
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   574
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   575
                        case 'm':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   576
                            mname = args[count++];
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   577
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   578
                        case '0':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   579
                            flag0 = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   580
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   581
                        case 'i':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   582
                            if (cflag || uflag || xflag || tflag) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   583
                                usageError(getMsg("error.multiple.main.operations"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   584
                                return false;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   585
                            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   586
                            // do not increase the counter, files will contain rootjar
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   587
                            rootjar = args[count++];
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   588
                            iflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   589
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   590
                        case 'n':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   591
                            nflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   592
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   593
                        case 'e':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   594
                            ename = args[count++];
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   595
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   596
                        case 'P':
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   597
                            pflag = true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   598
                            break;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   599
                        default:
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   600
                            usageError(formatMsg("error.illegal.option",
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   601
                                       String.valueOf(flags.charAt(i))));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   602
                            return false;
5620
ed56433bf5d6 4690407: JAR tool: option -i can't be combined with other options
sherman
parents: 3288
diff changeset
   603
                    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   604
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   605
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   606
        } catch (ArrayIndexOutOfBoundsException e) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   607
            usageError(getMsg("main.usage.summary"));
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   608
            return false;
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   609
        }
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   610
        if (!cflag && !tflag && !xflag && !uflag && !iflag && !dflag) {
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   611
            usageError(getMsg("error.bad.option"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   612
            return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   613
        }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   614
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   615
        /* parse file arguments */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   616
        int n = args.length - count;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   617
        if (n > 0) {
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   618
            int version = BASE_VERSION;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   619
            int k = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   620
            String[] nameBuf = new String[n];
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   621
            pathsMap.put(version, new HashSet<>());
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   622
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   623
                for (int i = count; i < args.length; i++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   624
                    if (args[i].equals("-C")) {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   625
                        if (dflag) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   626
                            // "--describe-module/-d" does not require file argument(s),
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   627
                            // but does accept --release
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   628
                            usageError(getMsg("error.bad.dflag"));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   629
                            return false;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   630
                        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
                        /* change the directory */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
                        String dir = args[++i];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
                        dir = (dir.endsWith(File.separator) ?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
                               dir : (dir + File.separator));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
                        dir = dir.replace(File.separatorChar, '/');
51188
630b5e06a947 8207395: jar has issues with UNC-path arguments for the jar -C parameter [windows]
mbaesken
parents: 50603
diff changeset
   636
630b5e06a947 8207395: jar has issues with UNC-path arguments for the jar -C parameter [windows]
mbaesken
parents: 50603
diff changeset
   637
                        boolean hasUNC = (File.separatorChar == '\\'&&  dir.startsWith("//"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
                        while (dir.indexOf("//") > -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
                            dir = dir.replace("//", "/");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
                        }
51188
630b5e06a947 8207395: jar has issues with UNC-path arguments for the jar -C parameter [windows]
mbaesken
parents: 50603
diff changeset
   641
                        if (hasUNC) { // Restore Windows UNC path.
630b5e06a947 8207395: jar has issues with UNC-path arguments for the jar -C parameter [windows]
mbaesken
parents: 50603
diff changeset
   642
                            dir = "/" + dir;
630b5e06a947 8207395: jar has issues with UNC-path arguments for the jar -C parameter [windows]
mbaesken
parents: 50603
diff changeset
   643
                        }
630b5e06a947 8207395: jar has issues with UNC-path arguments for the jar -C parameter [windows]
mbaesken
parents: 50603
diff changeset
   644
                        pathsMap.get(version).add(dir);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
                        nameBuf[k++] = dir + args[++i];
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   646
                    } else if (args[i].startsWith("--release")) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   647
                        int v = BASE_VERSION;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   648
                        try {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   649
                            v = Integer.valueOf(args[++i]);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   650
                        } catch (NumberFormatException x) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   651
                            error(formatMsg("error.release.value.notnumber", args[i]));
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   652
                            // this will fall into the next error, thus returning false
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   653
                        }
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   654
                        if (v < 9) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   655
                            usageError(formatMsg("error.release.value.toosmall", String.valueOf(v)));
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   656
                            return false;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   657
                        }
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   658
                        // associate the files, if any, with the previous version number
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   659
                        if (k > 0) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   660
                            String[] files = new String[k];
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   661
                            System.arraycopy(nameBuf, 0, files, 0, k);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   662
                            filesMap.put(version, files);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   663
                            isMultiRelease = version > BASE_VERSION;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   664
                        }
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   665
                        // reset the counters and start with the new version number
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   666
                        k = 0;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   667
                        nameBuf = new String[n];
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   668
                        version = v;
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   669
                        releaseValue = version;
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   670
                        pathsMap.put(version, new HashSet<>());
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
                    } else {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   672
                        if (dflag) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   673
                            // "--describe-module/-d" does not require file argument(s),
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   674
                            // but does accept --release
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   675
                            usageError(getMsg("error.bad.dflag"));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   676
                            return false;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   677
                        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   678
                        nameBuf[k++] = args[i];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   679
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   680
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   681
            } catch (ArrayIndexOutOfBoundsException e) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   682
                usageError(getMsg("error.bad.file.arg"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   683
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   684
            }
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   685
            // associate remaining files, if any, with a version
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   686
            if (k > 0) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   687
                String[] files = new String[k];
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   688
                System.arraycopy(nameBuf, 0, files, 0, k);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   689
                filesMap.put(version, files);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   690
                isMultiRelease = version > BASE_VERSION;
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   691
            }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   692
        } else if (cflag && (mname == null)) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   693
            usageError(getMsg("error.bad.cflag"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   694
            return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   695
        } else if (uflag) {
52723
f24ae8376e71 8210454: jar tool does not allow setting the module version without also setting the main class
lancea
parents: 51188
diff changeset
   696
            if ((mname != null) || (ename != null) || moduleVersion != null) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   697
                /* just want to update the manifest */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   698
                return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   699
            } else {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
   700
                usageError(getMsg("error.bad.uflag"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   701
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   702
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   703
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   704
        return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   705
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   706
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   707
    /*
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   708
     * Add the package of the given resource name if it's a .class
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   709
     * or a resource in a named package.
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   710
     */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   711
    void addPackageIfNamed(Set<String> packages, String name) {
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   712
        if (name.startsWith(VERSIONS_DIR)) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   713
            // trim the version dir prefix
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
   714
            int i0 = VERSIONS_DIR_LENGTH;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   715
            int i = name.indexOf('/', i0);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   716
            if (i <= 0) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   717
                warn(formatMsg("warn.release.unexpected.versioned.entry", name));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   718
                return;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   719
            }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   720
            while (i0 < i) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   721
                char c = name.charAt(i0);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   722
                if (c < '0' || c > '9') {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   723
                    warn(formatMsg("warn.release.unexpected.versioned.entry", name));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   724
                    return;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   725
                }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   726
                i0++;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   727
            }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   728
            name = name.substring(i + 1, name.length());
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   729
        }
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   730
        String pn = toPackageName(name);
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   731
        // add if this is a class or resource in a package
43712
5dfd0950317c 8173393: Module system implementation refresh (2/2017)
alanb
parents: 43109
diff changeset
   732
        if (Checks.isPackageName(pn)) {
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   733
            packages.add(pn);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   734
        }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   735
    }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   736
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   737
    private String toEntryName(String name, Set<String> cpaths, boolean isDir) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   738
        name = name.replace(File.separatorChar, '/');
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   739
        if (isDir) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   740
            name = name.endsWith("/") ? name : name + "/";
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   741
        }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   742
        String matchPath = "";
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   743
        for (String path : cpaths) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   744
            if (name.startsWith(path) && path.length() > matchPath.length()) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   745
                matchPath = path;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   746
            }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   747
        }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   748
        name = safeName(name.substring(matchPath.length()));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   749
        // the old implementaton doesn't remove
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   750
        // "./" if it was led by "/" (?)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   751
        if (name.startsWith("./")) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   752
            name = name.substring(2);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   753
        }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   754
        return name;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   755
    }
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   756
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   757
    private static String toVersionedName(String name, int version) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   758
        return version > BASE_VERSION
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   759
                ? VERSIONS_DIR + version + "/" + name : name;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   760
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   761
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   762
    private static String toPackageName(String path) {
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   763
        int index = path.lastIndexOf('/');
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   764
        if (index != -1) {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   765
            return path.substring(0, index).replace('/', '.');
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   766
        } else {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   767
            return "";
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   768
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   769
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   770
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   771
    private void expand() throws IOException {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   772
        for (int version : filesMap.keySet()) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   773
            String[] files = filesMap.get(version);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   774
            expand(null, files, pathsMap.get(version), version);
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   775
        }
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   776
    }
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   777
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   778
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   779
     * Expands list of files to process into full list of all files that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   780
     * can be found by recursively descending directories.
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   781
     *
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   782
     * @param dir    parent directory
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
   783
     * @param files  list of files to expand
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   784
     * @param cpaths set of directories specified by -C option for the files
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   785
     * @throws IOException if an I/O error occurs
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   786
     */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   787
    private void expand(File dir, String[] files, Set<String> cpaths, int version)
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   788
        throws IOException
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   789
    {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   790
        if (files == null)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   791
            return;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   792
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   793
        for (int i = 0; i < files.length; i++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   794
            File f;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   795
            if (dir == null)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   796
                f = new File(files[i]);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   797
            else
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   798
                f = new File(dir, files[i]);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   799
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   800
            boolean isDir = f.isDirectory();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   801
            String name = toEntryName(f.getPath(), cpaths, isDir);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   802
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   803
            if (version != BASE_VERSION) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   804
                if (name.startsWith(VERSIONS_DIR)) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   805
                    // the entry starts with VERSIONS_DIR and version != BASE_VERSION,
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   806
                    // which means the "[dirs|files]" in --release v [dirs|files]
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   807
                    // includes VERSIONS_DIR-ed entries --> warning and skip (?)
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   808
                    error(formatMsg2("error.release.unexpected.versioned.entry",
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   809
                                     name, String.valueOf(version)));
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   810
                    ok = false;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   811
                    return;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   812
                }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   813
                name = toVersionedName(name, version);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   814
            }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   815
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   816
            if (f.isFile()) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   817
                Entry e = new Entry(f, name, false);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   818
                if (isModuleInfoEntry(name)) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   819
                    moduleInfos.putIfAbsent(name, Files.readAllBytes(f.toPath()));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   820
                    if (uflag)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   821
                        entryMap.put(name, e);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   822
                } else if (entries.add(e)) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   823
                    if (uflag)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   824
                        entryMap.put(name, e);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   825
                }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   826
            } else if (isDir) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   827
                Entry e = new Entry(f, name, true);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   828
                if (entries.add(e)) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   829
                    // utilize entryMap for the duplicate dir check even in
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   830
                    // case of cflag == true.
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   831
                    // dir name confilict/duplicate could happen with -C option.
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   832
                    // just remove the last "e" from the "entries" (zos will fail
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   833
                    // with "duplicated" entries), but continue expanding the
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   834
                    // sub tree
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   835
                    if (entryMap.containsKey(name)) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   836
                        entries.remove(e);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   837
                    } else {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   838
                        entryMap.put(name, e);
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   839
                    }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   840
                    expand(f, f.list(), cpaths, version);
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   841
                }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   842
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   843
                error(formatMsg("error.nosuch.fileordir", String.valueOf(f)));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   844
                ok = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   845
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   846
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   847
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   848
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   849
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   850
     * Creates a new JAR file.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   851
     */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   852
    void create(OutputStream out, Manifest manifest) throws IOException
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   853
    {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   854
        try (ZipOutputStream zos = new JarOutputStream(out)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   855
            if (flag0) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   856
                zos.setMethod(ZipOutputStream.STORED);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   857
            }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   858
            // TODO: check module-info attributes against manifest ??
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   859
            if (manifest != null) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   860
                if (vflag) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   861
                    output(getMsg("out.added.manifest"));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   862
                }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   863
                ZipEntry e = new ZipEntry(MANIFEST_DIR);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   864
                e.setTime(System.currentTimeMillis());
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   865
                e.setSize(0);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   866
                e.setCrc(0);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   867
                zos.putNextEntry(e);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   868
                e = new ZipEntry(MANIFEST_NAME);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   869
                e.setTime(System.currentTimeMillis());
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   870
                if (flag0) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   871
                    crc32Manifest(e, manifest);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   872
                }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   873
                zos.putNextEntry(e);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   874
                manifest.write(zos);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   875
                zos.closeEntry();
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   876
            }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   877
            updateModuleInfo(moduleInfos, zos);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   878
            for (Entry entry : entries) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   879
                addFile(zos, entry);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   880
            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
   881
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   882
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   883
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   884
    private char toUpperCaseASCII(char c) {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   885
        return (c < 'a' || c > 'z') ? c : (char) (c + 'A' - 'a');
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   886
    }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   887
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   888
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   889
     * Compares two strings for equality, ignoring case.  The second
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   890
     * argument must contain only upper-case ASCII characters.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   891
     * We don't want case comparison to be locale-dependent (else we
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   892
     * have the notorious "turkish i bug").
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   893
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   894
    private boolean equalsIgnoreCase(String s, String upper) {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   895
        assert upper.toUpperCase(java.util.Locale.ENGLISH).equals(upper);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   896
        int len;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   897
        if ((len = s.length()) != upper.length())
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   898
            return false;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   899
        for (int i = 0; i < len; i++) {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   900
            char c1 = s.charAt(i);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   901
            char c2 = upper.charAt(i);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   902
            if (c1 != c2 && toUpperCaseASCII(c1) != c2)
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   903
                return false;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   904
        }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   905
        return true;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   906
    }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   907
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   908
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   909
     * Updates an existing jar file.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   910
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   911
    boolean update(InputStream in, OutputStream out,
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   912
                   InputStream newManifest,
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   913
                   Map<String,byte[]> moduleInfos,
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   914
                   JarIndex jarIndex) throws IOException
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   915
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   916
        ZipInputStream zis = new ZipInputStream(in);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   917
        ZipOutputStream zos = new JarOutputStream(out);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   918
        ZipEntry e = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   919
        boolean foundManifest = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   920
        boolean updateOk = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   921
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   922
        // All actual entries added/updated/existing, in the jar file (excl manifest
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   923
        // and module-info.class ).
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   924
        Set<String> jentries = new HashSet<>();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   925
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   926
        if (jarIndex != null) {
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   927
            addIndex(jarIndex, zos);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   928
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   929
90ce3da70b43 Initial load
duke
parents:
diff changeset
   930
        // put the old entries first, replace if necessary
90ce3da70b43 Initial load
duke
parents:
diff changeset
   931
        while ((e = zis.getNextEntry()) != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   932
            String name = e.getName();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   933
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   934
            boolean isManifestEntry = equalsIgnoreCase(name, MANIFEST_NAME);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   935
            boolean isModuleInfoEntry = isModuleInfoEntry(name);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   936
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   937
            if ((jarIndex != null && equalsIgnoreCase(name, INDEX_NAME))
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   938
                || (Mflag && isManifestEntry)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   939
                continue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   940
            } else if (isManifestEntry && ((newManifest != null) ||
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   941
                        (ename != null) || isMultiRelease)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   942
                foundManifest = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   943
                if (newManifest != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   944
                    // Don't read from the newManifest InputStream, as we
90ce3da70b43 Initial load
duke
parents:
diff changeset
   945
                    // might need it below, and we can't re-read the same data
90ce3da70b43 Initial load
duke
parents:
diff changeset
   946
                    // twice.
57488
94691d8e746f 8217375: jarsigner breaks old signature with long lines in manifest
weijun
parents: 52902
diff changeset
   947
                    try (FileInputStream fis = new FileInputStream(mname)) {
94691d8e746f 8217375: jarsigner breaks old signature with long lines in manifest
weijun
parents: 52902
diff changeset
   948
                        if (isAmbiguousMainClass(new Manifest(fis))) {
94691d8e746f 8217375: jarsigner breaks old signature with long lines in manifest
weijun
parents: 52902
diff changeset
   949
                            return false;
94691d8e746f 8217375: jarsigner breaks old signature with long lines in manifest
weijun
parents: 52902
diff changeset
   950
                        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   951
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   952
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   953
                // Update the manifest.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   954
                Manifest old = new Manifest(zis);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   955
                if (newManifest != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   956
                    old.read(newManifest);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   957
                }
15682
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
   958
                if (!updateManifest(old, zos)) {
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
   959
                    return false;
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
   960
                }
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
   961
            } else if (moduleInfos != null && isModuleInfoEntry) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   962
                moduleInfos.putIfAbsent(name, zis.readAllBytes());
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   963
            } else {
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   964
                boolean isDir = e.isDirectory();
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   965
                if (!entryMap.containsKey(name)) { // copy the old stuff
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   966
                    // do our own compression
90ce3da70b43 Initial load
duke
parents:
diff changeset
   967
                    ZipEntry e2 = new ZipEntry(name);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   968
                    e2.setMethod(e.getMethod());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   969
                    e2.setTime(e.getTime());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   970
                    e2.setComment(e.getComment());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   971
                    e2.setExtra(e.getExtra());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   972
                    if (e.getMethod() == ZipEntry.STORED) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   973
                        e2.setSize(e.getSize());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   974
                        e2.setCrc(e.getCrc());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   975
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   976
                    zos.putNextEntry(e2);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
   977
                    copy(zis, zos);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   978
                } else { // replace with the new files
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   979
                    Entry ent = entryMap.get(name);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
   980
                    addFile(zos, ent);
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
   981
                    entryMap.remove(name);
41221
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   982
                    entries.remove(ent);
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   983
                    isDir = ent.isDir;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   984
                }
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   985
                if (!isDir) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   986
                    jentries.add(name);
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
   987
                }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   988
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   989
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   990
90ce3da70b43 Initial load
duke
parents:
diff changeset
   991
        // add the remaining new files
41221
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   992
        for (Entry entry : entries) {
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
   993
            addFile(zos, entry);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   994
            if (!entry.isDir) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   995
                jentries.add(entry.name);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
   996
            }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   997
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   998
        if (!foundManifest) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   999
            if (newManifest != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1000
                Manifest m = new Manifest(newManifest);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1001
                updateOk = !isAmbiguousMainClass(m);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1002
                if (updateOk) {
15682
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1003
                    if (!updateManifest(m, zos)) {
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1004
                        updateOk = false;
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1005
                    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1006
                }
19409
d7c7b9d56631 8022921: Remove experimental Profile attribute
alanb
parents: 15682
diff changeset
  1007
            } else if (ename != null) {
15682
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1008
                if (!updateManifest(new Manifest(), zos)) {
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1009
                    updateOk = false;
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1010
                }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1011
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1012
        }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1013
        if (updateOk) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1014
            if (moduleInfos != null && !moduleInfos.isEmpty()) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1015
                Set<String> pkgs = new HashSet<>();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1016
                jentries.forEach( je -> addPackageIfNamed(pkgs, je));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1017
                addExtendedModuleAttributes(moduleInfos, pkgs);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1018
                updateOk = checkModuleInfo(moduleInfos.get(MODULE_INFO), jentries);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1019
                updateModuleInfo(moduleInfos, zos);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1020
                // TODO: check manifest main classes, etc
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1021
            } else if (moduleVersion != null || modulesToHash != null) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1022
                error(getMsg("error.module.options.without.info"));
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  1023
                updateOk = false;
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  1024
            }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1025
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1026
        zis.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1027
        zos.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1028
        return updateOk;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1029
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1030
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1031
    private void addIndex(JarIndex index, ZipOutputStream zos)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1032
        throws IOException
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1033
    {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1034
        ZipEntry e = new ZipEntry(INDEX_NAME);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1035
        e.setTime(System.currentTimeMillis());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1036
        if (flag0) {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1037
            CRC32OutputStream os = new CRC32OutputStream();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1038
            index.write(os);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1039
            os.updateEntry(e);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1040
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1041
        zos.putNextEntry(e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1042
        index.write(zos);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1043
        zos.closeEntry();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1044
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1045
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1046
    private void updateModuleInfo(Map<String,byte[]> moduleInfos, ZipOutputStream zos)
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1047
        throws IOException
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1048
    {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1049
        String fmt = uflag ? "out.update.module-info": "out.added.module-info";
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1050
        for (Map.Entry<String,byte[]> mi : moduleInfos.entrySet()) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1051
            String name = mi.getKey();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1052
            byte[] bytes = mi.getValue();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1053
            ZipEntry e = new ZipEntry(name);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1054
            e.setTime(System.currentTimeMillis());
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1055
            if (flag0) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1056
                crc32ModuleInfo(e, bytes);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1057
            }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1058
            zos.putNextEntry(e);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1059
            zos.write(bytes);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1060
            zos.closeEntry();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1061
            if (vflag) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1062
                output(formatMsg(fmt, name));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1063
            }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1064
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1065
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1066
15682
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1067
    private boolean updateManifest(Manifest m, ZipOutputStream zos)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1068
        throws IOException
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1069
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1070
        addVersion(m);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1071
        addCreatedBy(m);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1072
        if (ename != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1073
            addMainClass(m, ename);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1074
        }
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1075
        if (isMultiRelease) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1076
            addMultiRelease(m);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1077
        }
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1078
        ZipEntry e = new ZipEntry(MANIFEST_NAME);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1079
        e.setTime(System.currentTimeMillis());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1080
        if (flag0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1081
            crc32Manifest(e, m);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1082
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1083
        zos.putNextEntry(e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1084
        m.write(zos);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1085
        if (vflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1086
            output(getMsg("out.update.manifest"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1087
        }
15682
793a36de151d 8003255: (profiles) Update JAR file specification to support profiles
alanb
parents: 14504
diff changeset
  1088
        return true;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1089
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1090
29914
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1091
    private static final boolean isWinDriveLetter(char c) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1092
        return ((c >= 'a') && (c <= 'z')) || ((c >= 'A') && (c <= 'Z'));
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1093
    }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1094
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1095
    private String safeName(String name) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1096
        if (!pflag) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1097
            int len = name.length();
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1098
            int i = name.lastIndexOf("../");
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1099
            if (i == -1) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1100
                i = 0;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1101
            } else {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1102
                i += 3; // strip any dot-dot components
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1103
            }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1104
            if (File.separatorChar == '\\') {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1105
                // the spec requests no drive letter. skip if
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1106
                // the entry name has one.
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1107
                while (i < len) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1108
                    int off = i;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1109
                    if (i + 1 < len &&
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1110
                        name.charAt(i + 1) == ':' &&
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1111
                        isWinDriveLetter(name.charAt(i))) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1112
                        i += 2;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1113
                    }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1114
                    while (i < len && name.charAt(i) == '/') {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1115
                        i++;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1116
                    }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1117
                    if (i == off) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1118
                        break;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1119
                    }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1120
                }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1121
            } else {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1122
                while (i < len && name.charAt(i) == '/') {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1123
                    i++;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1124
                }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1125
            }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1126
            if (i != 0) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1127
                name = name.substring(i);
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1128
            }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1129
        }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1130
        return name;
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1131
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1132
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1133
    private void addVersion(Manifest m) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1134
        Attributes global = m.getMainAttributes();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1135
        if (global.getValue(Attributes.Name.MANIFEST_VERSION) == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1136
            global.put(Attributes.Name.MANIFEST_VERSION, VERSION);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1137
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1138
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1139
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1140
    private void addCreatedBy(Manifest m) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1141
        Attributes global = m.getMainAttributes();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1142
        if (global.getValue(new Attributes.Name("Created-By")) == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1143
            String javaVendor = System.getProperty("java.vendor");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1144
            String jdkVersion = System.getProperty("java.version");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1145
            global.put(new Attributes.Name("Created-By"), jdkVersion + " (" +
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1146
                        javaVendor + ")");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1147
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1148
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1149
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1150
    private void addMainClass(Manifest m, String mainApp) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1151
        Attributes global = m.getMainAttributes();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1152
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1153
        // overrides any existing Main-Class attribute
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1154
        global.put(Attributes.Name.MAIN_CLASS, mainApp);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1155
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1156
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1157
    private void addMultiRelease(Manifest m) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1158
        Attributes global = m.getMainAttributes();
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1159
        global.put(Attributes.Name.MULTI_RELEASE, "true");
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1160
    }
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1161
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1162
    private boolean isAmbiguousMainClass(Manifest m) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1163
        if (ename != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1164
            Attributes global = m.getMainAttributes();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1165
            if ((global.get(Attributes.Name.MAIN_CLASS) != null)) {
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
  1166
                usageError(getMsg("error.bad.eflag"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1167
                return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1168
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1169
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1170
        return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1171
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1172
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1173
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1174
     * Adds a new file entry to the ZIP output stream.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1175
     */
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1176
    void addFile(ZipOutputStream zos, Entry entry) throws IOException {
41221
eb0dc5ea2302 8165944: jar utility doesn't process more than one -C argument
sdrach
parents: 40251
diff changeset
  1177
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1178
        File file = entry.file;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1179
        String name = entry.name;
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1180
        boolean isDir = entry.isDir;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1181
52902
e3398b2e1ab0 8214971: Replace use of string.equals("") with isEmpty()
rriggs
parents: 52723
diff changeset
  1182
        if (name.isEmpty() || name.equals(".") || name.equals(zname)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1183
            return;
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1184
        } else if ((name.equals(MANIFEST_DIR) || name.equals(MANIFEST_NAME))
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1185
                   && !Mflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1186
            if (vflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1187
                output(formatMsg("out.ignore.entry", name));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1188
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1189
            return;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1190
        } else if (name.equals(MODULE_INFO)) {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1191
            throw new Error("Unexpected module info: " + name);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1192
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1193
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1194
        long size = isDir ? 0 : file.length();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1195
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1196
        if (vflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1197
            out.print(formatMsg("out.adding", name));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1198
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1199
        ZipEntry e = new ZipEntry(name);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1200
        e.setTime(file.lastModified());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1201
        if (size == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1202
            e.setMethod(ZipEntry.STORED);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1203
            e.setSize(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1204
            e.setCrc(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1205
        } else if (flag0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1206
            crc32File(e, file);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1207
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1208
        zos.putNextEntry(e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1209
        if (!isDir) {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1210
            copy(file, zos);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1211
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1212
        zos.closeEntry();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1213
        /* report how much compression occurred. */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1214
        if (vflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1215
            size = e.getSize();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1216
            long csize = e.getCompressedSize();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1217
            out.print(formatMsg2("out.size", String.valueOf(size),
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1218
                        String.valueOf(csize)));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1219
            if (e.getMethod() == ZipEntry.DEFLATED) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1220
                long ratio = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1221
                if (size != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1222
                    ratio = ((size - csize) * 100) / size;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1223
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1224
                output(formatMsg("out.deflated", String.valueOf(ratio)));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1225
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1226
                output(getMsg("out.stored"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1227
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1228
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1229
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1230
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1231
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1232
     * A buffer for use only by copy(InputStream, OutputStream).
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1233
     * Not as clean as allocating a new buffer as needed by copy,
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1234
     * but significantly more efficient.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1235
     */
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1236
    private byte[] copyBuf = new byte[8192];
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1237
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1238
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1239
     * Copies all bytes from the input stream to the output stream.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1240
     * Does not close or flush either stream.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1241
     *
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1242
     * @param from the input stream to read from
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1243
     * @param to the output stream to write to
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1244
     * @throws IOException if an I/O error occurs
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1245
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1246
    private void copy(InputStream from, OutputStream to) throws IOException {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1247
        int n;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1248
        while ((n = from.read(copyBuf)) != -1)
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1249
            to.write(copyBuf, 0, n);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1250
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1251
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1252
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1253
     * Copies all bytes from the input file to the output stream.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1254
     * Does not close or flush the output stream.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1255
     *
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1256
     * @param from the input file to read from
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1257
     * @param to the output stream to write to
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1258
     * @throws IOException if an I/O error occurs
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1259
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1260
    private void copy(File from, OutputStream to) throws IOException {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1261
        try (InputStream in = new FileInputStream(from)) {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1262
            copy(in, to);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1263
        }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1264
    }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1265
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1266
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1267
     * Copies all bytes from the input stream to the output file.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1268
     * Does not close the input stream.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1269
     *
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1270
     * @param from the input stream to read from
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1271
     * @param to the output file to write to
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1272
     * @throws IOException if an I/O error occurs
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1273
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1274
    private void copy(InputStream from, File to) throws IOException {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1275
        try (OutputStream out = new FileOutputStream(to)) {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1276
            copy(from, out);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1277
        }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1278
    }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1279
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1280
    /**
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1281
     * Computes the crc32 of a module-info.class.  This is necessary when the
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1282
     * ZipOutputStream is in STORED mode.
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1283
     */
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1284
    private void crc32ModuleInfo(ZipEntry e, byte[] bytes) throws IOException {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1285
        CRC32OutputStream os = new CRC32OutputStream();
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1286
        ByteArrayInputStream in = new ByteArrayInputStream(bytes);
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1287
        in.transferTo(os);
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1288
        os.updateEntry(e);
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1289
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1290
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1291
    /**
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1292
     * Computes the crc32 of a Manifest.  This is necessary when the
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1293
     * ZipOutputStream is in STORED mode.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1294
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1295
    private void crc32Manifest(ZipEntry e, Manifest m) throws IOException {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1296
        CRC32OutputStream os = new CRC32OutputStream();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1297
        m.write(os);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1298
        os.updateEntry(e);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1299
    }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1300
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1301
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1302
     * Computes the crc32 of a File.  This is necessary when the
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1303
     * ZipOutputStream is in STORED mode.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1304
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1305
    private void crc32File(ZipEntry e, File f) throws IOException {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1306
        CRC32OutputStream os = new CRC32OutputStream();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1307
        copy(f, os);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1308
        if (os.n != f.length()) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1309
            throw new JarException(formatMsg(
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1310
                        "error.incorrect.length", f.getPath()));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1311
        }
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1312
        os.updateEntry(e);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1313
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1314
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1315
    void replaceFSC(Map<Integer, String []> filesMap) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1316
        filesMap.keySet().forEach(version -> {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1317
            String[] files = filesMap.get(version);
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1318
            if (files != null) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1319
                for (int i = 0; i < files.length; i++) {
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1320
                    files[i] = files[i].replace(File.separatorChar, '/');
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1321
                }
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1322
            }
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1323
        });
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1324
    }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1325
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1326
    @SuppressWarnings("serial")
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1327
    Set<ZipEntry> newDirSet() {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1328
        return new HashSet<ZipEntry>() {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1329
            public boolean add(ZipEntry e) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1330
                return ((e == null || useExtractionTime) ? false : super.add(e));
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1331
            }};
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1332
    }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1333
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1334
    void updateLastModifiedTime(Set<ZipEntry> zes) throws IOException {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1335
        for (ZipEntry ze : zes) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1336
            long lastModified = ze.getTime();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1337
            if (lastModified != -1) {
29914
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1338
                String name = safeName(ze.getName().replace(File.separatorChar, '/'));
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1339
                if (name.length() != 0) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1340
                    File f = new File(name.replace('/', File.separatorChar));
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1341
                    f.setLastModified(lastModified);
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1342
                }
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1343
            }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1344
        }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1345
    }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1346
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1347
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1348
     * Extracts specified entries from JAR file.
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1349
     *
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1350
     * @return whether entries were found and successfully extracted
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1351
     * (indicating this was a zip file without "leading garbage")
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1352
     */
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1353
    boolean extract(InputStream in, String files[]) throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1354
        ZipInputStream zis = new ZipInputStream(in);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1355
        ZipEntry e;
1762
4659af392d07 6496274: jar seems to use more CPU than it should
sherman
parents: 2
diff changeset
  1356
        // Set of all directory entries specified in archive.  Disallows
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1357
        // null entries.  Disallows all entries if using pre-6.0 behavior.
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1358
        boolean entriesFound = false;
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1359
        Set<ZipEntry> dirs = newDirSet();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1360
        while ((e = zis.getNextEntry()) != null) {
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1361
            entriesFound = true;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1362
            if (files == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1363
                dirs.add(extractFile(zis, e));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1364
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1365
                String name = e.getName();
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1366
                for (String file : files) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1367
                    if (name.startsWith(file)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1368
                        dirs.add(extractFile(zis, e));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1369
                        break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1370
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1371
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1372
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1373
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1374
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1375
        // Update timestamps of directories specified in archive with their
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1376
        // timestamps as given in the archive.  We do this after extraction,
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1377
        // instead of during, because creating a file in a directory changes
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1378
        // that directory's timestamp.
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1379
        updateLastModifiedTime(dirs);
26714
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1380
3caa40031b3f 8058520: jar xf does not work on zip files with leading garbage
martin
parents: 25859
diff changeset
  1381
        return entriesFound;
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1382
    }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1383
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1384
    /**
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1385
     * Extracts specified entries from JAR file, via ZipFile.
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1386
     */
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1387
    void extract(String fname, String files[]) throws IOException {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1388
        ZipFile zf = new ZipFile(fname);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1389
        Set<ZipEntry> dirs = newDirSet();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1390
        Enumeration<? extends ZipEntry> zes = zf.entries();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1391
        while (zes.hasMoreElements()) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1392
            ZipEntry e = zes.nextElement();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1393
            if (files == null) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1394
                dirs.add(extractFile(zf.getInputStream(e), e));
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1395
            } else {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1396
                String name = e.getName();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1397
                for (String file : files) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1398
                    if (name.startsWith(file)) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1399
                        dirs.add(extractFile(zf.getInputStream(e), e));
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1400
                        break;
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1401
                    }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1402
                }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1403
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1404
        }
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1405
        zf.close();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1406
        updateLastModifiedTime(dirs);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1407
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1408
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1409
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1410
     * Extracts next entry from JAR file, creating directories as needed.  If
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1411
     * the entry is for a directory which doesn't exist prior to this
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1412
     * invocation, returns that entry, otherwise returns null.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1413
     */
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1414
    ZipEntry extractFile(InputStream is, ZipEntry e) throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1415
        ZipEntry rc = null;
29914
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1416
        // The spec requres all slashes MUST be forward '/', it is possible
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1417
        // an offending zip/jar entry may uses the backwards slash in its
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1418
        // name. It might cause problem on Windows platform as it skips
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1419
        // our "safe" check for leading slahs and dot-dot. So replace them
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1420
        // with '/'.
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1421
        String name = safeName(e.getName().replace(File.separatorChar, '/'));
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1422
        if (name.length() == 0) {
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1423
            return rc;    // leading '/' or 'dot-dot' only path
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1424
        }
48393dc87f88 8064601: Improve jar file handling
sherman
parents: 29369
diff changeset
  1425
        File f = new File(name.replace('/', File.separatorChar));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1426
        if (e.isDirectory()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1427
            if (f.exists()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1428
                if (!f.isDirectory()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1429
                    throw new IOException(formatMsg("error.create.dir",
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1430
                        f.getPath()));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1431
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1432
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1433
                if (!f.mkdirs()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1434
                    throw new IOException(formatMsg("error.create.dir",
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1435
                        f.getPath()));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1436
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1437
                    rc = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1438
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1439
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1440
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1441
            if (vflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1442
                output(formatMsg("out.create", name));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1443
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1444
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1445
            if (f.getParent() != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1446
                File d = new File(f.getParent());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1447
                if (!d.exists() && !d.mkdirs() || !d.isDirectory()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1448
                    throw new IOException(formatMsg(
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1449
                        "error.create.dir", d.getPath()));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1450
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1451
            }
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1452
            try {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1453
                copy(is, f);
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1454
            } finally {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1455
                if (is instanceof ZipInputStream)
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1456
                    ((ZipInputStream)is).closeEntry();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1457
                else
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1458
                    is.close();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1459
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1460
            if (vflag) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1461
                if (e.getMethod() == ZipEntry.DEFLATED) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1462
                    output(formatMsg("out.inflated", name));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1463
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1464
                    output(formatMsg("out.extracted", name));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1465
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1466
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1467
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1468
        if (!useExtractionTime) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1469
            long lastModified = e.getTime();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1470
            if (lastModified != -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1471
                f.setLastModified(lastModified);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1472
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1473
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1474
        return rc;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1475
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1476
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1477
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1478
     * Lists contents of JAR file.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1479
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1480
    void list(InputStream in, String files[]) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1481
        ZipInputStream zis = new ZipInputStream(in);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1482
        ZipEntry e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1483
        while ((e = zis.getNextEntry()) != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1484
            /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1485
             * In the case of a compressed (deflated) entry, the entry size
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1486
             * is stored immediately following the entry data and cannot be
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1487
             * determined until the entry is fully read. Therefore, we close
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1488
             * the entry first before printing out its attributes.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1489
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1490
            zis.closeEntry();
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1491
            printEntry(e, files);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1492
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1493
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1494
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1495
    /**
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1496
     * Lists contents of JAR file, via ZipFile.
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1497
     */
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1498
    void list(String fname, String files[]) throws IOException {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1499
        ZipFile zf = new ZipFile(fname);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1500
        Enumeration<? extends ZipEntry> zes = zf.entries();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1501
        while (zes.hasMoreElements()) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1502
            printEntry(zes.nextElement(), files);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1503
        }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1504
        zf.close();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1505
    }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1506
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1507
    /**
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1508
     * Outputs the class index table to the INDEX.LIST file of the
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1509
     * root jar file.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1510
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1511
    void dumpIndex(String rootjar, JarIndex index) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1512
        File jarFile = new File(rootjar);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1513
        Path jarPath = jarFile.toPath();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1514
        Path tmpPath = createTempFileInSameDirectoryAs(jarFile).toPath();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1515
        try {
8158
77d9c0f1c19f 7006126: (fs) Updates to file system API (1/2011)
alanb
parents: 7668
diff changeset
  1516
            if (update(Files.newInputStream(jarPath),
77d9c0f1c19f 7006126: (fs) Updates to file system API (1/2011)
alanb
parents: 7668
diff changeset
  1517
                       Files.newOutputStream(tmpPath),
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1518
                       null, null, index)) {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1519
                try {
8158
77d9c0f1c19f 7006126: (fs) Updates to file system API (1/2011)
alanb
parents: 7668
diff changeset
  1520
                    Files.move(tmpPath, jarPath, REPLACE_EXISTING);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1521
                } catch (IOException e) {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1522
                    throw new IOException(getMsg("error.write.file"), e);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1523
                }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1524
            }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1525
        } finally {
8158
77d9c0f1c19f 7006126: (fs) Updates to file system API (1/2011)
alanb
parents: 7668
diff changeset
  1526
            Files.deleteIfExists(tmpPath);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1527
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1528
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1529
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1530
    private HashSet<String> jarPaths = new HashSet<String>();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1531
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1532
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1533
     * Generates the transitive closure of the Class-Path attribute for
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1534
     * the specified jar file.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1535
     */
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1536
    List<String> getJarPath(String jar) throws IOException {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1537
        List<String> files = new ArrayList<String>();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1538
        files.add(jar);
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1539
        jarPaths.add(jar);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1540
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1541
        // take out the current path
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1542
        String path = jar.substring(0, Math.max(0, jar.lastIndexOf('/') + 1));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1543
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1544
        // class path attribute will give us jar file name with
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1545
        // '/' as separators, so we need to change them to the
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1546
        // appropriate one before we open the jar file.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1547
        JarFile rf = new JarFile(jar.replace('/', File.separatorChar));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1548
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1549
        if (rf != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1550
            Manifest man = rf.getManifest();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1551
            if (man != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1552
                Attributes attr = man.getMainAttributes();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1553
                if (attr != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1554
                    String value = attr.getValue(Attributes.Name.CLASS_PATH);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1555
                    if (value != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1556
                        StringTokenizer st = new StringTokenizer(value);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1557
                        while (st.hasMoreTokens()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1558
                            String ajar = st.nextToken();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1559
                            if (!ajar.endsWith("/")) {  // it is a jar file
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1560
                                ajar = path.concat(ajar);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1561
                                /* check on cyclic dependency */
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1562
                                if (! jarPaths.contains(ajar)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1563
                                    files.addAll(getJarPath(ajar));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1564
                                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1565
                            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1566
                        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1567
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1568
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1569
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1570
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1571
        rf.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1572
        return files;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1573
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1574
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1575
    /**
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1576
     * Generates class index file for the specified root jar file.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1577
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1578
    void genIndex(String rootjar, String[] files) throws IOException {
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1579
        List<String> jars = getJarPath(rootjar);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1580
        int njars = jars.size();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1581
        String[] jarfiles;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1582
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1583
        if (njars == 1 && files != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1584
            // no class-path attribute defined in rootjar, will
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1585
            // use command line specified list of jars
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1586
            for (int i = 0; i < files.length; i++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1587
                jars.addAll(getJarPath(files[i]));
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1588
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1589
            njars = jars.size();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1590
        }
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1591
        jarfiles = jars.toArray(new String[njars]);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1592
        JarIndex index = new JarIndex(jarfiles);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1593
        dumpIndex(rootjar, index);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1594
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1595
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1596
    /**
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1597
     * Prints entry information, if requested.
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1598
     */
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1599
    void printEntry(ZipEntry e, String[] files) throws IOException {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1600
        if (files == null) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1601
            printEntry(e);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1602
        } else {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1603
            String name = e.getName();
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1604
            for (String file : files) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1605
                if (name.startsWith(file)) {
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1606
                    printEntry(e);
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1607
                    return;
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1608
                }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1609
            }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1610
        }
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1611
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1612
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1613
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1614
     * Prints entry information.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1615
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1616
    void printEntry(ZipEntry e) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1617
        if (vflag) {
1816
f310e059bd83 6332094: "jar t" and "jar x" should use ZipFile, not ZipInputStream
sherman
parents: 1762
diff changeset
  1618
            StringBuilder sb = new StringBuilder();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1619
            String s = Long.toString(e.getSize());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1620
            for (int i = 6 - s.length(); i > 0; --i) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1621
                sb.append(' ');
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1622
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1623
            sb.append(s).append(' ').append(new Date(e.getTime()).toString());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1624
            sb.append(' ').append(e.getName());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1625
            output(sb.toString());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1626
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1627
            output(e.getName());
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1628
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1629
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1630
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1631
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1632
     * Prints usage message.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1633
     */
42463
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
  1634
    void usageError(String s) {
39c0bbf9d03c 8170952: jar's usage message output need some cleanup
sherman
parents: 42345
diff changeset
  1635
        err.println(s);
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  1636
        err.println(getMsg("main.usage.summary.try"));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1637
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1638
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1639
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1640
     * A fatal exception has been caught.  No recovery possible
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1641
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1642
    void fatalError(Exception e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1643
        e.printStackTrace();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1644
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1645
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1646
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1647
     * A fatal condition has been detected; message is "s".
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1648
     * No recovery possible
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1649
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1650
    void fatalError(String s) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1651
        error(program + ": " + s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1652
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1653
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1654
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1655
     * Print an output message; like verbose output and the like
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1656
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1657
    protected void output(String s) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1658
        out.println(s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1659
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1660
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1661
    /**
21278
ef8a3a2a72f2 8022746: List of spelling errors in API doc
malenkov
parents: 19409
diff changeset
  1662
     * Print an error message; like something is broken
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1663
     */
39313
0970a9a1de13 8114827: JDK 9 multi-release enabled jar tool
sdrach
parents: 38468
diff changeset
  1664
    void error(String s) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1665
        err.println(s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1666
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1667
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1668
    /**
41754
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1669
     * Print a warning message
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1670
     */
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1671
    void warn(String s) {
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1672
        err.println(s);
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1673
    }
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1674
8b7c8d5e9a0d 8164805: Fail to create a MR modular JAR with a versioned entry of a concealed package
sdrach
parents: 41484
diff changeset
  1675
    /**
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1676
     * Main routine to start program.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1677
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1678
    public static void main(String args[]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1679
        Main jartool = new Main(System.out, System.err, "jar");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1680
        System.exit(jartool.run(args) ? 0 : 1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1681
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1682
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1683
    /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1684
     * An OutputStream that doesn't send its output anywhere, (but could).
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1685
     * It's here to find the CRC32 of an input file, necessary for STORED
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1686
     * mode in ZIP.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1687
     */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1688
    private static class CRC32OutputStream extends java.io.OutputStream {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1689
        final CRC32 crc = new CRC32();
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1690
        long n = 0;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1691
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1692
        CRC32OutputStream() {}
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1693
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1694
        public void write(int r) throws IOException {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1695
            crc.update(r);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1696
            n++;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1697
        }
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1698
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1699
        public void write(byte[] b, int off, int len) throws IOException {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1700
            crc.update(b, off, len);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1701
            n += len;
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1702
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1703
3218
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1704
        /**
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1705
         * Updates a ZipEntry which describes the data read by this
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1706
         * output stream, in STORED mode.
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1707
         */
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1708
        public void updateEntry(ZipEntry e) {
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1709
            e.setMethod(ZipEntry.STORED);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1710
            e.setSize(n);
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1711
            e.setCrc(crc.getValue());
855a6257b729 6854795: Miscellaneous improvements to "jar"
martin
parents: 3056
diff changeset
  1712
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1713
    }
21348
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1714
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1715
    /**
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1716
     * Attempt to create temporary file in the system-provided temporary folder, if failed attempts
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1717
     * to create it in the same folder as the file in parameter (if any)
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1718
     */
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1719
    private File createTemporaryFile(String tmpbase, String suffix) {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1720
        File tmpfile = null;
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1721
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1722
        try {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1723
            tmpfile = File.createTempFile(tmpbase, suffix);
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1724
        } catch (IOException | SecurityException e) {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1725
            // Unable to create file due to permission violation or security exception
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1726
        }
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1727
        if (tmpfile == null) {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1728
            // Were unable to create temporary file, fall back to temporary file in the same folder
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1729
            if (fname != null) {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1730
                try {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1731
                    File tmpfolder = new File(fname).getAbsoluteFile().getParentFile();
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1732
                    tmpfile = File.createTempFile(fname, ".tmp" + suffix, tmpfolder);
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1733
                } catch (IOException ioe) {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1734
                    // Last option failed - fall gracefully
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1735
                    fatalError(ioe);
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1736
                }
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1737
            } else {
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1738
                // No options left - we can not compress to stdout without access to the temporary folder
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1739
                fatalError(new IOException(getMsg("error.create.tempfile")));
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1740
            }
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1741
        }
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1742
        return tmpfile;
e30c5696b4c5 8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
kizune
parents: 19409
diff changeset
  1743
    }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1744
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1745
    // Modular jar support
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1746
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1747
    /**
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1748
     * Associates a module descriptor's zip entry name along with its
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1749
     * bytes and an optional URI. Used when describing modules.
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1750
     */
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1751
    interface ModuleInfoEntry {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1752
       String name();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1753
       Optional<String> uriString();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1754
       InputStream bytes() throws IOException;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1755
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1756
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1757
    static class ZipFileModuleInfoEntry implements ModuleInfoEntry {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1758
        private final ZipFile zipFile;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1759
        private final ZipEntry entry;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1760
        ZipFileModuleInfoEntry(ZipFile zipFile, ZipEntry entry) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1761
            this.zipFile = zipFile;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1762
            this.entry = entry;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1763
        }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1764
        @Override public String name() { return entry.getName(); }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1765
        @Override public InputStream bytes() throws IOException {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1766
            return zipFile.getInputStream(entry);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1767
        }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1768
        /** Returns an optional containing the effective URI. */
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1769
        @Override public Optional<String> uriString() {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1770
            String uri = (Paths.get(zipFile.getName())).toUri().toString();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1771
            uri = "jar:" + uri + "/!" + entry.getName();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1772
            return Optional.of(uri);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1773
        }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1774
    }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1775
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1776
    static class StreamedModuleInfoEntry implements ModuleInfoEntry {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1777
        private final String name;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1778
        private final byte[] bytes;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1779
        StreamedModuleInfoEntry(String name, byte[] bytes) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1780
            this.name = name;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1781
            this.bytes = bytes;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1782
        }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1783
        @Override public String name() { return name; }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1784
        @Override public InputStream bytes() throws IOException {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1785
            return new ByteArrayInputStream(bytes);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1786
        }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1787
        /** Returns an empty optional. */
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1788
        @Override public Optional<String> uriString() {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1789
            return Optional.empty();  // no URI can be derived
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1790
        }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1791
    }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1792
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1793
    /** Describes a module from a given zip file. */
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1794
    private boolean describeModule(ZipFile zipFile) throws IOException {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1795
        ZipFileModuleInfoEntry[] infos = zipFile.stream()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1796
                .filter(e -> isModuleInfoEntry(e.getName()))
47958
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  1797
                .sorted(ENTRY_COMPARATOR)
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1798
                .map(e -> new ZipFileModuleInfoEntry(zipFile, e))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1799
                .toArray(ZipFileModuleInfoEntry[]::new);
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1800
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1801
        if (infos.length == 0) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1802
            // No module descriptor found, derive and describe the automatic module
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1803
            String fn = zipFile.getName();
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1804
            ModuleFinder mf = ModuleFinder.of(Paths.get(fn));
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1805
            try {
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1806
                Set<ModuleReference> mref = mf.findAll();
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1807
                if (mref.isEmpty()) {
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1808
                    output(formatMsg("error.unable.derive.automodule", fn));
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1809
                    return true;
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1810
                }
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1811
                ModuleDescriptor md = mref.iterator().next().descriptor();
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1812
                output(getMsg("out.automodule") + "\n");
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1813
                describeModule(md, null, null, "");
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1814
            } catch (FindException e) {
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1815
                String msg = formatMsg("error.unable.derive.automodule", fn);
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1816
                Throwable t = e.getCause();
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1817
                if (t != null)
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1818
                    msg = msg + "\n" + t.getMessage();
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1819
                output(msg);
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1820
            }
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1821
        } else {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1822
            return describeModuleFromEntries(infos);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1823
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1824
        return true;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1825
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1826
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1827
    private boolean describeModuleFromStream(FileInputStream fis)
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1828
        throws IOException
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1829
    {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1830
        List<ModuleInfoEntry> infos = new LinkedList<>();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1831
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1832
        try (BufferedInputStream bis = new BufferedInputStream(fis);
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1833
             ZipInputStream zis = new ZipInputStream(bis)) {
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1834
            ZipEntry e;
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1835
            while ((e = zis.getNextEntry()) != null) {
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
  1836
                String ename = e.getName();
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1837
                if (isModuleInfoEntry(ename)) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1838
                    infos.add(new StreamedModuleInfoEntry(ename, zis.readAllBytes()));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1839
                }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1840
            }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1841
        }
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1842
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1843
        if (infos.size() == 0)
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
  1844
            return false;
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1845
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1846
        ModuleInfoEntry[] sorted = infos.stream()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1847
                .sorted(Comparator.comparing(ModuleInfoEntry::name, ENTRYNAME_COMPARATOR))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1848
                .toArray(ModuleInfoEntry[]::new);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1849
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1850
        return describeModuleFromEntries(sorted);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1851
    }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1852
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1853
    private boolean lessThanEqualReleaseValue(ModuleInfoEntry entry) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1854
        return intVersionFromEntry(entry) <= releaseValue ? true : false;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1855
    }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1856
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1857
    private static String versionFromEntryName(String name) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1858
        String s = name.substring(VERSIONS_DIR_LENGTH);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1859
        return s.substring(0, s.indexOf("/"));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1860
    }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1861
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1862
    private static int intVersionFromEntry(ModuleInfoEntry entry) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1863
        String name = entry.name();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1864
        if (!name.startsWith(VERSIONS_DIR))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1865
            return BASE_VERSION;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1866
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1867
        String s = name.substring(VERSIONS_DIR_LENGTH);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1868
        s = s.substring(0, s.indexOf('/'));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1869
        return Integer.valueOf(s);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1870
    }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1871
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1872
    /**
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1873
     * Describes a single module descriptor, determined by the specified
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1874
     * --release, if any, from the given ordered entries.
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1875
     * The given infos must be ordered as per ENTRY_COMPARATOR.
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1876
     */
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1877
    private boolean describeModuleFromEntries(ModuleInfoEntry[] infos)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1878
        throws IOException
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1879
    {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1880
        assert infos.length > 0;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1881
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1882
        // Informative: output all non-root descriptors, if any
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1883
        String releases = Arrays.stream(infos)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1884
                .filter(e -> !e.name().equals(MODULE_INFO))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1885
                .map(ModuleInfoEntry::name)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1886
                .map(Main::versionFromEntryName)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1887
                .collect(joining(" "));
52902
e3398b2e1ab0 8214971: Replace use of string.equals("") with isEmpty()
rriggs
parents: 52723
diff changeset
  1888
        if (!releases.isEmpty())
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1889
            output("releases: " + releases + "\n");
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1890
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1891
        // Describe the operative descriptor for the specified --release, if any
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1892
        if (releaseValue != -1) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1893
            ModuleInfoEntry entry = null;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1894
            int i = 0;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1895
            while (i < infos.length && lessThanEqualReleaseValue(infos[i])) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1896
                entry = infos[i];
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1897
                i++;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1898
            }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1899
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1900
            if (entry == null) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1901
                output(formatMsg("error.no.operative.descriptor",
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1902
                                 String.valueOf(releaseValue)));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1903
                return false;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1904
            }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1905
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1906
            String uriString = entry.uriString().orElse("");
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1907
            try (InputStream is = entry.bytes()) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1908
                describeModule(is, uriString);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1909
            }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1910
        } else {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1911
            // no specific --release specified, output the root, if any
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1912
            if (infos[0].name().equals(MODULE_INFO)) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1913
                String uriString = infos[0].uriString().orElse("");
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1914
                try (InputStream is = infos[0].bytes()) {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1915
                    describeModule(is, uriString);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1916
                }
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1917
            } else {
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1918
                // no root, output message to specify --release
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1919
                output(getMsg("error.no.root.descriptor"));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1920
            }
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
  1921
        }
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
  1922
        return true;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1923
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1924
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1925
    static <T> String toLowerCaseString(Collection<T> set) {
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  1926
        if (set.isEmpty()) { return ""; }
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1927
        return " " + set.stream().map(e -> e.toString().toLowerCase(Locale.ROOT))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1928
                  .sorted().collect(joining(" "));
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  1929
    }
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  1930
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1931
    static <T> String toString(Collection<T> set) {
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1932
        if (set.isEmpty()) { return ""; }
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1933
        return " " + set.stream().map(e -> e.toString()).sorted().collect(joining(" "));
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1934
    }
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1935
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1936
    private void describeModule(InputStream entryInputStream, String uriString)
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1937
        throws IOException
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1938
    {
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  1939
        ModuleInfo.Attributes attrs = ModuleInfo.read(entryInputStream, null);
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  1940
        ModuleDescriptor md = attrs.descriptor();
44359
c6761862ca0b 8174823: Module system implementation refresh (3/2017)
alanb
parents: 44033
diff changeset
  1941
        ModuleTarget target = attrs.target();
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  1942
        ModuleHashes hashes = attrs.recordedHashes();
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  1943
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1944
        describeModule(md, target, hashes, uriString);
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1945
    }
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1946
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1947
    private void describeModule(ModuleDescriptor md,
44360
alanb
parents: 44267 44359
diff changeset
  1948
                                ModuleTarget target,
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1949
                                ModuleHashes hashes,
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1950
                                String uriString)
44267
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1951
        throws IOException
00898a596f31 8176772: jar tool support to report automatic module names
chegar
parents: 44033
diff changeset
  1952
    {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1953
        StringBuilder sb = new StringBuilder();
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1954
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1955
        sb.append(md.toNameAndVersion());
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1956
52902
e3398b2e1ab0 8214971: Replace use of string.equals("") with isEmpty()
rriggs
parents: 52723
diff changeset
  1957
        if (!uriString.isEmpty())
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1958
            sb.append(" ").append(uriString);
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1959
        if (md.isOpen())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1960
            sb.append(" open");
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1961
        if (md.isAutomatic())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1962
            sb.append(" automatic");
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1963
        sb.append("\n");
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
  1964
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1965
        // unqualified exports (sorted by package)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1966
        md.exports().stream()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1967
                .sorted(Comparator.comparing(Exports::source))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1968
                .filter(e -> !e.isQualified())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1969
                .forEach(e -> sb.append("exports ").append(e.source())
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1970
                                .append(toLowerCaseString(e.modifiers()))
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1971
                                .append("\n"));
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1972
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1973
        // dependences
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1974
        md.requires().stream().sorted()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1975
                .forEach(r -> sb.append("requires ").append(r.name())
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1976
                                .append(toLowerCaseString(r.modifiers()))
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1977
                                .append("\n"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1978
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1979
        // service use and provides
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1980
        md.uses().stream().sorted()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1981
                .forEach(s -> sb.append("uses ").append(s).append("\n"));
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1982
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1983
        md.provides().stream()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1984
                .sorted(Comparator.comparing(Provides::service))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1985
                .forEach(p -> sb.append("provides ").append(p.service())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1986
                                .append(" with")
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1987
                                .append(toString(p.providers()))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1988
                                .append("\n"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  1989
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1990
        // qualified exports
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1991
        md.exports().stream()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1992
                .sorted(Comparator.comparing(Exports::source))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1993
                .filter(Exports::isQualified)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1994
                .forEach(e -> sb.append("qualified exports ").append(e.source())
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  1995
                                .append(" to").append(toLowerCaseString(e.targets()))
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1996
                                .append("\n"));
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  1997
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1998
        // open packages
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  1999
        md.opens().stream()
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2000
                .sorted(Comparator.comparing(Opens::source))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2001
                .filter(o -> !o.isQualified())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2002
                .forEach(o -> sb.append("opens ").append(o.source())
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  2003
                                 .append(toLowerCaseString(o.modifiers()))
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2004
                                 .append("\n"));
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2005
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2006
        md.opens().stream()
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2007
                .sorted(Comparator.comparing(Opens::source))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2008
                .filter(Opens::isQualified)
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2009
                .forEach(o -> sb.append("qualified opens ").append(o.source())
50130
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  2010
                                 .append(toLowerCaseString(o.modifiers()))
aefd3c7f5373 8191533: jar --describe-module prints service provider class names in lower case
sherman
parents: 47958
diff changeset
  2011
                                 .append(" to").append(toLowerCaseString(o.targets()))
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2012
                                 .append("\n"));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2013
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2014
        // non-exported/non-open packages
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2015
        Set<String> concealed = new TreeSet<>(md.packages());
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2016
        md.exports().stream().map(Exports::source).forEach(concealed::remove);
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2017
        md.opens().stream().map(Opens::source).forEach(concealed::remove);
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2018
        concealed.forEach(p -> sb.append("contains ").append(p).append("\n"));
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2019
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2020
        md.mainClass().ifPresent(v -> sb.append("main-class ").append(v).append("\n"));
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2021
44359
c6761862ca0b 8174823: Module system implementation refresh (3/2017)
alanb
parents: 44033
diff changeset
  2022
        if (target != null) {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2023
            String targetPlatform = target.targetPlatform();
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2024
            if (!targetPlatform.isEmpty())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2025
                sb.append("platform ").append(targetPlatform).append("\n");
44359
c6761862ca0b 8174823: Module system implementation refresh (3/2017)
alanb
parents: 44033
diff changeset
  2026
       }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2027
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2028
       if (hashes != null) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2029
           hashes.names().stream().sorted().forEach(
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2030
                   mod -> sb.append("hashes ").append(mod).append(" ")
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2031
                            .append(hashes.algorithm()).append(" ")
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2032
                            .append(toHex(hashes.hashFor(mod)))
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 44360
diff changeset
  2033
                            .append("\n"));
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2034
        }
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2035
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2036
        output(sb.toString());
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2037
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2038
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2039
    private static String toHex(byte[] ba) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2040
        StringBuilder sb = new StringBuilder(ba.length << 1);
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2041
        for (byte b: ba) {
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2042
            sb.append(String.format("%02x", b & 0xff));
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2043
        }
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2044
        return sb.toString();
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2045
    }
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2046
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2047
    static String toBinaryName(String classname) {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2048
        return (classname.replace('.', '/')) + ".class";
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2049
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2050
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2051
    private boolean checkModuleInfo(byte[] moduleInfoBytes, Set<String> entries)
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2052
        throws IOException
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2053
    {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2054
        boolean ok = true;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2055
        if (moduleInfoBytes != null) {  // no root module-info.class if null
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2056
            try {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2057
                // ModuleDescriptor.read() checks open/exported pkgs vs packages
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2058
                ModuleDescriptor md = ModuleDescriptor.read(ByteBuffer.wrap(moduleInfoBytes));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2059
                // A module must have the implementation class of the services it 'provides'.
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2060
                if (md.provides().stream().map(Provides::providers).flatMap(List::stream)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2061
                      .filter(p -> !entries.contains(toBinaryName(p)))
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2062
                      .peek(p -> fatalError(formatMsg("error.missing.provider", p)))
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2063
                      .count() != 0) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2064
                    ok = false;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2065
                }
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2066
            } catch (InvalidModuleDescriptorException x) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2067
                fatalError(x.getMessage());
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2068
                ok = false;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2069
            }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2070
        }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2071
        return ok;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2072
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2073
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2074
    /**
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2075
     * Adds extended modules attributes to the given module-info's.  The given
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2076
     * Map values are updated in-place. Returns false if an error occurs.
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2077
     */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2078
    private void addExtendedModuleAttributes(Map<String,byte[]> moduleInfos,
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2079
                                                Set<String> packages)
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2080
        throws IOException
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2081
    {
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2082
        for (Map.Entry<String,byte[]> e: moduleInfos.entrySet()) {
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2083
            ModuleDescriptor md = ModuleDescriptor.read(ByteBuffer.wrap(e.getValue()));
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2084
            e.setValue(extendedInfoBytes(md, e.getValue(), packages));
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2085
        }
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2086
    }
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2087
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2088
    static boolean isModuleInfoEntry(String name) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2089
        // root or versioned module-info.class
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2090
        if (name.endsWith(MODULE_INFO)) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2091
            int end = name.length() - MODULE_INFO.length();
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2092
            if (end == 0)
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2093
                return true;
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2094
            if (name.startsWith(VERSIONS_DIR)) {
43731
bc7110b230c1 8165640: Enhance jar tool to allow module-info in versioned directories but not in base in modular multi-release jar files
sherman
parents: 43712
diff changeset
  2095
                int off = VERSIONS_DIR_LENGTH;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2096
                if (off == end)      // meta-inf/versions/module-info.class
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2097
                    return false;
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2098
                while (off < end - 1) {
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2099
                    char c = name.charAt(off++);
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2100
                    if (c < '0' || c > '9')
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2101
                        return false;
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2102
                }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2103
                return name.charAt(off) == '/';
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2104
            }
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2105
        }
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2106
        return false;
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2107
    }
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2108
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2109
    /**
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2110
     * Returns a byte array containing the given module-info.class plus any
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2111
     * extended attributes.
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2112
     *
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2113
     * If --module-version, --main-class, or other options were provided
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2114
     * then the corresponding class file attributes are added to the
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2115
     * module-info here.
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2116
     */
43095
336dfda4ae89 8172432: jar cleanup/update for module and mrm jar
sherman
parents: 42703
diff changeset
  2117
    private byte[] extendedInfoBytes(ModuleDescriptor md,
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2118
                                     byte[] miBytes,
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2119
                                     Set<String> packages)
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2120
        throws IOException
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2121
    {
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2122
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2123
        InputStream is = new ByteArrayInputStream(miBytes);
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2124
        ModuleInfoExtender extender = ModuleInfoExtender.newExtender(is);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2125
42338
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2126
        // Add (or replace) the Packages attribute
a60f280f803c 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41754
diff changeset
  2127
        extender.packages(packages);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2128
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2129
        // --main-class
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2130
        if (ename != null)
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2131
            extender.mainClass(ename);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2132
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2133
        // --module-version
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2134
        if (moduleVersion != null)
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2135
            extender.version(moduleVersion);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2136
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2137
        // --hash-modules
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2138
        if (modulesToHash != null) {
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2139
            String mn = md.name();
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2140
            Hasher hasher = new Hasher(md, fname);
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2141
            ModuleHashes moduleHashes = hasher.computeHashes(mn);
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2142
            if (moduleHashes != null) {
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2143
                extender.hashes(moduleHashes);
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2144
            } else {
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2145
                warn("warning: no module is recorded in hash in " + mn);
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2146
            }
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2147
        }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2148
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2149
        if (moduleResolution.value() != 0) {
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2150
            extender.moduleResolution(moduleResolution);
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2151
        }
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2152
38468
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2153
        extender.write(baos);
d459a0f8fe72 8156497: Add jar tool support for Multi-Release Modular JARs
chegar
parents: 38457
diff changeset
  2154
        return baos.toByteArray();
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2155
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2156
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2157
    /**
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2158
     * Compute and record hashes
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2159
     */
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2160
    private class Hasher {
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2161
        final ModuleHashesBuilder hashesBuilder;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2162
        final ModuleFinder finder;
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2163
        final Set<String> modules;
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2164
        Hasher(ModuleDescriptor descriptor, String fname) throws IOException {
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2165
            // Create a module finder that finds the modular JAR
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2166
            // being created/updated
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2167
            URI uri = Paths.get(fname).toUri();
42703
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2168
            ModuleReference mref = new ModuleReference(descriptor, uri) {
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2169
                @Override
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2170
                public ModuleReader open() {
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2171
                    throw new UnsupportedOperationException("should not reach here");
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2172
                }
20c39ea4a507 8170987: Module system implementation refresh (12/2016)
alanb
parents: 42463
diff changeset
  2173
            };
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2174
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2175
            // Compose a module finder with the module path and
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2176
            // the modular JAR being created or updated
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2177
            this.finder = ModuleFinder.compose(moduleFinder,
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2178
                new ModuleFinder() {
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2179
                    @Override
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2180
                    public Optional<ModuleReference> find(String name) {
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2181
                        if (descriptor.name().equals(name))
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2182
                            return Optional.of(mref);
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2183
                        else
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2184
                            return Optional.empty();
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2185
                    }
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2186
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2187
                    @Override
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2188
                    public Set<ModuleReference> findAll() {
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2189
                        return Collections.singleton(mref);
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2190
                    }
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2191
                });
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2192
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2193
            // Determine the modules that matches the pattern {@code modulesToHash}
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2194
            Set<String> roots = finder.findAll().stream()
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2195
                .map(ref -> ref.descriptor().name())
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2196
                .filter(mn -> modulesToHash.matcher(mn).find())
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2197
                .collect(Collectors.toSet());
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2198
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2199
            // use system module path unless it creates a modular JAR for
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2200
            // a module that is present in the system image e.g. upgradeable
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2201
            // module
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2202
            ModuleFinder system;
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2203
            String name = descriptor.name();
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2204
            if (name != null && ModuleFinder.ofSystem().find(name).isPresent()) {
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2205
                system = ModuleFinder.of();
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2206
            } else {
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2207
                system = ModuleFinder.ofSystem();
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2208
            }
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2209
            // get a resolved module graph
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2210
            Configuration config =
43712
5dfd0950317c 8173393: Module system implementation refresh (2/2017)
alanb
parents: 43109
diff changeset
  2211
                Configuration.empty().resolve(system, finder, roots);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2212
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2213
            // filter modules resolved from the system module finder
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2214
            this.modules = config.modules().stream()
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2215
                .map(ResolvedModule::name)
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2216
                .filter(mn -> roots.contains(mn) && !system.find(mn).isPresent())
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2217
                .collect(Collectors.toSet());
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2218
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2219
            this.hashesBuilder = new ModuleHashesBuilder(config, modules);
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2220
        }
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2221
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2222
        /**
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2223
         * Compute hashes of the specified module.
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2224
         *
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2225
         * It records the hashing modules that depend upon the specified
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2226
         * module directly or indirectly.
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2227
         */
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2228
        ModuleHashes computeHashes(String name) {
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2229
            if (hashesBuilder == null)
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2230
                return null;
37779
7c84df693837 8154956: Module system implementation refresh (4/2016)
alanb
parents: 36745
diff changeset
  2231
43109
fe275140c3f1 8160286: jmod hash is creating unlinkable modules
mchung
parents: 43099
diff changeset
  2232
            return hashesBuilder.computeHashes(Set.of(name)).get(name);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2233
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents: 29914
diff changeset
  2234
    }
47958
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2235
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2236
    // sort base entries before versioned entries, and sort entry classes with
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2237
    // nested classes so that the outter class appears before the associated
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2238
    // nested class
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2239
    static Comparator<String> ENTRYNAME_COMPARATOR = (s1, s2) ->  {
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2240
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2241
        if (s1.equals(s2)) return 0;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2242
        boolean b1 = s1.startsWith(VERSIONS_DIR);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2243
        boolean b2 = s2.startsWith(VERSIONS_DIR);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2244
        if (b1 && !b2) return 1;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2245
        if (!b1 && b2) return -1;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2246
        int n = 0; // starting char for String compare
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2247
        if (b1 && b2) {
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2248
            // normally strings would be sorted so "10" goes before "9", but
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2249
            // version number strings need to be sorted numerically
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2250
            n = VERSIONS_DIR.length();   // skip the common prefix
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2251
            int i1 = s1.indexOf('/', n);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2252
            int i2 = s2.indexOf('/', n);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2253
            if (i1 == -1) throw new Validator.InvalidJarException(s1);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2254
            if (i2 == -1) throw new Validator.InvalidJarException(s2);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2255
            // shorter version numbers go first
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2256
            if (i1 != i2) return i1 - i2;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2257
            // otherwise, handle equal length numbers below
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2258
        }
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2259
        int l1 = s1.length();
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2260
        int l2 = s2.length();
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2261
        int lim = Math.min(l1, l2);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2262
        for (int k = n; k < lim; k++) {
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2263
            char c1 = s1.charAt(k);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2264
            char c2 = s2.charAt(k);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2265
            if (c1 != c2) {
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2266
                // change natural ordering so '.' comes before '$'
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2267
                // i.e. outer classes come before nested classes
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2268
                if (c1 == '$' && c2 == '.') return 1;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2269
                if (c1 == '.' && c2 == '$') return -1;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2270
                return c1 - c2;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2271
            }
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2272
        }
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2273
        return l1 - l2;
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2274
    };
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2275
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2276
    static Comparator<ZipEntry> ENTRY_COMPARATOR =
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2277
        Comparator.comparing(ZipEntry::getName, ENTRYNAME_COMPARATOR);
d34958cb3163 8186087: jar tool fails to create a multi-release jar when validating nested classes
sherman
parents: 47216
diff changeset
  2278
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  2279
}