Enhance your Groovy DSL in a plugin-ish way

The base of a groovy DSL is the Script class or the DelegatingScript. To provide some functions to your DSL, you define them in this class, implement the missingMethod or enhance the metaclass. No matter which way you choose, it blows up your class with rising amount of functions. So after some time you might want to restructure your code. For some other reason you might want to enable other people to enhance the DSL without touching the base DSL class.

The idea is to provide a class that can be instantiated in you DSL with a create-method.

1. Define a marker interface to find the plugin class (you can do this with annotations as well)

Every class implementing this marker interface will be instantiable in your DSL

2. A class that enhances your DSL

We’ll have a Car class that has only one field describing the color of our car

 

3. The actual DSL-backing class

Now we’ll go to the bone. As mentioned above, there are several ways to enhance the DSL class with new functions. In this sample we’ll add new functions by enhancing DSL’s metaClass.

We’ll have two functions to get an instance of Car: newCar and newCarWith.

newCar is calling the Default-Constructor without arguments. newCarWith calls the “Named” Default-Constructor with a map. To achieve this, there are two closures defined in line 3 and 4. First one with two params (actual class and the map) and one with a single param (only the class to instantiate).

Now we need to find the classes to instantiate. I used the reflections-API to find all classes implementing the Instantiable interface (line 6). In this sample only classes of the package de.metacode.dslplugin are scanned. You can scann the whole known classes as well.

Line 7 and 8 is where the magic happens: Two new methods are added to metaClass. Note the curry-call on the newInstance-closures. What this does is called currying (or Schönfinkeln). It reduces a function with n arguments to a function with n-1 arguments. In this case curry generates a new closure without the first parameter Class clazz. So we have one closure without any parameter and another one with a map.

 

4. Writing some DSL-code

Now we can instantiate the Car without params or with named params:

The output is

 

That’s it

Here you’ll find executable wired-together code:

https://gist.github.com/codeeraser/31d031cd921e168049ac

You’ll might get some issues with reflections-API when running this code via command-line. To get reflections work like one would expect can be pretty annoying (see stackoverflow…). The shortest way is to run the sample in an IDE.

Groovy DSL: How to split up execution of method calls like gradle does

This is how you can implement a gradle-ish behavior of executing a designated method before everything else in a dsl script (gradle executes doFirst{} first). Actually, I have no clue how gradle does it. This approach is about manipulating the AST with an implementation of  CompilationCustomizer. I’ll describe two variations. First, how to execute the script twice, e.g. to set up an environment and execute the actual code sometimes later. And than how you can re-order the statements before execution. It’ll be helpful if you know about how AST transformations work (if you don’t, see the links at the end of this article).

Let’s say we have a DSL script like this:

First variation:

We’ll handle the DSL using DelegatingScript. The is the delegate for the first run (only doFirst will be executed):

For the second run this is doing everything else:

At this point we’ve got everything we need to execute the DSL. Now comes the AST transformation part. The following CompilerCustomizer removes every statement from the AST except a designated method. Because there can only be one (method), this is called the HighlanderCustomizer 😉

Bring it all together:

The console output should look like this:

 

Second variation:

Now the DSL will not be executed in two steps but rather in one single run. We’ll use a CompilationCustomizer again. This time the customizer finds a designated method call and changes the order of execution.

Our Delegate (this time only one is needed):

The ReOrderCustomizer:

Bring it all together:

Additionally, one should check that the designated method appears only once in the DSL script.

To understand how the AST looks like, just debug the customizer’s call method.

This article is based on a question I postet on Stackoverflow: http://stackoverflow.com/questions/29967886/groovy-dsl-how-can-i-let-two-delegating-classes-handle-different-parts-of-a-dsl

 

Links:

Great example of using Groovy to implement a DSL: https://groovy.codeplex.com/wikipage?title=Guillaume%20Laforge’s%20%22Mars%20Rover%22%20tutorial%20on%20Groovy%20DSL’s

About CompilationCustomizer: http://www.jroller.com/melix/entry/customizing_groovy_compilation_process

AST transformation workshop: http://melix.github.io/ast-workshop/

Joe’s series of AST articles: http://joesgroovyblog.blogspot.de