All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cn.nukkit.plugin.js.ESMFileSystem Maven / Gradle / Ivy

package cn.nukkit.plugin.js;

import cn.nukkit.Nukkit;
import cn.nukkit.Server;
import cn.nukkit.plugin.CommonJSPlugin;
import cn.nukkit.plugin.JavaPluginLoader;
import cn.nukkit.utils.SeekableInMemoryByteChannel;
import org.graalvm.polyglot.io.FileSystem;

import java.io.*;
import java.net.URI;
import java.nio.channels.SeekableByteChannel;
import java.nio.charset.StandardCharsets;
import java.nio.file.*;
import java.nio.file.attribute.FileAttribute;
import java.util.Map;
import java.util.Set;
import java.util.WeakHashMap;

import static cn.nukkit.plugin.js.JSClassLoader.javaClassCache;

public final class ESMFileSystem implements FileSystem {
    final File baseDir;
    private final CommonJSPlugin plugin;
    private ClassLoader mainClassLoader;

    private final static Map innerModuleCache = new WeakHashMap<>(1, 1f);

    public ESMFileSystem(File baseDir, CommonJSPlugin plugin) {
        this.baseDir = baseDir;
        this.plugin = plugin;
    }

    @Override
    public Path parsePath(URI uri) {
        return parsePath(uri.toString());
    }

    @Override
    public Path parsePath(String path) {
        Path resolvedPath = null;
        var feature = JSFeatures.getFeatureByModule(path);
        if (feature != null && plugin.usedFeatures.containsKey(feature.getName())) {
            resolvedPath = Path.of("jsFeature", feature.getName() + "@" + path);
        } else if (path.startsWith("@")) {
            if (!path.contains("/") && !path.contains("\\")) {
                resolvedPath = Path.of(Server.getInstance().getPluginPath(), path + "/index.js");
            } else {
                resolvedPath = Path.of(Server.getInstance().getPluginPath(), path);
            }
        } else if (path.startsWith(":")) {
            resolvedPath = Path.of("inner-module", path.substring(1));
        } else if ((!path.endsWith(".js") && !path.startsWith("./") && !path.startsWith("../") && getDots(path) > 1)) {
            if (mainClassLoader == null)
                mainClassLoader = Thread.currentThread().getContextClassLoader();
            try {
                if (javaClassCache.containsKey(path)) {
                    return Path.of("java-class", path);
                }
                var clazz = mainClassLoader.loadClass(path);
                if (clazz != null) {
                    javaClassCache.put(path, clazz);
                    resolvedPath = Path.of("java-class", path);
                } else {
                    outer:
                    for (var pl : Server.getInstance().getPluginManager().getFileAssociations().values()) {
                        if (pl instanceof JavaPluginLoader javaPluginLoader) {
                            for (var loader : javaPluginLoader.getClassLoaders().values()) {
                                clazz = loader.loadClass(path);
                                if (clazz != null) {
                                    javaClassCache.put(path, clazz);
                                    resolvedPath = Path.of("java-class", path);
                                    break outer;
                                }
                            }
                        }
                    }
                }
            } catch (ClassNotFoundException ignore) {
                outer2:
                for (var pl : Server.getInstance().getPluginManager().getFileAssociations().values()) {
                    if (pl instanceof JavaPluginLoader javaPluginLoader) {
                        for (var loader : javaPluginLoader.getClassLoaders().values()) {
                            try {
                                var clazz = loader.loadClass(path);
                                if (clazz != null) {
                                    javaClassCache.put(path, clazz);
                                    resolvedPath = Path.of("java-class", path);
                                    break outer2;
                                }
                            } catch (ClassNotFoundException ignore2) {

                            }
                        }
                    }
                }
            }
        }
        if (resolvedPath == null) {
            resolvedPath = baseDir.toPath().resolve(path);
        }
        if (!resolvedPath.startsWith("java-class") && !resolvedPath.startsWith("jsFeature") && !resolvedPath.startsWith("inner-module") && !Files.isRegularFile(resolvedPath)) {
            var tmpPath = Path.of(resolvedPath + ".js");
            if (Files.isRegularFile(tmpPath)) {
                resolvedPath = tmpPath;
            }
        }
        return resolvedPath;
    }

