From 4e897c9a482d5a210c15af91a4cbfce52bfdbd91 Mon Sep 17 00:00:00 2001 From: John Chilton Date: Fri, 3 Jan 2025 23:04:01 -0500 Subject: [PATCH] WIP: sample sheets... --- client/src/api/index.ts | 4 + .../Collections/CollectionCreatorModal.vue | 12 + .../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 +- .../HistoryOperations/SelectionOperations.vue | 1 + .../History/adapters/buildCollectionModal.ts | 3 +- .../RuleBuilder/rule-definitions.js | 29 ++ .../Editor/Forms/FormCollectionType.vue | 1 + .../Editor/Forms/FormColumnDefinition.vue | 235 ++++++++++++++++ .../Editor/Forms/FormColumnDefinitionType.vue | 52 ++++ .../Editor/Forms/FormColumnDefinitions.vue | 157 +++++++++++ .../Editor/Forms/FormInputCollection.vue | 16 +- .../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 | 254 ++++++++++++++++++ .../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 | 54 ++++ 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 | 114 ++++++++ 44 files changed, 2097 insertions(+), 22 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 bf4da433b8aa..d770abb0e746 100644 --- a/client/src/api/index.ts +++ b/client/src/api/index.ts @@ -314,4 +314,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 a196a777d560..b35dcffe7ce7 100644 --- a/client/src/components/Collections/CollectionCreatorModal.vue +++ b/client/src/components/Collections/CollectionCreatorModal.vue @@ -16,6 +16,7 @@ import type { CollectionType } from "../History/adapters/buildCollectionModal"; import ListCollectionCreator from "./ListCollectionCreator.vue"; import PairCollectionCreator from "./PairCollectionCreator.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"; @@ -31,6 +32,7 @@ interface Props { hideModalOnCreate?: boolean; filterText?: string; useBetaComponents?: boolean; + fileSourcesConfigured: boolean; } const props = defineProps(); @@ -277,6 +279,16 @@ function resetModal() { mode="modal" @on-create="createHDCA" @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/CurrentHistory/HistoryOperations/SelectionOperations.vue b/client/src/components/History/CurrentHistory/HistoryOperations/SelectionOperations.vue index c978c95b8cfc..899be92db15e 100644 --- a/client/src/components/History/CurrentHistory/HistoryOperations/SelectionOperations.vue +++ b/client/src/components/History/CurrentHistory/HistoryOperations/SelectionOperations.vue @@ -166,6 +166,7 @@ v-if="collectionModalType" :history-id="history.id" :collection-type="collectionModalType" + :file-sources-configured="config.file_sources_configured" :filter-text="filterText" :selected-items="collectionSelection" :use-beta-components="useBetaComponents" diff --git a/client/src/components/History/adapters/buildCollectionModal.ts b/client/src/components/History/adapters/buildCollectionModal.ts index cf1b24ba8ebe..35225d2cbbc9 100644 --- a/client/src/components/History/adapters/buildCollectionModal.ts +++ b/client/src/components/History/adapters/buildCollectionModal.ts @@ -14,8 +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"; interface HasName { name: string | null; } diff --git a/client/src/components/RuleBuilder/rule-definitions.js b/client/src/components/RuleBuilder/rule-definitions.js index 4d15512035f5..70810e65b17b 100644 --- a/client/src/components/RuleBuilder/rule-definitions.js +++ b/client/src/components/RuleBuilder/rule-definitions.js @@ -197,6 +197,35 @@ const RULES = { return { data, columns }; }, }, + add_column_by_sample_sheet_index: { + title: _l("Add Column By By Sample Sheet Index"), + display: (rule, colHeaders) => { + return `Add column for value of sample sheet index ${rule.value}.`; + }, + init: (component, rule) => { + if (!rule) { + component.addColumnBySampleSheetIndex = null; + } else { + component.addColumnBySampleSheetIndex = rule.value; + } + }, + save: (component, rule) => { + rule.value = component.addColumnBySampleSheetIndex; + }, + apply: (rule, data, sources, columns) => { + const ruleValue = rule.value; + const newRow = (row, index) => { + const newRow = row.slice(); + const columns = sources[index]["columns"]; + const value = columns[ruleValue]; + newRow.push(value); + return newRow; + }; + data = data.map(newRow); + columns.push(NEW_COLUMN); + return { data, columns }; + }, + }, add_column_group_tag_value: { title: _l("Add Column from Group Tag Value"), display: (rule, colHeaders) => { diff --git a/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue b/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue index f337d2d48c5a..8ff96a290b66 100644 --- a/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue +++ b/client/src/components/Workflow/Editor/Forms/FormCollectionType.vue @@ -26,6 +26,7 @@ const collectionTypeOptions = [ { value: "list:record", label: "List of Records" }, { value: "list:paired", label: "List of Dataset Pairs" }, { value: "list:paired_or_unpaired", label: "Mixed List of Paired and Unpaired Datasets" }, + { 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 ceda3e459fa5..1d7f389650ea 100644 --- a/client/src/components/Workflow/Editor/Forms/FormInputCollection.vue +++ b/client/src/components/Workflow/Editor/Forms/FormInputCollection.vue @@ -1,7 +1,7 @@