From 847285ba7191aa6d26fb3dccc06748e1c4a202b1 Mon Sep 17 00:00:00 2001 From: Konrad Szwarc Date: Fri, 27 Sep 2024 23:22:28 +0200 Subject: [PATCH] [EDR Workflows] Fix agent count for single agent policies (#194294) https://github.com/user-attachments/assets/2b64c1e0-0e6d-4ef5-952d-e4364b4403c4 The PR #193705 introduced an issue when counting active agents for integration policies with only one agent policy assigned. In such cases, `query.policyIds` was treated as a single string instead of an array of strings (as expected with multiple agent policy ids like `/?policyIds=x&policyIds=y`). This PR resolves the issue by ensuring consistent handling of policyIds, regardless of the number of associated agent policies. --- .../server/routes/agent/handlers.test.ts | 64 ++++++++++++++++--- .../fleet/server/routes/agent/handlers.ts | 11 +++- .../fleet/server/types/rest_spec/agent.ts | 2 +- .../middleware/policy_settings_middleware.ts | 2 +- .../management/services/policies/ingest.ts | 4 +- 5 files changed, 69 insertions(+), 14 deletions(-) diff --git a/x-pack/plugins/fleet/server/routes/agent/handlers.test.ts b/x-pack/plugins/fleet/server/routes/agent/handlers.test.ts index 8e064a92a96a1..e5071b5272c27 100644 --- a/x-pack/plugins/fleet/server/routes/agent/handlers.test.ts +++ b/x-pack/plugins/fleet/server/routes/agent/handlers.test.ts @@ -7,7 +7,9 @@ import { coreMock, httpServerMock } from '@kbn/core/server/mocks'; -import { getAvailableVersionsHandler } from './handlers'; +import { getAgentStatusForAgentPolicy } from '../../services/agents/status'; + +import { getAgentStatusForAgentPolicyHandler, getAvailableVersionsHandler } from './handlers'; jest.mock('../../services/agents/versions', () => { return { @@ -24,16 +26,60 @@ jest.mock('../../services/app_context', () => { }; }); -describe('getAvailableVersionsHandler', () => { - it('should return the value from getAvailableVersions', async () => { - const ctx = coreMock.createCustomRequestHandlerContext(coreMock.createRequestHandlerContext()); - const response = httpServerMock.createResponseFactory(); +jest.mock('../../services/agents/status', () => ({ + getAgentStatusForAgentPolicy: jest.fn(), +})); + +describe('Handlers', () => { + describe('getAgentStatusForAgentPolicyHandler', () => { + it.each([ + { requested: 'policy-id-1', called: ['policy-id-1'] }, + { requested: ['policy-id-2'], called: ['policy-id-2'] }, + { requested: ['policy-id-3', 'policy-id-4'], called: ['policy-id-3', 'policy-id-4'] }, + ...[undefined, '', []].map((requested) => ({ requested, called: undefined })), + ])('calls getAgentStatusForAgentPolicy with correct parameters', async (item) => { + const request = { + query: { + policyId: 'policy-id', + kuery: 'kuery', + policyIds: item.requested, + }, + }; + const response = httpServerMock.createResponseFactory(); + + await getAgentStatusForAgentPolicyHandler( + { + core: coreMock.createRequestHandlerContext(), + fleet: { internalSoClient: {} }, + } as any, + request as any, + response + ); + + expect(getAgentStatusForAgentPolicy).toHaveBeenCalledWith( + expect.anything(), + expect.anything(), + 'policy-id', + 'kuery', + undefined, + item.called + ); + }); + }); + + describe('getAvailableVersionsHandler', () => { + it('should return the value from getAvailableVersions', async () => { + const ctx = coreMock.createCustomRequestHandlerContext( + coreMock.createRequestHandlerContext() + ); + const response = httpServerMock.createResponseFactory(); - await getAvailableVersionsHandler(ctx, httpServerMock.createKibanaRequest(), response); + await getAvailableVersionsHandler(ctx, httpServerMock.createKibanaRequest(), response); - expect(response.ok).toBeCalled(); - expect(response.ok.mock.calls[0][0]?.body).toEqual({ - items: ['8.1.0', '8.0.0', '7.17.0'], + expect(response.ok).toBeCalled(); + expect(response.ok.mock.calls[0][0]?.body).toEqual({ + items: ['8.1.0', '8.0.0', '7.17.0'], + }); }); }); }); diff --git a/x-pack/plugins/fleet/server/routes/agent/handlers.ts b/x-pack/plugins/fleet/server/routes/agent/handlers.ts index 67703bba5caae..4f4b5592f2d04 100644 --- a/x-pack/plugins/fleet/server/routes/agent/handlers.ts +++ b/x-pack/plugins/fleet/server/routes/agent/handlers.ts @@ -323,13 +323,22 @@ export const getAgentStatusForAgentPolicyHandler: FleetRequestHandler< const [coreContext, fleetContext] = await Promise.all([context.core, context.fleet]); const esClient = coreContext.elasticsearch.client.asInternalUser; const soClient = fleetContext.internalSoClient; + + const parsePolicyIds = (policyIds: string | string[] | undefined): string[] | undefined => { + if (!policyIds || !policyIds.length) { + return undefined; + } + + return Array.isArray(policyIds) ? policyIds : [policyIds]; + }; + const results = await getAgentStatusForAgentPolicy( esClient, soClient, request.query.policyId, request.query.kuery, coreContext.savedObjects.client.getCurrentNamespace(), - request.query.policyIds + parsePolicyIds(request.query.policyIds) ); const body: GetAgentStatusResponse = { results }; diff --git a/x-pack/plugins/fleet/server/types/rest_spec/agent.ts b/x-pack/plugins/fleet/server/types/rest_spec/agent.ts index 82cae68602e94..8f44d5554501e 100644 --- a/x-pack/plugins/fleet/server/types/rest_spec/agent.ts +++ b/x-pack/plugins/fleet/server/types/rest_spec/agent.ts @@ -241,7 +241,7 @@ export const PostBulkUpdateAgentTagsRequestSchema = { export const GetAgentStatusRequestSchema = { query: schema.object({ policyId: schema.maybe(schema.string()), - policyIds: schema.maybe(schema.arrayOf(schema.string())), + policyIds: schema.maybe(schema.oneOf([schema.arrayOf(schema.string()), schema.string()])), kuery: schema.maybe( schema.string({ validate: (value: string) => { diff --git a/x-pack/plugins/security_solution/public/management/pages/policy/store/policy_details/middleware/policy_settings_middleware.ts b/x-pack/plugins/security_solution/public/management/pages/policy/store/policy_details/middleware/policy_settings_middleware.ts index e74403777523c..bf87eab2a7293 100644 --- a/x-pack/plugins/security_solution/public/management/pages/policy/store/policy_details/middleware/policy_settings_middleware.ts +++ b/x-pack/plugins/security_solution/public/management/pages/policy/store/policy_details/middleware/policy_settings_middleware.ts @@ -96,7 +96,7 @@ export const policySettingsMiddlewareRunner: MiddlewareRunner = async ( // Agent summary is secondary data, so its ok for it to come after the details // page is populated with the main content - if (policyItem.policy_id) { + if (policyItem.policy_ids?.length) { const { results } = await sendGetFleetAgentStatusForPolicy(http, policyItem.policy_ids); dispatch({ type: 'serverReturnedPolicyDetailsAgentSummaryData', diff --git a/x-pack/plugins/security_solution/public/management/services/policies/ingest.ts b/x-pack/plugins/security_solution/public/management/services/policies/ingest.ts index 523b1b9a858b1..c125464bffdb9 100644 --- a/x-pack/plugins/security_solution/public/management/services/policies/ingest.ts +++ b/x-pack/plugins/security_solution/public/management/services/policies/ingest.ts @@ -83,7 +83,7 @@ export const sendPutPackagePolicy = ( }; /** - * Get a status summary for all Agents that are currently assigned to a given agent policy + * Get a status summary for all Agents that are currently assigned to a given agent policies * * @param http * @param policyIds @@ -91,7 +91,7 @@ export const sendPutPackagePolicy = ( */ export const sendGetFleetAgentStatusForPolicy = ( http: HttpStart, - /** the Agent (fleet) policy id */ + /** the Agent (fleet) policy ids */ policyIds: string[], options: Exclude = {} ): Promise => {