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

grizzled.file.util.scala Maven / Gradle / Ivy

The newest version!
/*
  ---------------------------------------------------------------------------
  This software is released under a BSD license, adapted from
  http://opensource.org/licenses/bsd-license.php

  Copyright (c) 2009-2012, Brian M. Clapper
  All rights reserved.

  Redistribution and use in source and binary forms, with or without
  modification, are permitted provided that the following conditions are
  met:

  * Redistributions of source code must retain the above copyright notice,
    this list of conditions and the following disclaimer.

  * Redistributions in binary form must reproduce the above copyright
    notice, this list of conditions and the following disclaimer in the
    documentation and/or other materials provided with the distribution.

  * Neither the names "clapper.org", "Grizzled Scala Library", nor the
    names of its contributors may be used to endorse or promote products
    derived from this software without specific prior written permission.

  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
  IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
  THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
  CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  ---------------------------------------------------------------------------
*/

package grizzled.file

import scala.util.matching.Regex
import scala.annotation.tailrec

import grizzled.file.GrizzledFile._
import grizzled.io.RichInputStream._
import grizzled.io.RichReader._
import grizzled.sys.os
import grizzled.sys.OperatingSystem._

import java.io.{File, IOException}
import java.security.{SecureRandom => Random}

class FileDoesNotExistException(message: String) extends Exception

/**
 * Useful file-related utility functions.
 */
object util {
  import grizzled.string._ // Grizzled string functions

  val fileSeparator = File.separator
  val fileSeparatorChar = fileSeparator(0)

  /* ---------------------------------------------------------------------- *\
   Public Methods
   \* ---------------------------------------------------------------------- */

  /**
   * Get the directory name of a pathname.
   *
   * @param path     path (absolute or relative)
   * @param fileSep  the file separator to use. Defaults to the value of
   *                 the "file.separator" property.
   *
   * @return the directory portion
   */
  def dirname(path: String, fileSep: String = fileSeparator): String = {
    val components = splitPath(path, fileSep) 
    components match {
      case Nil => 
        ""

      case List("") =>
        ""

      case simple :: Nil if (! (simple startsWith fileSep)) =>
        "."

      case _ => {
        val len = components.length
        val result = components.take(len - 1) mkString fileSep
        if (result.length == 0)
          fileSep
        else
          result
      }
    }
  }

  /**
   * Get the basename (file name only) part of a path.
   *
   * @param path     the path (absolute or relative)
   * @param fileSep  the file separator to use. Defaults to the value of
   *                 the "file.separator" property.
   *
   * @return the file name portion
   */
  def basename(path: String, fileSep: String = fileSeparator): String = {
    val components = splitPath(path, fileSep)
    components match {
      case Nil =>
        ""
      case List("") =>
        ""
      case simple :: Nil if (! (simple startsWith fileSep)) =>
        path
      case _ =>
        components.drop(components.length - 1) mkString fileSep
    }
  }

  /**
   * Split a path into directory (dirname) and file (basename) components.
   * Analogous to Python's os.path.pathsplit() function.
   *
   * @param path     the path to split
   * @param fileSep  the file separator to use. Defaults to the value of
   *                 the "file.separator" property.
   *
   * @return a (dirname, basename) tuple of strings
   */
  def dirnameBasename(path: String,
                      fileSep: String = fileSeparator): (String, String) = {
    if ((path == null) || (path.length == 0))
      ("", "")
    else if ((path == ".") || (path == ".."))
      (path, "")
    else if (! (path contains fileSep))
      (".", path)
    else if (path == fileSep)
      (fileSep, "")
    else  {
      // Use a character to split, so it's not interpreted as a regular
      // expression (which causes problems with a Windows-style "\".
      // NOTE: We deliberately don't use splitPath() here.
      val components = (path split fileSep(0)).toList

      if (components.length == 1)
        (components(0), "")

      else {
        val listTuple = components splitAt (components.length - 1)
        val s: String = listTuple._1 mkString fileSep
        val prefix = 
          if ((s.length == 0) && (path startsWith fileSep))
            fileSep
          else
            s
        (prefix, listTuple._2 mkString fileSep)
      }
    }
  }

  private lazy val ExtRegexp = """^(.*)(\.[^.]+)$""".r

  /**
   * Split a pathname into the directory name, basename, and extension
   * pieces.
   *
   * @param pathname the pathname
   * @param fileSep  the file separator to use. Defaults to the value of
   *                 the "file.separator" property.
   *
   * @return a 3-tuple of (dirname, basename, extension)
   */
  def dirnameBasenameExtension(pathname: String, 
                               fileSep: String = fileSeparator):
  (String, String, String) = {

    val (path1, extension) = pathname match {
      case ExtRegexp(pathNoExt, ext) => (pathNoExt, ext)
      case _                         => (pathname, "")
    }

    val (dirname, basename) = dirnameBasename(path1, fileSep)
    (dirname, basename, extension)
  }

  /**
   * Return the current working directory, as an absolute path.
   *
   * @return the current working directory
   */
  def pwd: String = new File(".").getCanonicalPath

  /**
   * Calculate the relative path between two files.
   *
   * @param from  the starting file
   * @param to    the file to be converted to a relative path
   *
   * @return the (String) relative path
   */
  def relativePath(from: File, to: File): String = {
    if (from.getAbsolutePath == to.getAbsolutePath)
      basename(from.getPath)

    else {
      val fromPath = toPathArray(from)
      val toPath = toPathArray(to)
      val commonLength = commonPrefix(fromPath, toPath)
      val relativeTo = toPath.drop(commonLength)
      if (fromPath.length == commonLength)
        // It's right under the from path.
        relativeTo mkString fileSeparator
      else {
        val commonParentsTotal = (fromPath.length - commonLength - 1)
          require(commonParentsTotal >= 0)
        val up = (".." + fileSeparator) * commonParentsTotal
        relativeTo.mkString(up, fileSeparator, "")
      }
    }
  }

  /**
   * Return a list of paths matching a pathname pattern. The pattern may
   * contain simple shell-style wildcards. See fnmatch().
   *
   * @param path  The path to expand.
   *
   * @return a list of possibly expanded file names
   */
  def glob(path: String): List[String] = {
    // This method is essentially a direct translation of the Python
    // glob.glob() function.

    def glob1(dirname: String, pattern: String): List[String] = {
      val dir = if (dirname.length == 0) pwd else dirname
      val names = new File(dir).list.toList

      if (names == null)
        Nil

      else {
        val names2 =
          if (path(0) != '.')
            names.filter(_(0) != '.')
          else
            names

        names.filter(n => fnmatch(n, pattern))
      }
    }

    def glob0(dirname: String, basename: String): List[String] = {
      if (basename.length == 0) {
        if (new File(dirname).isDirectory)
          List[String](basename)
        else
          Nil
      }
        else {
          val path = dirname + fileSeparator + basename
          if (new File(path).exists())
            List[String](basename)
          else
            Nil
        }
    }

    val wildcards = """[\*\?\[]""".r
    if ((wildcards findFirstIn path) == None)
      List[String](path)

    else {
      val (dirname, basename) = dirnameBasename(path)
      if (dirname.length == 0)
        for (name <- glob1(pwd, basename)) yield name

      else {
        val dirs =
          if ((wildcards findFirstIn dirname) != None)
            glob(dirname)
          else
            List[String](dirname)
        val globber =
          if ((wildcards findFirstIn basename) != None)
            glob1 _
          else
            glob0 _
        for (d <- dirs; name <- globber(d, basename))
          yield d + fileSeparator + name
      }
    }
  }

  /**
   * An extended glob function that supports all the wildcards of
   * the glob() function, in addition to:
   *
   * 
    *
  • a leading "~", signifying the user's home directory *
  • a special "**" wildcard that recursively matches any directory. * (Think "ant".) *
* * "~user" is not supported, however. * * @param pattern the wildcard pattern * * @return list of matches, or an empty list for none */ def eglob(pattern: String): List[String] = { def doGlob(pieces: List[String], directory: String): List[String] = { import scala.collection.mutable.ArrayBuffer val result = new ArrayBuffer[String]() val piece = pieces(0) val last = (pieces.length == 1) if (piece == "**") { val remainingPieces = if (last) Nil else pieces.drop(1) for ((root, dirs, files) <- walk(directory, true)) { if (last) // At the end of a pattern, "**" just recursively // matches directories. result += root else // Recurse downward, trying to match the rest of // the pattern. result ++= doGlob(remainingPieces, root) } } else { // Regular glob pattern. val path = directory + fileSeparator + piece val matches = glob(path) if (matches.length > 0) { if (last) // Save the matches, and stop. result ++= matches else { // Must continue recursing. val remainingPieces = pieces.drop(1) for (m <- matches; if (new File(m).isDirectory)) { val subResult = doGlob(remainingPieces, m) for (partialPath <- subResult) result += partialPath } } } } result.toList } // Main eglob() logic // Account for leading "~" val adjustedPattern = if (pattern.length == 0) "." else if (pattern.startsWith("~")) normalizePath(joinPath(System.getProperty("user.home"), pattern drop 1)) else pattern // Determine leading directory, which is different per OS (because // of Windows' stupid drive letters). val (relativePattern, directory) = eglobPatternSplitter(adjustedPattern) // Do the actual globbing. val pieces = splitPath(relativePattern) val matches = doGlob(pieces.toList, directory) matches map (normalizePath _) } /* ---------------------------------------------------------------------- *\ Private Methods \* ---------------------------------------------------------------------- */ /** * Convert a file into a path array. Borrowed from SBT source code. */ private def toPathArray(file: File): Array[String] = { @tailrec def toPathList(f: File, current: List[String]): List[String] = { if (f == null) current else toPathList(f.getParentFile, f.getName :: current) } toPathList(file.getCanonicalFile, Nil).toArray } /** * Get the length of the common prefix between two arrays. */ private def commonPrefix[T](a: Array[T], b: Array[T]): Int = { @tailrec def common(count: Int): Int = { if ((count >= a.length) || (count >= b.length) || (a(count) != b(count))) count else common(count + 1) } common(0) } /** * For the eglob algorithm to work, the pattern needs to be split into a * (directory, subpattern) pair, where the subpattern is relative. This * splitting operating is operating system-dependent, largely because * of Windows' stupid drive letters. This variable holds a partially * applied function for the splitter, determined the first time it is * referenced. That way, eglob() doesn't do this same match on every * call. */ private lazy val eglobPatternSplitter = os match { case (Mac | Posix) => splitPosixEglobPattern(_) case Windows => splitWindowsEglobPattern(_) case _ => throw new UnsupportedOperationException("Unknown OS: " + os) } /** * Windows pattern splitter for eglob(). See description for the * eglobPatternSplitter value, above. * * @param pattern the pattern to split * * @return a (directory, subpattern) tuple */ private def splitWindowsEglobPattern(pattern: String): (String, String) = { splitDrivePath(pattern) match { case ("", "") => (".", ".") case ("", path) => (path, ".") case (drive, "") => (".", drive) case (drive, path) => { // Hack: Can't handle non-absolute paths in a drive. // Pretend a drive letter means "absolute". Note that // "drive" can be empty here, which is fine. if (path(0) == '\\') (path drop 1, drive + "\\") else (path, drive + "\\") } } } /** * Posix pattern splitter for eglob(). See description for the * eglobPatternSplitter value, above. * * @param pattern the pattern to split * * @return a (directory, subpattern) tuple */ private def splitPosixEglobPattern(pattern: String): (String, String) = { if (pattern.length == 0) (".", ".") else if (pattern(0) == fileSeparatorChar) (pattern drop 1, "/") else (pattern, ".") } /** * Similar to Python's fnmatch() function, this function * determines whether a string matches a wildcard pattern. Patterns * are Unix shell-style wildcards: * * * * * * * * * * * * * * * * * * *
*matches everything
?matches any single character
[set]matches any character in set
[!set]matches any character not in set
* * An initial period in filename is not special. Matches are * case-sensitive on Posix operating systems, case-insensitive elsewhere. * * @param name the name to match * @param pattern the wildcard pattern */ def fnmatch(name: String, pattern: String): Boolean = { // Convert to regular expression pattern. val caseConv: String => String = if (os == Posix) {s => s} else {s => s.toLowerCase} val regex = caseConv("^" + pattern.replace("\\", "\\\\") .replace(".", "\\.") .replace("*", ".*") .replace("[!", "[^") .replace("?", ".") + "$").r (regex findFirstIn caseConv(name)) != None } /** * Directory tree generator, adapted from Python's os.walk() * function. * *

For each directory in the directory tree rooted at top (including top * itself, but excluding '.' and '..'), yields a 3-tuple

* *
dirpath, dirnames, filenames
* *

dirpath is a string, the path to the directory. * dirnames is a list of the names of the subdirectories in * dirpath (excluding '.' and '..'). filenames is a list * of the names of the non-directory files in dirpath. Note that * the names in the lists are just names, with no path components. To * get a full path (which begins with top) to a file or directory in * dirpath, use dirpath + java.io.fileSeparator + name, * or use joinPath().

* *

If topdown is true, the triple for a directory is * generated before the triples for any of its subdirectories * (directories are generated top down). If topdown is * false, the triple for a directory is generated after the * triples for all of its subdirectories (directories are generated * bottom up).

* *

WARNING! This method does not grok symbolic links!

* * @param top name of starting directory * @param topdown true to do a top-down traversal, false * otherwise. * * @return List of triplets, as described above. */ def walk(top: String, topdown: Boolean = true): List[(String, List[String], List[String])] = { // This needs to be made more efficient, with some kind of generator. import scala.collection.mutable.ArrayBuffer val dirs = new ArrayBuffer[String]() val nondirs = new ArrayBuffer[String]() val result = new ArrayBuffer[(String, List[String], List[String])]() val fTop = new File(top) val names = fTop.list if (names != null) { for (name <- names) { val f = new File(top + fileSeparator + name) if (f.isDirectory) dirs += name else nondirs += name } if (topdown) result += ((top, dirs.toList, nondirs.toList)) for (name <- dirs) result ++= walk(top + fileSeparator + name, topdown) if (! topdown) result += ((top, dirs.toList, nondirs.toList)) } result.toList } /** * Split a path into its constituent components. If the path is * absolute, the first piece will have a file separator in the * beginning. Examples: * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
InputOutput
""List("")
"/"List("/") *
"foo"List("foo")
"foo/bar"List("foo", "bar")
"."List(".")
"../foo"List("..", "foo")
"./foo"List(".", "foo")
"/foo/bar/baz"List("/foo", "bar", "baz")
"foo/bar/baz"List("foo", "bar", "baz")
"/foo"List("/foo")
* * @param path the path * @param fileSep the file separator to use. Defaults to the value of * the "file.separator" property. * * @return the component pieces. */ def splitPath(path: String, fileSep: String = fileSeparator): List[String] = { // Split with the path separator character, rather than the path // separator string. Using the string causes Scala to interpret it // as a regular expression, which causes problems when the separator // is a backslash (as on Windows). We could escape the backslash, // but it's just as easy to split on the character, not the string, // Null guard. val nonNullPath = if (path == null) "" else path // Special case for Windows. (Stupid drive letters.) val (prefix, usePath) = if (fileSep == "\\") splitDrivePath(nonNullPath) else ("", nonNullPath) // If there are leading file separator characters, split() will // produce extra empty array elements. Prevent that. val subpath = usePath.foldLeft("") { (c1, c2) => // Note: c1 and c2 are strings, not characters if (c1 == fileSep) c2.toString else c1 + c2 } val absolute = path.startsWith(fileSep) || (prefix != "") // Split on the character, not the full string. Splitting on a string // uses regular expression semantics, which will fail if this is // Windows (and the file separator is "\"). Windows is a pain in the // ass. val pieces = (subpath split fileSep(0)).toList if (absolute) { if (pieces.length == 0) List[String](prefix + fileSep) else (prefix + fileSep + pieces.head) :: pieces.tail } else pieces } /** * Join components of a path together. * * @param fileSep the file separator to use * @param pieces path pieces * * @return a composite path */ def joinPath(fileSep: String, pieces: List[String]): String = pieces mkString fileSep /** * Join components of a path together, using the file separator of the * currently running system * * @param pieces path pieces * * @return a composite path */ def joinPath(pieces: String*): String = joinPath(fileSeparator, pieces.toList) /** * Join components of a path together, using the file separator of the * currently running system * * @param pieces path pieces * * @return a composite path */ def joinPath(pieces: File*): File = new File(joinPath(fileSeparator, pieces.toList.map(_.getName))) /** * Determine the temporary directory to use. * * @return the temporary directory */ def temporaryDirectory: File = { import grizzled.sys.OperatingSystem._ def guess: String = { grizzled.sys.os match { case Posix => "/tmp" case Mac => "/tmp" case (Windows | WindowsCE | OS2 | NetWare) => """C:\TEMP""" case _ => throw new UnsupportedOperationException } } val sysProp = System getProperty "java.io.tmpdir" val tempDirName = if (sysProp == null) guess else sysProp new File(tempDirName) } private lazy val random = new Random() /** * Create a temporary directory. * * @param prefix Prefix for directory name * @param maxTries Maximum number of times to try creating the * directory before giving up. * * @return the directory. Throws an IOException if it can't create * the directory. */ def createTemporaryDirectory(prefix: String, maxTries: Int = 3): File = { def createDirectory(dir: File): Option[File] = { if (! dir.exists) { if (! dir.mkdirs()) throw new IOException( "Failed to create directory \"" + dir.getAbsolutePath + "\"" ) Some(dir) } else { None } } @tailrec def create(tries: Int): File = { import java.lang.{Integer => JInt} if (tries > maxTries) throw new IOException( "Failed to create temporary directory " + "after " + maxTries + " attempts." ) val usePrefix = if (prefix == null) "" else prefix val randomName = usePrefix + JInt.toHexString(random.nextInt) val dir = new File (temporaryDirectory, randomName) createDirectory(dir) match { case Some(dir) if (dir.isEmpty) => dir case _ => create(tries + 1) } } create(0) } /** * Allow execution of a block of code within the context of a temporary * directory. The temporary directory is cleaned up after the operation * completes. * * @param prefix file name prefix to use * @param action action to perform * * @return whatever the action returns */ def withTemporaryDirectory[T](prefix: String)(action: File => T) = { val temp = createTemporaryDirectory(prefix) temp.deleteOnExit try { action(temp) } finally { temp.deleteRecursively } } /** * Copy multiple files to a target directory. Also see the version of this * method that takes only one file. * * @param files An Iterable of file names to be copied * @param targetDir Path name to target directory * @param createTarget true to create the target directory, * false to throw an exception if the * directory doesn't already exist. * * @throws FileDoesNotExistException a source file or the target directory * doesn't exist * @throws IOException cannot create target directory */ def copy(files: Iterable[String], targetDir: String, createTarget: Boolean = true): Unit = { val target = new File(targetDir) if ((! target.exists()) && (createTarget)) if (! target.mkdirs()) throw new IOException("Unable to create target directory \"" + targetDir + "\"") if (target.exists() && (! target.isDirectory())) throw new IOException("Cannot copy files to non-directory \"" + targetDir + "\"") if (! target.exists()) throw new FileDoesNotExistException("Target directory \"" + targetDir + "\" does not exist.") for (file <- files) copyFile(file, targetDir + fileSeparator + basename(file)) } /** * Copy a file to a directory. * * @param file Path name of the file to copy * @param targetDir Path name to target directory * @param createTarget true to create the target directory, * false to throw an exception if the * directory doesn't already exist. * * @throws FileDoesNotExistException source file or target directory * doesn't exist * @throws IOException cannot create target directory */ def copy(file: String, targetDir: String, createTarget: Boolean): Unit = copy(List[String](file), targetDir, createTarget) /** * Copy a file to a directory. If the target directory does not exist, * it is created. * * @param file Path name of the file to copy * @param targetDir Path name to target directory * * @throws FileDoesNotExistException a source file or the target directory * doesn't exist * @throws IOException cannot create target directory */ def copy(file: String, targetDir: String): Unit = copy(file, targetDir, true) /** * Copy a source file to a target file, using binary copying. The source * file must be a file. The target path can be a file or a directory; if * it is a directory, the target file will have the same base name as * as the source file. * * @param sourcePath path to the source file * @param targetPath path to the target file or directory * * @return the full path of the target file */ def copyFile(sourcePath: String, targetPath: String): String = copyFile(new File(sourcePath), new File(targetPath)).getPath /** * Copy a source file to a target file, using binary copying. The source * file must be a file. The target path can be a file or a directory; if * it is a directory, the target file will have the same base name as * as the source file. * * @param source path to the source file * @param target path to the target file or directory * * @return the full path of the target file */ def copyFile(source: File, target: File): File = { import java.io.{InputStream, OutputStream, BufferedInputStream, BufferedOutputStream, FileInputStream, FileOutputStream} val targetFile = if (target.isDirectory()) new File(joinPath(target.getPath, basename(source.getName))) else target val in = new BufferedInputStream(new FileInputStream(source)) val out = new BufferedOutputStream(new FileOutputStream(targetFile)) try { in.copyTo(out) targetFile } finally { in.close() out.close() } } /** * Recursively copy a source directory and its contents to a target * directory. Creates the target directory if it does not exist. * * @param sourceDir the source directory * @param targetDir the target directory */ def copyTree(sourceDir: String, targetDir: String): Unit = copyTree(new File(sourceDir), new File(targetDir)) /** * Recursively copy a source directory and its contents to a target * directory. Creates the target directory if it does not exist. * * @param sourceDir the source directory * @param targetDir the target directory */ def copyTree(sourceDir: File, targetDir: File): Unit = { if (! sourceDir.exists()) throw new FileDoesNotExistException(sourceDir.getPath) if (! sourceDir.isDirectory) throw new IOException("Source directory \"" + sourceDir.getPath + "\" is not a directory.") val files = sourceDir.list.map(f => (new File(sourceDir, f), new File(targetDir, f))) targetDir.mkdirs for ((src, target) <- files) { if (src.isDirectory) copyTree(src, target) else copyFile(src, target) } } /** * Recursively remove a directory tree. This function is conceptually * equivalent to rm -r on a Unix system. * * @param dir The directory */ def deleteTree(dir: String): Unit = deleteTree(new File(dir)) /** * Recursively remove a directory tree. This function is conceptually * equivalent to rm -r on a Unix system. * * @param dir The directory */ def deleteTree(dir: File): Unit = { if (! dir.isDirectory) throw new IOException("\"" + dir + "\" is not a directory.") for (f <- dir.listFiles) { if (f.isDirectory) deleteTree(f) else f.delete() } dir.delete() } /** * Similar to the Unix touch command, this function: * *
    *
  • updates the access and modification times for any existing files * in a list of files *
  • creates any non-existent files in the list of files *
* * If any file in the list is a directory, this method will throw an * exception. * * @param files Iterable of files to touch * @param time Set the last-modified time to this time, or to the current * time if this parameter is negative. * * @throws IOException on error */ def touchMany(files: Iterable[String], time: Long = -1): Unit = { val useTime = if (time < 0) System.currentTimeMillis else time for (name <- files) { val file = new File(name) if (file.isDirectory) throw new IOException("File \"" + name + "\" is a directory") if (! file.exists()) file.createNewFile() file.setLastModified(useTime) } } /** * Similar to the Unix touch command, this function: * *
    *
  • updates the access and modification times for a file *
  • creates the file if it does not exist *
* * If the file is a directory, this method will throw an exception. * * @param path The file to touch * @param time Set the last-modified time to this time, or to the current * time if this parameter is negative. * * @throws IOException on error */ def touch(path: String, time: Long = -1): Unit = touchMany(List[String](path), time) private lazy val DrivePathPattern = "^([A-Za-z]?:)?(.*)$".r /** * Split a Windows-style path into drive name and path portions. * * @param path the path * * @return a (drive, path) tuple, either component of which can be * * an empty string */ def splitDrivePath(path: String): (String, String) = { path match { case DrivePathPattern(driveSpec, path) => driveSpec match { case null => ("", path) case ":" => ("", path) case _ => (driveSpec, path) } case _ => ("", path) } } /** * Path normalization is operating system-specific. This value * holds the real path normalizer, determined once. */ private lazy val doPathNormalizing = os match { case (Mac | Posix) => normalizePosixPath(_) case Windows => normalizeWindowsPath(_) case _ => throw new UnsupportedOperationException("Unknown OS: " + os) } /** * Normalize a path, eliminating double slashes, resolving embedded * ".." strings (e.g., "/foo/../bar" becomes "/bar"), etc. Works for * Windows and Posix operating systems. * * @param path the path * * @return the normalized path */ def normalizePath(path: String): String = doPathNormalizing(path) /** * Shared between normalizeWindowsPath() and normalizePosixPath(), * this function normalizes the pieces of a path, handling embedded "..", * empty elements (from splitting when there are adjacent file separators), * etc. * * @param pieces path components, with no separators * * @return sanitized list of path components */ private def normalizePathPieces(pieces: List[String]): List[String] = { pieces match { case Nil => Nil case "" :: tail => normalizePathPieces(tail) case "." :: tail => normalizePathPieces(tail) case a :: ".." :: tail => normalizePathPieces(tail) case _ => List[String](pieces.head) ++ normalizePathPieces(pieces.tail) } } /** * Normalize a Windows path name. Handles UNC paths. Adapted from the * Python version of normpath() in Python's os.ntpath module. * * @param path the path * * @return the normalized path */ def normalizeWindowsPath(path: String): String = { // We need to be careful here. If the prefix is empty, and the path // starts with a backslash, it could either be an absolute path on // the current drive (\dir1\dir2\file) or a UNC filename // (\\server\mount\dir1\file). It is therefore imperative NOT to // collapse multiple backslashes blindly in that case. The code // below preserves multiple backslashes when there is no drive // letter. This means that the invalid filename \\\a\b is preserved // unchanged, where a\\\b is normalized to a\b. val (prefix, newPath) = splitDrivePath(path) match { case ("", path) => { // No drive letter - preserve initial backslashes (path takeWhile (_ == '\\') mkString "", path dropWhile (_ == '\\') mkString "") } case (prefix, path) => // We have a drive letter. (prefix + "\\", path dropWhile (_ == '\\') mkString "") } // Normalize the path pieces. Note: normalizePathPieces() doesn't // handle leading ".." in an absolute path, such as "\\..\\..". We // handle that later. val piecesTemp = normalizePathPieces(newPath.split("\\\\").toList) // Remove any leading ".." that shouldn't be there. val newPieces = if (prefix == "\\") piecesTemp dropWhile (_ == "..") else piecesTemp // If the path is now empty, substitute ".". if ((prefix.length == 0) && (newPieces.length == 0)) "." else prefix + (newPieces mkString "\\") } /** * Adapted from the Python version of normpath() in Python's * os.posixpath module. * * @param path the path * * @return the normalized path */ def normalizePosixPath(path: String): String = { import scala.collection.mutable.ListBuffer path match { case "" => "." case "." => "." case _ => { // POSIX allows one or two initial slashes, but treats // three or more as a single slash. We don't do that here. // Two initial slashes is also collapsed into one. val initialSlashes = if (path.startsWith("/")) 1 else 0 // Normalize the path pieces. Note: normalizePathPieces() // doesn't handle leading ".." in an absolute path, such as // "/../..". We handle that later. // // Note: Must also account for a single leading ".", which // must be preserved val pieces = path.split("/").toList match { case Nil => Nil case "." :: remainder => remainder case other => other } val normalizedPieces1 = normalizePathPieces(pieces) // Remove any leading ".." that shouldn't be there. val normalizedPieces2 = if (path startsWith "/") normalizedPieces1 dropWhile (_ == "..") else normalizedPieces1 val result = ("/" * initialSlashes) + (normalizedPieces2 mkString "/") // An empty string is "." if (result == "") "." else result } } } /** * Native-to-universal path conversion is operating system-specific. * These values hold the real converters, determined once. */ private lazy val makeUniversalPath: (String) => String = os match { case (Mac | Posix) => (path: String) => path case Windows => (path: String) => path.replace(fileSeparator, "/") case _ => throw new UnsupportedOperationException("Unknown OS: " + os) } private lazy val makeNativePath: (String) => String = os match { case (Mac | Posix) => (path: String) => path case Windows => (path: String) => path.replace("/", fileSeparator) case _ => throw new UnsupportedOperationException("Unknown OS: " + os) } /** * Converts a path name from its operating system-specific format to a * universal path notation. Universal path notation always uses a * Unix-style "/" to separate path elements. A universal path can be * converted to a native (operating system-specific) path via the * native_path() function. Note that on POSIX-compliant systems, * this function simply returns the path parameter unmodified. * * @param path the path to convert to universal path notation * * @return the universal path */ def universalPath(path: String): String = makeUniversalPath(path) /** * Converts a path name from universal path notation to the operating * system-specific format. Universal path notation always uses a * Unix-style "/" to separate path elements. A native path can be * converted to a universal path via the universal_path() * function. Note that on POSIX-compliant systems, this function simply * returns the path parameter unmodified. * * @param path the path to convert from universtal to native path notation * * @return the native path */ def nativePath(path: String): String = makeNativePath(path) }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy