package scala.tools.nsc
package plugins
import io.{ File, Path }
import java.net.URLClassLoader
import java.util.jar.JarFile
import java.util.zip.ZipException
import scala.collection.mutable
import mutable.ListBuffer
import scala.xml.XML
abstract class Plugin {
val name: String
val components: List[PluginComponent]
val description: String
val global: Global
def processOptions(options: List[String], error: String => Unit) {
if (!options.isEmpty)
error("Error: " + name + " has no options")
}
val optionsHelp: Option[String] = None
}
object Plugin {
private val PluginXML = "scalac-plugin.xml"
private def loaderFor(jarfiles: Seq[Path]): ClassLoader = {
val compilerLoader = classOf[Plugin].getClassLoader
val jarurls = jarfiles map (_.toURL)
new URLClassLoader(jarurls.toArray, compilerLoader)
}
private def loadDescription(jarfile: Path): Option[PluginDescription] =
if (!jarfile.exists) None
else try {
val jar = new JarFile(jarfile.jfile)
try {
jar getEntry PluginXML match {
case null => None
case entry =>
val in = jar getInputStream entry
val packXML = XML load in
in.close()
PluginDescription fromXML packXML
}
}
finally jar.close()
}
catch {
case _: ZipException => None
}
type AnyClass = Class[_]
def loadFrom(jarfile: Path, loader: ClassLoader): Option[AnyClass] =
loadDescription(jarfile) match {
case None =>
println("Warning: could not load descriptor for plugin %s".format(jarfile))
None
case Some(pdesc) =>
try Some(loader loadClass pdesc.classname) catch {
case _: Exception =>
println("Warning: class not found for plugin in %s (%s)".format(jarfile, pdesc.classname))
None
}
}
def loadAllFrom(
jars: List[Path],
dirs: List[Path],
ignoring: List[String]): List[AnyClass] =
{
val alljars = (jars ::: (for {
dir <- dirs if dir.isDirectory
entry <- dir.toDirectory.files.toList sortBy (_.name)
if Path.isJarOrZip(entry)
pdesc <- loadDescription(entry)
if !(ignoring contains pdesc.name)
} yield entry)).distinct
val loader = loaderFor(alljars)
alljars map (loadFrom(_, loader)) flatten
}
def instantiate(clazz: AnyClass, global: Global): Plugin = {
val constructor = clazz getConstructor classOf[Global]
(constructor newInstance global).asInstanceOf[Plugin]
}
}