Skip to content

Commit

Permalink
Allow describing wdl with zipped imports.
Browse files Browse the repository at this point in the history
  • Loading branch information
kshakir committed May 9, 2023
1 parent b1ec16a commit ca14a65
Show file tree
Hide file tree
Showing 18 changed files with 203 additions and 22 deletions.
10 changes: 10 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,15 @@
# Cromwell Change Log

## 86 Release Notes

### `/describe` endpoint support for `workflowDependencies`

Previously it was not possible to describe a `workflowSource` that required `workflowDependencies` as the `/describe`
endpoint did not allow specifying the additional zip file.

Now the endpoint will allow the user to upload the `workflowDependencies` zip file, will validate the top level
workflow plus the dependencies, then return the appropriate response including the defined workflow inputs and outputs.

## 85 Release Notes

### Migration of PKs to BIGINT
Expand Down
5 changes: 5 additions & 0 deletions CromIAM/src/main/resources/swagger/cromiam.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -838,6 +838,11 @@ paths:
required: false
type: file
in: formData
- name: workflowDependencies
description: ZIP file containing workflow source files that are used to resolve local imports. This zip bundle will be unpacked in a sandbox accessible to this workflow.
required: false
type: file
in: formData
- $ref: '#/parameters/workflowTypeParam'
- $ref: '#/parameters/workflowTypeVersionParam'
responses:
Expand Down
3 changes: 1 addition & 2 deletions centaur/src/main/scala/centaur/test/Test.scala
Original file line number Diff line number Diff line change
Expand Up @@ -228,8 +228,7 @@ object Operations extends StrictLogging {
override def run: IO[Unit] = {


// We can't describe workflows based on zipped imports, so don't try:
if (workflow.skipDescribeEndpointValidation || workflow.data.zippedImports.nonEmpty) {
if (workflow.skipDescribeEndpointValidation) {
IO.pure(())
} else {
checkDescriptionInner(0)
Expand Down
3 changes: 2 additions & 1 deletion centaur/src/main/scala/centaur/test/workflow/Workflow.scala
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,8 @@ final case class Workflow private(testName: String,
workflowUrl = data.workflowUrl,
workflowType = data.workflowType,
workflowTypeVersion = data.workflowTypeVersion,
inputsJson = data.inputs.map(_.unsafeRunSync())
inputsJson = data.inputs.map(_.unsafeRunSync()),
zippedImports = data.zippedImports,
)

def secondRun: Workflow = {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -261,7 +261,13 @@ object CromwellClient {
Multipart.FormData.BodyPart(name, HttpEntity(MediaTypes.`application/json`, ByteString(source)))
}

val multipartFormData = Multipart.FormData(sourceBodyParts.toSeq : _*)
val zipBodyParts = Map(
"workflowDependencies" -> describeRequest.zippedImports
) collect {
case (name, Some(file)) => Multipart.FormData.BodyPart.fromPath(name, MediaTypes.`application/zip`, file.path)
}

val multipartFormData = Multipart.FormData((sourceBodyParts ++ zipBodyParts).toSeq : _*)
multipartFormData.toEntity()
}

Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,11 @@
package cromwell.api.model

import better.files.File

final case class WorkflowDescribeRequest(workflowSource: Option[String],
workflowUrl: Option[String],
workflowType: Option[String],
workflowTypeVersion: Option[String],
inputsJson: Option[String])
inputsJson: Option[String],
zippedImports: Option[File],
)
3 changes: 2 additions & 1 deletion docs/api/RESTAPI.md

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 5 additions & 0 deletions engine/src/main/resources/swagger/cromwell.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -642,6 +642,11 @@ paths:
required: false
type: file
in: formData
- name: workflowDependencies
description: ZIP file containing workflow source files that are used to resolve local imports. This zip bundle will be unpacked in a sandbox accessible to this workflow.
required: false
type: file
in: formData
- $ref: '#/parameters/workflowTypeParam'
- $ref: '#/parameters/workflowTypeVersionParam'
responses:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ trait WomtoolRouteSupport extends WebServiceUtils {
val workflowInputs = data.get("workflowInputs").map(_.utf8String)
val workflowType = data.get("workflowType").map(_.utf8String)
val workflowVersion = data.get("workflowTypeVersion").map(_.utf8String)
val workflowDependencies = data.get("workflowDependencies").map(_.toArray)

val wsfc = WorkflowSourceFilesCollection(
workflowSource,
Expand All @@ -55,7 +56,7 @@ trait WomtoolRouteSupport extends WebServiceUtils {
workflowInputs.getOrElse(""),
workflowOptions = WorkflowOptions.empty,
labelsJson = "",
importsFile = None,
importsFile = workflowDependencies,
workflowOnHold = false,
warnings = Seq.empty,
requestedWorkflowId = None
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -670,7 +670,12 @@ object CromwellApiServiceSpec {
s"[reading back DescribeRequest contents] workflow url: ${sourceFiles.workflowUrl}",
s"[reading back DescribeRequest contents] inputs: ${sourceFiles.inputsJson}",
s"[reading back DescribeRequest contents] type: ${sourceFiles.workflowType}",
s"[reading back DescribeRequest contents] version: ${sourceFiles.workflowTypeVersion}"
s"[reading back DescribeRequest contents] version: ${sourceFiles.workflowTypeVersion}",
s"[reading back DescribeRequest contents] dependencies: ${
sourceFiles.importsZipFileOption.map(bytes =>
bytes.map(b => "0x%02X".format(b)).mkString("[", ", ", "]")
)
}",
)

sender() ! DescribeSuccess(description = WorkflowDescription(valid = true, errors = readBack, validWorkflow = true))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,11 @@ class WomtoolRouteSupportSpec extends AsyncFlatSpec with ScalatestRouteTest with
val workflowInputs = Multipart.FormData.BodyPart("workflowInputs", HttpEntity(MediaTypes.`application/json`, "{\"a\":\"is for apple\"}"))
val workflowType = Multipart.FormData.BodyPart("workflowType", HttpEntity(ContentTypes.`text/plain(UTF-8)`, "WDL"))
val workflowVersion = Multipart.FormData.BodyPart("workflowTypeVersion", HttpEntity(ContentTypes.`text/plain(UTF-8)`, "1.0"))
val workflowDependencies =
Multipart.FormData.BodyPart(
"workflowDependencies",
HttpEntity(MediaTypes.`application/zip`, Array[Byte](0x0A, 0x0B, 0x0C)),
)

val workflowSourceTriggerDescribeFailure =
Multipart.FormData.BodyPart("workflowSource", HttpEntity(ContentTypes.`text/plain(UTF-8)`, "fail to describe"))
Expand Down Expand Up @@ -82,7 +87,8 @@ class WomtoolRouteSupportSpec extends AsyncFlatSpec with ScalatestRouteTest with
"[reading back DescribeRequest contents] workflow url: None",
"[reading back DescribeRequest contents] inputs: ",
"[reading back DescribeRequest contents] type: None",
"[reading back DescribeRequest contents] version: None"
"[reading back DescribeRequest contents] version: None",
"[reading back DescribeRequest contents] dependencies: None",
),
validWorkflow = true
)
Expand All @@ -104,7 +110,8 @@ class WomtoolRouteSupportSpec extends AsyncFlatSpec with ScalatestRouteTest with
"[reading back DescribeRequest contents] workflow url: Some(https://raw.githubusercontent.com/broadinstitute/cromwell/develop/womtool/src/test/resources/validate/wdl_draft3/valid/callable_imports/my_workflow.wdl)",
"[reading back DescribeRequest contents] inputs: ",
"[reading back DescribeRequest contents] type: None",
"[reading back DescribeRequest contents] version: None"
"[reading back DescribeRequest contents] version: None",
"[reading back DescribeRequest contents] dependencies: None",
),
validWorkflow = true
)
Expand All @@ -126,7 +133,40 @@ class WomtoolRouteSupportSpec extends AsyncFlatSpec with ScalatestRouteTest with
"[reading back DescribeRequest contents] workflow url: None",
"[reading back DescribeRequest contents] inputs: {\"a\":\"is for apple\"}",
"[reading back DescribeRequest contents] type: Some(WDL)",
"[reading back DescribeRequest contents] version: Some(1.0)"
"[reading back DescribeRequest contents] version: Some(1.0)",
"[reading back DescribeRequest contents] dependencies: None",
),
validWorkflow = true
)
} { responseAs[WorkflowDescription] }
}
}

it should "include inputs, workflow type, workflow version, and workflow dependencies in the WorkflowSourceFilesCollection" in {
Post(
s"/womtool/$version/describe",
Multipart.FormData(
BodyParts.workflowSource,
BodyParts.workflowInputs,
BodyParts.workflowType,
BodyParts.workflowVersion,
BodyParts.workflowDependencies,
).toEntity()
) ~>
akkaHttpService.womtoolRoutes ~>
check {
status should be(StatusCodes.OK)

assertResult {
WorkflowDescription(valid = true,
errors = List(
"this is fake data from the mock SR actor",
"[reading back DescribeRequest contents] workflow hashcode: Some(580529622)",
"[reading back DescribeRequest contents] workflow url: None",
"[reading back DescribeRequest contents] inputs: {\"a\":\"is for apple\"}",
"[reading back DescribeRequest contents] type: Some(WDL)",
"[reading back DescribeRequest contents] version: Some(1.0)",
"[reading back DescribeRequest contents] dependencies: Some([0x0A, 0x0B, 0x0C])",
),
validWorkflow = true
)
Expand Down
19 changes: 17 additions & 2 deletions services/src/main/scala/cromwell/services/womtool/Describer.scala
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
package cromwell.services.womtool

import cats.data.Validated.{Invalid, Valid}
import cromwell.core.WorkflowSourceFilesCollection
import cats.syntax.traverse._
import common.validation.ErrorOr.ErrorOr
import cromwell.core.{WorkflowId, WorkflowSourceFilesCollection}
import cromwell.languages.util.ImportResolver.HttpResolver
import cromwell.languages.util.{ImportResolver, LanguageFactoryUtil}
import cromwell.languages.{LanguageFactory, ValidatedWomNamespace}
Expand All @@ -14,9 +16,22 @@ import wom.expression.NoIoFunctionSet
object Describer {

def describeWorkflow(wsfc: WorkflowSourceFilesCollection): DescribeResult = {
val zipResolverErrorOr: ErrorOr[Option[ImportResolver.ImportResolver]] =
wsfc.importsZipFileOption.map(ImportResolver.zippedImportResolver(_, WorkflowId.randomId())).sequence

val initialResolvers = List(HttpResolver(None, Map.empty))
val initialResolversErrorOr: ErrorOr[List[ImportResolver.ImportResolver]] =
zipResolverErrorOr map { zipResolverOption =>
zipResolverOption.toList ++ List(HttpResolver(None, Map.empty))
}

initialResolversErrorOr match {
case Valid(initialResolvers) => describeWorkflow(wsfc, initialResolvers)
case Invalid(errors) => DescribeFailure(errors.toList.mkString(", "))

Check warning on line 29 in services/src/main/scala/cromwell/services/womtool/Describer.scala

View check run for this annotation

Codecov / codecov/patch

services/src/main/scala/cromwell/services/womtool/Describer.scala#L29

Added line #L29 was not covered by tests
}
}

def describeWorkflow(wsfc: WorkflowSourceFilesCollection,
initialResolvers: List[ImportResolver.ImportResolver]): DescribeResult = {
// The HTTP resolver is used to pull down workflows submitted by URL
LanguageFactoryUtil.findWorkflowSource(wsfc.workflowSource, wsfc.workflowUrl, initialResolvers) match {
case Right((workflowSource: WorkflowSource, importResolvers: List[ImportResolver.ImportResolver])) =>
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
{
"valid" : true,
"errors" : [
],
"validWorkflow" : true,
"name" : "relative_imports",
"inputs" : [
],
"outputs" : [
{
"name" : "result",
"valueType" : {
"typeName" : "Int"
},
"typeDisplayName" : "Int"
}
],
"images" : [
],
"submittedDescriptorType" : {
"descriptorType" : "WDL",
"descriptorTypeVersion" : "Biscayne"
},
"importedDescriptorTypes" : [
],
"meta" : {

},
"parameterMeta" : {

},
"isRunnableWorkflow" : true
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
version development

import "sub_wfs/foo.wdl"

workflow relative_imports {
call foo.foo_wf

output {
Int result = foo_wf.unpacked
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
version development

struct MyStruct {
Int a
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
version development

import "../structs/my_struct.wdl"
import "tasks/add5.wdl" as a5

workflow foo_wf {
call a5.add5 { input: x = object { a: 100 } }
output {
Int unpacked = add5.five_added.a
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
version development

import "../../structs/my_struct.wdl"

task add5 {
input {
MyStruct x
}
command <<<
echo $((5 + ~{x.a}))
>>>
output {
MyStruct five_added = object { a: read_int(stdout()) }
}
runtime {
docker: "ubuntu:latest"
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ package cromwell.services.womtool

import common.assertion.CromwellTimeoutSpec
import cromwell.core.path._
import cromwell.core.{WorkflowOptions, WorkflowSourceFilesCollection, WorkflowSourceFilesWithoutImports}
import cromwell.core.{WorkflowOptions, WorkflowSourceFilesCollection}
import cromwell.languages.config.{CromwellLanguages, LanguageConfiguration}
import cromwell.services.womtool.DescriberSpec._
import cromwell.services.womtool.WomtoolServiceMessages.DescribeSuccess
Expand Down Expand Up @@ -37,26 +37,37 @@ class DescriberSpec extends AnyFlatSpec with CromwellTimeoutSpec with Matchers {
val workflowType = Try(caseDirectory.resolve("workflowType").contentAsString.stripLineEnd).toOption
val workflowTypeVersion =
Try(caseDirectory.resolve("workflowTypeVersion").contentAsString.stripLineEnd).toOption
val importsFile =
Try(caseDirectory.resolve("workflowDependencies")).filter(_.exists).map(_.zip()).toOption

val interimWsfc = WorkflowSourceFilesWithoutImports(
workflowSource = None,
workflowUrl = None,
val workflowSource = testCase match {
case FileAndDescription(file, _) => Option(file)
case _ => None
}

val workflowUrl = testCase match {
case UrlAndDescription(url, _) => Option(url)
case _ => None
}

val wsfc = WorkflowSourceFilesCollection(
workflowSource = workflowSource,
workflowUrl = workflowUrl,
workflowRoot = None,
workflowType = workflowType,
workflowTypeVersion = workflowTypeVersion,
inputsJson = "",
workflowOptions = WorkflowOptions.empty,
importsFile = importsFile.map(_.byteArray),
workflowOnHold = false,
labelsJson = "",
warnings = Seq.empty,
requestedWorkflowId = None
)

val wsfc = testCase match {
case FileAndDescription(file, _) => interimWsfc.copy(workflowSource = Option(file))
case UrlAndDescription(url, _) => interimWsfc.copy(workflowUrl = Option(url))
}

check(wsfc, parse(testCase.expectedDescription).toOption.get)

importsFile.map(_.delete(swallowIOExceptions = true))
}
}
}
Expand Down

0 comments on commit ca14a65

Please sign in to comment.