All Downloads are FREE. Search and download functionalities are using the official Maven repository.

nextflow.cli.CmdSecret.groovy Maven / Gradle / Ivy

Go to download

A DSL modelled around the UNIX pipe concept, that simplifies writing parallel and scalable pipelines in a portable manner

There is a newer version: 24.11.0-edge
Show newest version
/*
 * Copyright 2013-2024, Seqera Labs
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 */

package nextflow.cli

import com.beust.jcommander.Parameter
import com.beust.jcommander.Parameters
import groovy.transform.CompileStatic
import groovy.util.logging.Slf4j
import nextflow.exception.AbortOperationException
import nextflow.plugin.Plugins
import nextflow.secret.SecretsLoader
import nextflow.secret.SecretsProvider
/**
 * Implements the {@code secret} command
 *
 * @author Paolo Di Tommaso 
 */
@Slf4j
@CompileStatic
@Parameters(commandDescription = "Manage pipeline secrets")
class CmdSecret extends CmdBase implements UsageAware {

    interface SubCmd {
        String getName()
        void apply(List result)
        void usage(List result)
    }

    static public final String NAME = 'secrets'

    private List commands = []

    String getName() {
        return NAME
    }

    @Parameter(hidden = true)
    List args

    private SecretsProvider provider

    CmdSecret() {
        commands.add( new GetCmd() )
        commands.add( new SetCmd() )
        commands.add( new ListCmd() )
        commands.add( new DeleteCmd() )
    }

    /**
     * Print the command usage help
     */
    void usage() {
        usage(args)
    }

    /**
     * Print the command usage help
     *
     * @param args The arguments as entered by the user
     */
    void usage(List args) {

        List result = []
        if( !args ) {
            result << this.getClass().getAnnotation(Parameters).commandDescription()
            result << 'Usage: nextflow secrets  [options]'
            result << ''
            result << 'Commands:'
            commands.collect{ it.name }.sort().each { result << "  $it".toString()  }
            result << ''
        }
        else {
            def sub = commands.find { it.name == args[0] }
            if( sub )
                sub.usage(result)
            else {
                throw new AbortOperationException("Unknown secrets sub-command: ${args[0]}")
            }
        }

        println result.join('\n').toString()
    }

    /**
     * Main command entry point
     */
    @Override
    void run() {
        if( !args ) {
            usage()
            return
        }

        // setup the plugins system and load the secrets provider
        Plugins.init()
        provider = SecretsLoader.instance.load()

        // run the command
        try {
            getCmd(args).apply(args.drop(1))
        }
        finally {
            // close the provider
            provider?.close()
        }
    }

    protected SubCmd getCmd(List args) {

        def cmd = commands.find { it.name == args[0] }
        if( cmd ) {
            return cmd
        }

        def matches = commands.collect{ it.name }.closest(args[0])
        def msg = "Unknown cloud sub-command: ${args[0]}"
        if( matches )
            msg += " -- Did you mean one of these?\n" + matches.collect { "  $it"}.join('\n')
        throw new AbortOperationException(msg)
    }

    private void addOption(String fieldName, List result) {
        def annot = this.class.getDeclaredField(fieldName)?.getAnnotation(Parameter)
        if( annot ) {
            result << '  ' + annot.names().join(', ')
            result << '     ' + annot.description()
        }
        else {
            log.debug "Unknown help field: $fieldName"
        }
    }

    class SetCmd implements SubCmd {

        @Override
        String getName() { 'set' }

        @Override
        void apply(List result) {
            if( result.size() < 1 )
                throw new AbortOperationException("Missing secret name")
            if( result.size() < 2 )
                throw new AbortOperationException("Missing secret value")

            String secretName = result.first()
            String secretValue = result.last()
            provider.putSecret(secretName, secretValue)
        }

        @Override
        void usage(List result) {
            result << 'Set a key-pair in the secrets store'
            result << "Usage: nextflow secrets $name  ".toString()
            result << ''
            result << ''
        }
    }

    class GetCmd implements SubCmd {

        @Override
        String getName() { 'get' }

        @Override
        void apply(List result) {
            if( result.size() != 1 )
                throw new AbortOperationException("Wrong number of arguments")

            String secretName = result.first()
            if( !secretName )
                throw new AbortOperationException("Missing secret name")
            println provider.getSecret(secretName)?.value
        }

        @Override
        void usage(List result) {
            result << 'Get a secret value with the name'
            result << "Usage: nextflow secrets $name ".toString()
            result << ''
        }
    }

    /**
     * Implements the secret `list` sub-command
     */
    class ListCmd implements SubCmd {
        @Override
        String getName() { 'list' }

        @Override
        void apply(List result) {
            if( result.size()  )
                throw new AbortOperationException("Wrong number of arguments")

            final names = new ArrayList(provider.listSecretsNames()).sort()
            if( names ) {
                for( String it : names ) {
                    println it
                }
            }
            else {
                println "no secrets available"
            }
        }

        @Override
        void usage(List result) {
            result << 'List all names in the secrets store'
            result << "Usage: nextflow secrets $name".toString()
            result << ''
        }
    }

    /**
     * Implements the secret `remove` sub-command
     */
    class DeleteCmd implements SubCmd {
        @Override
        String getName() { 'delete' }

        @Override
        void apply(List result) {
            if( result.size() != 1 )
                throw new AbortOperationException("Wrong number of arguments")

            String secretName = result.first()

            if( !secretName )
                throw new AbortOperationException("Missing secret name")
            provider.removeSecret(secretName)
        }

        @Override
        void usage(List result) {
            result << 'Delete an entry from the secrets store'
            result << "Usage: nextflow secrets $name".toString()
            result << ''
            addOption('secretName', result)
            result << ''
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy