Source code for renga.models.cwl.workflow

# -*- coding: utf-8 -*-
#
# Copyright 2018 - Swiss Data Science Center (SDSC)
# A partnership between École Polytechnique Fédérale de Lausanne (EPFL) and
# Eidgenössische Technische Hochschule Zürich (ETHZ).
#
# 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.
"""Represent workflows from the Common Workflow Language."""

import fnmatch
import uuid

import attr

from ._ascwl import CWLClass, mapped
from .parameter import WorkflowOutputParameter
from .process import Process


[docs]@attr.s class WorkflowStep(object): """Define an executable element of a workflow.""" run = attr.ib() # string, Process id = attr.ib(default=attr.Factory(uuid.uuid4)) in_ = attr.ib(default=None) out = attr.ib(default=None)
[docs]@attr.s class Workflow(Process, CWLClass): """Define a workflow representation.""" outputs = mapped(WorkflowOutputParameter) steps = mapped(WorkflowStep)
[docs] def add_step(self, **kwargs): """Add a workflow step.""" self.steps.append(WorkflowStep(**kwargs))
[docs] def get_output_id(self, path): """Return an id of the matching path from default values.""" for output in self.outputs: if output.type != 'File': continue if output.outputSource: step_id, _, source = output.outputSource.partition('/') for workflow_step in self.steps: if workflow_step.id == step_id: break else: continue if source not in workflow_step.out: continue # TODO load step and call get_output_id elif output.outputBinding: glob = output.outputBinding.glob # TODO better support for Expression if glob.startswith('$(inputs.'): input_id = glob[len('$(inputs.'):-1] for input_ in self.inputs: if input_.id == input_id and input_.default == path: return output.id elif fnmatch.fnmatch(path, glob): return output.id