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

commonMain.okio.FileSystem.kt Maven / Gradle / Ivy

/*
 * Copyright (C) 2020 Square, 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 okio

/**
 * Read and write access to a hierarchical collection of files, addressed by [paths][Path]. This
 * is a natural interface to the current computer's local file system.
 *
 * Other implementations are possible:
 *
 *  * `FakeFileSystem` is an in-memory file system suitable for testing. Note that this class is
 *    included in the `okio-fakefilesystem` artifact.
 *
 *  * [ForwardingFileSystem] is a file system decorator. Use it to apply monitoring, encryption,
 *    compression, or filtering to another file system.
 *
 *  * A ZIP file system could provide access to the contents of a `.zip` file.
 *
 * For improved capability and testability, consider structuring your classes to dependency inject
 * a `FileSystem` rather than using `FileSystem.SYSTEM` directly.
 *
 * Small API
 * ---------
 *
 * This interface is deliberately limited in which features it supports.
 *
 * It is not suitable for high-latency or unreliable remote file systems. It lacks support for
 * retries, timeouts, cancellation, and bulk operations.
 *
 * It cannot create special file types like hard links, pipes, or mounts. Reading or writing these
 * files works as if they were regular files.
 *
 * It cannot read or write file access control features like the UNIX `chmod` and Windows access
 * control lists. It does honor these controls and will fail with an [IOException] if privileges
 * are insufficient!
 *
 * It cannot lock files or check which files are locked.
 *
 * It cannot watch the file system for changes.
 *
 * Applications that need rich file system features should use another API!
 *
 * Multiplatform
 * -------------
 *
 * This class supports a matrix of Kotlin platforms (JVM, Kotlin/Native, Kotlin/JS) and operating
 * systems (Linux, macOS, and Windows). It attempts to balance working similarly across platforms
 * with being consistent with the local operating system.
 *
 * This is a blocking API which limits its applicability on concurrent Node.js services. File
 * operations will block the event loop (and all JavaScript execution!) until they complete.
 *
 * It supports the path schemes of both Windows (like `C:\Users`) and UNIX (like `/home`). Note that
 * path resolution rules differ by platform.
 *
 * Differences vs. Java IO APIs
 * ----------------------------
 *
 * The `java.io.File` class is Java's original file system API. The `delete` and `renameTo` methods
 * return false if the operation failed. The `list` method returns null if the file isn't a
 * directory or could not be listed. This class always throws an [IOException] when an operation
 * doesn't succeed.
 *
 * The `java.nio.Path` and `java.nio.Files` classes are the entry points of Java's new file system
 * API. Each `Path` instance is scoped to a particular file system, though that is often implicit
 * because the `Paths.get()` function automatically uses the default (ie. system) file system.
 * In Okio's API paths are just identifiers; you must use a specific `FileSystem` object to do
 * I/O with.
 */
expect abstract class FileSystem() {

  /**
   * Resolves [path] against the current working directory and symlinks in this file system. The
   * returned path identifies the same file as [path], but with an absolute path that does not
   * include any symbolic links.
   *
   * This is similar to `File.getCanonicalFile()` on the JVM and `realpath` on POSIX. Unlike
   * `File.getCanonicalFile()`, this throws if the file doesn't exist.
   *
   * @throws IOException if [path] cannot be resolved. This will occur if the file doesn't exist,
   *     if the current working directory doesn't exist or is inaccessible, or if another failure
   *     occurs while resolving the path.
   */
  @Throws(IOException::class)
  abstract fun canonicalize(path: Path): Path

  /**
   * Returns metadata of the file, directory, or object identified by [path].
   *
   * @throws IOException if [path] does not exist or its metadata cannot be read.
   */
  @Throws(IOException::class)
  fun metadata(path: Path): FileMetadata

  /**
   * Returns metadata of the file, directory, or object identified by [path]. This returns null if
   * there is no file at [path].
   *
   * @throws IOException if [path] cannot be accessed due to a connectivity problem, permissions
   *     problem, or other issue.
   */
  @Throws(IOException::class)
  abstract fun metadataOrNull(path: Path): FileMetadata?

  /**
   * Returns true if [path] identifies an object on this file system.
   *
   * @throws IOException if [path] cannot be accessed due to a connectivity problem, permissions
   *     problem, or other issue.
   */
  @Throws(IOException::class)
  fun exists(path: Path): Boolean

  /**
   * Returns the children of [dir]. The returned list is sorted using natural ordering. If [dir] is
   * a relative path, the returned elements will also be relative paths. If it is an absolute path,
   * the returned elements will also be absolute paths.
   *
   * Note that a path does not need to be a [directory][FileMetadata.isDirectory] for this function
   * to return successfully. For example, mounted storage devices may have child files but do not
   * identify themselves as directories.
   *
   * @throws IOException if [dir] does not exist or cannot be listed. A path cannot be listed if the
   *     current process doesn't have access to [dir], or if there's a loop of symbolic links, or if
   *     any name is too long.
   */
  @Throws(IOException::class)
  abstract fun list(dir: Path): List

  /**
   * Returns the children of the directory identified by [dir]. The returned list is sorted using
   * natural ordering. If [dir] is a relative path, the returned elements will also be relative
   * paths. If it is an absolute path, the returned elements will also be absolute paths.
   *
   * This returns null if [dir] does not exist or cannot be listed. A directory cannot be listed if
   * the current process doesn't have access to [dir], or if there's a loop of symbolic links, or if
   * any name is too long.
   */
  abstract fun listOrNull(dir: Path): List?

  /**
   * Returns a sequence that **lazily** traverses the children of [dir] using repeated calls to
   * [list]. If none of [dir]'s children are directories this returns the same elements as [list].
   *
   * The returned sequence visits the tree of files in depth-first order. Parent paths are returned
   * before their children.
   *
   * Note that [listRecursively] does not throw exceptions but the returned sequence does. When it
   * is iterated, the returned sequence throws a [FileNotFoundException] if [dir] does not exist, or
   * an [IOException] if [dir] cannot be listed.
   *
   * @param followSymlinks true to follow symlinks while traversing the children. If [dir] itself is
   *     a symlink it will be followed even if this parameter is false.
   */
  open fun listRecursively(dir: Path, followSymlinks: Boolean = false): Sequence

  /**
   * Returns a handle to read [file]. This will fail if the file doesn't already exist.
   *
   * @throws IOException if [file] does not exist, is not a file, or cannot be accessed. A file
   *     cannot be accessed if the current process doesn't have sufficient permissions for [file],
   *     if there's a loop of symbolic links, or if any name is too long.
   */
  @Throws(IOException::class)
  abstract fun openReadOnly(file: Path): FileHandle

  /**
   * Returns a handle to read and write [file]. This will create the file if it doesn't already
   * exist.
   *
   * @param mustCreate true to throw an [IOException] instead of overwriting an existing file.
   *     This is equivalent to `O_EXCL` on POSIX and `CREATE_NEW` on Windows.
   * @param mustExist true to throw an [IOException] instead of creating a new file. This is
   *     equivalent to `r+` on POSIX and `OPEN_EXISTING` on Windows.
   * @throws IOException if [file] is not a file, or cannot be accessed. A file cannot be accessed
   *     if the current process doesn't have sufficient reading and writing permissions for [file],
   *     if there's a loop of symbolic links, or if any name is too long.
   */
  @Throws(IOException::class)
  abstract fun openReadWrite(
    file: Path,
    mustCreate: Boolean = false,
    mustExist: Boolean = false
  ): FileHandle

  /**
   * Returns a source that reads the bytes of [file] from beginning to end.
   *
   * @throws IOException if [file] does not exist, is not a file, or cannot be read. A file cannot
   *     be read if the current process doesn't have access to [file], if there's a loop of symbolic
   *     links, or if any name is too long.
   */
  @Throws(IOException::class)
  abstract fun source(file: Path): Source

  /**
   * Creates a source to read [file], executes [readerAction] to read it, and then closes the
   * source. This is a compact way to read the contents of a file.
   */
  @Throws(IOException::class)
  inline fun  read(file: Path, readerAction: BufferedSource.() -> T): T

  /**
   * Returns a sink that writes bytes to [file] from beginning to end. If [file] already exists it
   * will be replaced with the new data.
   *
   * @param mustCreate true to throw an [IOException] instead of overwriting an existing file.
   *     This is equivalent to `O_EXCL` on POSIX and `CREATE_NEW` on Windows.
   *
   * @throws IOException if [file] cannot be written. A file cannot be written if its enclosing
   *     directory does not exist, if the current process doesn't have access to [file], if there's
   *     a loop of symbolic links, or if any name is too long.
   */
  @Throws(IOException::class)
  abstract fun sink(file: Path, mustCreate: Boolean = false): Sink

  /**
   * Creates a sink to write [file], executes [writerAction] to write it, and then closes the sink.
   * This is a compact way to write a file.
   *
   * @param mustCreate true to throw an [IOException] instead of overwriting an existing file.
   *     This is equivalent to `O_EXCL` on POSIX and `CREATE_NEW` on Windows.
   */
  @Throws(IOException::class)
  inline fun  write(
    file: Path,
    mustCreate: Boolean = false,
    writerAction: BufferedSink.() -> T
  ): T

  /**
   * Returns a sink that appends bytes to the end of [file], creating it if it doesn't already
   * exist.
   *
   * @param mustExist true to throw an [IOException] instead of creating a new file. This is
   *     equivalent to `r+` on POSIX and `OPEN_EXISTING` on Windows.
   *
   * @throws IOException if [file] cannot be written. A file cannot be written if its enclosing
   *     directory does not exist, if the current process doesn't have access to [file], if there's
   *     a loop of symbolic links, or if any name is too long.
   */
  @Throws(IOException::class)
  abstract fun appendingSink(file: Path, mustExist: Boolean = false): Sink

  /**
   * Creates a directory at the path identified by [dir].
   *
   * @param mustCreate true to throw an [IOException] if the directory already exists.
   * @throws IOException if [dir]'s parent does not exist, is not a directory, or cannot be written.
   *     A directory cannot be created if the current process doesn't have access, if there's a loop
   *     of symbolic links, or if any name is too long.
   */
  @Throws(IOException::class)
  abstract fun createDirectory(dir: Path, mustCreate: Boolean = false)

  /**
   * Creates a directory at the path identified by [dir], and any enclosing parent path directories,
   * recursively.
   *
   * @param mustCreate true to throw an [IOException] instead of overwriting an existing directory.
   * @throws IOException if any [metadata] or [createDirectory] operation fails.
   */
  @Throws(IOException::class)
  fun createDirectories(dir: Path, mustCreate: Boolean = false)

  /**
   * Moves [source] to [target] in-place if the underlying file system supports it. If [target]
   * exists, it is first removed. If `source == target`, this operation does nothing. This may be
   * used to move a file or a directory.
   *
   * **Only as Atomic as the Underlying File System Supports**
   *
   * FAT and NTFS file systems cannot atomically move a file over an existing file. If the target
   * file already exists, the move is performed into two steps:
   *
   *  1. Atomically delete the target file.
   *  2. Atomically rename the source file to the target file.
   *
   * The delete step and move step are each atomic but not atomic in aggregate! If this process
   * crashes, the host operating system crashes, or the hardware fails it is possible that the
   * delete step will succeed and the rename will not.
   *
   * **Entire-file or nothing**
   *
   * These are the possible results of this operation:
   *
   *  * This operation returns normally, the source file is absent, and the target file contains the
   *    data previously held by the source file. This is the success case.
   *
   *  * The operation throws an [IOException] and the file system is unchanged. For example, this
   *    occurs if this process lacks permissions to perform the move.
   *
   *  * This operation throws an [IOException], the target file is deleted, but the source file is
   *    unchanged. This is the partial failure case described above and is only possible on
   *    file systems like FAT and NTFS that do not support atomic file replacement. Typically in
   *    such cases this operation won't return at all because the process or operating system has
   *    also crashed.
   *
   * There is no failure mode where the target file holds a subset of the bytes of the source file.
   * If the rename step cannot be performed atomically, this function will throw an [IOException]
   * before attempting a move. Typically this occurs if the source and target files are on different
   * physical volumes.
   *
   * **Non-Atomic Moves**
   *
   * If you need to move files across volumes, use [copy] followed by [delete], and change your
   * application logic to recover should the copy step suffer a partial failure.
   *
   * @throws IOException if the move cannot be performed, or cannot be performed atomically. Moves
   *     fail if the source doesn't exist, if the target is not writable, if the target already
   *     exists and cannot be replaced, or if the move would cause physical or quota limits to be
   *     exceeded. This list of potential problems is not exhaustive.
   */
  @Throws(IOException::class)
  abstract fun atomicMove(source: Path, target: Path)

  /**
   * Copies all the bytes from the file at [source] to the file at [target]. This does not copy
   * file metadata like last modified time, permissions, or extended attributes.
   *
   * This function is not atomic; a failure may leave [target] in an inconsistent state. For
   * example, [target] may be empty or contain only a prefix of [source].
   *
   * @throws IOException if [source] cannot be read or if [target] cannot be written.
   */
  @Throws(IOException::class)
  open fun copy(source: Path, target: Path)

  /**
   * Deletes the file or directory at [path].
   *
   * @param mustExist true to throw an [IOException] if there is nothing at [path] to delete.
   * @throws IOException if there is a file or directory but it could not be deleted. Deletes fail
   *     if the current process doesn't have access, if the file system is readonly, or if [path]
   *     is a non-empty directory. This list of potential problems is not exhaustive.
   */
  @Throws(IOException::class)
  abstract fun delete(path: Path, mustExist: Boolean = false)

  /**
   * Recursively deletes all children of [fileOrDirectory] if it is a directory, then deletes
   * [fileOrDirectory] itself.
   *
   * This function does not defend against race conditions. For example, if child files are created
   * or deleted in [fileOrDirectory] while this function is executing, this may fail with an
   * [IOException].
   *
   * @param mustExist true to throw an [IOException] if there is nothing at [fileOrDirectory] to
   *     delete.
   * @throws IOException if any [metadata], [list], or [delete] operation fails.
   */
  @Throws(IOException::class)
  open fun deleteRecursively(fileOrDirectory: Path, mustExist: Boolean = false)

  /**
   * Creates a symbolic link at [source] that resolves to [target]. If [target] is a relative path,
   * it is relative to `source.parent`.
   *
   * @throws IOException if [source] cannot be created. This may be because it already exists
   *     or because its storage doesn't support symlinks. This list of potential problems is not
   *     exhaustive.
   */
  @Throws(IOException::class)
  abstract fun createSymlink(source: Path, target: Path)

  companion object {
    /**
     * Returns a writable temporary directory on [SYSTEM].
     *
     * This is platform-specific.
     *
     *  * **JVM and Android**: the path in the `java.io.tmpdir` system property
     *  * **Linux, iOS, and macOS**: the path in the `TMPDIR` environment variable.
     *  * **Windows**: the first non-null of `TEMP`, `TMP`, and `USERPROFILE` environment variables.
     *
     * **Note that the returned directory is not generally private.** Other users or processes that
     * share this file system may read data that is written to this directory, or write malicious
     * data for this process to receive.
     */
    val SYSTEM_TEMPORARY_DIRECTORY: Path
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy