From 1422844b9ab1b6f2c4930def601d9e37acdf7120 Mon Sep 17 00:00:00 2001 From: John Chilton Date: Sun, 22 Dec 2024 15:24:56 -0500 Subject: [PATCH] WIP: sample sheets... --- client/src/api/index.ts | 4 + .../Collections/CollectionCreatorModal.vue | 10 + .../SampleSheetCollectionCreator.vue | 52 ++++ .../Collections/SampleSheetWizard.vue | 139 ++++++++++ .../sheet/DownloadWorkbookButton.vue | 22 ++ .../Collections/sheet/SampleSheetGrid.vue | 247 +++++++++++++++++ .../components/Collections/sheet/workbooks.ts | 33 +++ .../Form/Elements/FormData/FormData.vue | 7 +- .../History/adapters/buildCollectionModal.ts | 2 +- .../Editor/Forms/FormCollectionType.vue | 1 + .../Editor/Forms/FormColumnDefinition.vue | 235 ++++++++++++++++ .../Editor/Forms/FormColumnDefinitionType.vue | 52 ++++ .../Editor/Forms/FormColumnDefinitions.vue | 157 +++++++++++ .../Editor/Forms/FormInputCollection.vue | 15 + .../modules/collectionTypeDescription.ts | 2 +- client/src/stores/workflowStepStore.ts | 2 + lib/galaxy/managers/collections.py | 17 +- lib/galaxy/managers/collections_util.py | 6 + lib/galaxy/model/__init__.py | 16 +- .../model/dataset_collections/builder.py | 25 +- .../model/dataset_collections/registry.py | 2 + .../dataset_collections/types/sample_sheet.py | 30 ++ .../types/sample_sheet_util.py | 127 +++++++++ .../types/sample_sheet_workbook.py | 257 ++++++++++++++++++ .../ec25b23b08e2_implement_sample_sheets.py | 2 + .../unittest_utils/filled_in_workbook_1.xlsx | Bin 0 -> 6100 bytes lib/galaxy/schema/schema.py | 42 +++ lib/galaxy/tool_util/client/staging.py | 5 +- lib/galaxy/tool_util/cwl/util.py | 17 +- .../tool_util/parser/parameter_validators.py | 17 +- lib/galaxy/util/rules_dsl.py | 24 ++ lib/galaxy/util/rules_dsl_spec.yml | 22 ++ .../webapps/galaxy/api/dataset_collections.py | 56 ++++ lib/galaxy/workflow/modules.py | 5 + .../api/test_dataset_collections.py | 160 ++++++++++- lib/galaxy_test/api/test_tools.py | 3 + lib/galaxy_test/base/populators.py | 17 +- lib/galaxy_test/base/rules_test_data.py | 51 ++++ packages/data/setup.cfg | 1 + pyproject.toml | 1 + .../test_sample_sheet_util.py | 93 +++++++ .../test_sample_sheet_workbook.py | 119 ++++++++ 42 files changed, 2075 insertions(+), 20 deletions(-) create mode 100644 client/src/components/Collections/SampleSheetCollectionCreator.vue create mode 100644 client/src/components/Collections/SampleSheetWizard.vue create mode 100644 client/src/components/Collections/sheet/DownloadWorkbookButton.vue create mode 100644 client/src/components/Collections/sheet/SampleSheetGrid.vue create mode 100644 client/src/components/Collections/sheet/workbooks.ts create mode 100644 client/src/components/Workflow/Editor/Forms/FormColumnDefinition.vue create mode 100644 client/src/components/Workflow/Editor/Forms/FormColumnDefinitionType.vue create mode 100644 client/src/components/Workflow/Editor/Forms/FormColumnDefinitions.vue create mode 100644 lib/galaxy/model/dataset_collections/types/sample_sheet.py create mode 100644 lib/galaxy/model/dataset_collections/types/sample_sheet_util.py create mode 100644 lib/galaxy/model/dataset_collections/types/sample_sheet_workbook.py create mode 100644 lib/galaxy/model/unittest_utils/filled_in_workbook_1.xlsx create mode 100644 test/unit/data/dataset_collections/test_sample_sheet_util.py create mode 100644 test/unit/data/dataset_collections/test_sample_sheet_workbook.py diff --git a/client/src/api/index.ts b/client/src/api/index.ts index 6f9ce9d0213d..798bde4aa951 100644 --- a/client/src/api/index.ts +++ b/client/src/api/index.ts @@ -312,4 +312,8 @@ export type ObjectExportTaskResponse = components["schemas"]["ObjectExportTaskRe export type ExportObjectRequestMetadata = components["schemas"]["ExportObjectRequestMetadata"]; export type ExportObjectResultMetadata = components["schemas"]["ExportObjectResultMetadata"]; +export type SampleSheetColumnDefinition = components["schemas"]["SampleSheetColumnDefinitionModel"]; +export type SampleSheetColumnDefinitionType = SampleSheetColumnDefinition["type"]; +export type SampleSheetColumnDefinitions = SampleSheetColumnDefinition[] | null; + export type AsyncTaskResultSummary = components["schemas"]["AsyncTaskResultSummary"]; diff --git a/client/src/components/Collections/CollectionCreatorModal.vue b/client/src/components/Collections/CollectionCreatorModal.vue index 6bd17b96ab1e..4370f56b6863 100644 --- a/client/src/components/Collections/CollectionCreatorModal.vue +++ b/client/src/components/Collections/CollectionCreatorModal.vue @@ -17,6 +17,7 @@ import ListCollectionCreator from "./ListCollectionCreator.vue"; import PairCollectionCreator from "./PairCollectionCreator.vue"; import PairedListCollectionCreator from "./PairedListCollectionCreator.vue"; import PairedOrUnpairedListCollectionCreator from "./PairedOrUnpairedListCollectionCreator.vue"; +import SampleSheetCollectionCreator from "./SampleSheetCollectionCreator.vue"; import Heading from "@/components/Common/Heading.vue"; import GenericItem from "@/components/History/Content/GenericItem.vue"; import LoadingSpan from "@/components/LoadingSpan.vue"; @@ -323,6 +324,15 @@ function resetModal() { :extensions="props.extensions" @clicked-create="createPairedCollection" @on-cancel="hideModal" /> + diff --git a/client/src/components/Collections/SampleSheetCollectionCreator.vue b/client/src/components/Collections/SampleSheetCollectionCreator.vue new file mode 100644 index 000000000000..6a949c3f656e --- /dev/null +++ b/client/src/components/Collections/SampleSheetCollectionCreator.vue @@ -0,0 +1,52 @@ + + + diff --git a/client/src/components/Collections/SampleSheetWizard.vue b/client/src/components/Collections/SampleSheetWizard.vue new file mode 100644 index 000000000000..1511041243b5 --- /dev/null +++ b/client/src/components/Collections/SampleSheetWizard.vue @@ -0,0 +1,139 @@ + + + diff --git a/client/src/components/Collections/sheet/DownloadWorkbookButton.vue b/client/src/components/Collections/sheet/DownloadWorkbookButton.vue new file mode 100644 index 000000000000..aec1084ec9e8 --- /dev/null +++ b/client/src/components/Collections/sheet/DownloadWorkbookButton.vue @@ -0,0 +1,22 @@ + + + diff --git a/client/src/components/Collections/sheet/SampleSheetGrid.vue b/client/src/components/Collections/sheet/SampleSheetGrid.vue new file mode 100644 index 000000000000..22ef1f33caf5 --- /dev/null +++ b/client/src/components/Collections/sheet/SampleSheetGrid.vue @@ -0,0 +1,247 @@ + + + diff --git a/client/src/components/Collections/sheet/workbooks.ts b/client/src/components/Collections/sheet/workbooks.ts new file mode 100644 index 000000000000..021a77c5ff29 --- /dev/null +++ b/client/src/components/Collections/sheet/workbooks.ts @@ -0,0 +1,33 @@ +import { type SampleSheetColumnDefinition, type SampleSheetColumnDefinitions } from "@/api"; +import { withPrefix } from "@/utils/redirect"; + +export function getDownloadWorkbookUrl(columnDefinitions: SampleSheetColumnDefinitions, initialRows?: string[][]) { + const columnDefinitionsJson = JSON.stringify(columnDefinitions); + const columnDefinitionsJsonBase64 = Buffer.from(columnDefinitionsJson).toString("base64"); + let url = withPrefix(`/api/sample_sheet_workbook/generate?column_definitions=${columnDefinitionsJsonBase64}`); + if (initialRows) { + const initialRowsJson = JSON.stringify(initialRows); + const initialRowsJsonBase64 = Buffer.from(initialRowsJson).toString("base64"); + url = `${url}&initial_rows=${initialRowsJsonBase64}`; + } + return url; +} + +export function downloadWorkbook(columnDefinitions: SampleSheetColumnDefinitions, initialRows?: string[][]) { + const url = getDownloadWorkbookUrl(columnDefinitions, initialRows); + window.location.assign(url); +} + +export function initialValue(columnDefinition: SampleSheetColumnDefinition) { + switch (columnDefinition.type) { + case "int": + return 0; + case "float": + return 0.0; + case "boolean": + return false; + case "string": + default: + return ""; + } +} diff --git a/client/src/components/Form/Elements/FormData/FormData.vue b/client/src/components/Form/Elements/FormData/FormData.vue index b13dd87ae8e9..80663d75aae7 100644 --- a/client/src/components/Form/Elements/FormData/FormData.vue +++ b/client/src/components/Form/Elements/FormData/FormData.vue @@ -33,6 +33,7 @@ const COLLECTION_TYPE_TO_LABEL: Record = { list: "list", "list:paired": "list of dataset pairs", paired: "dataset pair", + sample_sheet: "sample sheet derived", }; type SelectOption = { @@ -88,7 +89,7 @@ const dragTarget: Ref = ref(null); // Collection creator modal settings const collectionModalShow = ref(false); -const collectionModalType = ref<"list" | "list:paired" | "paired">("list"); +const collectionModalType = ref<"list" | "list:paired" | "paired" | "sample_sheet">("list"); const { currentHistoryId } = storeToRefs(useHistoryStore()); const restrictsExtensions = computed(() => { const extensions = props.extensions; @@ -497,7 +498,7 @@ function canAcceptSrc(historyContentType: "dataset" | "dataset_collection", coll } } -const collectionTypesWithBuilders = ["list", "list:paired", "paired"]; +const collectionTypesWithBuilders = ["list", "list:paired", "paired", "list:paired_or_unpaired", "sample_sheet"]; /** Allowed collection types for collection creation */ const effectiveCollectionTypes = props.collectionTypes?.filter((collectionType) => @@ -508,7 +509,7 @@ function buildNewCollection(collectionType: string) { if (!collectionTypesWithBuilders.includes(collectionType)) { throw Error(`Unknown collection type: ${collectionType}`); } - collectionModalType.value = collectionType as "list" | "list:paired" | "paired"; + collectionModalType.value = collectionType as "list" | "list:paired" | "paired" | "sample_sheet"; collectionModalShow.value = true; } diff --git a/client/src/components/History/adapters/buildCollectionModal.ts b/client/src/components/History/adapters/buildCollectionModal.ts index 53a41b424914..9cb280476a57 100644 --- a/client/src/components/History/adapters/buildCollectionModal.ts +++ b/client/src/components/History/adapters/buildCollectionModal.ts @@ -14,7 +14,7 @@ import jQuery from "jquery"; import type { HDASummary, HistoryItemSummary } from "@/api"; import RULE_BASED_COLLECTION_CREATOR from "@/components/Collections/RuleBasedCollectionCreatorModal"; -export type CollectionType = "list" | "paired" | "list:paired" | "rules"; +export type CollectionType = "list" | "paired" | "list:paired" | "rules" | "list:paired_or_unpaired" | "sample_sheet"; export type DatasetPair = { forward: HDASummary; reverse: HDASummary; diff --git a/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue b/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue index a93cb5ded6f1..77b5aa8161f2 100644 --- a/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue +++ b/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue @@ -23,6 +23,7 @@ const collectionTypeOptions = [ { value: "list", label: "List of Datasets" }, { value: "paired", label: "Dataset Pair" }, { value: "list:paired", label: "List of Dataset Pairs" }, + { value: "sample_sheet", label: "Sample Sheet of Datasets" }, ]; function updateValue(newValue: string | undefined) { diff --git a/client/src/components/Workflow/Editor/Forms/FormColumnDefinition.vue b/client/src/components/Workflow/Editor/Forms/FormColumnDefinition.vue new file mode 100644 index 000000000000..0068bacbadbf --- /dev/null +++ b/client/src/components/Workflow/Editor/Forms/FormColumnDefinition.vue @@ -0,0 +1,235 @@ + + + + + diff --git a/client/src/components/Workflow/Editor/Forms/FormColumnDefinitionType.vue b/client/src/components/Workflow/Editor/Forms/FormColumnDefinitionType.vue new file mode 100644 index 000000000000..c8ec62eb08d1 --- /dev/null +++ b/client/src/components/Workflow/Editor/Forms/FormColumnDefinitionType.vue @@ -0,0 +1,52 @@ + + + diff --git a/client/src/components/Workflow/Editor/Forms/FormColumnDefinitions.vue b/client/src/components/Workflow/Editor/Forms/FormColumnDefinitions.vue new file mode 100644 index 000000000000..5512a4d56f79 --- /dev/null +++ b/client/src/components/Workflow/Editor/Forms/FormColumnDefinitions.vue @@ -0,0 +1,157 @@ + + + + + diff --git a/client/src/components/Workflow/Editor/Forms/FormInputCollection.vue b/client/src/components/Workflow/Editor/Forms/FormInputCollection.vue index ea17f09fb50d..ef5379b0e0d1 100644 --- a/client/src/components/Workflow/Editor/Forms/FormInputCollection.vue +++ b/client/src/components/Workflow/Editor/Forms/FormInputCollection.vue @@ -1,6 +1,7 @@