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

nextflow.executor.MoabExecutor.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.executor

import groovy.xml.XmlSlurper

import java.nio.file.Path

import groovy.transform.CompileDynamic
import groovy.transform.CompileStatic
import groovy.util.logging.Slf4j
import nextflow.processor.TaskRun
/**
 * Implements a executor for Moab batch scheduler cluster
 *
 * http://www.adaptivecomputing.com/moab-hpc-basic-edition/
 *
 * @author Paolo Di Tommaso 
 */
@Slf4j
@CompileStatic
class MoabExecutor extends AbstractGridExecutor {

    /**
     * Gets the directives to submit the specified task to the cluster for execution
     *
     * See http://docs.adaptivecomputing.com/maui/commands/msub.php
     *
     * @param task A {@link TaskRun} to be submitted
     * @param result The {@link List} instance to which add the job directives
     * @return A {@link List} containing all directive tokens and values.
     */
    protected List getDirectives( TaskRun task, List result ) {
        assert result !=null

        result << '-N' << getJobNameFor(task)
        result << '-o' << quote(task.workDir.resolve(TaskRun.CMD_LOG))
        result << '-j' << 'oe'

        // the requested queue name
        if( task.config.queue ) {
            result << '-q'  << (String)task.config.queue
        }

        if( task.config.getCpus() > 1 ) {
            result << '-l' << "nodes=1:ppn=${task.config.getCpus()}".toString()
        }

        // max task duration
        if( task.config.getTime() ) {
            final duration = task.config.getTime()
            result << "-l" << "walltime=${duration.format('HH:mm:ss')}".toString()
        }

        // task max memory
        if( task.config.getMemory() ) {
            // https://www.osc.edu/documentation/knowledge_base/out_of_memory_oom_or_excessive_memory_usage
            result << "-l" << "mem=${task.config.getMemory().toString().replaceAll(/[\s]/,'').toLowerCase()}".toString()
        }

        // -- at the end append the command script wrapped file name
        addClusterOptionsDirective(task.config, result)

        return result
    }

    /**
     * The command line to submit this job
     *
     * @param task The {@link TaskRun} instance to submit for execution to the cluster
     * @param scriptFile The file containing the job launcher script
     * @return A list representing the submit command line
     */
    List getSubmitCommandLine(TaskRun task, Path scriptFile ) {
        def cmd = new ArrayList(5)
        cmd << 'msub'
        cmd << '--xml'
        cmd << scriptFile.name
        return cmd
    }

    protected String getHeaderToken() { '#MSUB' }

    /**
     * Parse the string returned by the {@code qsub} command and extract the job ID string
     *
     * @param text The string returned when submitting the job
     * @return The actual job ID string
     */
    @Override
    @CompileDynamic
    def parseJobId( String text ) {
        String result
        try {
            result = new XmlSlurper()
                            .parseText(text)
                            .job[0]
                            .@JobID
                            .toString()
        }
        catch (Exception e) {
            throw new IllegalArgumentException("Invalid Moab submit response:\n$text\n\n", e)
        }

        if( !result )
            throw new IllegalArgumentException("Missing Moab submit job ID:\n$text\n\n")

        return result
    }

    @Override
    protected List getKillCommand() { ['mjobctl', '-c'] }

    @Override
    protected List queueStatusCommand(Object queue) {
        return ['showq', '--xml', '-w', "user="+System.getProperty('user.name')]
    }

    static private Map DECODE_STATUS = [
            'Running': QueueStatus.RUNNING,
            'Starting': QueueStatus.RUNNING,
            'Idle': QueueStatus.HOLD,
            'UserHold': QueueStatus.HOLD,
            'SystemHold': QueueStatus.HOLD,
            'Deferred': QueueStatus.HOLD,
            'NotQueued': QueueStatus.ERROR,
            'BatchHold': QueueStatus.ERROR,
    ]

    protected QueueStatus decode(String status) {
        DECODE_STATUS.get(status)
    }

    @Override
    @CompileDynamic
    protected Map parseQueueStatus(String xmlStatus) {
        // parse XML string and and decode job states
        final result = new LinkedHashMap()
        try {
            def data = new XmlSlurper().parseText(xmlStatus)
            for( def queue : data.queue ) {
                if( !queue.@count?.toInteger() )
                    continue

                def state = queue.@option?.toString()
                if( state=="active")
                    parseQueueJobNodes(queue, result, QueueStatus.RUNNING )
                else if( state=="eligible" )
                    parseQueueJobNodes(queue, result, QueueStatus.PENDING )
                else
                    parseQueueJobNodes(queue, result )
            }
        }
        catch (Exception e) {
            throw e
        }
        return result
    }

    @CompileDynamic
    protected void parseQueueJobNodes(queueNode, Map result, QueueStatus state=null) {
        for( def entry : queueNode.job ) {
            final value = state ?: decode(entry.@State?.toString())
            result.put( entry.@JobID?.toString(), value )
        }
    }


    protected List killTaskCommand(def jobId) {
        final result = getKillCommand()
        if( jobId instanceof Collection ) {
            result.add( jobId.join(',') )
            log.trace "Kill command: ${result}"
        }
        else {
            result.add(jobId.toString())
        }
        return result
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy