Skip to content

Commit

Permalink
Merge branch 'main' into fix-tests-pydot
Browse files Browse the repository at this point in the history
  • Loading branch information
lkk7 authored Aug 19, 2024
2 parents fcab10e + 73b742f commit c5ce229
Show file tree
Hide file tree
Showing 130 changed files with 12,872 additions and 374 deletions.
2 changes: 2 additions & 0 deletions .github/workflows/ci-tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -173,6 +173,8 @@ jobs:
- cwl-version: v1.2
container: docker
extras: "--fast-parser"
- cwl-version: v1.3.0-dev1
extras: "--relax-path-checks"

steps:
- uses: actions/checkout@v4
Expand Down
5 changes: 5 additions & 0 deletions MANIFEST.in
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ recursive-include mypy-stubs *.pyi *.py
include tests/*
include tests/cwl-conformance/cwltool-conftest.py
include tests/loop/*
include tests/loop-ext/*
include tests/tmp1/tmp2/tmp3/.gitkeep
include tests/tmp4/alpha/*
include tests/wf/*
Expand Down Expand Up @@ -54,6 +55,10 @@ include cwltool/schemas/v1.2/*.yml
include cwltool/schemas/v1.2/*.md
include cwltool/schemas/v1.2/salad/schema_salad/metaschema/*.yml
include cwltool/schemas/v1.2/salad/schema_salad/metaschema/*.md
include cwltool/schemas/v1.3.0-dev1/*.yml
include cwltool/schemas/v1.3.0-dev1/*.md
include cwltool/schemas/v1.3.0-dev1/salad/schema_salad/metaschema/*.yml
include cwltool/schemas/v1.3.0-dev1/salad/schema_salad/metaschema/*.md
include cwltool/extensions.yml
include cwltool/extensions-v1.1.yml
include cwltool/extensions-v1.2.yml
Expand Down
6 changes: 3 additions & 3 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -160,11 +160,11 @@ diff-cover.html: coverage.xml

## test : run the cwltool test suite
test: $(PYSOURCES)
python3 -m pytest -rs ${PYTEST_EXTRA}
python3 -m pytest -rsfE ${PYTEST_EXTRA}

## testcov : run the cwltool test suite and collect coverage
testcov: $(PYSOURCES)
python3 -m pytest -rs --cov --cov-config=.coveragerc --cov-report= ${PYTEST_EXTRA}
python3 -m pytest -rsfE --cov --cov-config=.coveragerc --cov-report= ${PYTEST_EXTRA}

sloccount.sc: $(PYSOURCES) Makefile
sloccount --duplicates --wide --details $^ > $@
Expand All @@ -183,7 +183,7 @@ mypy: $(PYSOURCES)

mypyc: $(PYSOURCES)
MYPYPATH=mypy-stubs CWLTOOL_USE_MYPYC=1 pip install --verbose -e . \
&& pytest -rs -vv ${PYTEST_EXTRA}
&& pytest -rsfE -vv ${PYTEST_EXTRA}

shellcheck: FORCE
shellcheck build-cwltool-docker.sh cwl-docker.sh release-test.sh conformance-test.sh \
Expand Down
2 changes: 1 addition & 1 deletion README.rst
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
#############################################################################################
``cwltool``: The reference reference implementation of the Common Workflow Language standards
``cwltool``: The reference implementation of the Common Workflow Language standards
#############################################################################################

|Linux Status| |Coverage Status| |Docs Status|
Expand Down
4 changes: 2 additions & 2 deletions conformance-test.sh
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ venv() {
# VERSION=v1.2 GIT_TARGET=main CONTAINER=podman ./conformance_test.sh

# Version of the standard to test against
# Current options: v1.0, v1.1, v1.2
# Current options: v1.0, v1.1, v1.2, v1.3.0-dev1
VERSION=${VERSION:-"v1.2"}

# Which commit of the standard's repo to use
Expand Down Expand Up @@ -121,7 +121,7 @@ if (( "${#exclusions[*]}" > 0 )); then
fi

# Build command
TEST_COMMAND="python -m pytest ${CONFORMANCE_TEST} -n logical --dist worksteal -rs --junit-xml=${TMP_DIR}/cwltool_conf_${VERSION}_${GIT_TARGET}_${CONTAINER}.xml -o junit_suite_name=cwltool_$(echo "${CWLTOOL_OPTIONS}" | tr "[:blank:]-" _)"
TEST_COMMAND="python -m pytest ${CONFORMANCE_TEST} -n logical --dist worksteal -rsfE --junit-xml=${TMP_DIR}/cwltool_conf_${VERSION}_${GIT_TARGET}_${CONTAINER}.xml -o junit_suite_name=cwltool_$(echo "${CWLTOOL_OPTIONS}" | tr "[:blank:]-" _)"
if [[ -n "${EXCLUDE}" ]] ; then
TEST_COMMAND="${TEST_COMMAND} --cwl-exclude ${EXCLUDE}"
fi
Expand Down
27 changes: 9 additions & 18 deletions cwltool/checker.py
Original file line number Diff line number Diff line change
Expand Up @@ -517,42 +517,33 @@ def is_conditional_step(param_to_step: Dict[str, CWLObjectType], parm_id: str) -


def is_all_output_method_loop_step(param_to_step: Dict[str, CWLObjectType], parm_id: str) -> bool:
"""Check if a step contains a http://commonwl.org/cwltool#Loop requirement with `all` outputMethod."""
"""Check if a step contains a `loop` directive with `all` outputMethod."""
source_step: Optional[MutableMapping[str, Any]] = param_to_step.get(parm_id)
if source_step is not None:
for requirement in source_step.get("requirements", []):
if (
requirement["class"] == "http://commonwl.org/cwltool#Loop"
and requirement.get("outputMethod") == "all"
):
return True
if source_step.get("loop") is not None and source_step.get("outputMethod") == "all":
return True
return False


def loop_checker(steps: Iterator[MutableMapping[str, Any]]) -> None:
"""
Check http://commonwl.org/cwltool#Loop requirement compatibility with other directives.
Check `loop` compatibility with other directives.
:raises ValidationException: If there is an incompatible combination between
cwltool:loop and 'scatter' or 'when'.
:raises ValidationException: If there is an incompatible combination between `loop` and `scatter`.
"""
exceptions = []
for step in steps:
requirements = {
**{h["class"]: h for h in step.get("hints", [])},
**{r["class"]: r for r in step.get("requirements", [])},
}
if "http://commonwl.org/cwltool#Loop" in requirements:
if "when" in step:
if "loop" in step:
if "when" not in step:
exceptions.append(
SourceLine(step, "id").makeError(
"The `cwltool:Loop` clause is not compatible with the `when` directive."
"The `when` clause is mandatory when the `loop` directive is defined."
)
)
if "scatter" in step:
exceptions.append(
SourceLine(step, "id").makeError(
"The `cwltool:Loop` clause is not compatible with the `scatter` directive."
"The `loop` clause is not compatible with the `scatter` directive."
)
)
if exceptions:
Expand Down
2 changes: 1 addition & 1 deletion cwltool/command_line_tool.py
Original file line number Diff line number Diff line change
Expand Up @@ -806,7 +806,7 @@ def remove_dirname(d: CWLObjectType) -> None:
def job(
self,
job_order: CWLObjectType,
output_callbacks: Optional[OutputCallbackType],
output_callbacks: OutputCallbackType,
runtimeContext: RuntimeContext,
) -> Generator[Union[JobBase, CallbackJob], None, None]:
workReuse, _ = self.get_requirement("WorkReuse")
Expand Down
20 changes: 14 additions & 6 deletions cwltool/errors.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,17 @@
class WorkflowException(Exception):
pass
"""Common errors.
WorkflowException and GraphTargetMissingException are aliased to
equivalent errors from cwl_utils.errors and re-exported by this module
to avoid breaking the interface for other code.
"""

# flake8: noqa: F401

from cwl_utils.errors import WorkflowException as WorkflowException


from cwl_utils.errors import GraphTargetMissingException as GraphTargetMissingException


class UnsupportedRequirement(WorkflowException):
Expand All @@ -8,7 +20,3 @@ class UnsupportedRequirement(WorkflowException):

class ArgumentException(Exception):
"""Mismatched command line arguments provided."""


class GraphTargetMissingException(WorkflowException):
"""When a ``$graph`` is encountered and there is no target and no ``main``/``#main``."""
121 changes: 121 additions & 0 deletions cwltool/extensions-v1.3.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
$base: http://commonwl.org/cwltool#
$namespaces:
cwl: "https://w3id.org/cwl/cwl#"
$graph:
- $import: https://w3id.org/cwl/CommonWorkflowLanguage.yml

- name: Secrets
type: record
inVocab: false
extends: cwl:ProcessRequirement
fields:
class:
type: string
doc: "Always 'Secrets'"
jsonldPredicate:
"_id": "@type"
"_type": "@vocab"
secrets:
type: string[]
doc: |
List one or more input parameters that are sensitive (such as passwords)
which will be deliberately obscured from logging.
jsonldPredicate:
"_type": "@id"
refScope: 0


- name: ProcessGenerator
type: record
inVocab: true
extends: cwl:Process
documentRoot: true
fields:
- name: class
jsonldPredicate:
"_id": "@type"
"_type": "@vocab"
type: string
- name: run
type: [string, cwl:Process]
jsonldPredicate:
_id: "cwl:run"
_type: "@id"
subscope: run
doc: |
Specifies the process to run.
- name: MPIRequirement
type: record
inVocab: false
extends: cwl:ProcessRequirement
doc: |
Indicates that a process requires an MPI runtime.
fields:
- name: class
type: string
doc: "Always 'MPIRequirement'"
jsonldPredicate:
"_id": "@type"
"_type": "@vocab"
- name: processes
type: [int, cwl:Expression]
doc: |
The number of MPI processes to start. If you give a string,
this will be evaluated as a CWL Expression and it must
evaluate to an integer.
- name: CUDARequirement
type: record
extends: cwl:ProcessRequirement
inVocab: false
doc: |
Require support for NVIDA CUDA (GPU hardware acceleration).
fields:
class:
type: string
doc: 'cwltool:CUDARequirement'
jsonldPredicate:
_id: "@type"
_type: "@vocab"
cudaVersionMin:
type: string
doc: |
Minimum CUDA version to run the software, in X.Y format. This
corresponds to a CUDA SDK release. When running directly on
the host (not in a container) the host must have a compatible
CUDA SDK (matching the exact version, or, starting with CUDA
11.3, matching major version). When run in a container, the
container image should provide the CUDA runtime, and the host
driver is injected into the container. In this case, because
CUDA drivers are backwards compatible, it is possible to
use an older SDK with a newer driver across major versions.
See https://docs.nvidia.com/deploy/cuda-compatibility/ for
details.
cudaComputeCapability:
type:
- 'string'
- 'string[]'
doc: |
CUDA hardware capability required to run the software, in X.Y
format.
* If this is a single value, it defines only the minimum
compute capability. GPUs with higher capability are also
accepted.
* If it is an array value, then only select GPUs with compute
capabilities that explicitly appear in the array.
cudaDeviceCountMin:
type: ['null', int, cwl:Expression]
default: 1
doc: |
Minimum number of GPU devices to request. If not specified,
same as `cudaDeviceCountMax`. If neither are specified,
default 1.
cudaDeviceCountMax:
type: ['null', int, cwl:Expression]
doc: |
Maximum number of GPU devices to request. If not specified,
same as `cudaDeviceCountMin`.
7 changes: 2 additions & 5 deletions cwltool/process.py
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,7 @@ def filter(self, record: logging.LogRecord) -> bool:
]

cwl_files = (
"Base.yml",
"Workflow.yml",
"CommandLineTool.yml",
"CommonWorkflowLanguage.yml",
Expand Down Expand Up @@ -1046,10 +1047,6 @@ def validate_hints(self, avsc_names: Names, hints: List[CWLObjectType], strict:
sl = SourceLine(hints, i, ValidationException, debug)
with sl:
classname = cast(str, r["class"])
if classname == "http://commonwl.org/cwltool#Loop":
raise ValidationException(
"http://commonwl.org/cwltool#Loop is valid only under requirements."
)
avroname = classname
if classname in self.doc_loader.vocab:
avroname = avro_type_name(self.doc_loader.vocab[classname])
Expand Down Expand Up @@ -1078,7 +1075,7 @@ def visit(self, op: Callable[[CommentedMap], None]) -> None:
def job(
self,
job_order: CWLObjectType,
output_callbacks: Optional[OutputCallbackType],
output_callbacks: OutputCallbackType,
runtimeContext: RuntimeContext,
) -> JobsGeneratorType:
pass
Expand Down
6 changes: 3 additions & 3 deletions cwltool/procgenerator.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ def receive_output(self, jobout: Optional[CWLObjectType], processStatus: str) ->
def job(
self,
job_order: CWLObjectType,
output_callbacks: Optional[OutputCallbackType],
output_callbacks: OutputCallbackType,
runtimeContext: RuntimeContext,
) -> JobsGeneratorType:
try:
Expand All @@ -41,7 +41,7 @@ def job(
while self.processStatus is None:
yield None

if self.processStatus != "success" and output_callbacks:
if self.processStatus != "success":
output_callbacks(self.jobout, self.processStatus)
return

Expand Down Expand Up @@ -89,7 +89,7 @@ def __init__(
def job(
self,
job_order: CWLObjectType,
output_callbacks: Optional[OutputCallbackType],
output_callbacks: OutputCallbackType,
runtimeContext: RuntimeContext,
) -> JobsGeneratorType:
return ProcessGeneratorJob(self).job(job_order, output_callbacks, runtimeContext)
Expand Down
Loading

0 comments on commit c5ce229

Please sign in to comment.