aboutsummaryrefslogtreecommitdiff

Join us on gitter

(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

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)

  1. Example for user scalajs project is in: $CBT_HOME/cbt/examples/build-scalajs
  2. $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
  3. $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