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

scala.tools.nsc.util.ClassPath.scala Maven / Gradle / Ivy

/* NSC -- new Scala compiler
 * Copyright 2006-2011 LAMP/EPFL
 * @author  Martin Odersky
 */


package scala.tools.nsc
package util

import java.net.URL
import scala.collection.{ mutable, immutable }
import io.{ File, Directory, Path, Jar, AbstractFile, ClassAndJarInfo }
import scala.tools.util.StringOps.splitWhere
import Jar.isJarOrZip
import File.pathSeparator

/** 

* This module provides star expansion of '-classpath' option arguments, behaves the same as * java, see [http://java.sun.com/javase/6/docs/technotes/tools/windows/classpath.html] *

* * @author Stepan Koltsov */ object ClassPath { def scalaLibrary = locate[ScalaObject] def scalaCompiler = locate[Global] def info[T: ClassManifest] = new ClassAndJarInfo[T] def locate[T: ClassManifest] = info[T] rootClasspath def locateJar[T: ClassManifest] = info[T].rootPossibles find (x => isJarOrZip(x)) map (x => File(x)) def locateDir[T: ClassManifest] = info[T].rootPossibles find (_.isDirectory) map (_.toDirectory) /** Expand single path entry */ private def expandS(pattern: String): List[String] = { val wildSuffix = File.separator + "*" /** Get all subdirectories, jars, zips out of a directory. */ def lsDir(dir: Directory, filt: String => Boolean = _ => true) = dir.list filter (x => filt(x.name) && (x.isDirectory || isJarOrZip(x))) map (_.path) toList def basedir(s: String) = if (s contains File.separator) s.substring(0, s.lastIndexOf(File.separator)) else "." if (pattern == "*") lsDir(Directory(".")) else if (pattern endsWith wildSuffix) lsDir(Directory(pattern dropRight 2)) else if (pattern contains '*') { val regexp = ("^%s$" format pattern.replaceAll("""\*""", """.*""")).r lsDir(Directory(pattern).parent, regexp findFirstIn _ isDefined) } else List(pattern) } /** Return duplicated classpath entries as * (name, list of origins) * in the order they occur on the path. */ // def findDuplicates(cp: ClassPath[_]) = { // def toFullName(x: (String, _, cp.AnyClassRep)) = x._1 + "." + x._3.name // def toOriginString(x: ClassPath[_]) = x.origin getOrElse x.name // // /** Flatten everything into tuples, recombine grouped by name, filter down to 2+ entries. */ // val flattened = ( // for ((pkgName, pkg) <- cp.allPackagesWithNames ; clazz <- pkg.classes) yield // (pkgName, pkg, clazz) // ) // val multipleAppearingEntries = flattened groupBy toFullName filter (_._2.size > 1) // // /** Extract results. */ // for (name <- flattened map toFullName distinct ; dups <- multipleAppearingEntries get name) yield // (name, dups map { case (_, cp, _) => toOriginString(cp) }) // } /** Split classpath using platform-dependent path separator */ def split(path: String): List[String] = (path split pathSeparator).toList filterNot (_ == "") distinct /** Join classpath using platform-dependent path separator */ def join(paths: String*): String = paths filterNot (_ == "") mkString pathSeparator /** Split the classpath, apply a transformation function, and reassemble it. */ def map(cp: String, f: String => String): String = join(split(cp) map f: _*) /** Split the classpath, filter according to predicate, and reassemble. */ def filter(cp: String, p: String => Boolean): String = join(split(cp) filter p: _*) /** Split the classpath and map them into Paths */ def toPaths(cp: String): List[Path] = split(cp) map (x => Path(x).toAbsolute) /** Make all classpath components absolute. */ def makeAbsolute(cp: String): String = fromPaths(toPaths(cp): _*) /** Join the paths as a classpath */ def fromPaths(paths: Path*): String = join(paths map (_.path): _*) def fromURLs(urls: URL*): String = fromPaths(urls map (x => Path(x.getPath)) : _*) /** Split the classpath and map them into URLs */ def toURLs(cp: String): List[URL] = toPaths(cp) map (_.toURL) /** Expand path and possibly expanding stars */ def expandPath(path: String, expandStar: Boolean = true): List[String] = if (expandStar) split(path) flatMap expandS else split(path) /** Expand dir out to contents, a la extdir */ def expandDir(extdir: String): List[String] = { AbstractFile getDirectory extdir match { case null => Nil case dir => dir filter (_.isClassContainer) map (x => new java.io.File(dir.file, x.name) getPath) toList } } /** A useful name filter. */ def isTraitImplementation(name: String) = name endsWith "$class.class" import java.net.MalformedURLException def specToURL(spec: String): Option[URL] = try Some(new URL(spec)) catch { case _: MalformedURLException => None } /** A class modeling aspects of a ClassPath which should be * propagated to any classpaths it creates. */ abstract class ClassPathContext[T] { /** A filter which can be used to exclude entities from the classpath * based on their name. */ def isValidName(name: String): Boolean = true /** From the representation to its identifier. */ def toBinaryName(rep: T): String /** Create a new classpath based on the abstract file. */ def newClassPath(file: AbstractFile): ClassPath[T] /** Creators for sub classpaths which preserve this context. */ def sourcesInPath(path: String): List[ClassPath[T]] = for (file <- expandPath(path, false) ; dir <- Option(AbstractFile getDirectory file)) yield new SourcePath[T](dir, this) def contentsOfDirsInPath(path: String): List[ClassPath[T]] = for (dir <- expandPath(path, false) ; name <- expandDir(dir) ; entry <- Option(AbstractFile getDirectory name)) yield newClassPath(entry) def classesAtAllURLS(path: String): List[ClassPath[T]] = (path split " ").toList flatMap classesAtURL def classesAtURL(spec: String) = for (url <- specToURL(spec).toList ; location <- Option(AbstractFile getURL url)) yield newClassPath(location) def classesInExpandedPath(path: String): IndexedSeq[ClassPath[T]] = classesInPathImpl(path, true).toIndexedSeq def classesInPath(path: String) = classesInPathImpl(path, false) // Internal private def classesInPathImpl(path: String, expand: Boolean) = for (file <- expandPath(path, expand) ; dir <- Option(AbstractFile getDirectory file)) yield newClassPath(dir) } class JavaContext extends ClassPathContext[AbstractFile] { def toBinaryName(rep: AbstractFile) = { val name = rep.name assert(endsClass(name), name) name.substring(0, name.length - 6) } def newClassPath(dir: AbstractFile) = new DirectoryClassPath(dir, this) } object DefaultJavaContext extends JavaContext { override def isValidName(name: String) = !isTraitImplementation(name) } @inline private def endsClass(s: String) = s.length > 6 && s.substring(s.length - 6) == ".class" @inline private def endsScala(s: String) = s.length > 6 && s.substring(s.length - 6) == ".scala" @inline private def endsJava(s: String) = s.length > 5 && s.substring(s.length - 5) == ".java" /** From the source file to its identifier. */ def toSourceName(f: AbstractFile): String = { val name = f.name if (endsScala(name)) name.substring(0, name.length - 6) else if (endsJava(name)) name.substring(0, name.length - 5) else throw new FatalError("Unexpected source file ending: " + name) } } import ClassPath._ /** * Represents a package which contains classes and other packages */ abstract class ClassPath[T] { type AnyClassRep = ClassPath[T]#ClassRep /** * The short name of the package (without prefix) */ def name: String /** * A String representing the origin of this classpath element, if known. * For example, the path of the directory or jar. */ def origin: Option[String] = None /** A list of URLs representing this classpath. */ def asURLs: List[URL] /** The whole classpath in the form of one String. */ def asClasspathString: String /** Info which should be propagated to any sub-classpaths. */ def context: ClassPathContext[T] /** Lists of entities. */ def classes: IndexedSeq[AnyClassRep] def packages: IndexedSeq[ClassPath[T]] def sourcepaths: IndexedSeq[AbstractFile] /** Information which entails walking the tree. This is probably only * necessary for tracking down problems - it's normally not used. */ // def allPackages: List[ClassPath[T]] = packages ::: (packages flatMap (_.allPackages)) // def allPackageNames: List[String] = { // def subpackages(prefix: String, cp: ClassPath[T]): List[String] = ( // (cp.packages map (prefix + _.name)) ::: // (cp.packages flatMap (x => subpackages(prefix + x.name + ".", x))) // ) // subpackages("", this) // } // def allPackagesWithNames: List[(String, ClassPath[T])] = { // val root = packages map (p => p.name -> p) // val subs = // for ((prefix, p) <- root ; (k, v) <- p.allPackagesWithNames) yield // (prefix + "." + k, v) // // root ::: subs // } /** * Represents classes which can be loaded with a ClassfileLoader/MSILTypeLoader * and / or a SourcefileLoader. */ case class ClassRep(binary: Option[T], source: Option[AbstractFile]) { def name: String = binary match { case Some(x) => context.toBinaryName(x) case _ => assert(source.isDefined) toSourceName(source.get) } } /** Filters for assessing validity of various entities. */ def validClassFile(name: String) = endsClass(name) && context.isValidName(name) def validPackage(name: String) = (name != "META-INF") && (name != "") && (name.charAt(0) != '.') def validSourceFile(name: String) = endsScala(name) || endsJava(name) /** * Find a ClassRep given a class name of the form "package.subpackage.ClassName". * Does not support nested classes on .NET */ def findClass(name: String): Option[AnyClassRep] = splitWhere(name, _ == '.', true) match { case Some((pkg, rest)) => val rep = packages find (_.name == pkg) flatMap (_ findClass rest) rep map { case x: ClassRep => x case x => throw new FatalError("Unexpected ClassRep '%s' found searching for name '%s'".format(x, name)) } case _ => classes find (_.name == name) } def findSourceFile(name: String): Option[AbstractFile] = findClass(name) match { case Some(ClassRep(Some(x: AbstractFile), _)) => Some(x) case _ => None } def sortString = join(split(asClasspathString).sorted: _*) override def equals(that: Any) = that match { case x: ClassPath[_] => this.sortString == x.sortString case _ => false } override def hashCode = sortString.hashCode() } /** * A Classpath containing source files */ class SourcePath[T](dir: AbstractFile, val context: ClassPathContext[T]) extends ClassPath[T] { def name = dir.name override def origin = dir.underlyingSource map (_.path) def asURLs = if (dir.file == null) Nil else List(dir.toURL) def asClasspathString = dir.path val sourcepaths: IndexedSeq[AbstractFile] = IndexedSeq(dir) private def traverse() = { val classBuf = immutable.Vector.newBuilder[ClassRep] val packageBuf = immutable.Vector.newBuilder[SourcePath[T]] dir foreach { f => if (!f.isDirectory && validSourceFile(f.name)) classBuf += ClassRep(None, Some(f)) else if (f.isDirectory && validPackage(f.name)) packageBuf += new SourcePath[T](f, context) } (packageBuf.result, classBuf.result) } lazy val (packages, classes) = traverse() override def toString() = "sourcepath: "+ dir.toString() } /** * A directory (or a .jar file) containing classfiles and packages */ class DirectoryClassPath(val dir: AbstractFile, val context: ClassPathContext[AbstractFile]) extends ClassPath[AbstractFile] { def name = dir.name override def origin = dir.underlyingSource map (_.path) def asURLs = if (dir.file == null) Nil else List(dir.toURL) def asClasspathString = dir.path val sourcepaths: IndexedSeq[AbstractFile] = IndexedSeq() // calculates (packages, classes) in one traversal. private def traverse() = { val classBuf = immutable.Vector.newBuilder[ClassRep] val packageBuf = immutable.Vector.newBuilder[DirectoryClassPath] dir foreach { f => if (!f.isDirectory && validClassFile(f.name)) classBuf += ClassRep(Some(f), None) else if (f.isDirectory && validPackage(f.name)) packageBuf += new DirectoryClassPath(f, context) } (packageBuf.result, classBuf.result) } lazy val (packages, classes) = traverse() override def toString() = "directory classpath: "+ origin.getOrElse("?") } /** * A classpath unifying multiple class- and sourcepath entries. */ class MergedClassPath[T]( val entries: IndexedSeq[ClassPath[T]], val context: ClassPathContext[T]) extends ClassPath[T] { def this(entries: TraversableOnce[ClassPath[T]], context: ClassPathContext[T]) = this(entries.toIndexedSeq, context) def name = entries.head.name def asURLs = entries flatMap (_.asURLs) toList lazy val sourcepaths: IndexedSeq[AbstractFile] = entries flatMap (_.sourcepaths) override def origin = Some(entries map (x => x.origin getOrElse x.name) mkString ("Merged(", ", ", ")")) override def asClasspathString: String = join(entries map (_.asClasspathString) : _*) lazy val classes: IndexedSeq[AnyClassRep] = { var count = 0 val indices = mutable.HashMap[String, Int]() val cls = new mutable.ArrayBuffer[AnyClassRep](1024) for (e <- entries; c <- e.classes) { val name = c.name if (indices contains name) { val idx = indices(name) val existing = cls(idx) if (existing.binary.isEmpty && c.binary.isDefined) cls(idx) = existing.copy(binary = c.binary) if (existing.source.isEmpty && c.source.isDefined) cls(idx) = existing.copy(source = c.source) } else { indices(name) = count cls += c count += 1 } } cls.toIndexedSeq } lazy val packages: IndexedSeq[ClassPath[T]] = { var count = 0 val indices = mutable.HashMap[String, Int]() val pkg = new mutable.ArrayBuffer[ClassPath[T]](256) for (e <- entries; p <- e.packages) { val name = p.name if (indices contains name) { val idx = indices(name) pkg(idx) = addPackage(pkg(idx), p) } else { indices(name) = count pkg += p count += 1 } } pkg.toIndexedSeq } private def addPackage(to: ClassPath[T], pkg: ClassPath[T]) = { val newEntries: IndexedSeq[ClassPath[T]] = to match { case cp: MergedClassPath[_] => cp.entries :+ pkg case _ => IndexedSeq(to, pkg) } new MergedClassPath[T](newEntries, context) } // // override def allPackages: List[ClassPath[T]] = entries flatMap (_.allPackages) // override def allPackageNames = entries flatMap (_.allPackageNames) // override def allPackagesWithNames = entries flatMap (_.allPackagesWithNames) // // def duplicatedClasses = { // def toFullName(x: (String, _, AnyClassRep)) = x._1 + "." + x._3.name // // /** Flatten everything into tuples, recombine grouped by name, filter down to 2+ entries. */ // val flattened = ( // for ((pkgName, pkg) <- allPackagesWithNames ; clazz <- pkg.classes) yield // (pkgName, pkg, clazz) // ) // val multipleAppearingEntries = flattened groupBy toFullName filter (_._2.size > 1) // // /** Using original name list as reference point, return duplicated entries as // * (name, list of origins) // * in the order they occur on the path. // */ // for (name <- flattened map toFullName distinct ; dups <- multipleAppearingEntries get name) yield // (name, dups map { // case (_, cp, _) if cp.origin.isDefined => cp.origin.get // case (_, cp, _) => cp.asURLs.mkString // }) // } // def show() { println("ClassPath %s has %d entries and results in:\n".format(name, entries.size)) asClasspathString split ':' foreach (x => println(" " + x)) } // def showDuplicates() = // ClassPath findDuplicates this foreach { // case (name, xs) => println(xs.mkString(name + ":\n ", "\n ", "\n")) // } // override def toString() = "merged classpath "+ entries.mkString("(", "\n", ")") } /** * The classpath when compiling with target:jvm. Binary files (classfiles) are represented * as AbstractFile. nsc.io.ZipArchive is used to view zip/jar archives as directories. */ class JavaClassPath( containers: IndexedSeq[ClassPath[AbstractFile]], context: JavaContext) extends MergedClassPath[AbstractFile](containers, context) { }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy