com.google.common.jimfs.Jimfs.java Source code

Java tutorial

Introduction

Here is the source code for com.google.common.jimfs.Jimfs.java

Source

/*
 * Copyright 2013 Google Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.google.common.jimfs;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.jimfs.SystemJimfsFileSystemProvider.FILE_SYSTEM_KEY;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableMap;

import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.util.UUID;

/**
 * Static factory methods for creating new Jimfs file systems. File systems may either be created
 * with a basic configuration matching the current operating system or by providing a specific
 * {@link Configuration}. Basic {@linkplain Configuration#unix() UNIX},
 * {@linkplain Configuration#osX() Mac OS X} and {@linkplain Configuration#windows() Windows}
 * configurations are provided.
 *
 * <p>Examples:
 *
 * <pre>
 *   // A file system with a configuration similar to the current OS
 *   FileSystem fileSystem = Jimfs.newFileSystem();
 *
 *   // A file system with paths and behavior generally matching that of Windows
 *   FileSystem windows = Jimfs.newFileSystem(Configuration.windows());  </pre>
 *
 * <p>Additionally, various behavior of the file system can be customized by creating a custom
 * {@link Configuration}. A modified version of one of the existing default configurations can be
 * created using {@link Configuration#toBuilder()} or a new configuration can be created from
 * scratch with {@link Configuration#builder(PathType)}. See {@link Configuration.Builder} for what
 * can be configured.
 *
 * <p>Examples:
 *
 * <pre>
 *   // Modify the default UNIX configuration
 *   FileSystem fileSystem = Jimfs.newFileSystem(Configuration.unix()
 *       .toBuilder()
 *       .setAttributeViews("basic", "owner", "posix", "unix")
 *       .setWorkingDirectory("/home/user")
 *       .setBlockSize(4096)
 *       .build());
 *
 *   // Create a custom configuration
 *   Configuration config = Configuration.builder(PathType.windows())
 *       .setRoots("C:\\", "D:\\", "E:\\")
 *       // ...
 *       .build();  </pre>
 *
 * @author Colin Decker
 */
public final class Jimfs {

    /**
     * The URI scheme for the Jimfs file system ("jimfs").
     */
    public static final String URI_SCHEME = "jimfs";

    /**
     * The key used for mapping to the {@link Configuration} in the {@code env} map when creating a
     * new file system instance using {@code FileSystems.newFileSystem()}.
     */
    public static final String CONFIG_KEY = "config";

    private Jimfs() {
    }

    /**
     * Creates a new in-memory file system with a default configuration appropriate to the current
     * operating system. More specifically, if the operating system is Windows,
     * {@link Configuration#windows()} is used; if the operating system is Mac OS X,
     * {@link Configuration#osX()} is used; otherwise, {@link Configuration#unix()} is used.
     */
    public static FileSystem newFileSystem() {
        return newFileSystem(newRandomFileSystemName());
    }

    /**
     * Creates a new in-memory file system with a default configuration appropriate to the current
     * operating system. More specifically, if the operating system is Windows,
     * {@link Configuration#windows()} is used; if the operating system is Mac OS X,
     * {@link Configuration#osX()} is used; otherwise, {@link Configuration#unix()} is used.
     *
     * <p>The returned file system uses the given name as the host part of its URI and the URIs of
     * paths in the file system. For example, given the name {@code my-file-system}, the file
     * system's URI will be {@code jimfs://my-file-system} and the URI of the path {@code /foo/bar}
     * will be {@code jimfs://my-file-system/foo/bar}.
     */
    public static FileSystem newFileSystem(String name) {
        String os = System.getProperty("os.name");

        Configuration config;
        if (os.contains("Windows")) {
            config = Configuration.windows();
        } else if (os.contains("OS X")) {
            config = Configuration.osX();
        } else {
            config = Configuration.unix();
        }

        return newFileSystem(name, config);
    }

    /**
     * Creates a new in-memory file system with the given configuration.
     */
    public static FileSystem newFileSystem(Configuration configuration) {
        return newFileSystem(newRandomFileSystemName(), configuration);
    }

    /**
     * Creates a new in-memory file system with the given configuration.
     *
     * <p>The returned file system uses the given name as the host part of its URI and the URIs of
     * paths in the file system. For example, given the name {@code my-file-system}, the file
     * system's URI will be {@code jimfs://my-file-system} and the URI of the path {@code /foo/bar}
     * will be {@code jimfs://my-file-system/foo/bar}.
     */
    public static FileSystem newFileSystem(String name, Configuration configuration) {
        try {
            URI uri = new URI(URI_SCHEME, name, null, null);
            return newFileSystem(uri, configuration);
        } catch (URISyntaxException e) {
            throw new IllegalArgumentException(e);
        }
    }

    @VisibleForTesting
    static FileSystem newFileSystem(URI uri, Configuration config) {
        checkArgument(URI_SCHEME.equals(uri.getScheme()), "uri (%s) must have scheme %s", uri, URI_SCHEME);

        try {
            // Create the FileSystem. It uses JimfsFileSystemProvider as its provider, as that is
            // the provider that actually implements the operations needed for Files methods to work.
            JimfsFileSystem fileSystem = JimfsFileSystems.newFileSystem(JimfsFileSystemProvider.instance(), uri,
                    config);

            // Now, call FileSystems.newFileSystem, passing it the FileSystem we just created. This
            // allows the system-loaded SystemJimfsFileSystemProvider instance to cache the FileSystem
            // so that methods like Paths.get(URI) work.
            // We do it in this awkward way to avoid issues when the classes in the API (this class
            // and Configuration, for example) are loaded by a different classloader than the one that
            // loads SystemJimfsFileSystemProvider using ServiceLoader. See
            // https://github.com/google/jimfs/issues/18 for gory details.
            ImmutableMap<String, ?> env = ImmutableMap.of(FILE_SYSTEM_KEY, fileSystem);
            FileSystems.newFileSystem(uri, env, SystemJimfsFileSystemProvider.class.getClassLoader());

            return fileSystem;
        } catch (IOException e) {
            throw new AssertionError(e);
        }
    }

    private static String newRandomFileSystemName() {
        return UUID.randomUUID().toString();
    }
}