    private static int getDots(String originStr) {
        var res = 0;
        var i = originStr.indexOf('.');
        while (i != -1) {
            i = originStr.indexOf('.', i + 1);
            res++;
        }
        return res;
    }

    @Override
    public void checkAccess(Path path, Set modes, LinkOption... linkOptions) throws IOException {
        if (path.startsWith("inner-module")) {
            for (var each : modes) {
                if (each != AccessMode.READ) {
                    throw new IOException("Inner module cannot be accessed.");
                }
            }
        } else if (path.startsWith("java-class")) {
            for (var each : modes) {
                if (each != AccessMode.READ) {
                    throw new IOException("Java class cannot be accessed.");
                }
            }
        } else if (path.startsWith("jsFeature")) {
            for (var each : modes) {
                if (each != AccessMode.READ) {
                    throw new IOException("JS Feature cannot be accessed.");
                }
            }
        } else {
            path = path.toRealPath(linkOptions);
            for (var each : modes) {
                if (each == AccessMode.READ && !Files.isReadable(path)) {
                    throw new AccessDeniedException(path.toString());
                } else if (each == AccessMode.WRITE && !Files.isWritable(path)) {
                    throw new AccessDeniedException(path.toString());
                } else if (each == AccessMode.EXECUTE && !Files.isExecutable(path)) {
                    throw new AccessDeniedException(path.toString());
                }
            }
        }
    }

    @Override
    public void createDirectory(Path dir, FileAttribute... attrs) throws IOException {
        if (dir.startsWith("inner-module") || dir.startsWith("java-class") || dir.startsWith("jsFeature")) {
            throw new IOException("Inner module cannot be accessed.");
        }
        Files.createDirectories(dir, attrs);
    }

    @Override
    public void delete(Path path) throws IOException {
        if (path.startsWith("inner-module") || path.startsWith("java-class") || path.startsWith("jsFeature")) {
            throw new IOException("Inner module cannot be accessed.");
        }
        Files.delete(path);
    }

    @Override
    public SeekableByteChannel newByteChannel(Path path, Set options, FileAttribute... attrs) throws IOException {
        if (path.startsWith("inner-module")) {
            byte[] contents = new byte[0];
            var moduleName = path.toString().substring(13);
            if ("plugin-id".equals(moduleName)) {
                contents = ("export const id = " + plugin.id).getBytes(StandardCharsets.UTF_8);
            } else {
                if (innerModuleCache.containsKey(moduleName)) {
                    contents = innerModuleCache.get(moduleName);
                } else {
                    try (var ins = Nukkit.class.getModule().getResourceAsStream("inner-module/" + moduleName + ".js")) {
                        if (ins != null)
                            contents = ins.readAllBytes();
                    }
                    if (contents.length != 0) {
                        innerModuleCache.put(moduleName, contents);
                    }
                }
            }
            return new SeekableInMemoryByteChannel(contents);
        } else if (path.startsWith("java-class")) {
            var className = path.toString().substring(11);
            return new SeekableInMemoryByteChannel(ESMJavaExporter.exportJava(javaClassCache.get(className)).getBytes(StandardCharsets.UTF_8));
        } else if (path.startsWith("jsFeature")) {
            var tmp = path.toString().substring(10);
            var index = tmp.lastIndexOf("@");
            var featureName = tmp.substring(0, index);
            var moduleName = tmp.substring(index + 1);
            var feature = plugin.usedFeatures.get(featureName);
            var module = feature.generateModule(moduleName, plugin.getJsContext());
            var codeBuilder = new StringBuilder("const JSFeaturesClass = Java.type('cn.nukkit.plugin.js.JSFeatures');\n");
            for (var each : module.entrySet()) {
                var id = JSFeatures.FEATURE_GENERATED_TMP_ID.getAndIncrement();
                JSFeatures.FEATURE_GENERATED_TMP_MAP.put(id, each.getValue());
                codeBuilder.append("export const ").append(each.getKey()).append(" = ").append("JSFeaturesClass.FEATURE_GENERATED_TMP_MAP.remove(").append(id).append(");\n");
            }
            return new SeekableInMemoryByteChannel(codeBuilder.toString().getBytes(StandardCharsets.UTF_8));
        }
        return Files.newByteChannel(path, options, attrs);
    }

    public Reader newReader(Path path) throws IOException {
        if (path.startsWith("inner-module")) {
            String contents = "";
            var moduleName = path.toString().substring(13);
            if ("plugin-id".equals(moduleName)) {
                contents = "export const id = " + plugin.id;
            } else {
                if (innerModuleCache.containsKey(moduleName)) {
                    contents = new String(innerModuleCache.get(moduleName));
                } else {
                    try (var ins = Nukkit.class.getModule().getResourceAsStream("inner-module/" + moduleName + ".js")) {
                        if (ins != null)
                            return new InputStreamReader(ins);
                    }
                }
            }
            return new StringReader(contents);
        } else if (path.startsWith("java-class")) {
            var className = path.toString().substring(11);
            return new StringReader(ESMJavaExporter.exportJava(javaClassCache.get(className)));
        } else if (path.startsWith("jsFeature")) {
            var tmp = path.toString().substring(10);
            var index = tmp.lastIndexOf("@");
            var featureName = tmp.substring(0, index);
            var moduleName = tmp.substring(index + 1);
            var feature = plugin.usedFeatures.get(featureName);
            var module = feature.generateModule(moduleName, plugin.getJsContext());
            var codeBuilder = new StringBuilder("const JSFeaturesClass = Java.type('cn.nukkit.plugin.js.JSFeatures');\n");
            for (var each : module.entrySet()) {
                var id = JSFeatures.FEATURE_GENERATED_TMP_ID.getAndIncrement();
                JSFeatures.FEATURE_GENERATED_TMP_MAP.put(id, each.getValue());
                codeBuilder.append("export const ").append(each.getKey()).append(" = ").append("JSFeaturesClass.FEATURE_GENERATED_TMP_MAP.remove(").append(id).append(");\n");
            }
            return new StringReader(codeBuilder.toString());
        }
        return Files.newBufferedReader(path, StandardCharsets.UTF_8);
    }

    @Override
    public DirectoryStream newDirectoryStream(Path dir, DirectoryStream.Filter filter) throws IOException {
        if (dir.startsWith("inner-module") || dir.startsWith("java-class") || dir.startsWith("jsFeature")) {
            throw new IOException("Inner module cannot be accessed.");
        }
        return Files.newDirectoryStream(dir, filter);
    }

    @Override
    public Path toAbsolutePath(Path path) {
        if (path.startsWith("inner-module") || path.startsWith("java-class") || path.startsWith("jsFeature")) {
            return path;
        }
        return path.toAbsolutePath();
    }

    @Override
    public Path toRealPath(Path path, LinkOption... linkOptions) throws IOException {
        if (path.startsWith("inner-module") || path.startsWith("java-class") || path.startsWith("jsFeature")) {
            return path;
        }
        return path.toRealPath(linkOptions);
    }

    @Override
    public Map readAttributes(Path path, String attributes, LinkOption... options) throws IOException {
        if (path.startsWith("inner-module") || path.startsWith("java-class") || path.startsWith("jsFeature")) {
            throw new IOException("Inner module cannot be accessed.");
        }
        return Files.readAttributes(path, attributes, options);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy