From c426c1e2c23f4793d4ffe3c9e6415be2eefb0894 Mon Sep 17 00:00:00 2001 From: John Chilton Date: Fri, 31 May 2024 10:32:38 -0400 Subject: [PATCH] Expand pre-checking of file source/object store configuration. - UI for detailed display of errors. - UI option to test configuration from management menu. - API + UI for checking configuration before upgrading to new version of template. - API + UI for checking configuration before updating current template's settings. - Add an option during update/upgrade to allow forcing the update even if configuration doesn't validate - I don't allow creation of invalid things, but if there are problems with an existing thing - admins and power users should have recourse. It is their data. --- client/src/api/configTemplates.ts | 6 + client/src/api/schema/schema.ts | 154 +++++++++ .../ConfigTemplates/ActionSummary.vue | 29 ++ .../ConfigTemplates/ConfigurationTestItem.vue | 32 ++ .../ConfigurationTestSummary.vue | 21 ++ .../ConfigurationTestSummaryModal.vue | 39 +++ .../ConfigTemplates/ForceActionButton.vue | 17 + .../ConfigTemplates/InstanceDropdown.test.ts | 4 +- .../ConfigTemplates/InstanceDropdown.vue | 9 +- .../ConfigTemplates/InstanceForm.vue | 13 +- .../components/ConfigTemplates/formUtil.ts | 3 +- .../src/components/ConfigTemplates/routing.ts | 18 + .../ConfigTemplates/test_fixtures.ts | 16 + .../useConfigurationTesting.ts | 311 ++++++++++++++++++ .../FileSources/Instances/CreateForm.vue | 50 +-- .../FileSources/Instances/EditInstance.vue | 82 ++--- .../Instances/InstanceDropdown.vue | 2 + .../FileSources/Instances/ManageIndex.vue | 12 +- .../FileSources/Instances/UpgradeForm.vue | 65 ++-- .../FileSources/Instances/routing.ts | 17 +- .../FileSources/Instances/services.ts | 5 + .../ObjectStore/Instances/CreateForm.test.ts | 26 +- .../ObjectStore/Instances/CreateForm.vue | 54 +-- .../ObjectStore/Instances/EditInstance.vue | 82 ++--- .../Instances/InstanceDropdown.vue | 2 + .../ObjectStore/Instances/ManageIndex.vue | 12 +- .../ObjectStore/Instances/UpgradeForm.test.ts | 13 +- .../ObjectStore/Instances/UpgradeForm.vue | 65 ++-- .../ObjectStore/Instances/routing.ts | 17 +- .../ObjectStore/Instances/services.ts | 8 + lib/galaxy/managers/_config_templates.py | 98 +++++- lib/galaxy/managers/file_source_instances.py | 96 ++++-- lib/galaxy/managers/object_store_instances.py | 87 ++++- lib/galaxy/webapps/galaxy/api/file_sources.py | 26 ++ lib/galaxy/webapps/galaxy/api/object_store.py | 26 ++ test/integration/objectstore/test_per_user.py | 12 + .../app/managers/test_user_file_sources.py | 96 ++++++ .../app/managers/test_user_object_stores.py | 111 ++++++- 38 files changed, 1408 insertions(+), 328 deletions(-) create mode 100644 client/src/components/ConfigTemplates/ActionSummary.vue create mode 100644 client/src/components/ConfigTemplates/ConfigurationTestItem.vue create mode 100644 client/src/components/ConfigTemplates/ConfigurationTestSummary.vue create mode 100644 client/src/components/ConfigTemplates/ConfigurationTestSummaryModal.vue create mode 100644 client/src/components/ConfigTemplates/ForceActionButton.vue create mode 100644 client/src/components/ConfigTemplates/routing.ts create mode 100644 client/src/components/ConfigTemplates/useConfigurationTesting.ts diff --git a/client/src/api/configTemplates.ts b/client/src/api/configTemplates.ts index 51b21382c505..187c3a5931a9 100644 --- a/client/src/api/configTemplates.ts +++ b/client/src/api/configTemplates.ts @@ -17,6 +17,12 @@ export type SecretData = { [key: string]: string }; export type PluginAspectStatus = components["schemas"]["PluginAspectStatus"]; export type PluginStatus = components["schemas"]["PluginStatus"]; +export type CreateInstancePayload = components["schemas"]["CreateInstancePayload"]; +export type UpgradeInstancePayload = components["schemas"]["UpgradeInstancePayload"]; +export type TestUpgradeInstancePayload = components["schemas"]["TestUpgradeInstancePayload"]; +export type UpdateInstancePayload = components["schemas"]["UpdateInstancePayload"]; +export type TestUpdateInstancePayload = components["schemas"]["TestUpdateInstancePayload"]; + export interface TemplateSummary { description: string | null; hidden?: boolean; diff --git a/client/src/api/schema/schema.ts b/client/src/api/schema/schema.ts index 2db6faeb1963..389a25f52387 100644 --- a/client/src/api/schema/schema.ts +++ b/client/src/api/schema/schema.ts @@ -322,6 +322,12 @@ export interface paths { /** Purge user file source instance. */ delete: operations["file_sources__instances_purge"]; }; + "/api/file_source_instances/{user_file_source_id}/test": { + /** Test a file source instance and return status. */ + get: operations["file_sources__instances_test_instance"]; + /** Test updating or upgrading user file source instance. */ + post: operations["file_sources__test_instances_update"]; + }; "/api/file_source_templates": { /** Get a list of file source templates available to build user defined file sources from */ get: operations["file_sources__templates_index"]; @@ -1273,6 +1279,12 @@ export interface paths { /** Purge user object store instance. */ delete: operations["object_stores__instances_purge"]; }; + "/api/object_store_instances/{user_object_store_id}/test": { + /** Get a persisted user object store instance. */ + get: operations["object_stores__instances_test_instance"]; + /** Test updating or upgrading user object source instance. */ + post: operations["object_stores__test_instances_update"]; + }; "/api/object_store_templates": { /** Get a list of object store templates available to build user defined object stores from */ get: operations["object_stores__templates_index"]; @@ -11996,6 +12008,26 @@ export interface components { */ type: "string"; }; + /** TestUpdateInstancePayload */ + TestUpdateInstancePayload: { + /** Variables */ + variables?: { + [key: string]: (string | boolean | number) | undefined; + } | null; + }; + /** TestUpgradeInstancePayload */ + TestUpgradeInstancePayload: { + /** Secrets */ + secrets: { + [key: string]: string | undefined; + }; + /** Template Version */ + template_version: number; + /** Variables */ + variables: { + [key: string]: (string | boolean | number) | undefined; + }; + }; /** ToolDataDetails */ ToolDataDetails: { /** @@ -15062,6 +15094,67 @@ export interface operations { }; }; }; + file_sources__instances_test_instance: { + /** Test a file source instance and return status. */ + parameters: { + /** @description The user ID that will be used to effectively make this API call. Only admins and designated users can make API calls on behalf of other users. */ + header?: { + "run-as"?: string | null; + }; + /** @description The UUID index for a persisted UserFileSourceStore object. */ + path: { + user_file_source_id: string; + }; + }; + responses: { + /** @description Successful Response */ + 200: { + content: { + "application/json": components["schemas"]["PluginStatus"]; + }; + }; + /** @description Validation Error */ + 422: { + content: { + "application/json": components["schemas"]["HTTPValidationError"]; + }; + }; + }; + }; + file_sources__test_instances_update: { + /** Test updating or upgrading user file source instance. */ + parameters: { + /** @description The user ID that will be used to effectively make this API call. Only admins and designated users can make API calls on behalf of other users. */ + header?: { + "run-as"?: string | null; + }; + /** @description The UUID index for a persisted UserFileSourceStore object. */ + path: { + user_file_source_id: string; + }; + }; + requestBody: { + content: { + "application/json": + | components["schemas"]["TestUpgradeInstancePayload"] + | components["schemas"]["TestUpdateInstancePayload"]; + }; + }; + responses: { + /** @description Successful Response */ + 200: { + content: { + "application/json": components["schemas"]["PluginStatus"]; + }; + }; + /** @description Validation Error */ + 422: { + content: { + "application/json": components["schemas"]["HTTPValidationError"]; + }; + }; + }; + }; file_sources__templates_index: { /** Get a list of file source templates available to build user defined file sources from */ parameters?: { @@ -20986,6 +21079,67 @@ export interface operations { }; }; }; + object_stores__instances_test_instance: { + /** Get a persisted user object store instance. */ + parameters: { + /** @description The user ID that will be used to effectively make this API call. Only admins and designated users can make API calls on behalf of other users. */ + header?: { + "run-as"?: string | null; + }; + /** @description The UUID used to identify a persisted UserObjectStore object. */ + path: { + user_object_store_id: string; + }; + }; + responses: { + /** @description Successful Response */ + 200: { + content: { + "application/json": components["schemas"]["PluginStatus"]; + }; + }; + /** @description Validation Error */ + 422: { + content: { + "application/json": components["schemas"]["HTTPValidationError"]; + }; + }; + }; + }; + object_stores__test_instances_update: { + /** Test updating or upgrading user object source instance. */ + parameters: { + /** @description The user ID that will be used to effectively make this API call. Only admins and designated users can make API calls on behalf of other users. */ + header?: { + "run-as"?: string | null; + }; + /** @description The UUID used to identify a persisted UserObjectStore object. */ + path: { + user_object_store_id: string; + }; + }; + requestBody: { + content: { + "application/json": + | components["schemas"]["TestUpgradeInstancePayload"] + | components["schemas"]["TestUpdateInstancePayload"]; + }; + }; + responses: { + /** @description Successful Response */ + 200: { + content: { + "application/json": components["schemas"]["PluginStatus"]; + }; + }; + /** @description Validation Error */ + 422: { + content: { + "application/json": components["schemas"]["HTTPValidationError"]; + }; + }; + }; + }; object_stores__templates_index: { /** Get a list of object store templates available to build user defined object stores from */ parameters?: { diff --git a/client/src/components/ConfigTemplates/ActionSummary.vue b/client/src/components/ConfigTemplates/ActionSummary.vue new file mode 100644 index 000000000000..6e4652a584fe --- /dev/null +++ b/client/src/components/ConfigTemplates/ActionSummary.vue @@ -0,0 +1,29 @@ + + + diff --git a/client/src/components/ConfigTemplates/ConfigurationTestItem.vue b/client/src/components/ConfigTemplates/ConfigurationTestItem.vue new file mode 100644 index 000000000000..3d360b35b708 --- /dev/null +++ b/client/src/components/ConfigTemplates/ConfigurationTestItem.vue @@ -0,0 +1,32 @@ + + + diff --git a/client/src/components/ConfigTemplates/ConfigurationTestSummary.vue b/client/src/components/ConfigTemplates/ConfigurationTestSummary.vue new file mode 100644 index 000000000000..be17f1593821 --- /dev/null +++ b/client/src/components/ConfigTemplates/ConfigurationTestSummary.vue @@ -0,0 +1,21 @@ + + + diff --git a/client/src/components/ConfigTemplates/ConfigurationTestSummaryModal.vue b/client/src/components/ConfigTemplates/ConfigurationTestSummaryModal.vue new file mode 100644 index 000000000000..22fa8a8b4736 --- /dev/null +++ b/client/src/components/ConfigTemplates/ConfigurationTestSummaryModal.vue @@ -0,0 +1,39 @@ + + + diff --git a/client/src/components/ConfigTemplates/ForceActionButton.vue b/client/src/components/ConfigTemplates/ForceActionButton.vue new file mode 100644 index 000000000000..55d42274f388 --- /dev/null +++ b/client/src/components/ConfigTemplates/ForceActionButton.vue @@ -0,0 +1,17 @@ + + + diff --git a/client/src/components/ConfigTemplates/InstanceDropdown.test.ts b/client/src/components/ConfigTemplates/InstanceDropdown.test.ts index e29d8d883113..abfca1e3070a 100644 --- a/client/src/components/ConfigTemplates/InstanceDropdown.test.ts +++ b/client/src/components/ConfigTemplates/InstanceDropdown.test.ts @@ -19,7 +19,7 @@ describe("InstanceDropdown", () => { }); const menu = wrapper.find(".dropdown-menu"); const links = menu.findAll("button.dropdown-item"); - expect(links.length).toBe(2); + expect(links.length).toBe(3); }); it("should render a drop down with upgrade if upgrade available as an option", async () => { @@ -35,6 +35,6 @@ describe("InstanceDropdown", () => { }); const menu = wrapper.find(".dropdown-menu"); const links = menu.findAll("button.dropdown-item"); - expect(links.length).toBe(3); + expect(links.length).toBe(4); }); }); diff --git a/client/src/components/ConfigTemplates/InstanceDropdown.vue b/client/src/components/ConfigTemplates/InstanceDropdown.vue index 95bbbf863cb8..f3fc2cebddd0 100644 --- a/client/src/components/ConfigTemplates/InstanceDropdown.vue +++ b/client/src/components/ConfigTemplates/InstanceDropdown.vue @@ -1,6 +1,6 @@ @@ -53,6 +54,10 @@ const emit = defineEmits<{ Edit configuration +