jdk/src/java.base/share/classes/jdk/internal/jrtfs/ExplodedImage.java
author sherman
Fri, 15 Apr 2016 13:05:52 -0700
changeset 37365 9cc4eb4d7491
parent 36511 jdk/src/java.base/share/classes/jdk/internal/jrtfs/JrtExplodedFileSystem.java@9d0388c6b336
child 39155 61d7981116d9
permissions -rw-r--r--
8147460: Clean-up jrtfs implementation Reviewed-by: alanb, jlaskey, sundar
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     1
/*
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     2
 * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     4
 *
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    10
 *
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    15
 * accompanied this code).
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    16
 *
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    20
 *
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    23
 * questions.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    24
 */
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    25
package jdk.internal.jrtfs;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    26
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    27
import java.io.IOException;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    28
import java.nio.file.DirectoryStream;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    29
import java.nio.file.FileSystem;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    30
import java.nio.file.FileSystemException;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    31
import java.nio.file.FileSystems;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    32
import java.nio.file.Files;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    33
import java.nio.file.Path;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    34
import java.nio.file.attribute.BasicFileAttributes;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    35
import java.util.ArrayList;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    36
import java.util.Collections;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    37
import java.util.HashMap;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    38
import java.util.List;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    39
import java.util.Map;
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    40
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    41
import jdk.internal.jimage.ImageReader.Node;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    42
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    43
/**
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    44
 * A jrt file system built on $JAVA_HOME/modules directory ('exploded modules
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    45
 * build')
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    46
 *
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    47
 * @implNote This class needs to maintain JDK 8 source compatibility.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    48
 *
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    49
 * It is used internally in the JDK to implement jimage/jrtfs access,
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    50
 * but also compiled and delivered as part of the jrtfs.jar to support access
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    51
 * to the jimage file provided by the shipped JDK by tools running on JDK 8.
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    52
 */
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    53
class ExplodedImage extends SystemImage {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    54
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    55
    private static final String MODULES = "/modules/";
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    56
    private static final String PACKAGES = "/packages/";
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    57
    private static final int PACKAGES_LEN = PACKAGES.length();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    58
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    59
    private final FileSystem defaultFS;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    60
    private final String separator;
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    61
    private final Map<String, PathNode> nodes = Collections.synchronizedMap(new HashMap<>());
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    62
    private final BasicFileAttributes modulesDirAttrs;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    63
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    64
    ExplodedImage(Path modulesDir) throws IOException {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    65
        defaultFS = FileSystems.getDefault();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    66
        String str = defaultFS.getSeparator();
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    67
        separator = str.equals("/") ? null : str;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    68
        modulesDirAttrs = Files.readAttributes(modulesDir, BasicFileAttributes.class);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    69
        initNodes();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    70
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    71
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    72
    // A Node that is backed by actual default file system Path
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    73
    private final class PathNode extends Node {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    74
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    75
        // Path in underlying default file system
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    76
        private Path path;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    77
        private PathNode link;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    78
        private List<Node> children;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    79
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    80
        PathNode(String name, Path path, BasicFileAttributes attrs) {  // path
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    81
            super(name, attrs);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    82
            this.path = path;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    83
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    84
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    85
        PathNode(String name, Node link) {              // link
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    86
            super(name, link.getFileAttributes());
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    87
            this.link = (PathNode)link;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    88
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    89
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    90
        PathNode(String name, List<Node> children) {    // dir
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    91
            super(name, modulesDirAttrs);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    92
            this.children = children;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    93
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    94
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
    95
        @Override
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    96
        public boolean isDirectory() {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    97
            return children != null ||
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    98
                   (link == null && getFileAttributes().isDirectory());
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
    99
        }
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   100
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   101
        @Override
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   102
        public boolean isLink() {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   103
            return link != null;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   104
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   105
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   106
        @Override
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   107
        public PathNode resolveLink(boolean recursive) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   108
            if (link == null)
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   109
                return this;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   110
            return recursive && link.isLink() ? link.resolveLink(true) : link;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   111
        }
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   112
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   113
        byte[] getContent() throws IOException {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   114
            if (!getFileAttributes().isRegularFile())
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   115
                throw new FileSystemException(getName() + " is not file");
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   116
            return Files.readAllBytes(path);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   117
        }
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   118
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   119
        @Override
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   120
        public List<Node> getChildren() {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   121
            if (!isDirectory())
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   122
                throw new IllegalArgumentException("not a directory: " + getNameString());
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   123
            if (children == null) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   124
                List<Node> list = new ArrayList<>();
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   125
                try (DirectoryStream<Path> stream = Files.newDirectoryStream(path)) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   126
                    for (Path p : stream) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   127
                        p = explodedModulesDir.relativize(p);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   128
                        String pName = MODULES + nativeSlashToFrontSlash(p.toString());
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   129
                        Node node = findNode(pName);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   130
                        if (node != null) {  // findNode may choose to hide certain files!
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   131
                            list.add(node);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   132
                        }
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   133
                    }
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   134
                } catch (IOException x) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   135
                    return null;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   136
                }
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   137
                children = list;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   138
            }
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   139
            return children;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   140
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   141
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   142
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   143
    @Override
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   144
    public void close() throws IOException {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   145
        nodes.clear();
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   146
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   147
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   148
    @Override
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   149
    public byte[] getResource(Node node) throws IOException {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   150
        return ((PathNode)node).getContent();
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   151
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   152
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   153
    // find Node for the given Path
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   154
    @Override
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   155
    public synchronized Node findNode(String str) {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   156
        Node node = findModulesNode(str);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   157
        if (node != null) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   158
            return node;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   159
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   160
        // lazily created for paths like /packages/<package>/<module>/xyz
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   161
        // For example /packages/java.lang/java.base/java/lang/
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   162
        if (str.startsWith(PACKAGES)) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   163
            // pkgEndIdx marks end of <package> part
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   164
            int pkgEndIdx = str.indexOf('/', PACKAGES_LEN);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   165
            if (pkgEndIdx != -1) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   166
                // modEndIdx marks end of <module> part
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   167
                int modEndIdx = str.indexOf('/', pkgEndIdx + 1);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   168
                if (modEndIdx != -1) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   169
                    // make sure we have such module link!
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   170
                    // ie., /packages/<package>/<module> is valid
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   171
                    Node linkNode = nodes.get(str.substring(0, modEndIdx));
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   172
                    if (linkNode == null || !linkNode.isLink()) {
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   173
                        return null;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   174
                    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   175
                    // map to "/modules/zyz" path and return that node
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   176
                    // For example, "/modules/java.base/java/lang" for
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   177
                    // "/packages/java.lang/java.base/java/lang".
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   178
                    String mod = MODULES + str.substring(pkgEndIdx + 1);
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   179
                    return findModulesNode(mod);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   180
                }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   181
            }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   182
        }
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   183
        return null;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   184
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   185
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   186
    // find a Node for a path that starts like "/modules/..."
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   187
    Node findModulesNode(String str) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   188
        PathNode node = nodes.get(str);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   189
        if (node != null) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   190
            return node;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   191
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   192
        // lazily created "/modules/xyz/abc/" Node
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   193
        // This is mapped to default file system path "<JDK_MODULES_DIR>/xyz/abc"
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   194
        Path p = underlyingPath(str);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   195
        if (p != null) {
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   196
            try {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   197
                BasicFileAttributes attrs = Files.readAttributes(p, BasicFileAttributes.class);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   198
                if (attrs.isRegularFile()) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   199
                    Path f = p.getFileName();
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   200
                    if (f.toString().startsWith("_the."))
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   201
                        return null;
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   202
                }
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   203
                node = new PathNode(str, p, attrs);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   204
                nodes.put(str, node);
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   205
                return node;
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   206
            } catch (IOException x) {
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   207
                // does not exists or unable to determine
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   208
            }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   209
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   210
        return null;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   211
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   212
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   213
    Path underlyingPath(String str) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   214
        if (str.startsWith(MODULES)) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   215
            str = frontSlashToNativeSlash(str.substring("/modules".length()));
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   216
            return defaultFS.getPath(explodedModulesDir.toString(), str);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   217
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   218
        return null;
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   219
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   220
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   221
    // convert "/" to platform path separator
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   222
    private String frontSlashToNativeSlash(String str) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   223
        return separator == null ? str : str.replace("/", separator);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   224
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   225
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   226
    // convert platform path separator to "/"
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   227
    private String nativeSlashToFrontSlash(String str) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   228
        return separator == null ? str : str.replace(separator, "/");
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   229
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   230
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   231
    // convert "/"s to "."s
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   232
    private String slashesToDots(String str) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   233
        return str.replace(separator != null ? separator : "/", ".");
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   234
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   235
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   236
    // initialize file system Nodes
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   237
    private void initNodes() throws IOException {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   238
        // same package prefix may exist in mutliple modules. This Map
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   239
        // is filled by walking "jdk modules" directory recursively!
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   240
        Map<String, List<String>> packageToModules = new HashMap<>();
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   241
        try (DirectoryStream<Path> stream = Files.newDirectoryStream(explodedModulesDir)) {
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   242
            for (Path module : stream) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   243
                if (Files.isDirectory(module)) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   244
                    String moduleName = module.getFileName().toString();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   245
                    // make sure "/modules/<moduleName>" is created
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   246
                    findModulesNode(MODULES + moduleName);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   247
                    Files.walk(module).filter(Files::isDirectory).forEach((p) -> {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   248
                        p = module.relativize(p);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   249
                        String pkgName = slashesToDots(p.toString());
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   250
                        // skip META-INFO and empty strings
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   251
                        if (!pkgName.isEmpty() && !pkgName.startsWith("META-INF")) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   252
                            List<String> moduleNames = packageToModules.get(pkgName);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   253
                            if (moduleNames == null) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   254
                                moduleNames = new ArrayList<>();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   255
                                packageToModules.put(pkgName, moduleNames);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   256
                            }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   257
                            moduleNames.add(moduleName);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   258
                        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   259
                    });
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   260
                }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   261
            }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   262
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   263
        // create "/modules" directory
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   264
        // "nodes" map contains only /modules/<foo> nodes only so far and so add all as children of /modules
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   265
        PathNode modulesDir = new PathNode("/modules", new ArrayList<>(nodes.values()));
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   266
        nodes.put(modulesDir.getName(), modulesDir);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   267
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   268
        // create children under "/packages"
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   269
        List<Node> packagesChildren = new ArrayList<>(packageToModules.size());
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   270
        for (Map.Entry<String, List<String>> entry : packageToModules.entrySet()) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   271
            String pkgName = entry.getKey();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   272
            List<String> moduleNameList = entry.getValue();
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   273
            List<Node> moduleLinkNodes = new ArrayList<>(moduleNameList.size());
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   274
            for (String moduleName : moduleNameList) {
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   275
                Node moduleNode = findModulesNode(MODULES + moduleName);
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   276
                PathNode linkNode = new PathNode(PACKAGES + pkgName + "/" + moduleName, moduleNode);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   277
                nodes.put(linkNode.getName(), linkNode);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   278
                moduleLinkNodes.add(linkNode);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   279
            }
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   280
            PathNode pkgDir = new PathNode(PACKAGES + pkgName, moduleLinkNodes);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   281
            nodes.put(pkgDir.getName(), pkgDir);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   282
            packagesChildren.add(pkgDir);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   283
        }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   284
        // "/packages" dir
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   285
        PathNode packagesDir = new PathNode("/packages", packagesChildren);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   286
        nodes.put(packagesDir.getName(), packagesDir);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   287
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   288
        // finally "/" dir!
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   289
        List<Node> rootChildren = new ArrayList<>();
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   290
        rootChildren.add(packagesDir);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   291
        rootChildren.add(modulesDir);
37365
9cc4eb4d7491 8147460: Clean-up jrtfs implementation
sherman
parents: 36511
diff changeset
   292
        PathNode root = new PathNode("/", rootChildren);
36511
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   293
        nodes.put(root.getName(), root);
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   294
    }
9d0388c6b336 8142968: Module System implementation
alanb
parents:
diff changeset
   295
}