(For a tutorial scroll down.)
Chris' Build Tool (CBT) for Scala
Easy to learn and master, lightning fast and backed by a thriving community of enthusiasts and contributors. For talks, development roadmap, projects using cbt, etc see the wiki.
What is CBT?
CBT is a build tool meaning it helps orchestrating compilation, code and documentation generation, packaging, deployment and custom tooling for your project. It mainly targets Scala projects but is not exclusive to them.
CBT builds are full programs written using vanilla Scala code. Familiar concepts make you feel right at home - build files are classes, tasks are defs, you customize using method overrides. You already know these things and everything behaves as expected. That way implementing any build related requirement becomes as easy as writing any other Scala code.
CBT is simple in the sense that it uses very few concepts. A single build uses classes, defs and inheritance. Builds and binary dependencies can be composed to model modules depending on each other.
CBT believes good integration with existing tools to be very helpful. In that spirit CBT aims for excellent integration with the command line and your shell.
CBT considers source files to be an excellent way to distribute code and has first class support for source and git dependencies.
How is CBT different from other build tools?
Not all build tools allow you to write builds in a full programming language. CBT is based on the assumption that builds are complex enough problems to warrant this and abstraction and re-use is better handled through libraries rather than some restricted, declarative DSL. CBT shares this philosophy with sbt. (This also means that integration with external tools such as an IDE better happens programmatically through an api rather than a static data representation such as xml.)
Like sbt, CBT chooses Scala as its language of choice, trying to appeal to Scala programmers allowing them to re-use their knowledge and the safety of the language.
Unlike sbt 0.11 and later, CBT maps task execution to JVM method invocations. sbt implements its own self-contained task graph model and interpreter. This allows sbt to have its model exactly fit the requirements. CBT instead uses existing JVM concepts for the solution and adds custom concepts only when necessary. CBT assumes this to lead to better ease of use due to familarity and better integration with existing tools such as interactive debuggers or stack traces because CBT's task call stack IS the JVM call stack.
sbt 0.7 shared this design decision as many may have forgotten. However, CBT is still quite a bit simpler than even sbt 0.7 as CBT gets away with fewer concepts. sbt 0.7 had notions of main vs test sources, multi-project builds, task-dependencies which weren't invocations and other concepts. CBT maps all of these to def invocations and build composition instead.
System requirements
CBT is best tested under OSX. People are also using it also under Ubuntu and Windows via cygwin. It should be easy to port CBT to other systems or drop the cygwin requirement. You will only have to touch the launcher bash or .bat scripts. Please contribute back if you fixed something :).
You currently need javac and realpath or gcc installed. nailgun is optional for speedup. gpg is required only for publishing maven artifacts.
Features
CBT supports the basic needs for Scala builds right now: Compiling, running, testing, packaging, publishing local and to sonatype, scaladoc, maven dependencies, source dependencies (e.g. for modularized projects), triggering tasks on file changes, cross-compilation, reproducible builds.
There is also a growing number of plugins in plugins/
and stage2/plugins/
,
but some things you'd like may still be missing. Consider writing
a plugin in that case. It's super easy, just a trait. Share it :).
Tutorial
This section explains how to get started with cbt step-by-step. There are also example projects with build files in examples/ and test/.
Installation
If you haven't cloned cbt yet, clone it now. Cloning is how you install cbt. We know that's a bit unusual, but roll with it, there are good reasons :). Open a shell, cd to the directory where you want to install cbt and execute:
$ git clone https://github.com/cvogt/cbt.git
There are a bash script cbt
and a cbt.bat
in the checkout directory.
Add one to your $PATH
, e.g. symlink it from ~/bin/cbt
.
Check that it works by calling cbt
. You should see CBT compiling itself
and showing a list of built-in tasks.
Great, you're all set up. Now, let's use cbt for a new example project. Follow the below steps. (There is also an experimental GUI described later to create a project, but going through the steps this time will help you understand what exactly is going on.)
Creating your first project
Create a new directory and cd into it. E.g. my-project
.
$ mkdir my-project
$ cd my-project
Let's create a tiny sample app. CBT can generate it for you. Just run:
$ cbt tools createMain
Running your code
Now there should be a file Main.scala
, which prints Hello World
when run.
So let's run it:
$ cbt run
You should see how CBT first compiles your project, then runs it and prints
Hello World
. CBT created the file Main.scala
top-level in your directory.
You can alternatively place .scala
or .java
files in src/
or any of its subdirectories.
Creating a build file
Without a build file, CBT just uses some default build settings. Let's make the build more concrete by creating a build file.
CBT can help you with that. Execute:
$ cbt tools createBuild
Now there should be a file build/build.scala
with a sample Build
class.
Btw., a build file can have its own build and so on recursively like in sbt.
When you create a file build/build/build.scala
and change Build
class in there
to extend BuildBuild
, it will be used to build your build/build.scala
. You can
add built-time dependencies like plugins this way.
Adding dependencies
In the generated build/build.scala
there are
several examples for dependencies. We recommend using the constructor syntax
ScalaDependency
(for automatically adding the scala version to the artifact id)
or MavenDependency
(for leaving the artifact id as is). The sbt-style %
-DSL
syntax is also supported for copy-and-paste convenience, but discouraged.
Alright, let's enable the override def dependencies
. Make sure to include
super.dependencies
, which currently only includes the Scala standard library.
Add a dependency of your choice, start using it from Main.scala
and cbt run
again.
As you can see CBT makes choice of the maven repository explicit. It does so for clarity.
Calling other tasks
Tasks are just defs. You can call any public zero-arguments method of your `Build class or its parents straight from the command line. To see how it works let's call the compile task.
$ cbt compile
Creating custom tasks
In order to create a custom task, simply add a new def to your Build class, e.g.
class Build...{
...
def foo = "asdf"
}
Now call the def from the command line:
$ cbt foo
As you can see it prints asdf
. Adding tasks is that easy.
Triggering tasks on file-changes
When you call a task, you can prefix it with loop
. You need to
have fswatch install (e.g. via brew install fswatch
).
CBT then watches the source files, the build files and even CBT's own
source code and re-runs the task when anything changes. If necessary,
this forces CBT to re-build itself, the project's dependencies and the project itself.
Let's try it. Let's loop the run task. Call this from the shell:
$ cbt loop run
Now change `Main.scala and see how cbt picks it up and re-runs it. CBT is fast. It may already be done re-compiling and re-running before you managed to change windows back from your editor to the shell.
Try changing the build file and see how CBT reacts to it as well.
To also clear the screen on each run use:
$ cbt loop clear run
To call and restart the main method on file change (like sbt-revolver)
$ cbt direct loop restart
Adding tests
The simplest way to add tests is putting a few assertions into the previously created Main.scala and be done with it. Alternatively you can add a test framework plugin to your build file to use something more sophisticated.
This however means that the class files of your tests will be included in the jar should you create one. If that's fine, you are done :). If it is not you need to create another project, which depends on your previous project. This project will be packaged separately or you can disable packaging there. Let's create such a project now.
Your project containing tests can be anywhere but a recommended location is a
sub-folder called test/
in your main project. Let's create it and create a
Main class and build file:
$ mkdir test
$ cd test
$ rm ../Main.scala
$ cbt tools createMain
$ cbt tools createBuild
We also deleted the main projects Main.scala, because now that we created a new one we would have two classes with the same name on the classpath which can be very confusing.
Now that we have a Main file in our test project, we can add some assertions to it.
In order for them to see the main projects code, we still need to do one more thing -
add a DirectoryDependency
to your test project's build file. There is a similar example
in the generated build.scala. What you need is this:
override def dependencies = super.dependencies ++ Seq(
DirectoryDependency( projectDirectory ++ "/.." )
)
This successfully makes your test project's code see the main projects code.
Add some class to your main project, e.g. case class Foo(i: Int = 5)
. Now
put an assertion into the Main class of your test project, e.g.
assert(Foo().i == 5)
and hit cbt run
inside your test project.
Make sure you deleted your main projects class Main when running your tests.
Congratulations, you successfully created a dependent project and ran your tests.
Dynamic overrides and eval
By making your Build extend trait CommandLineOverrides
you get access to the eval
and with
commands.
eval
allows you to evaluate scala expressions in the scope of your build and show the result.
You can use this to inspect your build.
$ cbt eval '1 + 1'
2
$ cbt eval scalaVersion
2.11.8
$ cbt eval 'sources.map(_.string).mkString(":")'
/a/b/c:/d/e/f
with
allows you to inject code into your build (or rather a dynamically generated subclass).
Follow the code with another task name and arguments if needed to run tasks of your modified build.
You can use this to override build settings dynamically.
$ cbt with 'def hello = "Hello"' hello
Hello
$ cbt with 'def hello = "Hello"; def world = "World"; def helloWorld = hello ++ " " ++ world' helloWorld
Hello World
$ cbt with 'def version = super.version ++ "-SNAPSHOT"' package
/a/b/c/e-SNAPSHOT.jar
Multi-projects Builds
A single build only handles a single project in CBT. So there isn't exactly such a things as a Multi-project Build. Instead you can simply write multiple projects that depend on each other. We have already done that with tests above, but you can do the exact same thing to modularize your project into multiple ones.
Reproducible builds
To achieve reproducible builds, you'll need to tie your build files to a particular
CBT-version. It doesn't matter what version of CBT you are actually running,
as long as the BuildInterface
is compatible (which should be true for a large number
of versions and we may find a better solution long term. If you see a compile error
during compilation of CBT itself that some method in BuildInterface was not
implemented or incorrectly implemented, you may be running an incompatible CBT
version. We'll try to fix that later, but for now you might have to checkout
the required hash of CBT by hand.).
When you specify a particular version, CBT will use that one instead of the installed one.
You can specify one by adding one line right before class Build
. It looks like this:
// cbt:https://github.com/cvogt/cbt.git#f11b8318b85f16843d8cfa0743f64c1576614ad6
class Build...
The URL points to any git repository containing one of CBT's forks. You currently have to use a stable reference - i.e. a hash or tag. (Checkouts are currently not updated. If you refer to a branch or tag which is moved on the remote, CBT will not realize that and keep using the old version).
Using CBT like a boss
Do you own your Build Tool or does your Build Tool own you? CBT makes it easy for YOU to be in control. We try to work on solid documentation, but even good documentation never tells the whole truth. Documentation can tell how to use something and why things are happening, but only the code can tell all the details of what exactly is happening. Reading the code can be intimidating for many Scala libraries, but not so with CBT. The source code is easy to read to the point that even Scala beginners will be able to understand it. So don't be afraid to actually look under the hood and check out what's happening.
And guess what, you already have the source code on your disk, because you installed CBT by cloning its git repository. You can even debug CBT and your build files in an interactive debugger like IntelliJ after some minor setup.
Finally, you can easily change CBT's code. Then CBT re-builds itself when you try to use it the next time. This means any changes you make are instantly reflected. This and the simple code make it super easy to fix bugs or add features yourself and feed them back into main line CBT.
When debugging things, it can help to enable CBT's debug logging by passing
-Dlog=all
to CBT (or a logger name instead of all
).
Other design decisions
CBT tries to couple its code very loosely. OO is used for configuration in build files. Interesting logic is in simple supporting library classes/objects, which can be used independently. You could even build a different configuration api than OO on top of them.
Known limitations
- currently CBT supports no generic task scoping. A solution is known, but not implemented. For now manually create intermediate tasks which serve as scoping for known situations and encode the scope in the name, e.g. fastOptScalaJsOptions
- currently CBT supports no dynamic overrides of tasks. A solution is known, but not implemented. scalaVersion and version are passed through the context instead for dynamic overrides.
- there is currently no built-in support for resources being added to a jar. Should be simple to add, consider a PR
- test framework support is currently a bit spotty but actively being worked on
- concurrent task and build execution is currently disabled
- CBT uses its own custom built maven resolver, which is really fast, but likely does not work in some edge cases. Those may or may not be easy to fix. We should add optional coursier integration back for a more complete solution.
Known bugs
- currently there is a bug in CBT where dependent builds may miss changes in the things they depend on. Deleting all target directories and starting from scratch helps.
- There are some yet unknown bugs which can be solved by killing the nailgun background process and/or re-running the desired cbt task multiple times until it succeeds.
- if you ever see no output from a command but expect some, make sure you are in the right directory and try to see if any of the above recommendations help
IDE Support
IntelliJ
IntelliJ IDEA has CBT support. Plugin for CBT support can be installed as any other IntelliJ plugin via JetBrains plugin repository. More information about IntellJ support and documentation can be found here
Plugin page on JetBrains plugin repository: http://plugins.jetbrains.com/plugin/10482-cbt
Shell completions
Bash completions
To auto-complete cbt task names in bash do this:
mkdir ~/.bash_completion.d/
cp shell-integration/cbt-completions.bash ~/.bash_completion.d/
Add this to your .bashrc
for f in ~/.bash_completion.d/*; do
source $f
done
Fish shell completions
copy this line into your fish configuration, on OSX: /.config/fish/config.fish
complete -c cbt -a '(cbt taskNames)'
Zsh completions
Manual installation
Add the following to your .zshrc
source /path/to/cbt/shell-integration/cbt-completions.zsh
oh-my-zsh
If using oh-my-zsh, you can install it as a plugin:
mkdir ~/.oh-my-zsh/custom/plugins/cbt
cp shell-integration/cbt-completions.zsh ~/.oh-my-zsh/custom/plugins/cbt/cbt.plugin.zsh
Then enable it in your .zshrc
:
plugins=( ... cbt)
Experimental GUI
Creating a project via the GUI
cd
into the directory inside of which you want to create a new project directory and run cbt tools gui
.
E.g.
$ cd ~/my-projects
$ cbt tools gui
This should start UI server at http://localhost:9080. There you can create Main class, CBT build,
add libraries, plugins, readme and other things. Let's say you choose my-project
as the project name.
The GUI will create ~/my-projects/my-project
for you.
Plugin-author guide
A CBT plugin is a trait that is mixed into a Build class. Only use this trait only for wiring things together. Don't put logic in there. Instead simply call methods on a separate class or object which serves as a library for your actual logic. It should be callable and testable outside of a Build class. This way the code of your plugin will be easier to test and easier to re-use. Feel free to make your logic rely on CBT's logger.
See plugins/
for examples.
Scala.js support
CBT supports cross-project Scala.js builds. It preserves same structure as in sbt (https://www.scala-js.org/doc/project/cross-build.html)
- Example for user scalajs project is in:
$CBT_HOME/cbt/examples/scalajs-*
$CBT_HOME/cbt compile
Will compile JVM and JS sources$CBT_HOME/cbt jsCompile
Will compile JS sources$CBT_HOME/cbt jvmCompile
Will compile JVM sources$CBT_HOME/cbt fastOptJS
and$CBT_HOME/cbt fullOptJS
Same as in Scala.js sbt project
Note: Scala.js support is under ongoing development.
Currently missing features:
- No support for jsDependencies: It means that all 3rd party dependencies should added manually, see scalajs build example
- No support for test
CBT productivity hacks
only show first 20 lines of type errors to catch the root ones
cbt c 2>&1 | head -n 20
Inheritance Pittfalls
trait Shared extends BaseBuild{
// this lowers the type from Seq[Dependency] to Seq[DirectoryDependency]
override def dependencies = Seq( DirectoryDependency(...) )
}
class Build(...) extends Shared{
// this now fails because GitDependency is not a DirectoryDependency
override def dependencies = Seq( GitDependency(...) )
}
// Solution: raise the type explicitly
trait Shared extends BaseBuild{
// this lowers the type from Seq[Dependency] to Seq[DirectoryDependency]
override def dependencies: Seq[Dependency] = Seq( DirectoryDependency(...) )
}
trait Shared extends BaseBuild{
override def dependencies: Seq[Dependency] = Seq() // removes all dependencies, does not inclide super.dependencies
}
trait SomePlugin extends BaseBuild{
// adds a dependency
override def dependencies: Seq[Dependency] = super.dependencies ++ Seq( baz )
}
class Build(...) extends Shared with SomePlugin{
// dependencies does now contain baz here, which can be surprising
}
// Solution can be being careful about the order and using traits instead of classes for mixins
class Build(...) extends SomePlugin with Shared