From 920f8b412a479ea190e0f1cdde649d6ea89dc6c4 Mon Sep 17 00:00:00 2001 From: John Chilton Date: Sun, 19 May 2024 16:41:48 -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 | 162 ++++++++- .../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 | 313 ++++++++++++++++++ .../FileSources/Instances/CreateForm.vue | 50 +-- .../FileSources/Instances/EditInstance.vue | 83 ++--- .../Instances/InstanceDropdown.vue | 2 + .../FileSources/Instances/ManageIndex.vue | 12 +- .../FileSources/Instances/UpgradeForm.vue | 66 ++-- .../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 | 83 ++--- .../Instances/InstanceDropdown.vue | 2 + .../ObjectStore/Instances/ManageIndex.vue | 12 +- .../ObjectStore/Instances/UpgradeForm.test.ts | 13 +- .../ObjectStore/Instances/UpgradeForm.vue | 66 ++-- .../ObjectStore/Instances/routing.ts | 17 +- .../ObjectStore/Instances/services.ts | 8 + lib/galaxy/managers/_config_templates.py | 98 +++++- lib/galaxy/managers/file_source_instances.py | 95 ++++-- lib/galaxy/managers/object_store_instances.py | 86 ++++- 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 | 105 ++++++ 38 files changed, 1413 insertions(+), 329 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 f11f1ad9751f..1208a1260e0d 100644 --- a/client/src/api/schema/schema.ts +++ b/client/src/api/schema/schema.ts @@ -329,13 +329,19 @@ export interface paths { post: operations["file_sources__test_new_instance_configuration"]; }; "/api/file_source_instances/{user_file_source_id}": { - /** Get a list of persisted file source instances defined by the requesting user. */ + /** Get a persisted user file source instance. */ get: operations["file_sources__instances_get"]; /** Update or upgrade user file source instance. */ put: operations["file_sources__instances_update"]; /** 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"]; @@ -1278,13 +1284,19 @@ export interface paths { post: operations["object_stores__test_new_instance_configuration"]; }; "/api/object_store_instances/{user_object_store_id}": { - /** Get a persisted object store instances owned by the requesting user. */ + /** Get a persisted user object store instance. */ get: operations["object_stores__instances_get"]; /** Update or upgrade user object store instance. */ put: operations["object_stores__instances_update"]; /** 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"]; @@ -12011,6 +12023,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: { /** @@ -15056,7 +15088,7 @@ export interface operations { }; }; file_sources__instances_get: { - /** Get a list of persisted file source instances defined by the requesting user. */ + /** Get a persisted 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?: { @@ -15140,6 +15172,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 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 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?: { @@ -20961,7 +21054,7 @@ export interface operations { }; }; object_stores__instances_get: { - /** Get a persisted object store instances owned by the requesting user. */ + /** 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?: { @@ -21045,6 +21138,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 identifier used to index 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 identifier used to index 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 da85c8ee60ff..98c73d2149c5 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 +