Write Plugins
Artifactory plugins are written as Groovy scripts in regular files and have a simple DSL to wrap users code in closures inside well-known extension points.
Scripts have a couple of helper objects that are globally bound (see the plugin script template).
Naming conventions
Note that Groovy scripts must follow the same naming conventions as those specified for Java.
The Artifactory Public API (PAPI)
Scripts have access to the full classpath of Artifactory. However, the only API supported for plugins is the Artifactory Public API, defined in artifactory-papi.jar.
You can find the artifactory-papi.jar under WEB-INF/lib folder inside artifactory.war.
For more information, see Plugin Code Template.
IDE code completion
All major IDEs have good Groovy editing and debugging capabilities.
In order to make your developing experience complete, we provide support for our own DSL for IntelliJ IDEA. IntelliJ IDEA's Groovy DSL script for Artifactory User Plugins can be found in our GitHub repo.
Eclipse DSLD file is also available courtesy of James Carnegie.
Globally Bound Variables
Variable Name | Variable Type | Comments |
|---|---|---|
log |
| Writes to Artifactory log logger name is the name of the script file |
repositories |
| Allows queries and operations on repositories and artifacts |
security |
| Provides information about current security context, (e.g., current user and their permissions) |
searches |
| API for searching for artifacts and builds Since 2.3.4 |
builds |
| Allows CRUD operations on builds Since 2.6 |
Closure Variables
Note! Declaring your own closure variables using the Groovy 'def ' keyword is considered best practice. Avoiding the "def" keyword is risky in terms of variable scoping, and will result in the variable being scoped globally, making it accessible from other closure executions.
Plugins Repository
Enhancing Artifactory with user plugins is community-driven effort.
If you are looking to go beyond Artifactory's out-of-the-box functionality take a look at already contributed plugins on GitHub, you might find what you are thinking about. If not, your contribution is very welcome!
Plugin Execution Points
The following table summarizes the available execution points. For more details about specific plugins, follow the section links.
Plugin Type | Code block name | When executed | Description |
|---|---|---|---|
Event Callback (without return value) | | On any upload | Handle before upload request events, executed before Artifactory starts to handle the original client request, useful for overriding the actual repo path during the Artifactory upload process. |
Event Callback (with return values) | | On any download | Provide an alternative response, by setting a success/error status code value and an optional error message or by setting new values for the inputStream and size context variables (For succeeded resolutions). |
| On any download | Provide an alternative response, by setting response headers, a success/error status code value and an optional error message or by setting new values for the inputStream and size context variables. This applies for GET/HEAD requests, for successful resolutions or Not Modified responses. | |
| When reaching out to remote repositories | Provides an alternative download path under the same remote repository, by setting a new value to the path variable. | |
| After fetching content from remote repositories | Provide an alternative download content, by setting new values for the inputStream and size context variables. | |
| After failing during content fetching from remote repositories | Provide an alternative response, by setting a success/error status code value and an optional error message or by setting new values for the inputStream and size context variables (For failed resolutions). | |
Event Callback (without return value) | | Before fetching content from remote repositories | Handle before remote download events. |
| After fetching content from remote repositories | Handle after remote download events. | |
| On any download | Handle before download events. | |
| On any download | Handle after download events. It is triggered after download execution, not download completion. | |
| On any download | Handle before download request events, executed before Artifactory starts to handle the original client request, useful for intercepting expirable resources (other than the default ones like maven-metadata.xml).
| |
Event Callback (without return value) | | Before / After selected storage operation | Handle events before and after Create, Delete, Move and Copy operations
|
Scheduled execution | any valid Groovy (Java) literal as execution name | According to provided interval/delay or cron expression | Job runs are controlled by the provided interval or cron expression, which are mutually exclusive. The actual code to run as part of the job should be part of the job's closure. |
User-driven execution | any valid Groovy (Java) literal as execution name | By REST call | External executions are invoked via REST requests. |
Event Callback (without return value) | any valid Groovy (Java) literal as realm name with nested blocks: | During user authentication | Newly added realms are added before any built-in realms (Artifactory internal realm, LDAP, Crowd etc.). User authentication will be attempted against these realms first, by the order they are defined.
|
Event Callback (without return value) | | Before the build info is saved in Artifactory | Handle before build info save events |
| After the build info is saved in Artifactory | Handle after build info save events | |
User or build server driven execution | any valid Groovy (Java) literal as promotion name | By REST call | Promotes integration (a.k.a. snapshot) build to be a release invoking any code associated with it. |
build server driven execution | any valid Groovy (Java) literal as staging strategy name. | During build server driven staging build configuration | The strategy provides the build server with the following information:
|
Event callback (with return value) | beforeFileReplication | Before file is replicated | Handle before file replication events. File replication can be skipped. |
beforeDirectoryReplication | Before directory is replicated | Handle before directory replication events. Directory replication can be skipped. | |
beforeDeleteReplication | Before file/directory is deleted | Handle before file or directory are deleted. | |
beforePropertyReplication | Before properties are replicated | Handle properties replication. |
User Plugin Execution Context
The Download, Storage, Execution ,and Build plugin types are executed under the identity of the user request that triggered them.
It is possible to force a block of plugin code to execute under the "system" role, which is not bound to any authorization rules and can therefore perform actions that are otherwise forbidden for the original user.
To run under the "system" role wrap your code with the asSystem closure:
... someCode ...
asSystem {
//This code runs as the system role
}
... someOtherCode ...
Including AQL Queries
User plugins may include AQL queries opening up the full set of search capabilities that AQL has to offer. AQL queries are implemented within the Searches object as shown in the example below.
import org.artifactory.repo.RepoPathFactory
import org.artifactory.search.Searches
import org.artifactory.search.aql.AqlResult
executions {
gemPropsPopulator() {
def repoKey = "gem-local"
((Searches) searches).aql(
"items.find({" +
"\"repo\": \"" + repoKey + "\"," +
"\"\$or\":[" +
"{\"property.key\":{\"\$ne\":\"gem.name\"}}," +
"{\"property.key\":{\"\$ne\":\"gem.version\"}}" +
"]})" +
".include(\"path\", \"name\")") {
AqlResult result ->
result.each {
...
...
...
}
}
}
}Updated 3 months ago
