/* sbt -- Simple Build Tool * Copyright 2008, 2009, 2010 Mark Harrah */ package sbt import java.io.File import java.net.{URI, URL} import scala.xml.NodeSeq import org.apache.ivy.plugins.resolver.IBiblioResolver import org.apache.ivy.util.url.CredentialsStore final case class ModuleID(organization: String, name: String, revision: String, configurations: Option[String], isChanging: Boolean, isTransitive: Boolean, explicitArtifacts: Seq[Artifact], extraAttributes: Map[String,String]) extends NotNull { override def toString = organization + ":" + name + ":" + revision // () required for chaining def notTransitive() = intransitive() def intransitive() = ModuleID(organization, name, revision, configurations, isChanging, false, explicitArtifacts, extraAttributes) def changing() = ModuleID(organization, name, revision, configurations, true, isTransitive, explicitArtifacts, extraAttributes) def from(url: String) = artifacts(Artifact(name, new URL(url))) def classifier(c: String) = artifacts(Artifact(name, c)) def artifacts(newArtifacts: Artifact*) = ModuleID(organization, name, revision, configurations, isChanging, isTransitive, newArtifacts ++ explicitArtifacts, extraAttributes) def extra(attributes: (String,String)*) = ModuleID(organization, name, revision, configurations, isChanging, isTransitive, explicitArtifacts, extraAttributes ++ ModuleID.checkE(attributes)) def sources() = artifacts(Artifact.sources(name)) def javadoc() = artifacts(Artifact.javadoc(name)) def withSources() = jarIfEmpty.sources() def withJavadoc() = jarIfEmpty.javadoc() private def jarIfEmpty = if(explicitArtifacts.isEmpty) jar() else this def jar() = artifacts(Artifact(name)) } object ModuleID { def apply(organization: String, name: String, revision: String): ModuleID = ModuleID(organization, name, revision, None) def apply(organization: String, name: String, revision: String, configurations: Option[String]): ModuleID = ModuleID(organization, name, revision, configurations, false, true) def apply(organization: String, name: String, revision: String, configurations: Option[String], isChanging: Boolean, isTransitive: Boolean): ModuleID = ModuleID(organization, name, revision, configurations, isChanging, isTransitive, Nil) def apply(organization: String, name: String, revision: String, configurations: Option[String], isChanging: Boolean, isTransitive: Boolean, explicitArtifacts: Seq[Artifact]): ModuleID = ModuleID(organization, name, revision, configurations, isChanging, isTransitive, explicitArtifacts, Map.empty) def checkE(attributes: Seq[(String, String)]) = for ( (key, value) <- attributes) yield if(key.startsWith("e:")) (key, value) else ("e:" + key, value) } sealed trait Resolver extends NotNull { def name: String } sealed case class MavenRepository(name: String, root: String) extends Resolver { override def toString = name + ": " + root } final class Patterns(val ivyPatterns: Seq[String], val artifactPatterns: Seq[String], val isMavenCompatible: Boolean) extends NotNull { private[sbt] def mavenStyle(): Patterns = Patterns(ivyPatterns, artifactPatterns, true) private[sbt] def withIvys(patterns: Seq[String]): Patterns = Patterns(patterns ++ ivyPatterns, artifactPatterns, isMavenCompatible) private[sbt] def withArtifacts(patterns: Seq[String]): Patterns = Patterns(ivyPatterns, patterns ++ artifactPatterns, isMavenCompatible) } object Patterns { def apply(artifactPatterns: String*): Patterns = Patterns(true, artifactPatterns : _*) def apply(isMavenCompatible: Boolean, artifactPatterns: String*): Patterns = Patterns(Nil, artifactPatterns, isMavenCompatible) def apply(ivyPatterns: Seq[String], artifactPatterns: Seq[String], isMavenCompatible: Boolean): Patterns = new Patterns(ivyPatterns, artifactPatterns, isMavenCompatible) } object RepositoryHelpers { final case class SshConnection(authentication: Option[SshAuthentication], hostname: Option[String], port: Option[Int]) extends NotNull { def copy(authentication: Option[SshAuthentication]) = SshConnection(authentication, hostname, port) } /** Configuration specific to an Ivy filesystem resolver. */ final case class FileConfiguration(isLocal: Boolean, isTransactional: Option[Boolean]) extends NotNull { def transactional() = FileConfiguration(isLocal, Some(true)) def nontransactional() = FileConfiguration(isLocal, Some(false)) def nonlocal() = FileConfiguration(false, isTransactional) } sealed trait SshAuthentication extends NotNull final case class PasswordAuthentication(user: String, password: Option[String]) extends SshAuthentication final case class KeyFileAuthentication(user: String, keyfile: File, password: Option[String]) extends SshAuthentication } import RepositoryHelpers.{SshConnection, FileConfiguration} import RepositoryHelpers.{KeyFileAuthentication, PasswordAuthentication, SshAuthentication} /** sbt interface to an Ivy repository based on patterns, which is most Ivy repositories.*/ sealed abstract class PatternsBasedRepository extends Resolver { type RepositoryType <: PatternsBasedRepository /** Should be implemented to create a new copy of this repository but with `patterns` as given.*/ protected def copy(patterns: Patterns): RepositoryType /** The object representing the configured patterns for this repository. */ def patterns: Patterns /** Enables maven 2 compatibility for this repository. */ def mavenStyle() = copy(patterns.mavenStyle()) /** Adds the given patterns for resolving/publishing Ivy files.*/ def ivys(ivyPatterns: String*): RepositoryType = copy(patterns.withIvys(ivyPatterns)) /** Adds the given patterns for resolving/publishing artifacts.*/ def artifacts(artifactPatterns: String*): RepositoryType = copy(patterns.withArtifacts(artifactPatterns)) } /** sbt interface for an Ivy filesystem repository. More convenient construction is done using Resolver.file. */ final case class FileRepository(name: String, configuration: FileConfiguration, patterns: Patterns) extends PatternsBasedRepository { type RepositoryType = FileRepository protected def copy(patterns: Patterns): FileRepository = FileRepository(name, configuration, patterns) private def copy(configuration: FileConfiguration) = FileRepository(name, configuration, patterns) def transactional() = copy(configuration.transactional()) def nonlocal() = copy(configuration.nonlocal()) } final case class URLRepository(name: String, patterns: Patterns) extends PatternsBasedRepository { type RepositoryType = URLRepository protected def copy(patterns: Patterns): URLRepository = URLRepository(name, patterns) } /** sbt interface for an Ivy ssh-based repository (ssh and sftp). Requires the Jsch library.. */ sealed abstract class SshBasedRepository extends PatternsBasedRepository { type RepositoryType <: SshBasedRepository protected def copy(connection: SshConnection): RepositoryType private def copy(authentication: SshAuthentication): RepositoryType = copy(connection.copy(Some(authentication))) /** The object representing the configured ssh connection for this repository. */ def connection: SshConnection /** Configures this to use the specified user name and password when connecting to the remote repository. */ def as(user: String, password: String): RepositoryType = as(user, Some(password)) def as(user: String): RepositoryType = as(user, None) def as(user: String, password: Option[String]) = copy(new PasswordAuthentication(user, password)) /** Configures this to use the specified keyfile and password for the keyfile when connecting to the remote repository. */ def as(user: String, keyfile: File): RepositoryType = as(user, keyfile, None) def as(user: String, keyfile: File, password: String): RepositoryType = as(user, keyfile, Some(password)) def as(user: String, keyfile: File, password: Option[String]): RepositoryType = copy(new KeyFileAuthentication(user, keyfile, password)) } /** sbt interface for an Ivy repository over ssh. More convenient construction is done using Resolver.ssh. */ final case class SshRepository(name: String, connection: SshConnection, patterns: Patterns, publishPermissions: Option[String]) extends SshBasedRepository { type RepositoryType = SshRepository protected def copy(patterns: Patterns): SshRepository = SshRepository(name, connection, patterns, publishPermissions) protected def copy(connection: SshConnection): SshRepository = SshRepository(name, connection, patterns, publishPermissions) /** Defines the permissions to set when publishing to this repository. */ def withPermissions(publishPermissions: String): SshRepository = withPermissions(Some(publishPermissions)) def withPermissions(publishPermissions: Option[String]): SshRepository = SshRepository(name, connection, patterns, publishPermissions) } /** sbt interface for an Ivy repository over sftp. More convenient construction is done using Resolver.sftp. */ final case class SftpRepository(name: String, connection: SshConnection, patterns: Patterns) extends SshBasedRepository { type RepositoryType = SftpRepository protected def copy(patterns: Patterns): SftpRepository = SftpRepository(name, connection, patterns) protected def copy(connection: SshConnection): SftpRepository = SftpRepository(name, connection, patterns) } import Resolver._ object ScalaToolsReleases extends MavenRepository(ScalaToolsReleasesName, ScalaToolsReleasesRoot) object ScalaToolsSnapshots extends MavenRepository(ScalaToolsSnapshotsName, ScalaToolsSnapshotsRoot) object DefaultMavenRepository extends MavenRepository("public", IBiblioResolver.DEFAULT_M2_ROOT) object JavaNet1Repository extends Resolver { def name = "java.net Maven1 Repository" } object Resolver { val ScalaToolsReleasesName = "Scala-Tools Maven2 Repository" val ScalaToolsSnapshotsName = "Scala-Tools Maven2 Snapshots Repository" val ScalaToolsReleasesRoot = "http://scala-tools.org/repo-releases" val ScalaToolsSnapshotsRoot = "http://scala-tools.org/repo-snapshots" /** Add the local, Maven Central, and Scala Tools releases repositories to the user repositories. */ def withDefaultResolvers(userResolvers: Seq[Resolver]): Seq[Resolver] = withDefaultResolvers(userResolvers, true) /** Add the local Ivy and Maven Central repositories to the user repositories. If `scalaTools` is true, add the Scala Tools releases repository as well. */ def withDefaultResolvers(userResolvers: Seq[Resolver], scalaTools: Boolean): Seq[Resolver] = withDefaultResolvers(userResolvers, true, scalaTools) /** Add the local Ivy repository to the user repositories. * If `scalaTools` is true, add the Scala Tools releases repository. * If `mavenCentral` is true, add the Maven Central repository. */ def withDefaultResolvers(userResolvers: Seq[Resolver], mavenCentral: Boolean, scalaTools: Boolean): Seq[Resolver] = Seq(Resolver.defaultLocal(None)) ++ userResolvers ++ single(DefaultMavenRepository, mavenCentral)++ single(ScalaToolsReleases, scalaTools) private def single[T](value: T, nonEmpty: Boolean): Seq[T] = if(nonEmpty) Seq(value) else Nil /** A base class for defining factories for interfaces to Ivy repositories that require a hostname , port, and patterns. */ sealed abstract class Define[RepositoryType <: SshBasedRepository] extends NotNull { /** Subclasses should implement this method to */ protected def construct(name: String, connection: SshConnection, patterns: Patterns): RepositoryType /** Constructs this repository type with the given `name`. `basePatterns` are the initial patterns to use. A ManagedProject * has an implicit defining these initial patterns based on a setting for either Maven or Ivy style patterns.*/ def apply(name: String)(implicit basePatterns: Patterns): RepositoryType = apply(name, None, None, None) /** Constructs this repository type with the given `name` and `hostname`. `basePatterns` are the initial patterns to use. * A ManagedProject has an implicit defining these initial patterns based on a setting for either Maven or Ivy style patterns.*/ def apply(name: String, hostname: String)(implicit basePatterns: Patterns): RepositoryType = apply(name, Some(hostname), None, None) /** Constructs this repository type with the given `name`, `hostname`, and the `basePath` against which the initial * patterns will be resolved. `basePatterns` are the initial patterns to use. * A ManagedProject has an implicit defining these initial patterns based on a setting for either Maven or Ivy style patterns.*/ def apply(name: String, hostname: String, basePath: String)(implicit basePatterns: Patterns): RepositoryType = apply(name, Some(hostname), None, Some(basePath)) /** Constructs this repository type with the given `name`, `hostname`, and `port`. `basePatterns` are the initial patterns to use. * A ManagedProject has an implicit defining these initial patterns based on a setting for either Maven or Ivy style patterns.*/ def apply(name: String, hostname: String, port: Int)(implicit basePatterns: Patterns): RepositoryType = apply(name, Some(hostname), Some(port), None) /** Constructs this repository type with the given `name`, `hostname`, `port`, and the `basePath` against which the initial * patterns will be resolved. `basePatterns` are the initial patterns to use. * A ManagedProject has an implicit defining these initial patterns based on a setting for either Maven or Ivy style patterns.*/ def apply(name: String, hostname: String, port: Int, basePath: String)(implicit basePatterns: Patterns): RepositoryType = apply(name, Some(hostname), Some(port), Some(basePath)) /** Constructs this repository type with the given `name`, `hostname`, `port`, and the `basePath` against which the initial * patterns will be resolved. `basePatterns` are the initial patterns to use. All but the `name` are optional (use None). * A ManagedProject has an implicit defining these initial patterns based on a setting for either Maven or Ivy style patterns.*/ def apply(name: String, hostname: Option[String], port: Option[Int], basePath: Option[String])(implicit basePatterns: Patterns): RepositoryType = construct(name, SshConnection(None, hostname, port), resolvePatterns(basePath, basePatterns)) } /** A factory to construct an interface to an Ivy SSH resolver.*/ object ssh extends Define[SshRepository] { protected def construct(name: String, connection: SshConnection, patterns: Patterns) = SshRepository(name, connection, patterns, None) } /** A factory to construct an interface to an Ivy SFTP resolver.*/ object sftp extends Define[SftpRepository] { protected def construct(name: String, connection: SshConnection, patterns: Patterns) = SftpRepository(name, connection, patterns) } /** A factory to construct an interface to an Ivy filesytem resolver. */ object file { /** Constructs a file resolver with the given name. The patterns to use must be explicitly specified * using the `ivys` or `artifacts` methods on the constructed resolver object.*/ def apply(name: String): FileRepository = FileRepository(name, defaultFileConfiguration, Patterns(false)) /** Constructs a file resolver with the given name and base directory. */ def apply(name: String, baseDirectory: File)(implicit basePatterns: Patterns): FileRepository = baseRepository(new File(baseDirectory.toURI.normalize) getAbsolutePath)(FileRepository(name, defaultFileConfiguration, _)) } object url { /** Constructs a URL resolver with the given name. The patterns to use must be explicitly specified * using the `ivys` or `artifacts` methods on the constructed resolver object.*/ def apply(name: String): URLRepository = URLRepository(name, Patterns(false)) /** Constructs a file resolver with the given name and base directory. */ def apply(name: String, baseURL: URL)(implicit basePatterns: Patterns): URLRepository = baseRepository(baseURL.toURI.normalize.toString)(URLRepository(name, _)) } private def baseRepository[T](base: String)(construct: Patterns => T)(implicit basePatterns: Patterns): T = construct(resolvePatterns(base, basePatterns)) /** If `base` is None, `patterns` is returned unchanged. * Otherwise, the ivy file and artifact patterns in `patterns` are resolved against the given base. */ private def resolvePatterns(base: Option[String], patterns: Patterns): Patterns = base match { case Some(path) => resolvePatterns(path, patterns) case None => patterns } /** Resolves the ivy file and artifact patterns in `patterns` against the given base. */ private def resolvePatterns(base: String, basePatterns: Patterns): Patterns = { val normBase = base.replace('\\', '/') def resolve(pattern: String) = if(normBase.endsWith("/") || pattern.startsWith("/")) normBase +pattern else normBase + "/" + pattern def resolveAll(patterns: Seq[String]) = patterns.map(resolve) Patterns(resolveAll(basePatterns.ivyPatterns), resolveAll(basePatterns.artifactPatterns), basePatterns.isMavenCompatible) } def defaultFileConfiguration = FileConfiguration(true, None) def mavenStylePatterns = Patterns(Nil, mavenStyleBasePattern :: Nil, true) def ivyStylePatterns = defaultIvyPatterns//Patterns(Nil, Nil, false) def defaultPatterns = mavenStylePatterns def mavenStyleBasePattern = "[organisation]/[module]/[revision]/[artifact]-[revision](-[classifier]).[ext]" def localBasePattern = "[organisation]/[module]/[revision]/[type]s/[artifact](-[classifier]).[ext]" def userRoot = System.getProperty("user.home") def userMavenRoot = userRoot + "/.m2/repository/" def userIvyRoot = userRoot + "/.ivy2/" private def userIvyRootFile = new File(userIvyRoot) def defaultLocal(ivyHome: Option[File]) = defaultUserFileRepository(ivyHome, "local") def defaultShared(ivyHome: Option[File]) = defaultUserFileRepository(ivyHome, "shared") def defaultUserFileRepository(ivyHome: Option[File], id: String) = file(id, new File(ivyHome.getOrElse(userIvyRootFile), id))(defaultIvyPatterns) def defaultIvyPatterns = { val pList = List(localBasePattern) Patterns(pList, pList, false) } } object Configurations { def config(name: String) = new Configuration(name) def defaultMavenConfigurations = Compile :: Runtime :: Test :: Provided :: System :: Optional :: Sources :: Javadoc :: Nil lazy val Default = config("default") lazy val Compile = config("compile") lazy val IntegrationTest = config("it") hide lazy val Provided = config("provided") lazy val Javadoc = config("javadoc") lazy val Runtime = config("runtime") lazy val Test = config("test") hide lazy val Sources = config("sources") lazy val System = config("system") lazy val Optional = config("optional") lazy val CompilerPlugin = config("plugin") hide private[sbt] val DefaultMavenConfiguration = defaultConfiguration(true) private[sbt] val DefaultIvyConfiguration = defaultConfiguration(false) private[sbt] def DefaultConfiguration(mavenStyle: Boolean) = if(mavenStyle) DefaultMavenConfiguration else DefaultIvyConfiguration private[sbt] def defaultConfiguration(mavenStyle: Boolean) = if(mavenStyle) Configurations.Compile else Configurations.Default private[sbt] def removeDuplicates(configs: Iterable[Configuration]) = Set(scala.collection.mutable.Map(configs.map(config => (config.name, config)).toSeq: _*).values.toList: _*) } /** Represents an Ivy configuration. */ final case class Configuration(name: String, description: String, isPublic: Boolean, extendsConfigs: List[Configuration], transitive: Boolean) extends NotNull { require(name != null && !name.isEmpty) require(description != null) def this(name: String) = this(name, "", true, Nil, true) def describedAs(newDescription: String) = Configuration(name, newDescription, isPublic, extendsConfigs, transitive) def extend(configs: Configuration*) = Configuration(name, description, isPublic, configs.toList ::: extendsConfigs, transitive) def notTransitive = intransitive def intransitive = Configuration(name, description, isPublic, extendsConfigs, false) def hide = Configuration(name, description, false, extendsConfigs, transitive) override def toString = name } final case class Artifact(name: String, `type`: String, extension: String, classifier: Option[String], configurations: Iterable[Configuration], url: Option[URL], extraAttributes: Map[String,String]) extends NotNull { def extra(attributes: (String,String)*) = Artifact(name, `type`, extension, classifier, configurations, url, extraAttributes ++ ModuleID.checkE(attributes)) } object Artifact { def apply(name: String): Artifact = Artifact(name, defaultType, defaultExtension, None, Nil, None) def apply(name: String, extra: Map[String,String]): Artifact = Artifact(name, defaultType, defaultExtension, None, Nil, None, extra) def apply(name: String, classifier: String): Artifact = Artifact(name, defaultType, defaultExtension, Some(classifier), Nil, None) def apply(name: String, `type`: String, extension: String): Artifact = Artifact(name, `type`, extension, None, Nil, None) def apply(name: String, `type`: String, extension: String, classifier: String): Artifact = Artifact(name, `type`, extension, Some(classifier), Nil, None) def apply(name: String, url: URL): Artifact =Artifact(name, extract(url, defaultType), extract(url, defaultExtension), None, Nil, Some(url)) def apply(name: String, `type`: String, extension: String, classifier: Option[String], configurations: Iterable[Configuration], url: Option[URL]): Artifact = Artifact(name, `type`, extension, classifier, configurations, url, Map.empty) val defaultExtension = "jar" val defaultType = "jar" def sources(name: String) = Artifact(name, "src", "jar", "sources") def javadoc(name: String) = Artifact(name, "doc", "jar", "javadoc") def extract(url: URL, default: String): String = extract(url.toString, default) def extract(name: String, default: String): String = { val i = name.lastIndexOf('.') if(i >= 0) name.substring(i+1) else default } def defaultArtifact(file: File) = { val name = file.getName val i = name.lastIndexOf('.') val base = if(i >= 0) name.substring(0, i) else name Artifact(name, extract(name, defaultType), extract(name, defaultExtension), None, Nil, Some(file.toURI.toURL)) } } final case class ModuleConfiguration(organization: String, name: String, revision: String, resolver: Resolver) extends NotNull object ModuleConfiguration { def apply(org: String, resolver: Resolver): ModuleConfiguration = apply(org, "*", "*", resolver) def apply(org: String, name: String, resolver: Resolver): ModuleConfiguration = ModuleConfiguration(org, name, "*", resolver) }