Skip to content

feat: add experiments subgraph#173

Open
jonathanirvine wants to merge 1 commit into
mainfrom
experiments-0.1.0
Open

feat: add experiments subgraph#173
jonathanirvine wants to merge 1 commit into
mainfrom
experiments-0.1.0

Conversation

@jonathanirvine
Copy link
Copy Markdown

Add ULIMS experiments service subgraph

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 6, 2026

diff --git a/tmp/old_sorted.json b/tmp/new_sorted.json
index d2d391c..a84648d 100644
--- a/tmp/old_sorted.json
+++ b/tmp/new_sorted.json
@@ -652,6 +652,183 @@
             "typeName": "Sample"
           }
         ]
+      },
+      {
+        "childNodes": [
+          {
+            "fieldNames": [
+              "edges",
+              "pageInfo"
+            ],
+            "typeName": "ExperimentConnection"
+          },
+          {
+            "fieldNames": [
+              "id",
+              "name",
+              "createdTime",
+              "updatedTime",
+              "data",
+              "dataSchemaUrl",
+              "proposalNumber",
+              "instrumentSessionNumber",
+              "instrumentSession",
+              "experiments"
+            ],
+            "typeName": "ExperimentDefinition"
+          },
+          {
+            "fieldNames": [
+              "edges",
+              "pageInfo"
+            ],
+            "typeName": "ExperimentDefinitionConnection"
+          },
+          {
+            "fieldNames": [
+              "cursor",
+              "node"
+            ],
+            "typeName": "ExperimentDefinitionEdge"
+          },
+          {
+            "fieldNames": [
+              "createExperiments"
+            ],
+            "typeName": "ExperimentDefinitionMutations"
+          },
+          {
+            "fieldNames": [
+              "cursor",
+              "node"
+            ],
+            "typeName": "ExperimentEdge"
+          },
+          {
+            "fieldNames": [
+              "startCursor",
+              "endCursor",
+              "hasNextPage",
+              "hasPreviousPage"
+            ],
+            "typeName": "PageInfo"
+          }
+        ],
+        "customGraphql": {
+          "federation": {
+            "enabled": true,
+            "serviceSdl": "schema @link(url: \"https://specs.apollo.dev/federation/v2.7\", import: [\"@external\", \"@key\", \"@provides\", \"@shareable\"]) {\n  query: Query\n  mutation: Mutation\n}\n\n\"\"\"Values required to create an experiment definition\"\"\"\ninput CreateExperimentDefinitionInput {\n  name: String!\n  data: JSON!\n  dataSchemaUrl: String!\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n}\n\n\"\"\"Values required for the createExperiments mutation\"\"\"\ninput CreateExperimentsInput {\n  experiments: [ExperimentInput!]!\n}\n\n\"\"\"Date with time (isoformat)\"\"\"\nscalar DateTime\n\ntype Experiment @key(fields: \"id\") {\n  id: UUID!\n  name: String!\n  createdTime: DateTime!\n  updatedTime: DateTime!\n  experimentDefinition: ExperimentDefinition!\n\n  \"\"\"The sample that this experiment is associated with\"\"\"\n  sample: Sample! @provides(fields: \"id\")\n}\n\ntype ExperimentConnection @shareable {\n  edges: [ExperimentEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype ExperimentDefinition {\n  id: UUID!\n  name: String!\n  createdTime: DateTime!\n  updatedTime: DateTime!\n  data: JSON!\n  dataSchemaUrl: String!\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n\n  \"\"\"\n  The instrument session that this experiment definition is associated with\n  \"\"\"\n  instrumentSession: InstrumentSession! @provides(fields: \"instrumentSessionNumber proposal{ proposalNumber }\")\n\n  \"\"\"Experiments associated with this experiment definition\"\"\"\n  experiments: [Experiment!]!\n}\n\ntype ExperimentDefinitionConnection @shareable {\n  edges: [ExperimentDefinitionEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype ExperimentDefinitionEdge @shareable {\n  cursor: String!\n  node: ExperimentDefinition!\n}\n\n\"\"\"Mutations for a given experiment defintion\"\"\"\ntype ExperimentDefinitionMutations {\n  createExperiments(input: CreateExperimentsInput!): [Experiment!]!\n}\n\ntype ExperimentEdge @shareable {\n  cursor: String!\n  node: Experiment!\n}\n\n\"\"\"Values required to create an experiment\"\"\"\ninput ExperimentInput {\n  name: String!\n  sampleId: UUID!\n}\n\ntype InstrumentSession @key(fields: \"instrumentSessionNumber proposal { proposalNumber }\") {\n  instrumentSessionNumber: Int! @external\n  proposal: Proposal @external\n\n  \"\"\"Experiment Definitions\"\"\"\n  experimentDefinitions(first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentDefinitionConnection!\n\n  \"\"\"Experiments associated with this session\"\"\"\n  experiments(first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentConnection!\n}\n\n\"\"\"Values required to uniquely identify an instrument session\"\"\"\ninput InstrumentSessionInput {\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n}\n\n\"\"\"\nThe `JSON` scalar type represents JSON values as specified by [ECMA-404](https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf).\n\"\"\"\nscalar JSON @specifiedBy(url: \"https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf\")\n\ntype Mutation {\n  createExperimentDefinition(input: CreateExperimentDefinitionInput!): ExperimentDefinition\n  experimentDefinition(id: UUID!): ExperimentDefinitionMutations\n}\n\ntype PageInfo @shareable {\n  startCursor: String\n  endCursor: String\n  hasNextPage: Boolean!\n  hasPreviousPage: Boolean!\n}\n\ntype Proposal @key(fields: \"proposalNumber\") {\n  proposalNumber: Int! @external\n}\n\ntype Query {\n  _entities(representations: [_Any!]!): [_Entity]!\n  _service: _Service!\n  experimentDefinition(id: UUID!): ExperimentDefinition\n  experimentDefinitions(instrumentSessions: [InstrumentSessionInput!]!, first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentDefinitionConnection\n  experiment(id: UUID!): Experiment\n  experiments(instrumentSessions: [InstrumentSessionInput!]!, first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentConnection\n}\n\ntype Sample @key(fields: \"id\") {\n  id: UUID! @external\n  experiments: [Experiment!]!\n}\n\nscalar UUID\n\nscalar _Any\n\nunion _Entity = Experiment | InstrumentSession | Proposal | Sample\n\ntype _Service {\n  sdl: String!\n}\n"
+          },
+          "fetch": {
+            "baseUrl": {},
+            "body": {},
+            "method": "POST",
+            "path": {},
+            "url": {
+              "staticVariableContent": "https://api.experiment-definition.diamond.ac.uk/graphql"
+            }
+          },
+          "subscription": {
+            "enabled": true,
+            "protocol": "GRAPHQL_SUBSCRIPTION_PROTOCOL_WS",
+            "url": {
+              "staticVariableContent": "https://api.experiment-definition.diamond.ac.uk/graphql"
+            },
+            "websocketSubprotocol": "GRAPHQL_WEBSOCKET_SUBPROTOCOL_AUTO"
+          },
+          "upstreamSchema": {
+            "key": "60b3f57ff30215a4bb01d6659c2b5ac5e2a97765"
+          }
+        },
+        "id": "4",
+        "keys": [
+          {
+            "selectionSet": "id",
+            "typeName": "Experiment"
+          },
+          {
+            "selectionSet": "instrumentSessionNumber proposal { proposalNumber }",
+            "typeName": "InstrumentSession"
+          },
+          {
+            "selectionSet": "proposalNumber",
+            "typeName": "Proposal"
+          },
+          {
+            "selectionSet": "id",
+            "typeName": "Sample"
+          }
+        ],
+        "kind": "GRAPHQL",
+        "overrideFieldPathFromAlias": true,
+        "provides": [
+          {
+            "fieldName": "sample",
+            "selectionSet": "id",
+            "typeName": "Experiment"
+          },
+          {
+            "fieldName": "instrumentSession",
+            "selectionSet": "instrumentSessionNumber proposal { proposalNumber }",
+            "typeName": "ExperimentDefinition"
+          }
+        ],
+        "requestTimeoutSeconds": "10",
+        "rootNodes": [
+          {
+            "fieldNames": [
+              "id",
+              "name",
+              "createdTime",
+              "updatedTime",
+              "experimentDefinition",
+              "sample"
+            ],
+            "typeName": "Experiment"
+          },
+          {
+            "externalFieldNames": [
+              "instrumentSessionNumber",
+              "proposal"
+            ],
+            "fieldNames": [
+              "experimentDefinitions",
+              "experiments"
+            ],
+            "typeName": "InstrumentSession"
+          },
+          {
+            "fieldNames": [
+              "createExperimentDefinition",
+              "experimentDefinition"
+            ],
+            "typeName": "Mutation"
+          },
+          {
+            "externalFieldNames": [
+              "proposalNumber"
+            ],
+            "typeName": "Proposal"
+          },
+          {
+            "fieldNames": [
+              "experimentDefinition",
+              "experimentDefinitions",
+              "experiment",
+              "experiments"
+            ],
+            "typeName": "Query"
+          },
+          {
+            "externalFieldNames": [
+              "id"
+            ],
+            "fieldNames": [
+              "experiments"
+            ],
+            "typeName": "Sample"
+          }
+        ]
       }
     ],
     "defaultFlushInterval": "500",
@@ -718,6 +895,26 @@
         "fieldName": "sample",
         "typeName": "Mutation"
       },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "input",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "createExperimentDefinition",
+        "typeName": "Mutation"
+      },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "id",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experimentDefinition",
+        "typeName": "Mutation"
+      },
       {
         "argumentsConfiguration": [
           {
@@ -954,6 +1151,78 @@
         "fieldName": "samples",
         "typeName": "Query"
       },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "id",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experimentDefinition",
+        "typeName": "Query"
+      },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "instrumentSessions",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "first",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "last",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "after",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "before",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experimentDefinitions",
+        "typeName": "Query"
+      },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "id",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experiment",
+        "typeName": "Query"
+      },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "instrumentSessions",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "first",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "last",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "after",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "before",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experiments",
+        "typeName": "Query"
+      },
       {
         "argumentsConfiguration": [
           {
@@ -1016,6 +1285,50 @@
         "fieldName": "samples",
         "typeName": "InstrumentSession"
       },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "first",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "last",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "after",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "before",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experimentDefinitions",
+        "typeName": "InstrumentSession"
+      },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "first",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "last",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "after",
+            "sourceType": "FIELD_ARGUMENT"
+          },
+          {
+            "name": "before",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "experiments",
+        "typeName": "InstrumentSession"
+      },
       {
         "argumentsConfiguration": [
           {
@@ -1143,10 +1456,21 @@
         ],
         "fieldName": "createSampleImageUploadUrl",
         "typeName": "SampleMutations"
+      },
+      {
+        "argumentsConfiguration": [
+          {
+            "name": "input",
+            "sourceType": "FIELD_ARGUMENT"
+          }
+        ],
+        "fieldName": "createExperiments",
+        "typeName": "ExperimentDefinitionMutations"
       }
     ],
-    "graphqlSchema": "schema {\n  query: Query\n  mutation: Mutation\n  subscription: Subscription\n}\n\ntype Artifact {\n  \"\"\"The file name of the artifact\"\"\"\n  name: String!\n  \"\"\"The download URL for the artifact\"\"\"\n  url: Url!\n  \"\"\"The MIME type of the artifact data\"\"\"\n  mimeType: String!\n}\n\nscalar Creator\n\n\"\"\"\nImplement the DateTime<Utc> scalar\n\nThe input/output is a string in RFC3339 format.\n\"\"\"\nscalar DateTime\n\n\"\"\"\nThe `JSON` scalar type represents JSON values as specified by [ECMA-404](https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf).\n\"\"\"\nscalar JSON\n\n\"\"\"A scalar that can represent any JSON Object value.\"\"\"\nscalar JSONObject\n\n\"\"\"A single log line streamed from a pod\"\"\"\ntype LogEntry {\n  \"\"\"The log line content\"\"\"\n  content: String!\n  \"\"\"The name of the pod producing the log\"\"\"\n  podName: String!\n}\n\n\"\"\"The root mutation of the service\"\"\"\ntype Mutation {\n  submitWorkflowTemplate(name: String!, visit: VisitInput!, parameters: JSON!): Workflow!\n  instrumentSession(proposalNumber: Int!, instrumentSessionNumber: Int!): InstrumentSessionMutations\n  createOrValidateSamples(input: CreateOrValidateSampleInput!): CreateSamplesResponse!\n  createSamples(input: CreateSampleInput!): [Sample!]! @deprecated(reason: \"Will be replaced by createOrValidateSamples\")\n  sample(sampleId: UUID!): SampleMutations\n}\n\n\"\"\"Represents Relay Node types\"\"\"\nunion NodeValue = Workflow\n\n\"\"\"Information about pagination in a connection\"\"\"\ntype PageInfo {\n  \"\"\"When paginating backwards, are there more items?\"\"\"\n  hasPreviousPage: Boolean!\n  \"\"\"When paginating forwards, are there more items?\"\"\"\n  hasNextPage: Boolean!\n  \"\"\"When paginating backwards, the cursor to continue.\"\"\"\n  startCursor: String\n  \"\"\"When paginating forwards, the cursor to continue.\"\"\"\n  endCursor: String\n}\n\n\"\"\"The root query of the service\"\"\"\ntype Query {\n  node(id: ID!): NodeValue\n  \"\"\"Get a single [`Workflow`] by proposal, visit, and name\"\"\"\n  workflow(visit: VisitInput!, name: String!): Workflow!\n  workflows(visit: VisitInput!, cursor: String, limit: Int, filter: WorkflowFilter): WorkflowConnection!\n  workflowTemplate(name: String!): WorkflowTemplate!\n  workflowTemplates(cursor: String, limit: Int, filter: WorkflowTemplatesFilter): WorkflowTemplateConnection!\n  \"\"\"Get a proposal by its number\"\"\"\n  proposal(proposalNumber: Int!): Proposal\n  \"\"\"Get a list of proposals\"\"\"\n  proposals(proposalCategory: String = null, first: Int = null, last: Int = null, after: String = null, before: String = null): ProposalConnection!\n  \"\"\"Get a instrument session\"\"\"\n  instrumentSession(proposalNumber: Int!, instrumentSessionNumber: Int!): InstrumentSession\n  \"\"\"Get a instrument session\"\"\"\n  instrumentSessions(proposalNumber: Int = null, proposalCategory: String = null): [InstrumentSession!]\n  \"\"\"Get an instrument\"\"\"\n  instrument(instrumentName: String!): Instrument\n  \"\"\"Get a list of instruments\"\"\"\n  instruments(scienceGroup: String = null): [Instrument!]!\n  \"\"\"Get an account\"\"\"\n  account(username: String!): Account\n  jsonSchema(url: String!): JSONSchema\n  jsonSchemas(type: String = null, instrument: String = null): [JSONSchema!]!\n  \"\"\"Get a sample by its id\"\"\"\n  sample(sampleId: UUID!): Sample\n  \"\"\"Get a list of samples associated with a given instrument session\"\"\"\n  samples(first: Int!, instrumentSessions: [InstrumentSessionInput!] = null, filter: SampleFilterInput! = {schemaUrl: null, createdTime: null, updatedTime: null, name: null, data: null}, before: String = null, after: String = null, last: Int = null, orderBy: SampleOrder! = {name: null, createdTime: null, updatedTime: null}): SampleConnection!\n}\n\n\"\"\"Supported DLS science groups\"\"\"\nenum ScienceGroup {\n  \"\"\"Macromolecular Crystallography\"\"\"\n  MX\n  \"\"\"Workflows Examples\"\"\"\n  EXAMPLES\n  \"\"\"Magnetic Materials\"\"\"\n  MAGNETIC_MATERIALS\n  \"\"\"Soft Condensed Matter\"\"\"\n  CONDENSED_MATTER\n  \"\"\"Imaging and Microscopy\"\"\"\n  IMAGING\n  \"\"\"Biological Cryo-Imaging\"\"\"\n  BIO_CRYO_IMAGING\n  \"\"\"Structures and Surfaces\"\"\"\n  SURFACES\n  \"\"\"Crystallography\"\"\"\n  CRYSTALLOGRAPHY\n  \"\"\"Spectroscopy\"\"\"\n  SPECTROSCOPY\n}\n\n\"\"\"The root mutation of the service\"\"\"\ntype Subscription {\n  \"\"\"Processing to subscribe to logs for a single pod of a workflow\"\"\"\n  logs(visit: VisitInput!, workflowName: String!, taskId: String!): LogEntry!\n  \"\"\"Processing to subscribe to data for all workflows in a session\"\"\"\n  workflow(visit: VisitInput!, name: String!): Workflow!\n}\n\ntype Task {\n  \"\"\"Unique name of the task\"\"\"\n  id: String!\n  \"\"\"Display name of the task\"\"\"\n  name: String!\n  \"\"\"Current status of a task\"\"\"\n  status: TaskStatus!\n  \"\"\"Parent of a task\"\"\"\n  depends: [String!]!\n  \"\"\"Children of a task\"\"\"\n  dependencies: [String!]!\n  \"\"\"Artifacts produced by a task\"\"\"\n  artifacts: [Artifact!]!\n  \"\"\"Node type - Pod, DAG, etc\"\"\"\n  stepType: String!\n  \"\"\"Start time for a task on a workflow\"\"\"\n  startTime: DateTime\n  \"\"\"End time for a task on a workflow\"\"\"\n  endTime: DateTime\n  \"\"\"\n  A human readable message indicating details about why this step is in this condition\n  \"\"\"\n  message: String\n}\n\nenum TaskStatus {\n  PENDING\n  RUNNING\n  SUCCEEDED\n  SKIPPED\n  FAILED\n  ERROR\n  OMITTED\n}\n\nscalar Template\n\n\"\"\"Information about where the template is stored\"\"\"\ntype TemplateSource {\n  \"\"\"The URL of the GitHub repository\"\"\"\n  repositoryUrl: String!\n  \"\"\"The path to the template within the repository\"\"\"\n  path: String!\n  \"\"\"The current tracked branch of the repository\"\"\"\n  targetRevision: String!\n}\n\n\"\"\"\nURL is a String implementing the [URL Standard](http://url.spec.whatwg.org/)\n\"\"\"\nscalar Url\n\n\"\"\"A visit to an instrument as part of a session\"\"\"\ntype Visit {\n  \"\"\"Project Proposal Code\"\"\"\n  proposalCode: String!\n  \"\"\"Project Proposal Number\"\"\"\n  proposalNumber: Int!\n  \"\"\"Session visit Number\"\"\"\n  number: Int!\n}\n\n\"\"\"A visit to an instrument as part of a session\"\"\"\ninput VisitInput {\n  \"\"\"Project Proposal Code\"\"\"\n  proposalCode: String!\n  \"\"\"Project Proposal Number\"\"\"\n  proposalNumber: Int!\n  \"\"\"Session visit Number\"\"\"\n  number: Int!\n}\n\ntype Workflow {\n  \"\"\"The unique ID derived from the visit and name\"\"\"\n  id: ID!\n  \"\"\"The name given to the workflow, unique within a given visit\"\"\"\n  name: String!\n  \"\"\"The visit the Workflow was run against\"\"\"\n  visit: Visit!\n  \"\"\"The current status of the workflow\"\"\"\n  status: WorkflowStatus\n  \"\"\"The top-level workflow parameters\"\"\"\n  parameters: JSONObject\n  \"\"\"The name of the template used to run the workflow\"\"\"\n  templateRef: String\n  \"\"\"The workflow creator\"\"\"\n  creator: WorkflowCreator!\n}\n\ntype WorkflowConnection {\n  \"\"\"Information to aid in pagination.\"\"\"\n  pageInfo: PageInfo!\n  \"\"\"A list of edges.\"\"\"\n  edges: [WorkflowEdge!]!\n  \"\"\"A list of nodes.\"\"\"\n  nodes: [Workflow!]!\n}\n\n\"\"\"Information about the creator of a workflow.\"\"\"\ntype WorkflowCreator {\n  \"\"\"\n  An identifier unique to the creator of the workflow.\n  Typically this is the creator's Fed-ID.\n  \"\"\"\n  creatorId: String!\n}\n\n\"\"\"An edge in a connection.\"\"\"\ntype WorkflowEdge {\n  \"\"\"The item at the end of the edge\"\"\"\n  node: Workflow!\n  \"\"\"A cursor for use in pagination\"\"\"\n  cursor: String!\n}\n\n\"\"\"All tasks in the workflow have errored\"\"\"\ntype WorkflowErroredStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"All tasks in the workflow have failed\"\"\"\ntype WorkflowFailedStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"All the supported Workflows filters\"\"\"\ninput WorkflowFilter {\n  \"\"\"The status field for a workflow\"\"\"\n  workflowStatusFilter: WorkflowStatusFilter\n  \"\"\"The fedid of the user who created the workflow\"\"\"\n  creator: Creator\n  \"\"\"The name of the workflow template\"\"\"\n  template: Template\n}\n\ntype WorkflowPendingStatus {\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n}\n\ntype WorkflowRunningStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"The status of a workflow\"\"\"\nunion WorkflowStatus = WorkflowPendingStatus | WorkflowRunningStatus | WorkflowSucceededStatus | WorkflowFailedStatus | WorkflowErroredStatus\n\n\"\"\"Represents workflow status filters\"\"\"\ninput WorkflowStatusFilter {\n  pending: Boolean! = false\n  running: Boolean! = false\n  succeeded: Boolean! = false\n  failed: Boolean! = false\n  error: Boolean! = false\n}\n\n\"\"\"All tasks in the workflow have succeded\"\"\"\ntype WorkflowSucceededStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\ntype WorkflowTemplate {\n  \"\"\"The name given to the workflow template, globally unique\"\"\"\n  name: String!\n  \"\"\"The group who maintains the workflow template\"\"\"\n  maintainer: String!\n  \"\"\"A human readable title for the workflow template\"\"\"\n  title: String\n  \"\"\"A human readable description of the workflow which is created\"\"\"\n  description: String\n  \"\"\"The repository storing the code associated with this template.\"\"\"\n  repository: String\n  \"\"\"A JSON Schema describing the arguments of a Workflow Template\"\"\"\n  arguments: JSON!\n  \"\"\"\n  A JSON Forms UI Schema describing how to render the arguments of the Workflow Template\n  \"\"\"\n  uiSchema: JSON\n  \"\"\"Information about where the template is obtained from\"\"\"\n  templateSource: TemplateSource\n}\n\ntype WorkflowTemplateConnection {\n  \"\"\"Information to aid in pagination.\"\"\"\n  pageInfo: PageInfo!\n  \"\"\"A list of edges.\"\"\"\n  edges: [WorkflowTemplateEdge!]!\n  \"\"\"A list of nodes.\"\"\"\n  nodes: [WorkflowTemplate!]!\n}\n\n\"\"\"An edge in a connection.\"\"\"\ntype WorkflowTemplateEdge {\n  \"\"\"The item at the end of the edge\"\"\"\n  node: WorkflowTemplate!\n  \"\"\"A cursor for use in pagination\"\"\"\n  cursor: String!\n}\n\n\"\"\"Supported label filters for ClusterWorkflowTemplates\"\"\"\ninput WorkflowTemplatesFilter {\n  \"\"\"The science group owning the template eg imaging\"\"\"\n  scienceGroup: [ScienceGroup!]\n}\n\ntype Account {\n  accountId: Int!\n  username: String!\n  emailAddress: String\n  title: String\n  givenName: String\n  familyName: String\n  type: AccountType!\n  state: AccountState!\n  proposalRoles: [ProposalAccount!]!\n  instrumentSessionRoles: [InstrumentSessionRole!]!\n}\n\nenum AccountState {\n  enabled\n  disabled\n}\n\nenum AccountType {\n  user\n  staff\n  functional\n}\n\ntype Instrument {\n  name: String!\n  scienceGroup: String\n  description: String\n  proposals: [Proposal!]!\n  instrumentSessions: [InstrumentSession!]!\n}\n\ntype InstrumentSession {\n  instrumentSessionId: Int!\n  instrumentSessionNumber: Int!\n  startTime: DateTime\n  endTime: DateTime\n  type: String\n  state: String\n  riskRating: String\n  proposal: Proposal\n  instrument: Instrument!\n  roles: [InstrumentSessionRole!]!\n  \"\"\"Samples associated with a given instrument session\"\"\"\n  samples(first: Int!, filter: SampleFilterInput! = {schemaUrl: null, createdTime: null, updatedTime: null, name: null, data: null}, before: String = null, after: String = null, last: Int = null, orderBy: SampleOrder! = {name: null, createdTime: null, updatedTime: null}): SampleConnection!\n}\n\ntype InstrumentSessionMutations {\n  instrumentSessionNumber: Int!\n  proposalNumber: Int!\n  \"\"\"Create or validate samples associated with this instrument session\"\"\"\n  createOrValidateSamples(input: CreateOrValidateSampleInputBase!): CreateSamplesResponse!\n}\n\ntype InstrumentSessionRole {\n  instrumentSession: InstrumentSession!\n  account: Account!\n  role: String!\n  onSite: Boolean!\n}\n\ntype Proposal {\n  proposalNumber: Int!\n  proposalCategory: String\n  title: String\n  summary: String\n  state: ProposalState!\n  instrumentSessions: [InstrumentSession!]!\n  instruments: [Instrument!]!\n  roles: [ProposalAccount!]!\n}\n\ntype ProposalAccount {\n  proposal: Proposal!\n  account: Account!\n  role: String!\n}\n\ntype ProposalConnection {\n  edges: [ProposalEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype ProposalEdge {\n  cursor: String!\n  node: Proposal!\n}\n\nenum ProposalState {\n  Open\n  Closed\n  Cancelled\n}\n\n\"\"\"A JSON schema\"\"\"\ntype JSONSchema {\n  \"\"\"The identifier of the schema\"\"\"\n  id: String!\n  \"\"\"A URL from which the schema can be accessed\"\"\"\n  url: String!\n  \"\"\"The type of object the shema describes (if known)\"\"\"\n  type: String\n  \"\"\"The title of the schema\"\"\"\n  title: String\n  \"\"\"The version of the schema\"\"\"\n  version: String\n  \"\"\"The instrument the schema was created for\"\"\"\n  instrument: String\n  \"\"\"The description og the schema\"\"\"\n  description: String\n}\n\ninput AddSampleEventInput {\n  description: String!\n}\n\ninput CreateOrValidateSampleInput {\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String!\n  \"\"\"Samples to be created\"\"\"\n  samples: [SampleIn!]!\n  \"\"\"\n  Whether or not the provided samples should only be validated and not created\n  \"\"\"\n  validateOnly: Boolean! = false\n  \"\"\"Number of the proposal the samples should be associated with\"\"\"\n  proposalNumber: Int!\n  \"\"\"Number of the instrument session the samples should be associated with\"\"\"\n  instrumentSessionNumber: Int!\n}\n\ninput CreateOrValidateSampleInputBase {\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String!\n  \"\"\"Samples to be created\"\"\"\n  samples: [SampleIn!]!\n  \"\"\"\n  Whether or not the provided samples should only be validated and not created\n  \"\"\"\n  validateOnly: Boolean! = false\n}\n\ninput CreateSampleInput {\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n  samples: [SampleInLegacy!]!\n  validateOnly: Boolean! = false\n}\n\n\"\"\"Return type when creating or validating samples\"\"\"\ntype CreateSamplesResponse {\n  \"\"\"Whether the operation has succeeded without validation errors\"\"\"\n  success: Boolean!\n  \"\"\"Samples that have been created\"\"\"\n  samples: [Sample!]!\n  \"\"\"Errors that occurred during sample validation\"\"\"\n  errors: [SampleValidationError!]!\n}\n\ninput DatetimeOperatorInput {\n  \"\"\"\n  Will filter to items where the `DateTime` field is greater than (i.e. after) the provided value\n  \"\"\"\n  gt: DateTime = null\n  \"\"\"\n  Will filter to items where the `DateTime` field is less than (i.e. before) the provided value\n  \"\"\"\n  lt: DateTime = null\n}\n\n\"\"\"The details of sample validation error\"\"\"\ntype ErrorDetails {\n  \"\"\"The type of error that occurred\"\"\"\n  type: String!\n  \"\"\"\n  Tuple of strings identifying where in the sample schema the error occurred.\n  \"\"\"\n  location: [String!]!\n  \"\"\"A human readable error message.\"\"\"\n  message: String!\n}\n\ninput InstrumentSessionInput {\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n}\n\ninput JSONOperator @oneOf {\n  stringOperator: StringOperatorInput = null\n  datetimeOperator: DatetimeOperatorInput = null\n  numericOperator: NumericOperatorInput = null\n}\n\ninput JSONOperatorInput {\n  \"\"\"A JSON path specifying the value to filter. Must start with '$.'\"\"\"\n  path: String!\n  \"\"\"The operator to apply to the JSON field\"\"\"\n  operator: JSONOperator!\n}\n\ninput NumericOperatorInput {\n  \"\"\"\n  Will filter to items where the numeric field is greater than the provided value\n  \"\"\"\n  gt: Float = null\n  \"\"\"\n  Will filter to items where the numeric field is less than the provided value\n  \"\"\"\n  lt: Float = null\n}\n\ntype Sample {\n  id: UUID!\n  name: String!\n  data: JSON!\n  createdTime: DateTime!\n  updatedTime: DateTime!\n  dataSchemaUrl: String!\n  \"\"\"Samples from which this sample is derived\"\"\"\n  parents(first: Int = null, before: String = null, after: String = null, last: Int = null): SampleConnection!\n  \"\"\"Samples derived from this sample\"\"\"\n  children(first: Int = null, before: String = null, after: String = null, last: Int = null): SampleConnection!\n  \"\"\"Events linked to this sample\"\"\"\n  events(first: Int = null, before: String = null, after: String = null, last: Int = null): SampleEventConnection!\n  \"\"\"The JSON schema that the sample's `data` conforms to\"\"\"\n  dataSchema: JSON!\n  \"\"\"The instrument sessions that this sample is associated with\"\"\"\n  instrumentSessions: [InstrumentSession!]!\n  images: [SampleImage!]!\n}\n\ntype SampleConnection {\n  edges: [SampleEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype SampleEdge {\n  cursor: String!\n  node: Sample!\n}\n\ntype SampleEvent {\n  id: UUID!\n  timestamp: DateTime!\n  description: String!\n}\n\ntype SampleEventConnection {\n  edges: [SampleEventEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype SampleEventEdge {\n  cursor: String!\n  node: SampleEvent!\n}\n\ninput SampleFilterInput {\n  \"\"\"Filter on the `schemaUrl` field of `Sample`\"\"\"\n  schemaUrl: StringOperatorInput = null\n  \"\"\"Filter on the `createdTime` field of `Sample`\"\"\"\n  createdTime: DatetimeOperatorInput = null\n  \"\"\"Filter on the `createdTime` field of `Sample`\"\"\"\n  updatedTime: DatetimeOperatorInput = null\n  \"\"\"Filter on the `name` field of `Sample`\"\"\"\n  name: StringOperatorInput = null\n  \"\"\"Filter on the `data` field of `Sample`\"\"\"\n  data: [JSONOperatorInput!] = null\n}\n\ntype SampleImage {\n  url: String!\n  filename: String!\n}\n\ninput SampleIn {\n  \"\"\"Name of the sample\"\"\"\n  name: String!\n  \"\"\"Data of the sample\"\"\"\n  data: JSON!\n}\n\ninput SampleInLegacy {\n  name: String!\n  data: JSON!\n  dataSchemaUrl: String!\n  parentIds: [Int!] = null\n  children: [SampleInLegacy!] = null\n}\n\ntype SampleMutations {\n  sampleId: UUID!\n  updateSample(input: UpdateSampleInput!): UpdateSampleResponse!\n  linkInstrumentSessionToSample(proposalNumber: Int!, instrumentSessionNumber: Int!): Void\n  addSampleEvent(sampleEvent: AddSampleEventInput!): SampleEvent!\n  createSampleImageUploadUrl(filename: String!, contentType: String!, contentLength: Int!): String!\n}\n\ninput SampleOrder {\n  name: SortingOrder = null\n  createdTime: SortingOrder = null\n  updatedTime: SortingOrder = null\n}\n\n\"\"\"The details of errors occurred when validating a sample\"\"\"\ntype SampleValidationError {\n  \"\"\"\n  The index of the sample in CreateSampleInput.samples for which the error occurred\n  \"\"\"\n  index: Int!\n  \"\"\"Errors that occurred when validating the sample\"\"\"\n  errors: [ErrorDetails!]!\n}\n\nenum SortingOrder {\n  ASC\n  DESC\n}\n\n\"\"\"Conditions used to filter results based on the value of a String field\"\"\"\ninput StringOperatorInput {\n  \"\"\"\n  Will filter to items where the `String` field is equal to the provided value\n  \"\"\"\n  eq: String = null\n  \"\"\"\n  Will filter to items where the `String` field is not equal to the provided value\n  \"\"\"\n  ne: String = null\n  \"\"\"\n  Will filter to items where the `String` field is a member of the provided value\n  \"\"\"\n  in: [String!] = null\n  \"\"\"\n  Will filter to items where the `String` field is not a member of the provided value\n  \"\"\"\n  nin: [String!] = null\n  \"\"\"\n  Will filter to items where the `String` field is contains the provided value\n  \"\"\"\n  contains: String = null\n}\n\nscalar UUID\n\ninput UpdateSampleInput {\n  \"\"\"Name of the sample\"\"\"\n  name: String\n  \"\"\"Data of the sample\"\"\"\n  data: JSON\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String\n}\n\n\"\"\"Return type when creating or validating samples\"\"\"\ntype UpdateSampleResponse {\n  \"\"\"Whether the operation has succeeded without validation errors\"\"\"\n  success: Boolean!\n  \"\"\"Sample that has been updated\"\"\"\n  sample: Sample\n  \"\"\"Errors that occurred during sample validation\"\"\"\n  errors: [SampleValidationError!]!\n}\n\n\"\"\"Represents NULL values\"\"\"\nscalar Void",
+    "graphqlSchema": "schema {\n  query: Query\n  mutation: Mutation\n  subscription: Subscription\n}\n\ntype Artifact {\n  \"\"\"The file name of the artifact\"\"\"\n  name: String!\n  \"\"\"The download URL for the artifact\"\"\"\n  url: Url!\n  \"\"\"The MIME type of the artifact data\"\"\"\n  mimeType: String!\n}\n\nscalar Creator\n\n\"\"\"\nImplement the DateTime<Utc> scalar\n\nThe input/output is a string in RFC3339 format.\n\"\"\"\nscalar DateTime\n\n\"\"\"\nThe `JSON` scalar type represents JSON values as specified by [ECMA-404](https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf).\n\"\"\"\nscalar JSON\n\n\"\"\"A scalar that can represent any JSON Object value.\"\"\"\nscalar JSONObject\n\n\"\"\"A single log line streamed from a pod\"\"\"\ntype LogEntry {\n  \"\"\"The log line content\"\"\"\n  content: String!\n  \"\"\"The name of the pod producing the log\"\"\"\n  podName: String!\n}\n\n\"\"\"The root mutation of the service\"\"\"\ntype Mutation {\n  submitWorkflowTemplate(name: String!, visit: VisitInput!, parameters: JSON!): Workflow!\n  instrumentSession(proposalNumber: Int!, instrumentSessionNumber: Int!): InstrumentSessionMutations\n  createOrValidateSamples(input: CreateOrValidateSampleInput!): CreateSamplesResponse!\n  createSamples(input: CreateSampleInput!): [Sample!]! @deprecated(reason: \"Will be replaced by createOrValidateSamples\")\n  sample(sampleId: UUID!): SampleMutations\n  createExperimentDefinition(input: CreateExperimentDefinitionInput!): ExperimentDefinition\n  experimentDefinition(id: UUID!): ExperimentDefinitionMutations\n}\n\n\"\"\"Represents Relay Node types\"\"\"\nunion NodeValue = Workflow\n\n\"\"\"Information about pagination in a connection\"\"\"\ntype PageInfo {\n  \"\"\"When paginating backwards, are there more items?\"\"\"\n  hasPreviousPage: Boolean!\n  \"\"\"When paginating forwards, are there more items?\"\"\"\n  hasNextPage: Boolean!\n  \"\"\"When paginating backwards, the cursor to continue.\"\"\"\n  startCursor: String\n  \"\"\"When paginating forwards, the cursor to continue.\"\"\"\n  endCursor: String\n}\n\n\"\"\"The root query of the service\"\"\"\ntype Query {\n  node(id: ID!): NodeValue\n  \"\"\"Get a single [`Workflow`] by proposal, visit, and name\"\"\"\n  workflow(visit: VisitInput!, name: String!): Workflow!\n  workflows(visit: VisitInput!, cursor: String, limit: Int, filter: WorkflowFilter): WorkflowConnection!\n  workflowTemplate(name: String!): WorkflowTemplate!\n  workflowTemplates(cursor: String, limit: Int, filter: WorkflowTemplatesFilter): WorkflowTemplateConnection!\n  \"\"\"Get a proposal by its number\"\"\"\n  proposal(proposalNumber: Int!): Proposal\n  \"\"\"Get a list of proposals\"\"\"\n  proposals(proposalCategory: String = null, first: Int = null, last: Int = null, after: String = null, before: String = null): ProposalConnection!\n  \"\"\"Get a instrument session\"\"\"\n  instrumentSession(proposalNumber: Int!, instrumentSessionNumber: Int!): InstrumentSession\n  \"\"\"Get a instrument session\"\"\"\n  instrumentSessions(proposalNumber: Int = null, proposalCategory: String = null): [InstrumentSession!]\n  \"\"\"Get an instrument\"\"\"\n  instrument(instrumentName: String!): Instrument\n  \"\"\"Get a list of instruments\"\"\"\n  instruments(scienceGroup: String = null): [Instrument!]!\n  \"\"\"Get an account\"\"\"\n  account(username: String!): Account\n  jsonSchema(url: String!): JSONSchema\n  jsonSchemas(type: String = null, instrument: String = null): [JSONSchema!]!\n  \"\"\"Get a sample by its id\"\"\"\n  sample(sampleId: UUID!): Sample\n  \"\"\"Get a list of samples associated with a given instrument session\"\"\"\n  samples(first: Int!, instrumentSessions: [InstrumentSessionInput!] = null, filter: SampleFilterInput! = {schemaUrl: null, createdTime: null, updatedTime: null, name: null, data: null}, before: String = null, after: String = null, last: Int = null, orderBy: SampleOrder! = {name: null, createdTime: null, updatedTime: null}): SampleConnection!\n  experimentDefinition(id: UUID!): ExperimentDefinition\n  experimentDefinitions(instrumentSessions: [InstrumentSessionInput!]!, first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentDefinitionConnection\n  experiment(id: UUID!): Experiment\n  experiments(instrumentSessions: [InstrumentSessionInput!]!, first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentConnection\n}\n\n\"\"\"Supported DLS science groups\"\"\"\nenum ScienceGroup {\n  \"\"\"Macromolecular Crystallography\"\"\"\n  MX\n  \"\"\"Workflows Examples\"\"\"\n  EXAMPLES\n  \"\"\"Magnetic Materials\"\"\"\n  MAGNETIC_MATERIALS\n  \"\"\"Soft Condensed Matter\"\"\"\n  CONDENSED_MATTER\n  \"\"\"Imaging and Microscopy\"\"\"\n  IMAGING\n  \"\"\"Biological Cryo-Imaging\"\"\"\n  BIO_CRYO_IMAGING\n  \"\"\"Structures and Surfaces\"\"\"\n  SURFACES\n  \"\"\"Crystallography\"\"\"\n  CRYSTALLOGRAPHY\n  \"\"\"Spectroscopy\"\"\"\n  SPECTROSCOPY\n}\n\n\"\"\"The root mutation of the service\"\"\"\ntype Subscription {\n  \"\"\"Processing to subscribe to logs for a single pod of a workflow\"\"\"\n  logs(visit: VisitInput!, workflowName: String!, taskId: String!): LogEntry!\n  \"\"\"Processing to subscribe to data for all workflows in a session\"\"\"\n  workflow(visit: VisitInput!, name: String!): Workflow!\n}\n\ntype Task {\n  \"\"\"Unique name of the task\"\"\"\n  id: String!\n  \"\"\"Display name of the task\"\"\"\n  name: String!\n  \"\"\"Current status of a task\"\"\"\n  status: TaskStatus!\n  \"\"\"Parent of a task\"\"\"\n  depends: [String!]!\n  \"\"\"Children of a task\"\"\"\n  dependencies: [String!]!\n  \"\"\"Artifacts produced by a task\"\"\"\n  artifacts: [Artifact!]!\n  \"\"\"Node type - Pod, DAG, etc\"\"\"\n  stepType: String!\n  \"\"\"Start time for a task on a workflow\"\"\"\n  startTime: DateTime\n  \"\"\"End time for a task on a workflow\"\"\"\n  endTime: DateTime\n  \"\"\"\n  A human readable message indicating details about why this step is in this condition\n  \"\"\"\n  message: String\n}\n\nenum TaskStatus {\n  PENDING\n  RUNNING\n  SUCCEEDED\n  SKIPPED\n  FAILED\n  ERROR\n  OMITTED\n}\n\nscalar Template\n\n\"\"\"Information about where the template is stored\"\"\"\ntype TemplateSource {\n  \"\"\"The URL of the GitHub repository\"\"\"\n  repositoryUrl: String!\n  \"\"\"The path to the template within the repository\"\"\"\n  path: String!\n  \"\"\"The current tracked branch of the repository\"\"\"\n  targetRevision: String!\n}\n\n\"\"\"\nURL is a String implementing the [URL Standard](http://url.spec.whatwg.org/)\n\"\"\"\nscalar Url\n\n\"\"\"A visit to an instrument as part of a session\"\"\"\ntype Visit {\n  \"\"\"Project Proposal Code\"\"\"\n  proposalCode: String!\n  \"\"\"Project Proposal Number\"\"\"\n  proposalNumber: Int!\n  \"\"\"Session visit Number\"\"\"\n  number: Int!\n}\n\n\"\"\"A visit to an instrument as part of a session\"\"\"\ninput VisitInput {\n  \"\"\"Project Proposal Code\"\"\"\n  proposalCode: String!\n  \"\"\"Project Proposal Number\"\"\"\n  proposalNumber: Int!\n  \"\"\"Session visit Number\"\"\"\n  number: Int!\n}\n\ntype Workflow {\n  \"\"\"The unique ID derived from the visit and name\"\"\"\n  id: ID!\n  \"\"\"The name given to the workflow, unique within a given visit\"\"\"\n  name: String!\n  \"\"\"The visit the Workflow was run against\"\"\"\n  visit: Visit!\n  \"\"\"The current status of the workflow\"\"\"\n  status: WorkflowStatus\n  \"\"\"The top-level workflow parameters\"\"\"\n  parameters: JSONObject\n  \"\"\"The name of the template used to run the workflow\"\"\"\n  templateRef: String\n  \"\"\"The workflow creator\"\"\"\n  creator: WorkflowCreator!\n}\n\ntype WorkflowConnection {\n  \"\"\"Information to aid in pagination.\"\"\"\n  pageInfo: PageInfo!\n  \"\"\"A list of edges.\"\"\"\n  edges: [WorkflowEdge!]!\n  \"\"\"A list of nodes.\"\"\"\n  nodes: [Workflow!]!\n}\n\n\"\"\"Information about the creator of a workflow.\"\"\"\ntype WorkflowCreator {\n  \"\"\"\n  An identifier unique to the creator of the workflow.\n  Typically this is the creator's Fed-ID.\n  \"\"\"\n  creatorId: String!\n}\n\n\"\"\"An edge in a connection.\"\"\"\ntype WorkflowEdge {\n  \"\"\"The item at the end of the edge\"\"\"\n  node: Workflow!\n  \"\"\"A cursor for use in pagination\"\"\"\n  cursor: String!\n}\n\n\"\"\"All tasks in the workflow have errored\"\"\"\ntype WorkflowErroredStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"All tasks in the workflow have failed\"\"\"\ntype WorkflowFailedStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"All the supported Workflows filters\"\"\"\ninput WorkflowFilter {\n  \"\"\"The status field for a workflow\"\"\"\n  workflowStatusFilter: WorkflowStatusFilter\n  \"\"\"The fedid of the user who created the workflow\"\"\"\n  creator: Creator\n  \"\"\"The name of the workflow template\"\"\"\n  template: Template\n}\n\ntype WorkflowPendingStatus {\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n}\n\ntype WorkflowRunningStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"The status of a workflow\"\"\"\nunion WorkflowStatus = WorkflowPendingStatus | WorkflowRunningStatus | WorkflowSucceededStatus | WorkflowFailedStatus | WorkflowErroredStatus\n\n\"\"\"Represents workflow status filters\"\"\"\ninput WorkflowStatusFilter {\n  pending: Boolean! = false\n  running: Boolean! = false\n  succeeded: Boolean! = false\n  failed: Boolean! = false\n  error: Boolean! = false\n}\n\n\"\"\"All tasks in the workflow have succeded\"\"\"\ntype WorkflowSucceededStatus {\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\ntype WorkflowTemplate {\n  \"\"\"The name given to the workflow template, globally unique\"\"\"\n  name: String!\n  \"\"\"The group who maintains the workflow template\"\"\"\n  maintainer: String!\n  \"\"\"A human readable title for the workflow template\"\"\"\n  title: String\n  \"\"\"A human readable description of the workflow which is created\"\"\"\n  description: String\n  \"\"\"The repository storing the code associated with this template.\"\"\"\n  repository: String\n  \"\"\"A JSON Schema describing the arguments of a Workflow Template\"\"\"\n  arguments: JSON!\n  \"\"\"\n  A JSON Forms UI Schema describing how to render the arguments of the Workflow Template\n  \"\"\"\n  uiSchema: JSON\n  \"\"\"Information about where the template is obtained from\"\"\"\n  templateSource: TemplateSource\n}\n\ntype WorkflowTemplateConnection {\n  \"\"\"Information to aid in pagination.\"\"\"\n  pageInfo: PageInfo!\n  \"\"\"A list of edges.\"\"\"\n  edges: [WorkflowTemplateEdge!]!\n  \"\"\"A list of nodes.\"\"\"\n  nodes: [WorkflowTemplate!]!\n}\n\n\"\"\"An edge in a connection.\"\"\"\ntype WorkflowTemplateEdge {\n  \"\"\"The item at the end of the edge\"\"\"\n  node: WorkflowTemplate!\n  \"\"\"A cursor for use in pagination\"\"\"\n  cursor: String!\n}\n\n\"\"\"Supported label filters for ClusterWorkflowTemplates\"\"\"\ninput WorkflowTemplatesFilter {\n  \"\"\"The science group owning the template eg imaging\"\"\"\n  scienceGroup: [ScienceGroup!]\n}\n\ntype Account {\n  accountId: Int!\n  username: String!\n  emailAddress: String\n  title: String\n  givenName: String\n  familyName: String\n  type: AccountType!\n  state: AccountState!\n  proposalRoles: [ProposalAccount!]!\n  instrumentSessionRoles: [InstrumentSessionRole!]!\n}\n\nenum AccountState {\n  enabled\n  disabled\n}\n\nenum AccountType {\n  user\n  staff\n  functional\n}\n\ntype Instrument {\n  name: String!\n  scienceGroup: String\n  description: String\n  proposals: [Proposal!]!\n  instrumentSessions: [InstrumentSession!]!\n}\n\ntype InstrumentSession {\n  instrumentSessionId: Int!\n  instrumentSessionNumber: Int!\n  startTime: DateTime\n  endTime: DateTime\n  type: String\n  state: String\n  riskRating: String\n  proposal: Proposal\n  instrument: Instrument!\n  roles: [InstrumentSessionRole!]!\n  \"\"\"Samples associated with a given instrument session\"\"\"\n  samples(first: Int!, filter: SampleFilterInput! = {schemaUrl: null, createdTime: null, updatedTime: null, name: null, data: null}, before: String = null, after: String = null, last: Int = null, orderBy: SampleOrder! = {name: null, createdTime: null, updatedTime: null}): SampleConnection!\n  \"\"\"Experiment Definitions\"\"\"\n  experimentDefinitions(first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentDefinitionConnection!\n  \"\"\"Experiments associated with this session\"\"\"\n  experiments(first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentConnection!\n}\n\ntype InstrumentSessionMutations {\n  instrumentSessionNumber: Int!\n  proposalNumber: Int!\n  \"\"\"Create or validate samples associated with this instrument session\"\"\"\n  createOrValidateSamples(input: CreateOrValidateSampleInputBase!): CreateSamplesResponse!\n}\n\ntype InstrumentSessionRole {\n  instrumentSession: InstrumentSession!\n  account: Account!\n  role: String!\n  onSite: Boolean!\n}\n\ntype Proposal {\n  proposalNumber: Int!\n  proposalCategory: String\n  title: String\n  summary: String\n  state: ProposalState!\n  instrumentSessions: [InstrumentSession!]!\n  instruments: [Instrument!]!\n  roles: [ProposalAccount!]!\n}\n\ntype ProposalAccount {\n  proposal: Proposal!\n  account: Account!\n  role: String!\n}\n\ntype ProposalConnection {\n  edges: [ProposalEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype ProposalEdge {\n  cursor: String!\n  node: Proposal!\n}\n\nenum ProposalState {\n  Open\n  Closed\n  Cancelled\n}\n\n\"\"\"A JSON schema\"\"\"\ntype JSONSchema {\n  \"\"\"The identifier of the schema\"\"\"\n  id: String!\n  \"\"\"A URL from which the schema can be accessed\"\"\"\n  url: String!\n  \"\"\"The type of object the shema describes (if known)\"\"\"\n  type: String\n  \"\"\"The title of the schema\"\"\"\n  title: String\n  \"\"\"The version of the schema\"\"\"\n  version: String\n  \"\"\"The instrument the schema was created for\"\"\"\n  instrument: String\n  \"\"\"The description og the schema\"\"\"\n  description: String\n}\n\ninput AddSampleEventInput {\n  description: String!\n}\n\ninput CreateOrValidateSampleInput {\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String!\n  \"\"\"Samples to be created\"\"\"\n  samples: [SampleIn!]!\n  \"\"\"\n  Whether or not the provided samples should only be validated and not created\n  \"\"\"\n  validateOnly: Boolean! = false\n  \"\"\"Number of the proposal the samples should be associated with\"\"\"\n  proposalNumber: Int!\n  \"\"\"Number of the instrument session the samples should be associated with\"\"\"\n  instrumentSessionNumber: Int!\n}\n\ninput CreateOrValidateSampleInputBase {\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String!\n  \"\"\"Samples to be created\"\"\"\n  samples: [SampleIn!]!\n  \"\"\"\n  Whether or not the provided samples should only be validated and not created\n  \"\"\"\n  validateOnly: Boolean! = false\n}\n\ninput CreateSampleInput {\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n  samples: [SampleInLegacy!]!\n  validateOnly: Boolean! = false\n}\n\n\"\"\"Return type when creating or validating samples\"\"\"\ntype CreateSamplesResponse {\n  \"\"\"Whether the operation has succeeded without validation errors\"\"\"\n  success: Boolean!\n  \"\"\"Samples that have been created\"\"\"\n  samples: [Sample!]!\n  \"\"\"Errors that occurred during sample validation\"\"\"\n  errors: [SampleValidationError!]!\n}\n\ninput DatetimeOperatorInput {\n  \"\"\"\n  Will filter to items where the `DateTime` field is greater than (i.e. after) the provided value\n  \"\"\"\n  gt: DateTime = null\n  \"\"\"\n  Will filter to items where the `DateTime` field is less than (i.e. before) the provided value\n  \"\"\"\n  lt: DateTime = null\n}\n\n\"\"\"The details of sample validation error\"\"\"\ntype ErrorDetails {\n  \"\"\"The type of error that occurred\"\"\"\n  type: String!\n  \"\"\"\n  Tuple of strings identifying where in the sample schema the error occurred.\n  \"\"\"\n  location: [String!]!\n  \"\"\"A human readable error message.\"\"\"\n  message: String!\n}\n\n\"\"\"Values required to uniquely identify an instrument session\"\"\"\ninput InstrumentSessionInput {\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n}\n\ninput JSONOperator @oneOf {\n  stringOperator: StringOperatorInput = null\n  datetimeOperator: DatetimeOperatorInput = null\n  numericOperator: NumericOperatorInput = null\n}\n\ninput JSONOperatorInput {\n  \"\"\"A JSON path specifying the value to filter. Must start with '$.'\"\"\"\n  path: String!\n  \"\"\"The operator to apply to the JSON field\"\"\"\n  operator: JSONOperator!\n}\n\ninput NumericOperatorInput {\n  \"\"\"\n  Will filter to items where the numeric field is greater than the provided value\n  \"\"\"\n  gt: Float = null\n  \"\"\"\n  Will filter to items where the numeric field is less than the provided value\n  \"\"\"\n  lt: Float = null\n}\n\ntype Sample {\n  id: UUID!\n  name: String!\n  data: JSON!\n  createdTime: DateTime!\n  updatedTime: DateTime!\n  dataSchemaUrl: String!\n  \"\"\"Samples from which this sample is derived\"\"\"\n  parents(first: Int = null, before: String = null, after: String = null, last: Int = null): SampleConnection!\n  \"\"\"Samples derived from this sample\"\"\"\n  children(first: Int = null, before: String = null, after: String = null, last: Int = null): SampleConnection!\n  \"\"\"Events linked to this sample\"\"\"\n  events(first: Int = null, before: String = null, after: String = null, last: Int = null): SampleEventConnection!\n  \"\"\"The JSON schema that the sample's `data` conforms to\"\"\"\n  dataSchema: JSON!\n  \"\"\"The instrument sessions that this sample is associated with\"\"\"\n  instrumentSessions: [InstrumentSession!]!\n  images: [SampleImage!]!\n  experiments: [Experiment!]!\n}\n\ntype SampleConnection {\n  edges: [SampleEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype SampleEdge {\n  cursor: String!\n  node: Sample!\n}\n\ntype SampleEvent {\n  id: UUID!\n  timestamp: DateTime!\n  description: String!\n}\n\ntype SampleEventConnection {\n  edges: [SampleEventEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype SampleEventEdge {\n  cursor: String!\n  node: SampleEvent!\n}\n\ninput SampleFilterInput {\n  \"\"\"Filter on the `schemaUrl` field of `Sample`\"\"\"\n  schemaUrl: StringOperatorInput = null\n  \"\"\"Filter on the `createdTime` field of `Sample`\"\"\"\n  createdTime: DatetimeOperatorInput = null\n  \"\"\"Filter on the `createdTime` field of `Sample`\"\"\"\n  updatedTime: DatetimeOperatorInput = null\n  \"\"\"Filter on the `name` field of `Sample`\"\"\"\n  name: StringOperatorInput = null\n  \"\"\"Filter on the `data` field of `Sample`\"\"\"\n  data: [JSONOperatorInput!] = null\n}\n\ntype SampleImage {\n  url: String!\n  filename: String!\n}\n\ninput SampleIn {\n  \"\"\"Name of the sample\"\"\"\n  name: String!\n  \"\"\"Data of the sample\"\"\"\n  data: JSON!\n}\n\ninput SampleInLegacy {\n  name: String!\n  data: JSON!\n  dataSchemaUrl: String!\n  parentIds: [Int!] = null\n  children: [SampleInLegacy!] = null\n}\n\ntype SampleMutations {\n  sampleId: UUID!\n  updateSample(input: UpdateSampleInput!): UpdateSampleResponse!\n  linkInstrumentSessionToSample(proposalNumber: Int!, instrumentSessionNumber: Int!): Void\n  addSampleEvent(sampleEvent: AddSampleEventInput!): SampleEvent!\n  createSampleImageUploadUrl(filename: String!, contentType: String!, contentLength: Int!): String!\n}\n\ninput SampleOrder {\n  name: SortingOrder = null\n  createdTime: SortingOrder = null\n  updatedTime: SortingOrder = null\n}\n\n\"\"\"The details of errors occurred when validating a sample\"\"\"\ntype SampleValidationError {\n  \"\"\"\n  The index of the sample in CreateSampleInput.samples for which the error occurred\n  \"\"\"\n  index: Int!\n  \"\"\"Errors that occurred when validating the sample\"\"\"\n  errors: [ErrorDetails!]!\n}\n\nenum SortingOrder {\n  ASC\n  DESC\n}\n\n\"\"\"Conditions used to filter results based on the value of a String field\"\"\"\ninput StringOperatorInput {\n  \"\"\"\n  Will filter to items where the `String` field is equal to the provided value\n  \"\"\"\n  eq: String = null\n  \"\"\"\n  Will filter to items where the `String` field is not equal to the provided value\n  \"\"\"\n  ne: String = null\n  \"\"\"\n  Will filter to items where the `String` field is a member of the provided value\n  \"\"\"\n  in: [String!] = null\n  \"\"\"\n  Will filter to items where the `String` field is not a member of the provided value\n  \"\"\"\n  nin: [String!] = null\n  \"\"\"\n  Will filter to items where the `String` field is contains the provided value\n  \"\"\"\n  contains: String = null\n}\n\nscalar UUID\n\ninput UpdateSampleInput {\n  \"\"\"Name of the sample\"\"\"\n  name: String\n  \"\"\"Data of the sample\"\"\"\n  data: JSON\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String\n}\n\n\"\"\"Return type when creating or validating samples\"\"\"\ntype UpdateSampleResponse {\n  \"\"\"Whether the operation has succeeded without validation errors\"\"\"\n  success: Boolean!\n  \"\"\"Sample that has been updated\"\"\"\n  sample: Sample\n  \"\"\"Errors that occurred during sample validation\"\"\"\n  errors: [SampleValidationError!]!\n}\n\n\"\"\"Represents NULL values\"\"\"\nscalar Void\n\n\"\"\"Values required to create an experiment definition\"\"\"\ninput CreateExperimentDefinitionInput {\n  name: String!\n  data: JSON!\n  dataSchemaUrl: String!\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n}\n\n\"\"\"Values required for the createExperiments mutation\"\"\"\ninput CreateExperimentsInput {\n  experiments: [ExperimentInput!]!\n}\n\ntype Experiment {\n  id: UUID!\n  name: String!\n  createdTime: DateTime!\n  updatedTime: DateTime!\n  experimentDefinition: ExperimentDefinition!\n  \"\"\"The sample that this experiment is associated with\"\"\"\n  sample: Sample!\n}\n\ntype ExperimentConnection {\n  edges: [ExperimentEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype ExperimentDefinition {\n  id: UUID!\n  name: String!\n  createdTime: DateTime!\n  updatedTime: DateTime!\n  data: JSON!\n  dataSchemaUrl: String!\n  proposalNumber: Int!\n  instrumentSessionNumber: Int!\n  \"\"\"\n  The instrument session that this experiment definition is associated with\n  \"\"\"\n  instrumentSession: InstrumentSession!\n  \"\"\"Experiments associated with this experiment definition\"\"\"\n  experiments: [Experiment!]!\n}\n\ntype ExperimentDefinitionConnection {\n  edges: [ExperimentDefinitionEdge!]!\n  pageInfo: PageInfo!\n}\n\ntype ExperimentDefinitionEdge {\n  cursor: String!\n  node: ExperimentDefinition!\n}\n\n\"\"\"Mutations for a given experiment defintion\"\"\"\ntype ExperimentDefinitionMutations {\n  createExperiments(input: CreateExperimentsInput!): [Experiment!]!\n}\n\ntype ExperimentEdge {\n  cursor: String!\n  node: Experiment!\n}\n\n\"\"\"Values required to create an experiment\"\"\"\ninput ExperimentInput {\n  name: String!\n  sampleId: UUID!\n}",
     "stringStorage": {
+      "60b3f57ff30215a4bb01d6659c2b5ac5e2a97765": "schema @link(url: \"https://specs.apollo.dev/federation/v2.7\", import: [\"@external\", \"@key\", \"@provides\", \"@shareable\"]) {\n  query: Query\n  mutation: Mutation\n}\n\ndirective @external on FIELD_DEFINITION | OBJECT\n\ndirective @key(fields: openfed__FieldSet!, resolvable: Boolean = true) repeatable on INTERFACE | OBJECT\n\ndirective @link(as: String, for: link__Purpose, import: [link__Import], url: String!) repeatable on SCHEMA\n\ndirective @provides(fields: openfed__FieldSet!) on FIELD_DEFINITION\n\ndirective @shareable repeatable on FIELD_DEFINITION | OBJECT\n\n\"\"\"Values required to create an experiment definition\"\"\"\ninput CreateExperimentDefinitionInput {\n  data: JSON!\n  dataSchemaUrl: String!\n  instrumentSessionNumber: Int!\n  name: String!\n  proposalNumber: Int!\n}\n\n\"\"\"Values required for the createExperiments mutation\"\"\"\ninput CreateExperimentsInput {\n  experiments: [ExperimentInput!]!\n}\n\n\"\"\"Date with time (isoformat)\"\"\"\nscalar DateTime\n\ntype Experiment @key(fields: \"id\") {\n  createdTime: DateTime!\n  experimentDefinition: ExperimentDefinition!\n  id: UUID!\n  name: String!\n  \"\"\"The sample that this experiment is associated with\"\"\"\n  sample: Sample! @provides(fields: \"id\")\n  updatedTime: DateTime!\n}\n\ntype ExperimentConnection {\n  edges: [ExperimentEdge!]! @shareable\n  pageInfo: PageInfo! @shareable\n}\n\ntype ExperimentDefinition {\n  createdTime: DateTime!\n  data: JSON!\n  dataSchemaUrl: String!\n  \"\"\"Experiments associated with this experiment definition\"\"\"\n  experiments: [Experiment!]!\n  id: UUID!\n  \"\"\"\n  The instrument session that this experiment definition is associated with\n  \"\"\"\n  instrumentSession: InstrumentSession! @provides(fields: \"instrumentSessionNumber proposal{ proposalNumber }\")\n  instrumentSessionNumber: Int!\n  name: String!\n  proposalNumber: Int!\n  updatedTime: DateTime!\n}\n\ntype ExperimentDefinitionConnection {\n  edges: [ExperimentDefinitionEdge!]! @shareable\n  pageInfo: PageInfo! @shareable\n}\n\ntype ExperimentDefinitionEdge {\n  cursor: String! @shareable\n  node: ExperimentDefinition! @shareable\n}\n\n\"\"\"Mutations for a given experiment defintion\"\"\"\ntype ExperimentDefinitionMutations {\n  createExperiments(input: CreateExperimentsInput!): [Experiment!]!\n}\n\ntype ExperimentEdge {\n  cursor: String! @shareable\n  node: Experiment! @shareable\n}\n\n\"\"\"Values required to create an experiment\"\"\"\ninput ExperimentInput {\n  name: String!\n  sampleId: UUID!\n}\n\ntype InstrumentSession @key(fields: \"instrumentSessionNumber proposal { proposalNumber }\") {\n  \"\"\"Experiment Definitions\"\"\"\n  experimentDefinitions(after: String = null, before: String = null, first: Int = null, last: Int = null): ExperimentDefinitionConnection!\n  \"\"\"Experiments associated with this session\"\"\"\n  experiments(after: String = null, before: String = null, first: Int = null, last: Int = null): ExperimentConnection!\n  instrumentSessionNumber: Int! @external\n  proposal: Proposal @external\n}\n\n\"\"\"Values required to uniquely identify an instrument session\"\"\"\ninput InstrumentSessionInput {\n  instrumentSessionNumber: Int!\n  proposalNumber: Int!\n}\n\n\"\"\"\nThe `JSON` scalar type represents JSON values as specified by [ECMA-404](https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf).\n\"\"\"\nscalar JSON @specifiedBy(url: \"https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf\")\n\ntype Mutation {\n  createExperimentDefinition(input: CreateExperimentDefinitionInput!): ExperimentDefinition\n  experimentDefinition(id: UUID!): ExperimentDefinitionMutations\n}\n\ntype PageInfo {\n  endCursor: String @shareable\n  hasNextPage: Boolean! @shareable\n  hasPreviousPage: Boolean! @shareable\n  startCursor: String @shareable\n}\n\ntype Proposal @key(fields: \"proposalNumber\") {\n  proposalNumber: Int! @external\n}\n\ntype Query {\n  experiment(id: UUID!): Experiment\n  experimentDefinition(id: UUID!): ExperimentDefinition\n  experimentDefinitions(after: String = null, before: String = null, first: Int = null, instrumentSessions: [InstrumentSessionInput!]!, last: Int = null): ExperimentDefinitionConnection\n  experiments(after: String = null, before: String = null, first: Int = null, instrumentSessions: [InstrumentSessionInput!]!, last: Int = null): ExperimentConnection\n}\n\ntype Sample @key(fields: \"id\") {\n  experiments: [Experiment!]!\n  id: UUID! @external\n}\n\nscalar UUID\n\nscalar link__Import\n\nenum link__Purpose {\n  EXECUTION\n  SECURITY\n}\n\nscalar openfed__FieldSet",
       "636412ec8e6e8e278df46ce7da59047a6182733c": "schema @link(url: \"https://specs.apollo.dev/federation/v2.7\", import: [\"@key\", \"@shareable\"]) {\n  query: Query\n  mutation: Mutation\n}\n\ndirective @key(fields: openfed__FieldSet!, resolvable: Boolean = true) repeatable on INTERFACE | OBJECT\n\ndirective @link(as: String, for: link__Purpose, import: [link__Import], url: String!) repeatable on SCHEMA\n\ndirective @shareable repeatable on FIELD_DEFINITION | OBJECT\n\ntype Account {\n  accountId: Int!\n  emailAddress: String\n  familyName: String\n  givenName: String\n  instrumentSessionRoles: [InstrumentSessionRole!]!\n  proposalRoles: [ProposalAccount!]!\n  state: AccountState!\n  title: String\n  type: AccountType!\n  username: String!\n}\n\nenum AccountState {\n  disabled\n  enabled\n}\n\nenum AccountType {\n  functional\n  staff\n  user\n}\n\n\"\"\"Date with time (isoformat)\"\"\"\nscalar DateTime\n\ntype Instrument {\n  description: String\n  instrumentSessions: [InstrumentSession!]!\n  name: String!\n  proposals: [Proposal!]!\n  scienceGroup: String\n}\n\ntype InstrumentSession @key(fields: \"instrumentSessionNumber proposal {proposalNumber}\") {\n  endTime: DateTime\n  instrument: Instrument!\n  instrumentSessionId: Int!\n  instrumentSessionNumber: Int!\n  proposal: Proposal\n  riskRating: String\n  roles: [InstrumentSessionRole!]!\n  startTime: DateTime\n  state: String\n  type: String\n}\n\ntype InstrumentSessionMutations @key(fields: \"instrumentSessionNumber proposalNumber\") {\n  instrumentSessionNumber: Int!\n  proposalNumber: Int!\n}\n\ntype InstrumentSessionRole {\n  account: Account!\n  instrumentSession: InstrumentSession!\n  onSite: Boolean!\n  role: String!\n}\n\ntype Mutation {\n  instrumentSession(instrumentSessionNumber: Int!, proposalNumber: Int!): InstrumentSessionMutations\n}\n\ntype PageInfo {\n  endCursor: String @shareable\n  hasNextPage: Boolean! @shareable\n  hasPreviousPage: Boolean! @shareable\n  startCursor: String @shareable\n}\n\ntype Proposal @key(fields: \"proposalNumber\") {\n  instrumentSessions: [InstrumentSession!]!\n  instruments: [Instrument!]!\n  proposalCategory: String\n  proposalNumber: Int!\n  roles: [ProposalAccount!]!\n  state: ProposalState!\n  summary: String\n  title: String\n}\n\ntype ProposalAccount {\n  account: Account!\n  proposal: Proposal!\n  role: String!\n}\n\ntype ProposalConnection {\n  edges: [ProposalEdge!]! @shareable\n  pageInfo: PageInfo! @shareable\n}\n\ntype ProposalEdge {\n  cursor: String! @shareable\n  node: Proposal! @shareable\n}\n\nenum ProposalState {\n  Cancelled\n  Closed\n  Open\n}\n\ntype Query {\n  \"\"\"Get an account\"\"\"\n  account(username: String!): Account\n  \"\"\"Get an instrument\"\"\"\n  instrument(instrumentName: String!): Instrument\n  \"\"\"Get a instrument session\"\"\"\n  instrumentSession(instrumentSessionNumber: Int!, proposalNumber: Int!): InstrumentSession\n  \"\"\"Get a instrument session\"\"\"\n  instrumentSessions(proposalCategory: String = null, proposalNumber: Int = null): [InstrumentSession!]\n  \"\"\"Get a list of instruments\"\"\"\n  instruments(scienceGroup: String = null): [Instrument!]!\n  \"\"\"Get a proposal by its number\"\"\"\n  proposal(proposalNumber: Int!): Proposal\n  \"\"\"Get a list of proposals\"\"\"\n  proposals(after: String = null, before: String = null, first: Int = null, last: Int = null, proposalCategory: String = null): ProposalConnection!\n}\n\nscalar link__Import\n\nenum link__Purpose {\n  EXECUTION\n  SECURITY\n}\n\nscalar openfed__FieldSet",
       "7e5c4524e9c7fbcc79549ca50a9bec18f0194dc2": "schema @link(url: \"https://specs.apollo.dev/federation/v2.7\", import: [\"@external\", \"@key\", \"@provides\", \"@shareable\"]) {\n  query: Query\n  mutation: Mutation\n}\n\ndirective @external on FIELD_DEFINITION | OBJECT\n\ndirective @key(fields: openfed__FieldSet!, resolvable: Boolean = true) repeatable on INTERFACE | OBJECT\n\ndirective @link(as: String, for: link__Purpose, import: [link__Import], url: String!) repeatable on SCHEMA\n\ndirective @provides(fields: openfed__FieldSet!) on FIELD_DEFINITION\n\ndirective @shareable repeatable on FIELD_DEFINITION | OBJECT\n\ninput AddSampleEventInput {\n  description: String!\n}\n\ninput CreateOrValidateSampleInput {\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String!\n  \"\"\"Number of the instrument session the samples should be associated with\"\"\"\n  instrumentSessionNumber: Int!\n  \"\"\"Number of the proposal the samples should be associated with\"\"\"\n  proposalNumber: Int!\n  \"\"\"Samples to be created\"\"\"\n  samples: [SampleIn!]!\n  \"\"\"\n  Whether or not the provided samples should only be validated and not created\n  \"\"\"\n  validateOnly: Boolean! = false\n}\n\ninput CreateOrValidateSampleInputBase {\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String!\n  \"\"\"Samples to be created\"\"\"\n  samples: [SampleIn!]!\n  \"\"\"\n  Whether or not the provided samples should only be validated and not created\n  \"\"\"\n  validateOnly: Boolean! = false\n}\n\ninput CreateSampleInput {\n  instrumentSessionNumber: Int!\n  proposalNumber: Int!\n  samples: [SampleInLegacy!]!\n  validateOnly: Boolean! = false\n}\n\n\"\"\"Return type when creating or validating samples\"\"\"\ntype CreateSamplesResponse {\n  \"\"\"Errors that occurred during sample validation\"\"\"\n  errors: [SampleValidationError!]!\n  \"\"\"Samples that have been created\"\"\"\n  samples: [Sample!]!\n  \"\"\"Whether the operation has succeeded without validation errors\"\"\"\n  success: Boolean!\n}\n\n\"\"\"Date with time (isoformat)\"\"\"\nscalar DateTime\n\ninput DatetimeOperatorInput {\n  \"\"\"\n  Will filter to items where the `DateTime` field is greater than (i.e. after) the provided value\n  \"\"\"\n  gt: DateTime = null\n  \"\"\"\n  Will filter to items where the `DateTime` field is less than (i.e. before) the provided value\n  \"\"\"\n  lt: DateTime = null\n}\n\n\"\"\"The details of sample validation error\"\"\"\ntype ErrorDetails {\n  \"\"\"\n  Tuple of strings identifying where in the sample schema the error occurred.\n  \"\"\"\n  location: [String!]!\n  \"\"\"A human readable error message.\"\"\"\n  message: String!\n  \"\"\"The type of error that occurred\"\"\"\n  type: String!\n}\n\ntype InstrumentSession @key(fields: \"instrumentSessionNumber proposal { proposalNumber }\") {\n  instrumentSessionNumber: Int! @external\n  proposal: Proposal @external\n  \"\"\"Samples associated with a given instrument session\"\"\"\n  samples(after: String = null, before: String = null, filter: SampleFilterInput! = {createdTime: null, data: null, name: null, schemaUrl: null, updatedTime: null}, first: Int!, last: Int = null, orderBy: SampleOrder! = {createdTime: null, name: null, updatedTime: null}): SampleConnection!\n}\n\ninput InstrumentSessionInput {\n  instrumentSessionNumber: Int!\n  proposalNumber: Int!\n}\n\ntype InstrumentSessionMutations @key(fields: \"instrumentSessionNumber proposalNumber\") {\n  \"\"\"Create or validate samples associated with this instrument session\"\"\"\n  createOrValidateSamples(input: CreateOrValidateSampleInputBase!): CreateSamplesResponse!\n  instrumentSessionNumber: Int! @external\n  proposalNumber: Int! @external\n}\n\n\"\"\"\nThe `JSON` scalar type represents JSON values as specified by [ECMA-404](https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf).\n\"\"\"\nscalar JSON @specifiedBy(url: \"https://ecma-international.org/wp-content/uploads/ECMA-404_2nd_edition_december_2017.pdf\")\n\ninput JSONOperator @oneOf {\n  datetimeOperator: DatetimeOperatorInput = null\n  numericOperator: NumericOperatorInput = null\n  stringOperator: StringOperatorInput = null\n}\n\ninput JSONOperatorInput {\n  \"\"\"The operator to apply to the JSON field\"\"\"\n  operator: JSONOperator!\n  \"\"\"A JSON path specifying the value to filter. Must start with '$.'\"\"\"\n  path: String!\n}\n\ntype Mutation {\n  createOrValidateSamples(input: CreateOrValidateSampleInput!): CreateSamplesResponse!\n  createSamples(input: CreateSampleInput!): [Sample!]! @deprecated(reason: \"Will be replaced by createOrValidateSamples\")\n  sample(sampleId: UUID!): SampleMutations\n}\n\ninput NumericOperatorInput {\n  \"\"\"\n  Will filter to items where the numeric field is greater than the provided value\n  \"\"\"\n  gt: Float = null\n  \"\"\"\n  Will filter to items where the numeric field is less than the provided value\n  \"\"\"\n  lt: Float = null\n}\n\ntype PageInfo {\n  endCursor: String @shareable\n  hasNextPage: Boolean! @shareable\n  hasPreviousPage: Boolean! @shareable\n  startCursor: String @shareable\n}\n\ntype Proposal @key(fields: \"proposalNumber\") {\n  proposalNumber: Int! @external\n}\n\ntype Query {\n  \"\"\"Get a sample by its id\"\"\"\n  sample(sampleId: UUID!): Sample\n  \"\"\"Get a list of samples associated with a given instrument session\"\"\"\n  samples(after: String = null, before: String = null, filter: SampleFilterInput! = {createdTime: null, data: null, name: null, schemaUrl: null, updatedTime: null}, first: Int!, instrumentSessions: [InstrumentSessionInput!] = null, last: Int = null, orderBy: SampleOrder! = {createdTime: null, name: null, updatedTime: null}): SampleConnection!\n}\n\ntype Sample @key(fields: \"id\") {\n  \"\"\"Samples derived from this sample\"\"\"\n  children(after: String = null, before: String = null, first: Int = null, last: Int = null): SampleConnection!\n  createdTime: DateTime!\n  data: JSON!\n  \"\"\"The JSON schema that the sample's `data` conforms to\"\"\"\n  dataSchema: JSON!\n  dataSchemaUrl: String!\n  \"\"\"Events linked to this sample\"\"\"\n  events(after: String = null, before: String = null, first: Int = null, last: Int = null): SampleEventConnection!\n  id: UUID!\n  images: [SampleImage!]!\n  \"\"\"The instrument sessions that this sample is associated with\"\"\"\n  instrumentSessions: [InstrumentSession!]! @provides(fields: \"instrumentSessionNumber proposal{ proposalNumber }\")\n  name: String!\n  \"\"\"Samples from which this sample is derived\"\"\"\n  parents(after: String = null, before: String = null, first: Int = null, last: Int = null): SampleConnection!\n  updatedTime: DateTime!\n}\n\ntype SampleConnection {\n  edges: [SampleEdge!]! @shareable\n  pageInfo: PageInfo! @shareable\n}\n\ntype SampleEdge {\n  cursor: String! @shareable\n  node: Sample! @shareable\n}\n\ntype SampleEvent {\n  description: String!\n  id: UUID!\n  timestamp: DateTime!\n}\n\ntype SampleEventConnection {\n  edges: [SampleEventEdge!]! @shareable\n  pageInfo: PageInfo! @shareable\n}\n\ntype SampleEventEdge {\n  cursor: String! @shareable\n  node: SampleEvent! @shareable\n}\n\ninput SampleFilterInput {\n  \"\"\"Filter on the `createdTime` field of `Sample`\"\"\"\n  createdTime: DatetimeOperatorInput = null\n  \"\"\"Filter on the `data` field of `Sample`\"\"\"\n  data: [JSONOperatorInput!] = null\n  \"\"\"Filter on the `name` field of `Sample`\"\"\"\n  name: StringOperatorInput = null\n  \"\"\"Filter on the `schemaUrl` field of `Sample`\"\"\"\n  schemaUrl: StringOperatorInput = null\n  \"\"\"Filter on the `createdTime` field of `Sample`\"\"\"\n  updatedTime: DatetimeOperatorInput = null\n}\n\ntype SampleImage {\n  filename: String!\n  url: String!\n}\n\ninput SampleIn {\n  \"\"\"Data of the sample\"\"\"\n  data: JSON!\n  \"\"\"Name of the sample\"\"\"\n  name: String!\n}\n\ninput SampleInLegacy {\n  children: [SampleInLegacy!] = null\n  data: JSON!\n  dataSchemaUrl: String!\n  name: String!\n  parentIds: [Int!] = null\n}\n\ntype SampleMutations {\n  addSampleEvent(sampleEvent: AddSampleEventInput!): SampleEvent!\n  createSampleImageUploadUrl(contentLength: Int!, contentType: String!, filename: String!): String!\n  linkInstrumentSessionToSample(instrumentSessionNumber: Int!, proposalNumber: Int!): Void\n  sampleId: UUID!\n  updateSample(input: UpdateSampleInput!): UpdateSampleResponse!\n}\n\ninput SampleOrder {\n  createdTime: SortingOrder = null\n  name: SortingOrder = null\n  updatedTime: SortingOrder = null\n}\n\n\"\"\"The details of errors occurred when validating a sample\"\"\"\ntype SampleValidationError {\n  \"\"\"Errors that occurred when validating the sample\"\"\"\n  errors: [ErrorDetails!]!\n  \"\"\"\n  The index of the sample in CreateSampleInput.samples for which the error occurred\n  \"\"\"\n  index: Int!\n}\n\nenum SortingOrder {\n  ASC\n  DESC\n}\n\n\"\"\"Conditions used to filter results based on the value of a String field\"\"\"\ninput StringOperatorInput {\n  \"\"\"\n  Will filter to items where the `String` field is contains the provided value\n  \"\"\"\n  contains: String = null\n  \"\"\"\n  Will filter to items where the `String` field is equal to the provided value\n  \"\"\"\n  eq: String = null\n  \"\"\"\n  Will filter to items where the `String` field is a member of the provided value\n  \"\"\"\n  in: [String!] = null\n  \"\"\"\n  Will filter to items where the `String` field is not equal to the provided value\n  \"\"\"\n  ne: String = null\n  \"\"\"\n  Will filter to items where the `String` field is not a member of the provided value\n  \"\"\"\n  nin: [String!] = null\n}\n\nscalar UUID\n\ninput UpdateSampleInput {\n  \"\"\"Data of the sample\"\"\"\n  data: JSON\n  \"\"\"URL of the JSON schema the samples' `data` should be validated against\"\"\"\n  dataSchemaUrl: String\n  \"\"\"Name of the sample\"\"\"\n  name: String\n}\n\n\"\"\"Return type when creating or validating samples\"\"\"\ntype UpdateSampleResponse {\n  \"\"\"Errors that occurred during sample validation\"\"\"\n  errors: [SampleValidationError!]!\n  \"\"\"Sample that has been updated\"\"\"\n  sample: Sample\n  \"\"\"Whether the operation has succeeded without validation errors\"\"\"\n  success: Boolean!\n}\n\n\"\"\"Represents NULL values\"\"\"\nscalar Void\n\nscalar link__Import\n\nenum link__Purpose {\n  EXECUTION\n  SECURITY\n}\n\nscalar openfed__FieldSet",
       "9a94e595184badd514b2996e26945dd3a01edbf4": "schema {\n  query: Query\n  mutation: Mutation\n  subscription: Subscription\n}\n\ndirective @link(as: String, for: link__Purpose, import: [link__Import], url: String!) repeatable on SCHEMA\n\ndirective @shareable repeatable on FIELD_DEFINITION | OBJECT\n\ntype Artifact {\n  \"\"\"The MIME type of the artifact data\"\"\"\n  mimeType: String!\n  \"\"\"The file name of the artifact\"\"\"\n  name: String!\n  \"\"\"The download URL for the artifact\"\"\"\n  url: Url!\n}\n\nscalar Creator\n\n\"\"\"\nImplement the DateTime<Utc> scalar\n\nThe input/output is a string in RFC3339 format.\n\"\"\"\nscalar DateTime\n\n\"\"\"A scalar that can represent any JSON value.\"\"\"\nscalar JSON\n\n\"\"\"A scalar that can represent any JSON Object value.\"\"\"\nscalar JSONObject\n\n\"\"\"A single log line streamed from a pod\"\"\"\ntype LogEntry {\n  \"\"\"The log line content\"\"\"\n  content: String!\n  \"\"\"The name of the pod producing the log\"\"\"\n  podName: String!\n}\n\n\"\"\"The root mutation of the service\"\"\"\ntype Mutation {\n  submitWorkflowTemplate(name: String!, parameters: JSON!, visit: VisitInput!): Workflow!\n}\n\n\"\"\"Represents Relay Node types\"\"\"\nunion NodeValue = Workflow\n\n\"\"\"Information about pagination in a connection\"\"\"\ntype PageInfo {\n  \"\"\"When paginating forwards, the cursor to continue.\"\"\"\n  endCursor: String @shareable\n  \"\"\"When paginating forwards, are there more items?\"\"\"\n  hasNextPage: Boolean! @shareable\n  \"\"\"When paginating backwards, are there more items?\"\"\"\n  hasPreviousPage: Boolean! @shareable\n  \"\"\"When paginating backwards, the cursor to continue.\"\"\"\n  startCursor: String @shareable\n}\n\n\"\"\"The root query of the service\"\"\"\ntype Query {\n  node(id: ID!): NodeValue\n  \"\"\"Get a single [`Workflow`] by proposal, visit, and name\"\"\"\n  workflow(name: String!, visit: VisitInput!): Workflow!\n  workflowTemplate(name: String!): WorkflowTemplate!\n  workflowTemplates(cursor: String, filter: WorkflowTemplatesFilter, limit: Int): WorkflowTemplateConnection!\n  workflows(cursor: String, filter: WorkflowFilter, limit: Int, visit: VisitInput!): WorkflowConnection!\n}\n\n\"\"\"Supported DLS science groups\"\"\"\nenum ScienceGroup {\n  \"\"\"Biological Cryo-Imaging\"\"\"\n  BIO_CRYO_IMAGING\n  \"\"\"Soft Condensed Matter\"\"\"\n  CONDENSED_MATTER\n  \"\"\"Crystallography\"\"\"\n  CRYSTALLOGRAPHY\n  \"\"\"Workflows Examples\"\"\"\n  EXAMPLES\n  \"\"\"Imaging and Microscopy\"\"\"\n  IMAGING\n  \"\"\"Magnetic Materials\"\"\"\n  MAGNETIC_MATERIALS\n  \"\"\"Macromolecular Crystallography\"\"\"\n  MX\n  \"\"\"Spectroscopy\"\"\"\n  SPECTROSCOPY\n  \"\"\"Structures and Surfaces\"\"\"\n  SURFACES\n}\n\n\"\"\"The root mutation of the service\"\"\"\ntype Subscription {\n  \"\"\"Processing to subscribe to logs for a single pod of a workflow\"\"\"\n  logs(taskId: String!, visit: VisitInput!, workflowName: String!): LogEntry!\n  \"\"\"Processing to subscribe to data for all workflows in a session\"\"\"\n  workflow(name: String!, visit: VisitInput!): Workflow!\n}\n\ntype Task {\n  \"\"\"Artifacts produced by a task\"\"\"\n  artifacts: [Artifact!]!\n  \"\"\"Children of a task\"\"\"\n  dependencies: [String!]!\n  \"\"\"Parent of a task\"\"\"\n  depends: [String!]!\n  \"\"\"End time for a task on a workflow\"\"\"\n  endTime: DateTime\n  \"\"\"Unique name of the task\"\"\"\n  id: String!\n  \"\"\"\n  A human readable message indicating details about why this step is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Display name of the task\"\"\"\n  name: String!\n  \"\"\"Start time for a task on a workflow\"\"\"\n  startTime: DateTime\n  \"\"\"Current status of a task\"\"\"\n  status: TaskStatus!\n  \"\"\"Node type - Pod, DAG, etc\"\"\"\n  stepType: String!\n}\n\nenum TaskStatus {\n  ERROR\n  FAILED\n  OMITTED\n  PENDING\n  RUNNING\n  SKIPPED\n  SUCCEEDED\n}\n\nscalar Template\n\n\"\"\"Information about where the template is stored\"\"\"\ntype TemplateSource {\n  \"\"\"The path to the template within the repository\"\"\"\n  path: String!\n  \"\"\"The URL of the GitHub repository\"\"\"\n  repositoryUrl: String!\n  \"\"\"The current tracked branch of the repository\"\"\"\n  targetRevision: String!\n}\n\n\"\"\"\nURL is a String implementing the [URL Standard](http://url.spec.whatwg.org/)\n\"\"\"\nscalar Url\n\n\"\"\"A visit to an instrument as part of a session\"\"\"\ntype Visit {\n  \"\"\"Session visit Number\"\"\"\n  number: Int!\n  \"\"\"Project Proposal Code\"\"\"\n  proposalCode: String!\n  \"\"\"Project Proposal Number\"\"\"\n  proposalNumber: Int!\n}\n\n\"\"\"A visit to an instrument as part of a session\"\"\"\ninput VisitInput {\n  \"\"\"Session visit Number\"\"\"\n  number: Int!\n  \"\"\"Project Proposal Code\"\"\"\n  proposalCode: String!\n  \"\"\"Project Proposal Number\"\"\"\n  proposalNumber: Int!\n}\n\ntype Workflow {\n  \"\"\"The workflow creator\"\"\"\n  creator: WorkflowCreator!\n  \"\"\"The unique ID derived from the visit and name\"\"\"\n  id: ID!\n  \"\"\"The name given to the workflow, unique within a given visit\"\"\"\n  name: String!\n  \"\"\"The top-level workflow parameters\"\"\"\n  parameters: JSONObject\n  \"\"\"The current status of the workflow\"\"\"\n  status: WorkflowStatus\n  \"\"\"The name of the template used to run the workflow\"\"\"\n  templateRef: String\n  \"\"\"The visit the Workflow was run against\"\"\"\n  visit: Visit!\n}\n\ntype WorkflowConnection {\n  \"\"\"A list of edges.\"\"\"\n  edges: [WorkflowEdge!]! @shareable\n  \"\"\"A list of nodes.\"\"\"\n  nodes: [Workflow!]! @shareable\n  \"\"\"Information to aid in pagination.\"\"\"\n  pageInfo: PageInfo! @shareable\n}\n\n\"\"\"Information about the creator of a workflow.\"\"\"\ntype WorkflowCreator {\n  \"\"\"\n  An identifier unique to the creator of the workflow.\n  Typically this is the creator's Fed-ID.\n  \"\"\"\n  creatorId: String!\n}\n\n\"\"\"An edge in a connection.\"\"\"\ntype WorkflowEdge {\n  \"\"\"A cursor for use in pagination\"\"\"\n  cursor: String! @shareable\n  \"\"\"The item at the end of the edge\"\"\"\n  node: Workflow! @shareable\n}\n\n\"\"\"All tasks in the workflow have errored\"\"\"\ntype WorkflowErroredStatus {\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"All tasks in the workflow have failed\"\"\"\ntype WorkflowFailedStatus {\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"All the supported Workflows filters\"\"\"\ninput WorkflowFilter {\n  \"\"\"The fedid of the user who created the workflow\"\"\"\n  creator: Creator\n  \"\"\"The name of the workflow template\"\"\"\n  template: Template\n  \"\"\"The status field for a workflow\"\"\"\n  workflowStatusFilter: WorkflowStatusFilter\n}\n\ntype WorkflowPendingStatus {\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n}\n\ntype WorkflowRunningStatus {\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\n\"\"\"The status of a workflow\"\"\"\nunion WorkflowStatus = WorkflowErroredStatus | WorkflowFailedStatus | WorkflowPendingStatus | WorkflowRunningStatus | WorkflowSucceededStatus\n\n\"\"\"Represents workflow status filters\"\"\"\ninput WorkflowStatusFilter {\n  error: Boolean! = false\n  failed: Boolean! = false\n  pending: Boolean! = false\n  running: Boolean! = false\n  succeeded: Boolean! = false\n}\n\n\"\"\"All tasks in the workflow have succeded\"\"\"\ntype WorkflowSucceededStatus {\n  \"\"\"Time at which this workflow completed\"\"\"\n  endTime: DateTime!\n  \"\"\"\n  A human readable message indicating details about why the workflow is in this condition\n  \"\"\"\n  message: String\n  \"\"\"Time at which this workflow started\"\"\"\n  startTime: DateTime!\n  \"\"\"Tasks created by the workflow\"\"\"\n  tasks: [Task!]!\n}\n\ntype WorkflowTemplate {\n  \"\"\"A JSON Schema describing the arguments of a Workflow Template\"\"\"\n  arguments: JSON!\n  \"\"\"A human readable description of the workflow which is created\"\"\"\n  description: String\n  \"\"\"The group who maintains the workflow template\"\"\"\n  maintainer: String!\n  \"\"\"The name given to the workflow template, globally unique\"\"\"\n  name: String!\n  \"\"\"The repository storing the code associated with this template.\"\"\"\n  repository: String\n  \"\"\"Information about where the template is obtained from\"\"\"\n  templateSource: TemplateSource\n  \"\"\"A human readable title for the workflow template\"\"\"\n  title: String\n  \"\"\"\n  A JSON Forms UI Schema describing how to render the arguments of the Workflow Template\n  \"\"\"\n  uiSchema: JSON\n}\n\ntype WorkflowTemplateConnection {\n  \"\"\"A list of edges.\"\"\"\n  edges: [WorkflowTemplateEdge!]! @shareable\n  \"\"\"A list of nodes.\"\"\"\n  nodes: [WorkflowTemplate!]! @shareable\n  \"\"\"Information to aid in pagination.\"\"\"\n  pageInfo: PageInfo! @shareable\n}\n\n\"\"\"An edge in a connection.\"\"\"\ntype WorkflowTemplateEdge {\n  \"\"\"A cursor for use in pagination\"\"\"\n  cursor: String! @shareable\n  \"\"\"The item at the end of the edge\"\"\"\n  node: WorkflowTemplate! @shareable\n}\n\n\"\"\"Supported label filters for ClusterWorkflowTemplates\"\"\"\ninput WorkflowTemplatesFilter {\n  \"\"\"The science group owning the template eg imaging\"\"\"\n  scienceGroup: [ScienceGroup!]\n}\n\nscalar link__Import\n\nenum link__Purpose {\n  EXECUTION\n  SECURITY\n}",
@@ -1173,6 +1497,11 @@
       "id": "3",
       "name": "samples",
       "routingUrl": "https://sample-information.diamond.ac.uk/api/graphql"
+    },
+    {
+      "id": "4",
+      "name": "experiments",
+      "routingUrl": "https://api.experiment-definition.diamond.ac.uk/graphql"
     }
   ],
   "version": "00000000-0000-0000-0000-000000000000"

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 6, 2026

diff --git a/tmp/old_supergraph.graphql b/tmp/new_supergraph.graphql
index 5ace86a..945a9f6 100644
--- a/tmp/old_supergraph.graphql
+++ b/tmp/new_supergraph.graphql
@@ -42,6 +42,8 @@ type Mutation {
   createOrValidateSamples(input: CreateOrValidateSampleInput!): CreateSamplesResponse!
   createSamples(input: CreateSampleInput!): [Sample!]! @deprecated(reason: "Will be replaced by createOrValidateSamples")
   sample(sampleId: UUID!): SampleMutations
+  createExperimentDefinition(input: CreateExperimentDefinitionInput!): ExperimentDefinition
+  experimentDefinition(id: UUID!): ExperimentDefinitionMutations
 }
 
 """Represents Relay Node types"""
@@ -100,6 +102,10 @@ type Query {
 
   """Get a list of samples associated with a given instrument session"""
   samples(first: Int!, instrumentSessions: [InstrumentSessionInput!] = null, filter: SampleFilterInput! = {schemaUrl: null, createdTime: null, updatedTime: null, name: null, data: null}, before: String = null, after: String = null, last: Int = null, orderBy: SampleOrder! = {name: null, createdTime: null, updatedTime: null}): SampleConnection!
+  experimentDefinition(id: UUID!): ExperimentDefinition
+  experimentDefinitions(instrumentSessions: [InstrumentSessionInput!]!, first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentDefinitionConnection
+  experiment(id: UUID!): Experiment
+  experiments(instrumentSessions: [InstrumentSessionInput!]!, first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentConnection
 }
 
 """Supported DLS science groups"""
@@ -475,6 +481,12 @@ type InstrumentSession {
 
   """Samples associated with a given instrument session"""
   samples(first: Int!, filter: SampleFilterInput! = {schemaUrl: null, createdTime: null, updatedTime: null, name: null, data: null}, before: String = null, after: String = null, last: Int = null, orderBy: SampleOrder! = {name: null, createdTime: null, updatedTime: null}): SampleConnection!
+
+  """Experiment Definitions"""
+  experimentDefinitions(first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentDefinitionConnection!
+
+  """Experiments associated with this session"""
+  experiments(first: Int = null, last: Int = null, after: String = null, before: String = null): ExperimentConnection!
 }
 
 type InstrumentSessionMutations {
@@ -630,6 +642,7 @@ type ErrorDetails {
   message: String!
 }
 
+"""Values required to uniquely identify an instrument session"""
 input InstrumentSessionInput {
   proposalNumber: Int!
   instrumentSessionNumber: Int!
@@ -684,6 +697,7 @@ type Sample {
   """The instrument sessions that this sample is associated with"""
   instrumentSessions: [InstrumentSession!]!
   images: [SampleImage!]!
+  experiments: [Experiment!]!
 }
 
 type SampleConnection {
@@ -834,4 +848,79 @@ type UpdateSampleResponse {
 }
 
 """Represents NULL values"""
-scalar Void
\ No newline at end of file
+scalar Void
+
+"""Values required to create an experiment definition"""
+input CreateExperimentDefinitionInput {
+  name: String!
+  data: JSON!
+  dataSchemaUrl: String!
+  proposalNumber: Int!
+  instrumentSessionNumber: Int!
+}
+
+"""Values required for the createExperiments mutation"""
+input CreateExperimentsInput {
+  experiments: [ExperimentInput!]!
+}
+
+type Experiment {
+  id: UUID!
+  name: String!
+  createdTime: DateTime!
+  updatedTime: DateTime!
+  experimentDefinition: ExperimentDefinition!
+
+  """The sample that this experiment is associated with"""
+  sample: Sample!
+}
+
+type ExperimentConnection {
+  edges: [ExperimentEdge!]!
+  pageInfo: PageInfo!
+}
+
+type ExperimentDefinition {
+  id: UUID!
+  name: String!
+  createdTime: DateTime!
+  updatedTime: DateTime!
+  data: JSON!
+  dataSchemaUrl: String!
+  proposalNumber: Int!
+  instrumentSessionNumber: Int!
+
+  """
+  The instrument session that this experiment definition is associated with
+  """
+  instrumentSession: InstrumentSession!
+
+  """Experiments associated with this experiment definition"""
+  experiments: [Experiment!]!
+}
+
+type ExperimentDefinitionConnection {
+  edges: [ExperimentDefinitionEdge!]!
+  pageInfo: PageInfo!
+}
+
+type ExperimentDefinitionEdge {
+  cursor: String!
+  node: ExperimentDefinition!
+}
+
+"""Mutations for a given experiment defintion"""
+type ExperimentDefinitionMutations {
+  createExperiments(input: CreateExperimentsInput!): [Experiment!]!
+}
+
+type ExperimentEdge {
+  cursor: String!
+  node: Experiment!
+}
+
+"""Values required to create an experiment"""
+input ExperimentInput {
+  name: String!
+  sampleId: UUID!
+}
\ No newline at end of file

@MattPrit MattPrit requested a review from EmsArnold May 7, 2026 08:09
Copy link
Copy Markdown
Collaborator

@EmsArnold EmsArnold left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good to me - just a quick question for my own curiosity.

Comment thread supergraph-config.yaml
schema:
file: schema/samples.graphql
- name: experiments
routing_url: https://api.experiment-definition.diamond.ac.uk/graphql
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Out of curiosity, it's slightly odd that this doesn't match the previous ULIMS routing_urls - is there a reason for this?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants