From 334a03823ff13299e1356ad9da4328b7aeaddfc0 Mon Sep 17 00:00:00 2001 From: Marc-Antoine Hinse Date: Wed, 27 Nov 2024 14:14:34 -0500 Subject: [PATCH] Moved the business logic of the configuration steps in the specific steps - Added an API key validation to determine if the key is valid instead of fetching the user tenants - Makefile: No need to check if we have python tests anymore, we do have them - Removed the splunk.min.js file since it's injected by splunk, removed the compilation of javascript files - Created constants file for typescript since it was getting all over the place - Added disabled step for buttons to block the user from submitting before the step is loaded --- Makefile | 4 +- packages/flare/bin/cron_job_ingest_events.py | 4 +- packages/flare/bin/flare.py | 5 + packages/flare/bin/flare_external_requests.py | 30 ++- .../resources/splunk/default/restmap.conf | 5 + .../main/resources/splunk/default/web.conf | 4 + .../flare/tests/bin/test_flare_wrapper.py | 19 +- packages/react-components/package.json | 6 +- .../src/ConfigurationScreen.css | 1 + .../src/ConfigurationScreen.tsx | 208 ++++-------------- .../react-components/src/StatusScreen.tsx | 11 +- .../src/components/Button.css | 5 + .../src/components/Button.tsx | 18 +- .../components/ConfigurationCompletedStep.tsx | 27 ++- .../components/ConfigurationGlobalStep.css | 2 +- .../components/ConfigurationInitialStep.tsx | 62 +++++- .../ConfigurationUserPreferencesStep.css | 9 +- .../ConfigurationUserPreferencesStep.tsx | 148 +++++++++---- .../react-components/src/components/Label.css | 2 +- .../src/components/ToastManager.tsx | 5 + .../src/components/Tooltip.css | 4 +- .../src/components/Tooltip.tsx | 8 +- .../react-components/src/models/constants.ts | 27 +++ packages/react-components/src/models/flare.ts | 2 +- .../react-components/src/models/splunk.ts | 35 ++- .../src/utils/setupConfiguration.ts | 134 +++++------ .../react-components/src/vendor/splunk.min.js | 1 - 27 files changed, 431 insertions(+), 355 deletions(-) create mode 100644 packages/react-components/src/models/constants.ts delete mode 100644 packages/react-components/src/vendor/splunk.min.js diff --git a/Makefile b/Makefile index af41741..305c229 100644 --- a/Makefile +++ b/Makefile @@ -75,9 +75,7 @@ inspect-tags: .PHONY: test test: venv-tools - @if test -d "./packages/flare/tests" ; then \ - venv-tools/bin/pytest ./packages/flare/tests/**/*.py -vv ; \ - fi + venv-tools/bin/pytest ./packages/flare/tests/**/*.py -vv ; .PHONY: format setup-web format: venv-tools diff --git a/packages/flare/bin/cron_job_ingest_events.py b/packages/flare/bin/cron_job_ingest_events.py index 036f7a7..6b287c1 100644 --- a/packages/flare/bin/cron_job_ingest_events.py +++ b/packages/flare/bin/cron_job_ingest_events.py @@ -289,7 +289,9 @@ def fetch_feed( start_date = get_start_date(kvstore=kvstore) logger.info(f"Fetching {tenant_id=}, {next=}, {start_date=}") for event_next in flare_api.fetch_feed_events( - next=next, start_date=start_date, ingest_metadata_only=ingest_metadata_only + next=next, + start_date=start_date, + ingest_metadata_only=ingest_metadata_only, ): yield event_next except Exception as e: diff --git a/packages/flare/bin/flare.py b/packages/flare/bin/flare.py index e1177e9..6b1086a 100644 --- a/packages/flare/bin/flare.py +++ b/packages/flare/bin/flare.py @@ -98,6 +98,11 @@ def _fetch_full_event_from_uid(self, *, uid: str) -> dict: self.logger.debug(event) return event + def fetch_api_key_validation(self) -> requests.Response: + return self.flare_client.get( + url="/tokens/test", + ) + def fetch_tenants(self) -> requests.Response: return self.flare_client.get( url="/firework/v2/me/tenants", diff --git a/packages/flare/bin/flare_external_requests.py b/packages/flare/bin/flare_external_requests.py index 4eb4eed..0d09214 100644 --- a/packages/flare/bin/flare_external_requests.py +++ b/packages/flare/bin/flare_external_requests.py @@ -12,18 +12,32 @@ from logger import Logger +class FlareValidateApiKey(splunk.rest.BaseRestHandler): + def handle_POST(self) -> None: + payload = self.request["payload"] + params = parse.parse_qs(payload) + + if "apiKey" not in params: + raise Exception("API Key is required") + + flare_api = FlareAPI(api_key=params["apiKey"][0]) + flare_api.fetch_api_key_validation() + self.response.setHeader("Content-Type", "application/json") + self.response.write(json.dumps({})) + + class FlareUserTenants(splunk.rest.BaseRestHandler): def handle_POST(self) -> None: logger = Logger(class_name=__file__) payload = self.request["payload"] params = parse.parse_qs(payload) - if "apiKey" in params: - flare_api = FlareAPI(api_key=params["apiKey"][0]) - user_tenants_response = flare_api.fetch_tenants() - tenants_response = user_tenants_response.json() - logger.debug(tenants_response) - self.response.setHeader("Content-Type", "application/json") - self.response.write(json.dumps(tenants_response)) - else: + if "apiKey" not in params: raise Exception("API Key is required") + + flare_api = FlareAPI(api_key=params["apiKey"][0]) + response = flare_api.fetch_tenants() + response_json = response.json() + logger.debug(f"FlareUserTenants: {response_json}") + self.response.setHeader("Content-Type", "application/json") + self.response.write(json.dumps(response_json)) diff --git a/packages/flare/src/main/resources/splunk/default/restmap.conf b/packages/flare/src/main/resources/splunk/default/restmap.conf index e5b7847..14e7562 100644 --- a/packages/flare/src/main/resources/splunk/default/restmap.conf +++ b/packages/flare/src/main/resources/splunk/default/restmap.conf @@ -1,3 +1,8 @@ +[script:flare_external_requests_api_key_validation] +match=/fetch_api_key_validation +handler=flare_external_requests.FlareValidateApiKey +python.version = python3 + [script:flare_external_requests_user_tenants] match=/fetch_user_tenants handler=flare_external_requests.FlareUserTenants diff --git a/packages/flare/src/main/resources/splunk/default/web.conf b/packages/flare/src/main/resources/splunk/default/web.conf index e310a5d..1563964 100644 --- a/packages/flare/src/main/resources/splunk/default/web.conf +++ b/packages/flare/src/main/resources/splunk/default/web.conf @@ -1,3 +1,7 @@ +[expose:flare_external_requests_api_key_validation] +pattern=fetch_api_key_validation +methods=POST + [expose:flare_external_requests_user_tenants] pattern=fetch_user_tenants methods=POST diff --git a/packages/flare/tests/bin/test_flare_wrapper.py b/packages/flare/tests/bin/test_flare_wrapper.py index 67894ed..cbabe47 100644 --- a/packages/flare/tests/bin/test_flare_wrapper.py +++ b/packages/flare/tests/bin/test_flare_wrapper.py @@ -41,7 +41,9 @@ def test_flare_full_data_without_metadata( events: list[dict] = [] for event, next_token in flare_api.fetch_feed_events( - next=None, start_date=None, ingest_metadata_only=True + next=None, + start_date=None, + ingest_metadata_only=True, ): assert next_token == expected_return_value["next"] events.append(event) @@ -100,7 +102,9 @@ def test_flare_full_data_with_metadata( events: list[dict] = [] for event, next_token in flare_api.fetch_feed_events( - next=None, start_date=None, ingest_metadata_only=False + next=None, + start_date=None, + ingest_metadata_only=False, ): assert next_token == expected_return_value["next"] events.append(event) @@ -138,9 +142,12 @@ def test_flare_full_data_with_metadata_and_exception( flare_api = FlareAPI(api_key="some_key", tenant_id=111) with pytest.raises(KeyError, match="metadata"): - for _, _ in flare_api.fetch_feed_events( - next=None, start_date=None, ingest_metadata_only=False - ): - pass + next( + flare_api.fetch_feed_events( + next=None, + start_date=None, + ingest_metadata_only=False, + ) + ) fetch_event_feed_metadata_mock.assert_called_once() diff --git a/packages/react-components/package.json b/packages/react-components/package.json index 4b7beb3..131cf97 100644 --- a/packages/react-components/package.json +++ b/packages/react-components/package.json @@ -4,12 +4,12 @@ "license": "UNLICENSED", "scripts": { "build": "node build.js build", - "eslint": "eslint src --ext \".js,.tsx,.ts\"", - "eslint:fix": "eslint src --ext \".js,.tsx,.ts\" --fix", + "eslint": "eslint src --ext \".tsx,.ts\"", + "eslint:fix": "eslint src --ext \".tsx,.ts\" --fix", "lint": "yarn run eslint && yarn run stylelint", "lint:ci": "yarn run eslint:ci && yarn run stylelint", "start": "webpack --watch", - "stylelint": "stylelint \"src/**/*.{js,jsx}\" --config stylelint.config.js" + "stylelint": "stylelint \"src/**/*.{ts,tsx}\" --config stylelint.config.js" }, "exports": { "./configuration-screen": "./ConfigurationScreen.js", diff --git a/packages/react-components/src/ConfigurationScreen.css b/packages/react-components/src/ConfigurationScreen.css index 40d7782..20b846b 100644 --- a/packages/react-components/src/ConfigurationScreen.css +++ b/packages/react-components/src/ConfigurationScreen.css @@ -17,6 +17,7 @@ width: 800px; gap: 1.5rem; align-self: center; + padding-bottom: 2rem; } .content-step { diff --git a/packages/react-components/src/ConfigurationScreen.tsx b/packages/react-components/src/ConfigurationScreen.tsx index 35f8996..1c9f75d 100644 --- a/packages/react-components/src/ConfigurationScreen.tsx +++ b/packages/react-components/src/ConfigurationScreen.tsx @@ -1,156 +1,44 @@ -import React, { useEffect, useState, FC } from 'react'; -import { - appName, - createFlareIndex, - redirectToHomepage, - fetchApiKey, - fetchAvailableIndexNames, - fetchCurrentIndexName, - fetchIngestMetadataOnly, - fetchTenantId, - fetchUserTenants, - saveConfiguration, -} from './utils/setupConfiguration'; -import { ConfigurationSteps, Tenant } from './models/flare'; -import './global.css'; +import React, { FC, useEffect, useState } from 'react'; import './ConfigurationScreen.css'; +import ConfigurationCompletedStep from './components/ConfigurationCompletedStep'; +import ConfigurationInitialStep from './components/ConfigurationInitialStep'; +import ConfigurationUserPreferencesStep from './components/ConfigurationUserPreferencesStep'; import LoadingBar from './components/LoadingBar'; +import { toastManager } from './components/ToastManager'; import DoneIcon from './components/icons/DoneIcon'; import ExternalLinkIcon from './components/icons/ExternalLinkIcon'; -import { toastManager } from './components/ToastManager'; import ToolIcon from './components/icons/ToolIcon'; -import ConfigurationInitialStep from './components/ConfigurationInitialStep'; -import ConfigurationUserPreferencesStep from './components/ConfigurationUserPreferencesStep'; -import ConfigurationCompletedStep from './components/ConfigurationCompletedStep'; - -const TOAST_API_KEY_ERROR = 'api_key_error'; -const TOAST_TENANT_SUCCESS = 'tenant_success'; +import './global.css'; +import { ConfigurationStep } from './models/flare'; +import { createFlareIndex, fetchApiKey, redirectToHomepage } from './utils/setupConfiguration'; const ConfigurationScreen: FC<{ theme: string }> = ({ theme }) => { + const [configurationStep, setConfigurationStep] = useState(ConfigurationStep.Initial); const [apiKey, setApiKey] = useState(''); - const [tenantId, setTenantId] = useState(-1); - const [indexName, setIndexName] = useState(appName); - const [errorMessage, setErrorMessage] = useState(''); - const [tenants, setUserTenants] = useState([]); - const [isIngestingMetadataOnly, setIsIngestingMetadataOnly] = useState(false); - const [isLoading, setIsLoading] = useState(false); - const [isCompleted, setIsCompleted] = useState(false); - const [indexNames, setIndexNames] = useState([]); toastManager.setTheme(theme); - function reset(): void { - setApiKey(''); - setTenantId(-1); - setUserTenants([]); - setIsLoading(false); - setIsCompleted(false); - } - - function getCurrentConfigurationStep(): ConfigurationSteps { - if (tenants.length === 0) { - return ConfigurationSteps.Initial; - } - if (!isCompleted) { - return ConfigurationSteps.UserPreferences; - } - - return ConfigurationSteps.Completed; - } - - const handleApiKeyChange = (e): void => setApiKey(e.target.value); - const handleTenantIdChange = (e): void => setTenantId(parseInt(e.target.value, 10)); - const handleIndexNameChange = (e): void => setIndexName(e.target.value); - const handleIsIngestingMetadataChange = (e): void => { - setIsIngestingMetadataOnly(e.target.checked); - }; - const handleBackButton = (): void => { - const currentConfigurationStep = getCurrentConfigurationStep(); - if (currentConfigurationStep === ConfigurationSteps.Initial) { - redirectToHomepage(); - } else if (currentConfigurationStep === ConfigurationSteps.UserPreferences) { - setUserTenants([]); - } else if (currentConfigurationStep === ConfigurationSteps.Completed) { - reset(); + switch (configurationStep) { + case ConfigurationStep.Initial: + redirectToHomepage(); + break; + case ConfigurationStep.UserPreferences: + case ConfigurationStep.Completed: + setConfigurationStep(ConfigurationStep.Initial); + break; + default: + throw new Error(`Back button not implemented for ${configurationStep}`); } }; - const handleSubmitApiKey = (): void => { - setIsLoading(true); - fetchUserTenants( - apiKey, - (userTenants: Tenant[]) => { - if (tenantId === -1 && userTenants.length > 0) { - setTenantId(userTenants[0].id); - } - setErrorMessage(''); - setUserTenants(userTenants); - setIsLoading(false); - }, - (error: string) => { - setErrorMessage(error); - setIsLoading(false); - toastManager.show({ - id: TOAST_API_KEY_ERROR, - isError: true, - content: 'Something went wrong. Please review your form.', - }); - } - ); - }; - - const handleSubmitTenant = (): void => { - setIsLoading(true); - saveConfiguration(apiKey, tenantId, indexName, isIngestingMetadataOnly) - .then(() => { - setIsLoading(false); - setIsCompleted(true); - toastManager.destroy(TOAST_API_KEY_ERROR); - toastManager.show({ - id: TOAST_TENANT_SUCCESS, - content: 'Configured Flare Account', - }); - }) - .catch((e: any) => { - setIsLoading(false); - toastManager.show({ - id: TOAST_API_KEY_ERROR, - isError: true, - content: `Something went wrong. ${e.responseText}`, - }); - }); - }; - - function getSelectedTenantName(): string { - const filteredTenants = tenants.filter((tenant: Tenant) => tenant.id === tenantId); - if (filteredTenants.length > 0) { - return filteredTenants[0].name; - } - - return 'unknown'; - } - useEffect(() => { - if (isCompleted) { - return; - } - createFlareIndex().then(() => { - Promise.all([ - fetchApiKey(), - fetchTenantId(), - fetchIngestMetadataOnly(), - fetchCurrentIndexName(), - fetchAvailableIndexNames(), - ]).then(([key, id, ingestMetadataOnly, index, availableIndexNames]) => { + if (configurationStep === ConfigurationStep.Initial) { + Promise.all([fetchApiKey(), createFlareIndex()]).then(([key]) => { setApiKey(key); - setTenantId(id); - setIsIngestingMetadataOnly(ingestMetadataOnly); - setIndexName(index); - setIndexNames(availableIndexNames); }); - }); - }, [isCompleted]); + } + }, [configurationStep]); useEffect(() => { const container = document.getElementById('container') as HTMLDivElement; @@ -160,51 +48,37 @@ const ConfigurationScreen: FC<{ theme: string }> = ({ theme }) => { } }, [theme]); - const currentConfigurationStep = getCurrentConfigurationStep(); - return (
- +
-
diff --git a/packages/react-components/src/components/ConfigurationUserPreferencesStep.css b/packages/react-components/src/components/ConfigurationUserPreferencesStep.css index c367223..67d1cf9 100644 --- a/packages/react-components/src/components/ConfigurationUserPreferencesStep.css +++ b/packages/react-components/src/components/ConfigurationUserPreferencesStep.css @@ -2,13 +2,8 @@ font-size: 0.75rem; color: var(--secondary-text-color); margin-top: 5px; - margin-bottom: 1rem; } -.switch-layout { - display: flex; - flex-direction: row; - gap: 0.5rem; - align-items: center; - margin-top: 1rem; +.switch-container { + margin-top: 0.5rem; } diff --git a/packages/react-components/src/components/ConfigurationUserPreferencesStep.tsx b/packages/react-components/src/components/ConfigurationUserPreferencesStep.tsx index 92eaf9c..02fac2f 100644 --- a/packages/react-components/src/components/ConfigurationUserPreferencesStep.tsx +++ b/packages/react-components/src/components/ConfigurationUserPreferencesStep.tsx @@ -1,47 +1,112 @@ -import React, { ChangeEvent, FC } from 'react'; -import Label from './Label'; +import React, { FC, useEffect, useState } from 'react'; +import { ConfigurationStep, Tenant } from '../models/flare'; import Button from './Button'; -import { Tenant } from '../models/flare'; +import Label from './Label'; import Select from './Select'; +import { APP_NAME } from '../models/constants'; +import { + fetchAvailableIndexNames, + fetchCurrentIndexName, + fetchIngestMetadataOnly, + fetchTenantId, + fetchUserTenants, + saveConfiguration, +} from '../utils/setupConfiguration'; import './ConfigurationGlobalStep.css'; import './ConfigurationUserPreferencesStep.css'; import Switch from './Switch'; +import { ToastKeys, toastManager } from './ToastManager'; +import Tooltip from './Tooltip'; const ConfigurationUserPreferencesStep: FC<{ show: boolean; - tenants: Tenant[]; - selectedTenantId: number; - indexNames: string[]; - selectedIndexName: string; - isLoading: boolean; - isIngestingMetadataOnly: boolean; + configurationStep: ConfigurationStep; + apiKey: string; onNavigateBackClick: () => void; - onSubmitUserPreferencesClick: () => void; - onTenantIdChange: (e: ChangeEvent) => void; - onIndexNameChange: (e: ChangeEvent) => void; - onIngestingMetadataChange: (e: ChangeEvent) => void; -}> = ({ - show, - tenants, - selectedTenantId, - indexNames, - selectedIndexName, - isLoading, - isIngestingMetadataOnly, - onNavigateBackClick, - onSubmitUserPreferencesClick, - onTenantIdChange, - onIndexNameChange, - onIngestingMetadataChange, -}) => { + onUserPreferencesSaved: () => void; +}> = ({ show, configurationStep, apiKey, onNavigateBackClick, onUserPreferencesSaved }) => { + const [tenantId, setTenantId] = useState(undefined); + const [tenants, setUserTenants] = useState([]); + const [indexName, setIndexName] = useState(''); + const [indexNames, setIndexNames] = useState([]); + const [isIngestingMetadataOnly, setIsIngestingMetadataOnly] = useState(false); + const [isLoading, setIsLoading] = useState(false); + + const handleTenantIdChange = (e): void => setTenantId(parseInt(e.target.value, 10)); + const handleIndexNameChange = (e): void => setIndexName(e.target.value); + const handleIsIngestingMetadataChange = (e): void => + setIsIngestingMetadataOnly(e.target.checked); + + const handleSubmitUserPreferences = (): void => { + setIsLoading(true); + + saveConfiguration(apiKey, Number(tenantId), indexName, isIngestingMetadataOnly) + .then(() => { + setIsLoading(false); + toastManager.destroy(ToastKeys.ERROR); + toastManager.show({ + id: ToastKeys.SUCCESS, + content: 'Configured Flare Account', + }); + onUserPreferencesSaved(); + }) + .catch((e: any) => { + setIsLoading(false); + toastManager.show({ + id: ToastKeys.ERROR, + isError: true, + content: `Something went wrong. ${e.responseText}`, + }); + }); + }; + + useEffect(() => { + if (configurationStep === ConfigurationStep.UserPreferences) { + Promise.all([ + fetchTenantId(), + fetchIngestMetadataOnly(), + fetchCurrentIndexName(), + fetchUserTenants(apiKey), + fetchAvailableIndexNames(), + ]) + .then(([id, ingestMetadataOnly, index, userTenants, availableIndexNames]) => { + setTenantId(id); + setIsIngestingMetadataOnly(ingestMetadataOnly); + setIndexName(index); + if (id === -1 && userTenants.length > 0) { + setTenantId(userTenants[0].id); + } + setUserTenants(userTenants); + setIndexNames(availableIndexNames); + }) + .catch(() => { + toastManager.show({ + id: ToastKeys.ERROR, + isError: true, + content: 'Something went wrong.', + }); + }); + } else { + setTenantId(undefined); + setIndexName(APP_NAME); + setIndexNames([]); + setUserTenants([]); + setIsLoading(false); + } + }, [configurationStep, apiKey]); + + const isFormValid = (): boolean => { + return tenantId !== undefined; + }; + return (