|
6 | 6 | */ |
7 | 7 |
|
8 | 8 | import { AuthenticatedUser } from '@kbn/core-security-common'; |
9 | | - |
10 | | -import { getAttackDiscoveryStats } from './helpers'; |
| 9 | +import moment from 'moment/moment'; |
| 10 | +import { getAttackDiscoveryStats, updateAttackDiscoveries } from './helpers'; |
11 | 11 | import { AttackDiscoveryDataClient } from '../../../lib/attack_discovery/persistence'; |
12 | 12 | import { transformESSearchToAttackDiscovery } from '../../../lib/attack_discovery/persistence/transforms/transforms'; |
13 | 13 | import { getAttackDiscoverySearchEsMock } from '../../../__mocks__/attack_discovery_schema.mock'; |
| 14 | +import { mockAnonymizedAlerts } from '../../../lib/attack_discovery/evaluation/__mocks__/mock_anonymized_alerts'; |
| 15 | +import { mockAttackDiscoveries } from '../../../lib/attack_discovery/evaluation/__mocks__/mock_attack_discoveries'; |
| 16 | +import { coreMock } from '@kbn/core/server/mocks'; |
| 17 | +import { loggerMock } from '@kbn/logging-mocks'; |
14 | 18 |
|
15 | 19 | jest.mock('lodash/fp', () => ({ |
16 | 20 | uniq: jest.fn((arr) => Array.from(new Set(arr))), |
@@ -270,4 +274,169 @@ describe('helpers', () => { |
270 | 274 | ]); |
271 | 275 | }); |
272 | 276 | }); |
| 277 | + |
| 278 | + describe('updateAttackDiscoveries', () => { |
| 279 | + const mockTelemetry = coreMock.createSetup().analytics; |
| 280 | + const mockLogger = loggerMock.create(); |
| 281 | + const mockStartTime = moment('2024-03-28T22:27:28.000Z'); |
| 282 | + const mockApiConfig = { |
| 283 | + actionTypeId: '.gen-ai', |
| 284 | + connectorId: 'my-gen-ai', |
| 285 | + model: 'gpt-4', |
| 286 | + }; |
| 287 | + const mockReplacements = {}; |
| 288 | + |
| 289 | + beforeEach(() => { |
| 290 | + jest.clearAllMocks(); |
| 291 | + }); |
| 292 | + |
| 293 | + it('should update attack discovery successfully', async () => { |
| 294 | + getAttackDiscovery.mockResolvedValue(mockCurrentAd); |
| 295 | + updateAttackDiscovery.mockResolvedValue({}); |
| 296 | + |
| 297 | + await updateAttackDiscoveries({ |
| 298 | + anonymizedAlerts: mockAnonymizedAlerts, |
| 299 | + apiConfig: mockApiConfig, |
| 300 | + attackDiscoveries: mockAttackDiscoveries, |
| 301 | + attackDiscoveryId: 'attack-discovery-id', |
| 302 | + authenticatedUser: mockAuthenticatedUser, |
| 303 | + dataClient: mockDataClient, |
| 304 | + hasFilter: false, |
| 305 | + end: 'now', |
| 306 | + latestReplacements: mockReplacements, |
| 307 | + logger: mockLogger, |
| 308 | + size: 10, |
| 309 | + start: 'now-24h', |
| 310 | + startTime: mockStartTime, |
| 311 | + telemetry: mockTelemetry, |
| 312 | + }); |
| 313 | + |
| 314 | + expect(updateAttackDiscovery).toHaveBeenCalledWith({ |
| 315 | + attackDiscoveryUpdateProps: expect.objectContaining({ |
| 316 | + status: 'succeeded', |
| 317 | + }), |
| 318 | + authenticatedUser: mockAuthenticatedUser, |
| 319 | + }); |
| 320 | + expect(mockTelemetry.reportEvent).toHaveBeenCalledWith('attack_discovery_success', { |
| 321 | + actionTypeId: '.gen-ai', |
| 322 | + alertsContextCount: 2, |
| 323 | + alertsCount: 8, |
| 324 | + configuredAlertsCount: 10, |
| 325 | + dateRangeDuration: 24, |
| 326 | + discoveriesGenerated: 1, |
| 327 | + durationMs: 0, |
| 328 | + hasFilter: false, |
| 329 | + isDefaultDateRange: true, |
| 330 | + model: 'gpt-4', |
| 331 | + }); |
| 332 | + }); |
| 333 | + it('should detect non-default time range', async () => { |
| 334 | + getAttackDiscovery.mockResolvedValue(mockCurrentAd); |
| 335 | + updateAttackDiscovery.mockResolvedValue({}); |
| 336 | + |
| 337 | + await updateAttackDiscoveries({ |
| 338 | + anonymizedAlerts: mockAnonymizedAlerts, |
| 339 | + apiConfig: mockApiConfig, |
| 340 | + attackDiscoveries: mockAttackDiscoveries, |
| 341 | + attackDiscoveryId: 'attack-discovery-id', |
| 342 | + authenticatedUser: mockAuthenticatedUser, |
| 343 | + dataClient: mockDataClient, |
| 344 | + hasFilter: false, |
| 345 | + end: 'now', |
| 346 | + latestReplacements: mockReplacements, |
| 347 | + logger: mockLogger, |
| 348 | + size: 10, |
| 349 | + start: 'now-1w', |
| 350 | + startTime: mockStartTime, |
| 351 | + telemetry: mockTelemetry, |
| 352 | + }); |
| 353 | + |
| 354 | + expect(updateAttackDiscovery).toHaveBeenCalledWith({ |
| 355 | + attackDiscoveryUpdateProps: expect.objectContaining({ |
| 356 | + status: 'succeeded', |
| 357 | + }), |
| 358 | + authenticatedUser: mockAuthenticatedUser, |
| 359 | + }); |
| 360 | + expect(mockTelemetry.reportEvent).toHaveBeenCalledWith('attack_discovery_success', { |
| 361 | + actionTypeId: '.gen-ai', |
| 362 | + alertsContextCount: 2, |
| 363 | + alertsCount: 8, |
| 364 | + configuredAlertsCount: 10, |
| 365 | + dateRangeDuration: 168, |
| 366 | + discoveriesGenerated: 1, |
| 367 | + durationMs: 0, |
| 368 | + hasFilter: false, |
| 369 | + isDefaultDateRange: false, |
| 370 | + model: 'gpt-4', |
| 371 | + }); |
| 372 | + }); |
| 373 | + it('hasFilter should be true when filter exists', async () => { |
| 374 | + getAttackDiscovery.mockResolvedValue(mockCurrentAd); |
| 375 | + updateAttackDiscovery.mockResolvedValue({}); |
| 376 | + |
| 377 | + await updateAttackDiscoveries({ |
| 378 | + anonymizedAlerts: mockAnonymizedAlerts, |
| 379 | + apiConfig: mockApiConfig, |
| 380 | + attackDiscoveries: mockAttackDiscoveries, |
| 381 | + attackDiscoveryId: 'attack-discovery-id', |
| 382 | + authenticatedUser: mockAuthenticatedUser, |
| 383 | + dataClient: mockDataClient, |
| 384 | + hasFilter: true, |
| 385 | + end: 'now', |
| 386 | + latestReplacements: mockReplacements, |
| 387 | + logger: mockLogger, |
| 388 | + size: 10, |
| 389 | + start: 'now-24h', |
| 390 | + startTime: mockStartTime, |
| 391 | + telemetry: mockTelemetry, |
| 392 | + }); |
| 393 | + |
| 394 | + expect(updateAttackDiscovery).toHaveBeenCalledWith({ |
| 395 | + attackDiscoveryUpdateProps: expect.objectContaining({ |
| 396 | + status: 'succeeded', |
| 397 | + }), |
| 398 | + authenticatedUser: mockAuthenticatedUser, |
| 399 | + }); |
| 400 | + expect(mockTelemetry.reportEvent).toHaveBeenCalledWith('attack_discovery_success', { |
| 401 | + actionTypeId: '.gen-ai', |
| 402 | + alertsContextCount: 2, |
| 403 | + alertsCount: 8, |
| 404 | + configuredAlertsCount: 10, |
| 405 | + dateRangeDuration: 24, |
| 406 | + discoveriesGenerated: 1, |
| 407 | + durationMs: 0, |
| 408 | + hasFilter: true, |
| 409 | + isDefaultDateRange: true, |
| 410 | + model: 'gpt-4', |
| 411 | + }); |
| 412 | + }); |
| 413 | + |
| 414 | + it('should handle error during update', async () => { |
| 415 | + const mockError = new Error('Update failed'); |
| 416 | + getAttackDiscovery.mockRejectedValue(mockError); |
| 417 | + |
| 418 | + await updateAttackDiscoveries({ |
| 419 | + anonymizedAlerts: mockAnonymizedAlerts, |
| 420 | + apiConfig: mockApiConfig, |
| 421 | + attackDiscoveries: mockAttackDiscoveries, |
| 422 | + attackDiscoveryId: 'attack-discovery-id', |
| 423 | + authenticatedUser: mockAuthenticatedUser, |
| 424 | + dataClient: mockDataClient, |
| 425 | + hasFilter: false, |
| 426 | + end: 'now', |
| 427 | + latestReplacements: mockReplacements, |
| 428 | + logger: mockLogger, |
| 429 | + size: 10, |
| 430 | + start: 'now-24h', |
| 431 | + startTime: mockStartTime, |
| 432 | + telemetry: mockTelemetry, |
| 433 | + }); |
| 434 | + |
| 435 | + expect(mockLogger.error).toHaveBeenCalledWith(mockError); |
| 436 | + expect(mockTelemetry.reportEvent).toHaveBeenCalledWith( |
| 437 | + 'attack_discovery_error', |
| 438 | + expect.any(Object) |
| 439 | + ); |
| 440 | + }); |
| 441 | + }); |
273 | 442 | }); |
0 commit comments