Quick Start
Grape lets you quickly add maven repository dependencies to your classpath. Here are the most common solutions:
Add a Dependency
| Code Block |
|---|
@Grab(group='org.springframework', module='spring', version='2.5.6') import org.springframework.jdbc.core.JdbcTemplate |
Yes, you can annotate an import in Groovy. You can also search for dependencies on mvnrepository.com and it will provide you the @Grab annotation form of the pom.xml entry.
Specify Additional Repositories
Not all dependencies are in maven central. You can add new ones like this:
| Code Block |
|---|
@GrabResolver(name='restlet', root='http://maven.restlet.org/') @Grab(group='org.restlet', module='org.restlet', version='1.1.6') |
Maven Classifiers
Some maven dependencies need classifiers in order to be able to resolve. You can fix that like this:
| Code Block |
|---|
@Grab(group='net.sf.json-lib', module='json-lib', version='2.2.3', classifier='jdk15') |
Excluding Transitive Dependencies
Sometimes you will want to exclude transitive dependencies as you might be already using a slightly different but compatible version of some artifact. You can do this as follows:
| Code Block |
|---|
@Grab('net.sourceforge.htmlunit:htmlunit:2.8')
@GrabExclude('xml-apis:xml-apis')
|
JDBC Drivers
Because of the way JDBC drivers are loaded, you'll need to configure Grape to attach JDBC driver dependencies to the system class loader. I.e:
| Code Block |
|---|
@GrabConfig(systemClassLoader=true) @Grab(group='mysql', module='mysql-connector-java', version='5.1.6') |
Using Grape From the Groovy Shell
From groovysh use the method call variant:
| Code Block |
|---|
groovy.grape.Grape.grab([group:'org.springframework', module:'spring', version:'2.5.6']) |
Proxy settings
If you are behind a firewall and/or need to use Groovy/Grape through a proxy server, you can specify those settings on the command like via the http.proxyHost and http.proxyPort system properties:
| Code Block |
|---|
groovy -Dhttp.proxyHost=yourproxy -Dhttp.proxyPort=8080 yourscript.groovy |
Or you can make this system wide by adding these properties to your JAVA_OPTS environment variable:
| Code Block |
|---|
JAVA_OPTS = -Dhttp.proxyHost=yourproxy -Dhttp.proxyPort=8080 |
Detail
Grape (The Groovy Adaptable Packaging Engine or Groovy Advanced Packaging Engine) is the infrastructure enabling the grab() calls in Groovy, a set of classes leveraging Ivy to allow for a repository driven module system for Groovy. This allows a developer to write a script with an essentially arbitrary library requirement, and ship just the script. Grape will, at runtime, download as needed and link the named libraries and all dependencies forming a transitive closure when the script is run from existing repositories such as Ibiblio, Codehaus, and java.net.
Grape follows the Ivy conventions for module version identification, with naming change.
group- Which module group the module comes from. Translates directly to a Maven groupId or an Ivy Organization. Any group matching/groovy[x][\..*]^/is reserved and may have special meaning to the groovy endorsed modules.module- The name of the module to load. Translated directly to a Maven artifactId or an Ivy artifact.version- The version of the module to use. Either a literal version '1.1-RC3' or an Ivy Range '[2.2.1,)' meaning 2.2.1 or any greater version).
The downloaded modules will be stored according to Ivy's standard mechanism with a cache root of ~/.groovy/grape
Usage
Annotation
One or more groovy.lang.Grab annotations can be added at any place that annotations are accepted to tell the compiler that this code relies on the specific library. This will have the effect of adding the library to the classloader of the groovy compiler. This annotation is detected and evaluated before any other resolution of classes in the script, so imported classes can be properly resolved by a @Grab annotation.
| Code Block |
|---|
import com.jidesoft.swing.JideSplitButton
@Grab(group='com.jidesoft', module='jide-oss', version='[2.2.1,2.3.0)')
public class TestClassAnnotation {
public static String testMethod () {
return JideSplitButton.class.name
}
}
|
An appropriate grab(...) call will be added to the static initializer of the class of the containing class (or script class in the case of an annotated script element).
Multiple Grape Annotations
In order to use a Grape annotation multiple times you must use the Grapes annotation, e.g.:
| Code Block |
|---|
@Grapes([
@Grab(group='commons-primitives', module='commons-primitives', version='1.0'),
@Grab(group='org.ccil.cowan.tagsoup', module='tagsoup', version='0.9.7')])
class Example {
// ...
}
|
Otherwise you'll encounter the following error:
| Code Block |
|---|
Cannot specify duplicate annotation on the same member |
.
Method call
Typically a call to grab will occur early in the script or in class initialization. This is to insure that the libraries are made available to the ClassLoader before the groovy code relies on the code. A couple of typical calls may appear as follows:
| Code Block |
|---|
import groovy.grape.Grape
// random maven library
Grape.grab(group:'com.jidesoft', module:'jide-oss', version:'[2.2.0,)')
Grape.grab([group:'org.apache.ivy', module:'ivy', version:'2.0.0-beta1', conf:['default', 'optional']],
[group:'org.apache.ant', module:'ant', version:'1.7.0'])
// endorsed Groovy Module
// FUTURE grab('Scriptom')
|
* Multiple calls to grab in the same context with the same parameters should be idempotent. However, if the same code is called with a different ClassLoader context then resolution may be re-run.
grabis disabled by default. Starting callingGrape.initGrape()will enable grab. Any calls to grab beforeinitGrape()is called will be ignored. Hence Grape managed classloading is opt in only. Multiple calls tiGrape.initGrape()after the first successful call are ignored.- If the
argsmap passed into thegrabcall has an attributenoExceptionsthat evaluates true no exceptions will be thrown. grabrequires that a RootLoader or GroovyClassLoader be specified or be in the ClassLoader chain of the calling class. By default failure to have such a ClassLoader available will result in module resolution and an exception being thrown (ifinitGrape()has been called).- The ClassLoader passed in via the
classLoader:argument and it's parent classloaders. - The ClassLoader of the object passed in as the
referenceObject:argument, and it's parent classloaders. - The ClassLoader of the class issuing the call to
grab
- The ClassLoader passed in via the
grab(HashMap) Parameters
group:- <String> - Which module group the module comes from. Translates directly to a Maven groupId. Any group matching/groovy(|\..|x|x\..)/is reserved and may have special meaning to the groovy endorsed modules.module:- <String> - The name of the module to load. Translated directly to a Maven artifactId.version:- <String> and possibly <Range> - The version of the module to use. Either a literal version '1.1-RC3' or an Ivy Range '[2.2.1,)' meaning 2.2.1 or any greater version).classifier:- <String> - The Maven classifier to resolve by.conf:- <String>, default 'default' - The configuration or scope of the module to download. The default conf isdefault:which maps to the mavenruntimeandmasterscopes.force:- <boolean>, defaults true - Used to indicate that this revision must be used in case of conflicts, independently of- conflicts manager
changing:- <boolean>, default false - Whether the artifact can change without it's version designation changing.transitive:- <boolean>, default true - Whether to resolve other dependencies this module has or not.
There are two principal variants of grab, one with a single Map and one with an arguments Map and multiple dependencies map. A call to the single map grab is the same as calling grab with the same map passed in twice, so grab arguments and dependencies can be mixed in the same map, and grab can be called as a single method with named parameters.
There are synonyms for these parameters. Submitting more than one is a runtime exception.
group:,groupId:,organisation:,organization:,org:module:,artifactId:,artifact:version:,revision:,rev:conf:,scope:,configuration:
Arguments Map arguments
classLoader:- <GroovyClassLaoder> or <RootClassLoader> - The ClassLoader to add resolved Jars torefObject:- <Object> - The closest parent ClassLoader for the object's class will be treated as though it were passed in asclassLoader:validate:- <boolean>, default false - Should poms or ivy files be validated (true), or should we trust the cache (false).noExceptions:- <boolean>, default false - If ClassLoader resolution or repository querying fails, should we throw an exception (false) or fail silently (true).
Command Line Tool
Grape added a command line executable 'grape' that allows for the inspection and management of the local grape cache.
| Code Block |
|---|
grape install <groupId> <artifactId> [<version>] |
This installs the specified groovy module or maven artifact. If a version is specified that specific version will be installed, otherwise the most recent version will be used (as if '*' we passed in).
| Code Block |
|---|
grape list |
Lists locally installed modules (with their full maven name in the case of groovy modules) and versions.
| Code Block |
|---|
grape resolve (<groupId> <artifactId> <version>)+ |
This returns the file locations of the jars representing the artifcats for the specified module(s) and the respective transitive dependencies. You may optionally pass in -ant, -dos, or -shell to get the dependencies expressed in a format applicable for an ant script, windows batch file, or unix shell script respectively. -ivy may be passed to see the dependencies expressed in an ivy like format.
Advanced configuration
Repository Directory
If you need to change the directory grape uses for downloading libraries you can specify the grape.root system property to change the default (which is ~/.groovy/grape)
| Code Block |
|---|
groovy -Dgrape.root=/repo/grape yourscript.groovy |
Customize Ivy settings
//TODO expand on discussion of grapeConfig.xml
You can customize the ivy settings that Grape uses by creating a ~/.groovy/grapeConfig.xml file. If no such file exists, here are the default settings used by Grape:
| Code Block |
|---|
<ivysettings>
<settings defaultResolver="downloadGrapes"/>
<resolvers>
<chain name="downloadGrapes">
<filesystem name="cachedGrapes">
<ivy pattern="${user.home}/.groovy/grapes/[organisation]/[module]/ivy-[revision].xml"/>
<artifact pattern="${user.home}/.groovy/grapes/[organisation]/[module]/[type]s/[artifact]-[revision].[ext]"/>
</filesystem>
<!-- todo add 'endorsed groovy extensions' resolver here -->
<ibiblio name="codehaus" root="http://repository.codehaus.org/" m2compatible="true"/>
<ibiblio name="ibiblio" m2compatible="true"/>
<ibiblio name="java.net2" root="http://download.java.net/maven/2/" m2compatible="true"/>
</chain>
</resolvers>
</ivysettings>
|
For more information on how to customize these settings, please refer to the Ivy documentation.
Add your local Maven2 repository
If you find yourself wanting to reuse artifacts that you already have locally in your Maven2 repository, then you can add this line to your ~/.groovy/grapeConfig.xml:
| Code Block |
|---|
<ibiblio name="local" root="file:${user.home}/.m2/repository/" m2compatible="true"/> |
And further customize your Grape configuration:
| Code Block |
|---|
<?xml version="1.0"?>
<ivysettings>
<settings defaultResolver="downloadGrapes"/>
<resolvers>
<chain name="downloadGrapes">
<!-- todo add 'endorsed groovy extensions' resolver here -->
<ibiblio name="local" root="file:${user.home}/.m2/repository/" m2compatible="true"/>
<filesystem name="cachedGrapes">
<ivy pattern="${user.home}/.groovy/grapes/[organisation]/[module]/ivy-[revision].xml"/>
<artifact pattern="${user.home}/.groovy/grapes/[organisation]/[module]/[type]s/[artifact]-[revision].[ext]"/>
</filesystem>
<ibiblio name="codehaus" root="http://repository.codehaus.org/" m2compatible="true"/>
<ibiblio name="ibiblio" m2compatible="true"/>
<ibiblio name="java.net2" root="http://download.java.net/maven/2/" m2compatible="true"/>
</chain>
</resolvers>
</ivysettings>
|
More Examples
Using Apache Commons Collections:
| Code Block |
|---|
// create and use a primitive array
import org.apache.commons.collections.primitives.ArrayIntList
@Grab(group='commons-primitives', module='commons-primitives', version='1.0')
def createEmptyInts() { new ArrayIntList() }
def ints = createEmptyInts()
ints.add(0, 42)
assert ints.size() == 1
assert ints.get(0) == 42
|
Using TagSoup:
| Code Block |
|---|
// find the PDF links in the Java 1.5.0 documentation
@Grab(group='org.ccil.cowan.tagsoup', module='tagsoup', version='0.9.7')
def getHtml() {
def parser = new XmlParser(new org.ccil.cowan.tagsoup.Parser())
parser.parse("http://java.sun.com/j2se/1.5.0/download-pdf.html")
}
html.body.'**'.a.@href.grep(~/.*\.pdf/).each{ println it }
|
Using Google Collections:
| Code Block |
|---|
// Google Collections example
import com.google.common.collect.HashBiMap
@Grab(group='com.google.code.google-collections', module='google-collect', version='snapshot-20080530')
def getFruit() { [grape:'purple', lemon:'yellow', orange:'orange'] as HashBiMap }
assert fruit.lemon == 'yellow'
assert fruit.inverse().yellow == 'lemon'
|
Launching a Jetty server to server Groovy templates:
| Code Block |
|---|
import org.mortbay.jetty.Server
import org.mortbay.jetty.servlet.*
import groovy.servlet.*
@Grab(group = 'org.mortbay.jetty', module = 'jetty-embedded', version = '6.1.0')
def runServer(duration) {
def server = new Server(8080)
def context = new Context(server, "/", Context.SESSIONS);
context.resourceBase = "."
context.addServlet(TemplateServlet, "*.gsp")
server.start()
sleep duration
server.stop()
}
runServer(10000)
|
Grape will download Jetty and its dependencies on first launch of this script, and cache them. We're creating a new Jetty Server on port 8080, then expose Groovy's TemplateServlet at the root of the context — Groovy comes with its own powerful template engine mechanism. We start the server and let it run for a certain duration. Each time someone will hit http://localhost:8080/somepage.gsp, it will display the somepage.gsp template to the user — those template pages should be situated in the same directory as this server script.
See Also:
Using Hibernate with Groovy
http://stackoverflow.com/questions/192432/getting-groovys-grape-going