diff --git a/README.md b/README.md index fd9decb..40ebc89 100644 --- a/README.md +++ b/README.md @@ -56,6 +56,7 @@ Existing service plugins enabled on boostrap-plugin * LogFilter * NodeExecutor * Orchestrator +* Option #### for Script Plugins: * ResourceModelSource @@ -64,6 +65,7 @@ Existing service plugins enabled on boostrap-plugin * NodeExecutor * FileCopier * NodeExecutorFileCopier: Generate both, Node Executor and File Copier service +* Option #### for UI plugins * UI diff --git a/build.gradle b/build.gradle index edb29b1..36e1287 100644 --- a/build.gradle +++ b/build.gradle @@ -23,6 +23,8 @@ dependencies { compile 'com.github.rundeck.cli-toolbelt:toolbelt:0.2.2' compile 'com.github.rundeck.cli-toolbelt:toolbelt-jewelcli:0.2.2' compile 'org.apache.commons:commons-text:1.4' + compile 'info.picocli:picocli:4.0.0-alpha-2' + testCompile 'org.spockframework:spock-core:1.0-groovy-2.4' } diff --git a/src/main/groovy/com/rundeck/plugin/Generator.groovy b/src/main/groovy/com/rundeck/plugin/Generator.groovy index fc59d54..4bda330 100644 --- a/src/main/groovy/com/rundeck/plugin/Generator.groovy +++ b/src/main/groovy/com/rundeck/plugin/Generator.groovy @@ -1,13 +1,14 @@ package com.rundeck.plugin -import com.lexicalscope.jewel.cli.Option import com.rundeck.plugin.template.FilesystemArtifactTemplateGenerator import com.rundeck.plugin.template.PluginType -import org.rundeck.toolbelt.Command -import org.rundeck.toolbelt.CommandRunFailure -import org.rundeck.toolbelt.SubCommand -import org.rundeck.toolbelt.ToolBelt -import org.rundeck.toolbelt.input.jewelcli.JewelInput +import com.rundeck.plugin.template.ServiceType +import picocli.CommandLine +import picocli.CommandLine.Option +import picocli.CommandLine.Command + + +import java.util.concurrent.Callable /* * Copyright 2018 Rundeck, Inc. (http://rundeck.com) @@ -25,36 +26,36 @@ import org.rundeck.toolbelt.input.jewelcli.JewelInput * limitations under the License. */ -@SubCommand -class Generator { +@Command(description = "Create a Rundeck plugin artifact.", + name = "plugin-bootstrap", mixinStandardHelpOptions = true, version = "1.1") +class Generator implements Callable{ - private static final List VALID_PLUGIN_TYPES = ["java","script","ui"] - - public static void main(String[] args) throws IOException, CommandRunFailure { - ToolBelt.with("plugin-bootstrap", new JewelInput(), new Generator()).runMain(args, true); + static void main(String[] args) throws Exception { + try{ + CommandLine.call(new Generator(), args) + }catch(Exception e){ + println(e.getMessage()) + } } - @Command(description = "Create a Rundeck plugin artifact") - public void create(CreateOpts createOpts) { - if(!VALID_PLUGIN_TYPES.contains(createOpts.pluginType)) { - println "Artifact type must be one of: ${VALID_PLUGIN_TYPES.join("|")}" - return - } + @Option(names = [ "-n", "--pluginName" ], description = "Plugin Name." , required = true) + String pluginName; + @Option(names = [ "-t", "--pluginType" ] ,description = 'Plugin Type: ${COMPLETION-CANDIDATES}' , required = true) + PluginType pluginType; + @Option(names = [ "-s", "--serviceType" ],description = 'Rundeck Service Type: ${COMPLETION-CANDIDATES}', required = true) + ServiceType serviceType + @Option(names = [ "-d", "--destinationDirectory" ],description = "The directory in which the artifact directory will be generated", required = true) + String destinationDirectory + + @Override + Void call() throws Exception { FilesystemArtifactTemplateGenerator generator = new FilesystemArtifactTemplateGenerator() - println generator.generate(createOpts.pluginName, - PluginType.valueOf(createOpts.pluginType), - createOpts.serviceType, - createOpts.destinationDirectory) - } - interface CreateOpts { - @Option(shortName = "n",description = "Plugin Name") - String getPluginName() - @Option(shortName = "t",description = "Plugin Type") - String getPluginType() - @Option(shortName = "s",description = "Rundeck Service Type") - String getServiceType() - @Option(shortName = "d",description = "The directory in which the artifact directory will be generated") - String getDestinationDirectory() + println generator.generate(this.pluginName, + this.pluginType, + this.serviceType.toString(), + this.destinationDirectory) + + return null } } diff --git a/src/main/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGenerator.groovy b/src/main/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGenerator.groovy index 8e2a048..bcbedcf 100644 --- a/src/main/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGenerator.groovy +++ b/src/main/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGenerator.groovy @@ -24,7 +24,7 @@ class JavaPluginTemplateGenerator extends AbstractTemplateGenerator { private static final String TEMPLATE_BASE = "templates/java-plugin/" private static final String JAVA_STRUCTURE = "java-plugin.structure" - private static final List ALLOWED_TEMPLATES = ["ResourceModelSource","Notification","WorkflowStep","WorkflowNodeStep","LogFilter","NodeExecutor","Orchestrator"] + private static final List ALLOWED_TEMPLATES = ["ResourceModelSource","Notification","WorkflowStep","WorkflowNodeStep","LogFilter","NodeExecutor","Orchestrator","Option"] @Override Map makeTemplateProperties(final String pluginName, final String providedService) { @@ -51,7 +51,7 @@ class JavaPluginTemplateGenerator extends AbstractTemplateGenerator { @Override void preTemplateValidations(String providedService) { - if(!ALLOWED_TEMPLATES.contains(providedService))throw new Exception("Only "+ALLOWED_TEMPLATES.toString()+" plugins generation are supported at this time") + if(!ALLOWED_TEMPLATES.contains(providedService))throw new Exception("Java plugin does not support this service : ${providedService}. Only "+ALLOWED_TEMPLATES.toString()+" are supported") } @Override diff --git a/src/main/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGenerator.groovy b/src/main/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGenerator.groovy index fdd4d00..f6f8d1e 100644 --- a/src/main/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGenerator.groovy +++ b/src/main/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGenerator.groovy @@ -24,7 +24,7 @@ class ScriptPluginTemplateGenerator extends AbstractTemplateGenerator { private static final String TEMPLATE_BASE = "templates/script-plugin/" private static final String SCRIPT_STRUCTURE = "script-plugin.structure" - private static final List ALLOWED_SERVICE_TYPES = ["NodeExecutor","FileCopier","ResourceModelSource","WorkflowNodeStep","RemoteScriptNodeStep","NodeExecutorFileCopier"] + private static final List ALLOWED_SERVICE_TYPES = ["NodeExecutor","FileCopier","ResourceModelSource","WorkflowNodeStep","RemoteScriptNodeStep","NodeExecutorFileCopier","Option"] @Override Map makeTemplateProperties(final String pluginName, final String providedService) { @@ -51,7 +51,7 @@ class ScriptPluginTemplateGenerator extends AbstractTemplateGenerator { @Override void preTemplateValidations(String providedService) { if(!ALLOWED_SERVICE_TYPES.contains(providedService)) { - throw new Exception("Script plugins do not support serivice: ${providedService}. Allowed types are: ${ALLOWED_SERVICE_TYPES.join(", ")}") + throw new Exception("Script plugin does not support this service: ${providedService}. Allowed types are: ${ALLOWED_SERVICE_TYPES.join(", ")}") } } @@ -73,6 +73,9 @@ class ScriptPluginTemplateGenerator extends AbstractTemplateGenerator { case "NodeExecutorFileCopier": path="nodeexecutor-filecopier" break + default: + path=providedService.toLowerCase() + break } diff --git a/src/main/groovy/com/rundeck/plugin/template/ServiceType.groovy b/src/main/groovy/com/rundeck/plugin/template/ServiceType.groovy new file mode 100644 index 0000000..db5a127 --- /dev/null +++ b/src/main/groovy/com/rundeck/plugin/template/ServiceType.groovy @@ -0,0 +1,16 @@ +package com.rundeck.plugin.template + +enum ServiceType { + ResourceModelSource, + Notification, + WorkflowStep, + WorkflowNodeStep, + LogFilter, + NodeExecutor, + Orchestrator, + FileCopier, + RemoteScriptNodeStep, + NodeExecutorFileCopier, + Option, + UI +} diff --git a/src/main/resources/templates/java-plugin/logfilter/build.gradle.template b/src/main/resources/templates/java-plugin/logfilter/build.gradle.template index 3119ea3..760cc19 100644 --- a/src/main/resources/templates/java-plugin/logfilter/build.gradle.template +++ b/src/main/resources/templates/java-plugin/logfilter/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' repositories { @@ -13,24 +19,51 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" testCompile "org.spockframework:spock-core:1.0-groovy-2.4" } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' +// task to copy plugin libs to output/lib dir +task copyToLib(type: Copy) { + into "\$buildDir/output/lib" + from configurations.pluginLibs +} + jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' - attributes 'Rundeck-Plugin-Tags': 'java,notification' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Tags': 'java,logfilter' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/java-plugin/nodeexecutor/build.gradle.template b/src/main/resources/templates/java-plugin/nodeexecutor/build.gradle.template index 131b5cd..67e4f7c 100644 --- a/src/main/resources/templates/java-plugin/nodeexecutor/build.gradle.template +++ b/src/main/resources/templates/java-plugin/nodeexecutor/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' repositories { @@ -13,8 +19,21 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" @@ -23,16 +42,30 @@ dependencies { testCompile group: 'org.objenesis', name: 'objenesis', version: '1.2' } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' +// task to copy plugin libs to output/lib dir +task copyToLib(type: Copy) { + into "\$buildDir/output/lib" + from configurations.pluginLibs +} + jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' - attributes 'Rundeck-Plugin-Tags': 'java,notification' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Tags': 'java,executor' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/java-plugin/notification/build.gradle.template b/src/main/resources/templates/java-plugin/notification/build.gradle.template index 3119ea3..7c0f203 100644 --- a/src/main/resources/templates/java-plugin/notification/build.gradle.template +++ b/src/main/resources/templates/java-plugin/notification/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' repositories { @@ -13,24 +19,45 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" testCompile "org.spockframework:spock-core:1.0-groovy-2.4" } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' attributes 'Rundeck-Plugin-Tags': 'java,notification' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/java-plugin/option/Plugin.java.template b/src/main/resources/templates/java-plugin/option/Plugin.java.template new file mode 100644 index 0000000..1d67154 --- /dev/null +++ b/src/main/resources/templates/java-plugin/option/Plugin.java.template @@ -0,0 +1,80 @@ +package com.plugin.${javaPluginClass.toLowerCase()}; + +import com.dtolabs.rundeck.core.plugins.Plugin; +import com.dtolabs.rundeck.core.plugins.configuration.Describable; +import com.dtolabs.rundeck.core.plugins.configuration.Description; +import com.dtolabs.rundeck.plugins.ServiceNameConstants; +import com.dtolabs.rundeck.plugins.descriptions.PluginDescription; +import com.dtolabs.rundeck.plugins.option.OptionValue; +import com.dtolabs.rundeck.plugins.option.OptionValuesPlugin; +import com.dtolabs.rundeck.plugins.util.DescriptionBuilder; +import com.dtolabs.rundeck.plugins.util.PropertyBuilder; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; + +@Plugin(service=ServiceNameConstants.OptionValues,name="${sanitizedPluginName}") +@PluginDescription(title="${pluginName}", description="My Option plugin description") +public class ${javaPluginClass} implements OptionValuesPlugin, Describable{ + + public static final String SERVICE_PROVIDER_NAME = "${sanitizedPluginName}"; + + /** + * Overriding this method gives the plugin a chance to take part in building the {@link + * com.dtolabs.rundeck.core.plugins.configuration.Description} presented by this plugin. This subclass can use the + * {@link DescriptionBuilder} to modify all aspects of the description, add or remove properties, etc. + */ + @Override + public Description getDescription() { + return DescriptionBuilder.builder() + .name(SERVICE_PROVIDER_NAME) + .title("${pluginName}") + .description("Example Workflow Step") + .property(PropertyBuilder.builder() + .string("example") + .title("Example String") + .description("Example description") + .required(false) + .build() + ) + .property(PropertyBuilder.builder() + .booleanType("exampleBoolean") + .title("Example Boolean") + .description("Example Boolean?") + .required(false) + .defaultValue("false") + .build() + ) + .build(); + } + + @Override + public List getOptionValues(final Map config) { + List options = new ArrayList<>(); + options.add(new StandardOptionValue("Alpha","alpha")); + options.add(new StandardOptionValue("Beta","beta")); + options.add(new StandardOptionValue("Gamma","gamma")); + return options; + } + + class StandardOptionValue implements OptionValue { + + private String name; + private String value; + StandardOptionValue(String name, String value) { + this.name = name; + this.value = value; + } + @Override + public String getName() { + return name; + } + + @Override + public String getValue() { + return value; + } + } + +} \ No newline at end of file diff --git a/src/main/resources/templates/java-plugin/option/PluginSpec.groovy.template b/src/main/resources/templates/java-plugin/option/PluginSpec.groovy.template new file mode 100644 index 0000000..dd22a17 --- /dev/null +++ b/src/main/resources/templates/java-plugin/option/PluginSpec.groovy.template @@ -0,0 +1,30 @@ +package com.plugin.${javaPluginClass.toLowerCase()} + +import com.dtolabs.rundeck.plugins.step.PluginStepContext +import com.dtolabs.rundeck.core.execution.workflow.steps.StepException +import com.dtolabs.rundeck.plugins.PluginLogger +import spock.lang.Specification + +class ${javaPluginClass}Spec extends Specification { + + def getContext(PluginLogger logger){ + Mock(PluginStepContext){ + getLogger()>>logger + } + } + + def "get options"(){ + given: + + def example = new ${javaPluginClass}() + def configuration = [example:"example123",exampleBoolean:"false",] + + when: + def options = example.getOptionValues(configuration) + + then: + options.size() > 0 + } + + +} \ No newline at end of file diff --git a/src/main/resources/templates/java-plugin/option/README.md.template b/src/main/resources/templates/java-plugin/option/README.md.template new file mode 100644 index 0000000..c125df2 --- /dev/null +++ b/src/main/resources/templates/java-plugin/option/README.md.template @@ -0,0 +1,4 @@ +# ${pluginName} Rundeck Plugin + +This is a option plugin. + diff --git a/src/main/resources/templates/java-plugin/option/build.gradle.template b/src/main/resources/templates/java-plugin/option/build.gradle.template new file mode 100644 index 0000000..7072701 --- /dev/null +++ b/src/main/resources/templates/java-plugin/option/build.gradle.template @@ -0,0 +1,68 @@ +plugins { + id 'groovy' + id 'java' +} + +version = '0.1.0' +defaultTasks 'clean','build' +sourceCompatibility = 1.8 +ext.rundeckPluginVersion= '2.0' +ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' + +repositories { + mavenLocal() + mavenCentral() +} + +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + +dependencies { + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' + + testCompile 'junit:junit:4.12' + testCompile "org.codehaus.groovy:groovy-all:2.4.15" + testCompile "org.spockframework:spock-core:1.0-groovy-2.4" +} + +// task to copy plugin libs to output/lib dir +task copyToLib(type: Copy) { + into "\$buildDir/output/lib" + from configurations.pluginLibs +} + +jar { + from "\$buildDir/output" + manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + + attributes 'Rundeck-Plugin-Classnames': pluginClassNames + attributes 'Rundeck-Plugin-File-Version': version + attributes 'Rundeck-Plugin-Name': '${pluginName}' + attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' + attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' + attributes 'Rundeck-Plugin-Tags': 'java,option' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + + } +} + +wrapper { + gradleVersion = '4.4.1' +} diff --git a/src/main/resources/templates/java-plugin/option/icon.png b/src/main/resources/templates/java-plugin/option/icon.png new file mode 100644 index 0000000..7aaf7e8 Binary files /dev/null and b/src/main/resources/templates/java-plugin/option/icon.png differ diff --git a/src/main/resources/templates/java-plugin/option/java-plugin.structure b/src/main/resources/templates/java-plugin/option/java-plugin.structure new file mode 100644 index 0000000..d0ce03e --- /dev/null +++ b/src/main/resources/templates/java-plugin/option/java-plugin.structure @@ -0,0 +1,6 @@ +build.gradle.template->build.gradle +README.md.template->README.md +icon.png->src/main/resources/resources/icon.png +Plugin.java.template->src/main/java/com/plugin/${javaPluginClass.toLowerCase()}/${javaPluginClass}.java +PluginSpec.groovy.template->src/test/groovy/com/plugin/${javaPluginClass.toLowerCase()}/${javaPluginClass}Spec.groovy + diff --git a/src/main/resources/templates/java-plugin/orchestrator/build.gradle.template b/src/main/resources/templates/java-plugin/orchestrator/build.gradle.template index 7e9afbb..01e54b1 100644 --- a/src/main/resources/templates/java-plugin/orchestrator/build.gradle.template +++ b/src/main/resources/templates/java-plugin/orchestrator/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' repositories { @@ -13,24 +19,45 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" testCompile "org.spockframework:spock-core:1.0-groovy-2.4" } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' attributes 'Rundeck-Plugin-Tags': 'java,orchestrator' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/java-plugin/resourcemodelsource/build.gradle.template b/src/main/resources/templates/java-plugin/resourcemodelsource/build.gradle.template index dbe5f90..7a2ca6c 100644 --- a/src/main/resources/templates/java-plugin/resourcemodelsource/build.gradle.template +++ b/src/main/resources/templates/java-plugin/resourcemodelsource/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}Factory' repositories { @@ -13,23 +19,45 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" testCompile "org.spockframework:spock-core:1.0-groovy-2.4" } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}Factory' jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' + attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' - attributes 'Rundeck-Plugin-Tags': 'java,notification' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Tags': 'java,resourceModel' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/java-plugin/workflownodestep/build.gradle.template b/src/main/resources/templates/java-plugin/workflownodestep/build.gradle.template index 3119ea3..b2146e4 100644 --- a/src/main/resources/templates/java-plugin/workflownodestep/build.gradle.template +++ b/src/main/resources/templates/java-plugin/workflownodestep/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' repositories { @@ -13,24 +19,45 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" testCompile "org.spockframework:spock-core:1.0-groovy-2.4" } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' - attributes 'Rundeck-Plugin-Tags': 'java,notification' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Tags': 'java,NodeStep' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/java-plugin/workflowstep/build.gradle.template b/src/main/resources/templates/java-plugin/workflowstep/build.gradle.template index 3119ea3..fa940c5 100644 --- a/src/main/resources/templates/java-plugin/workflowstep/build.gradle.template +++ b/src/main/resources/templates/java-plugin/workflowstep/build.gradle.template @@ -1,3 +1,8 @@ +plugins { + id 'groovy' + id 'java' +} + version = '0.1.0' defaultTasks 'clean','build' apply plugin: 'java' @@ -6,6 +11,7 @@ apply plugin: 'idea' sourceCompatibility = 1.8 ext.rundeckPluginVersion= '2.0' ext.rundeckVersion= '${rundeckVersion}' +ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' repositories { @@ -13,24 +19,45 @@ repositories { mavenCentral() } +configurations{ + //declare custom pluginLibs configuration to include only libs for this plugin + pluginLibs + + //declare compile to extend from pluginLibs so it inherits the dependencies + compile{ + extendsFrom pluginLibs + } +} + dependencies { - compile 'org.rundeck:rundeck-core:3.0.1+' + compile 'org.rundeck:rundeck-core:3.0.14+' + + //use pluginLibs to add dependecies, example: + //pluginLibs group: 'com.google.code.gson', name: 'gson', version: '2.8.2' testCompile 'junit:junit:4.12' testCompile "org.codehaus.groovy:groovy-all:2.4.15" testCompile "org.spockframework:spock-core:1.0-groovy-2.4" } -ext.pluginClassNames='com.plugin.${sanitizedPluginName}.${javaPluginClass}' jar { + from "\$buildDir/output" manifest { + def libList = configurations.pluginLibs.collect{'lib/'+it.name}.join(' ') + attributes 'Rundeck-Plugin-Classnames': pluginClassNames attributes 'Rundeck-Plugin-File-Version': version attributes 'Rundeck-Plugin-Name': '${pluginName}' attributes 'Rundeck-Plugin-Description': 'Provide a short description of your plugin here.' attributes 'Rundeck-Plugin-Rundeck-Compatibility-Version': '3.x' - attributes 'Rundeck-Plugin-Tags': 'java,notification' - attributes 'Rundeck-Plugin-Version': rundeckPluginVersion, 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Tags': 'java,WorkflowStep' + attributes 'Rundeck-Plugin-License': 'Apache 2.0' + attributes 'Rundeck-Plugin-Source-Link': 'Please put the link to your source repo here' + attributes 'Rundeck-Plugin-Target-Host-Compatibility': 'all' + attributes 'Rundeck-Plugin-Version': rundeckPluginVersion + attributes 'Rundeck-Plugin-Archive': 'true' + attributes 'Rundeck-Plugin-Libs': "\${libList}" + } } diff --git a/src/main/resources/templates/script-plugin/option/Makefile.template b/src/main/resources/templates/script-plugin/option/Makefile.template new file mode 100644 index 0000000..9bfae30 --- /dev/null +++ b/src/main/resources/templates/script-plugin/option/Makefile.template @@ -0,0 +1,11 @@ +all: install + +clean: + rm -rf build + +build: + mkdir -p build/libs build/zip-content/${sanitizedPluginName} + cp -r contents resources plugin.yaml build/zip-content/${sanitizedPluginName} + cd build/zip-content; zip -r ${sanitizedPluginName}.zip * + mv build/zip-content/${sanitizedPluginName}.zip build/libs + diff --git a/src/main/resources/templates/script-plugin/option/README.md.template b/src/main/resources/templates/script-plugin/option/README.md.template new file mode 100644 index 0000000..322765d --- /dev/null +++ b/src/main/resources/templates/script-plugin/option/README.md.template @@ -0,0 +1,22 @@ +# ${pluginName} Rundeck Plugin + +This is a ${providedService} plugin. + +## Build + +* Using gradle +``` +gradle clean build +``` + +* Using make + +``` +make clean build +``` + +## Install + +``` +cp build/libs/${sanitizedPluginName}.zip \$RDECK_BASE/libext +``` \ No newline at end of file diff --git a/src/main/resources/templates/script-plugin/option/build.gradle.template b/src/main/resources/templates/script-plugin/option/build.gradle.template new file mode 100644 index 0000000..fd9b92a --- /dev/null +++ b/src/main/resources/templates/script-plugin/option/build.gradle.template @@ -0,0 +1,21 @@ +buildscript { + repositories { + mavenCentral() + } +} +plugins { + id 'pl.allegro.tech.build.axion-release' version '1.7.0' +} + +ext.pluginName = '${pluginName}' +ext.pluginDescription = "Provide a short description of your plugin here" +ext.sopsCopyright = "© 2018, Rundeck, Inc." +ext.sopsUrl = "http://rundeck.com" +ext.buildDateString=new Date().format("yyyy-MM-dd'T'HH:mm:ssX") +ext.archivesBaseName = "${sanitizedPluginName}" +ext.pluginBaseFolder = "." + +project.version = "0.1.0-SNAPSHOT" +ext.archiveFilename = ext.archivesBaseName + '-' + version + +apply from: 'https://raw.githubusercontent.com/rundeck-plugins/build-zip/master/build.gradle' \ No newline at end of file diff --git a/src/main/resources/templates/script-plugin/option/icon.png b/src/main/resources/templates/script-plugin/option/icon.png new file mode 100644 index 0000000..7aaf7e8 Binary files /dev/null and b/src/main/resources/templates/script-plugin/option/icon.png differ diff --git a/src/main/resources/templates/script-plugin/option/option.template b/src/main/resources/templates/script-plugin/option/option.template new file mode 100644 index 0000000..51f937c --- /dev/null +++ b/src/main/resources/templates/script-plugin/option/option.template @@ -0,0 +1,7 @@ +#!/usr/bin/env bash +set -eu + +echo "==START_OPTIONS==" +echo "opt1:First Option" +echo "opt2:Second Option" +echo "==END_OPTIONS==" \ No newline at end of file diff --git a/src/main/resources/templates/script-plugin/option/plugin.yaml.template b/src/main/resources/templates/script-plugin/option/plugin.yaml.template new file mode 100644 index 0000000..692312c --- /dev/null +++ b/src/main/resources/templates/script-plugin/option/plugin.yaml.template @@ -0,0 +1,30 @@ +name: ${pluginName} +rundeckPluginVersion: 2.0 +author: Rundeck Dev +description: Describe your plugin here +rundeckCompatibilityVersion: 3.x +targetHostCompatibility: unix +license: Apache 2.0 +tags: + - option + - ${providedService} +date: ${currentDate} +version: 1.0.0 +providers: + - name: ${sanitizedPluginName} + service: OptionValues + title: ${pluginName} + description: The description of ${sanitizedPluginName} plugin + plugin-type: script + script-interpreter: /bin/bash + script-file: option + config: + - type: String + name: example + title: 'Example String' + description: 'Example String' + required: false + - type: Boolean + name: debug + title: Debug? + description: 'Write debug messages to stderr' \ No newline at end of file diff --git a/src/main/resources/templates/script-plugin/option/script-plugin.structure b/src/main/resources/templates/script-plugin/option/script-plugin.structure new file mode 100644 index 0000000..063a534 --- /dev/null +++ b/src/main/resources/templates/script-plugin/option/script-plugin.structure @@ -0,0 +1,7 @@ +plugin.yaml.template->plugin.yaml +option.template->contents/option +icon.png->resources/icon.png +README.md.template->README.md +build.gradle.template->build.gradle +Makefile.template->Makefile + diff --git a/src/test/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGeneratorTest.groovy b/src/test/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGeneratorTest.groovy index b0818c1..b2debb2 100644 --- a/src/test/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGeneratorTest.groovy +++ b/src/test/groovy/com/rundeck/plugin/generator/JavaPluginTemplateGeneratorTest.groovy @@ -117,4 +117,20 @@ class JavaPluginTemplateGeneratorTest extends Specification { new File(tmpDir,"/my-nodeexecutor-plugin/src/test/groovy/com/plugin/mynodeexecutorplugin/MyNodeexecutorPluginSpec.groovy").exists() } + def "Create Option Template"() { + when: + File tmpDir= File.createTempDir() + JavaPluginTemplateGenerator generator = new JavaPluginTemplateGenerator() + generator.createTemplate("My Option Plugin","Option",tmpDir.absolutePath) + int compileResult = TestUtils.buildGradle(new File(tmpDir,"my-option-plugin")) + + then: + compileResult == 0 + new File(tmpDir,"/my-option-plugin/build.gradle").exists() + new File(tmpDir,"/my-option-plugin/src/main/resources/resources/icon.png").exists() + new File(tmpDir,"/my-option-plugin/README.md").exists() + new File(tmpDir,"/my-option-plugin/src/main/java/com/plugin/myoptionplugin/MyOptionPlugin.java").exists() + new File(tmpDir,"/my-option-plugin/src/test/groovy/com/plugin/myoptionplugin/MyOptionPluginSpec.groovy").exists() + } + } diff --git a/src/test/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGeneratorTest.groovy b/src/test/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGeneratorTest.groovy index eecdeb2..887701d 100644 --- a/src/test/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGeneratorTest.groovy +++ b/src/test/groovy/com/rundeck/plugin/generator/ScriptPluginTemplateGeneratorTest.groovy @@ -34,4 +34,20 @@ class ScriptPluginTemplateGeneratorTest extends Specification { new File(destDir,"test-script-plugin/README.md").exists() } + + def "Generate Option Script Plugin From Template"() { + setup: + File destDir = File.createTempDir() + + when: + ScriptPluginTemplateGenerator generator = new ScriptPluginTemplateGenerator() + generator.createTemplate("Option Script Plugin","Option",destDir.absolutePath) + + then: + new File(destDir,"option-script-plugin/plugin.yaml").exists() + new File(destDir,"option-script-plugin/contents/option").exists() + new File(destDir,"option-script-plugin/resources/icon.png").exists() + new File(destDir,"option-script-plugin/README.md").exists() + + } }