From caa3916e7a49b3ffef1235c177afe42df8942efe Mon Sep 17 00:00:00 2001 From: Jeramy Soucy Date: Fri, 2 Feb 2024 14:22:56 -0500 Subject: [PATCH] [7.17] Updates test file wrapper to deterministically detect file write completion (#176115) (#176167) # Backport This will backport the following commits from `main` to `7.17`: - [Updates test file wrapper to deterministically detect file write completion (#176115)](https://github.com/elastic/kibana/pull/176115) ### Questions ? Please refer to the [Backport tool documentation](https://github.com/sqren/backport) --- .../tests/audit/audit_log.ts | 23 +++++++++++-------- 1 file changed, 13 insertions(+), 10 deletions(-) diff --git a/x-pack/test/security_api_integration/tests/audit/audit_log.ts b/x-pack/test/security_api_integration/tests/audit/audit_log.ts index 65ceaa46dd44a..a54e10623b4d8 100644 --- a/x-pack/test/security_api_integration/tests/audit/audit_log.ts +++ b/x-pack/test/security_api_integration/tests/audit/audit_log.ts @@ -12,6 +12,7 @@ import { RetryService } from '../../../../../test/common/services/retry'; import { FtrProviderContext } from '../../ftr_provider_context'; class FileWrapper { + delay = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms)); constructor(private readonly path: string, private readonly retry: RetryService) {} async reset() { // "touch" each file to ensure it exists and is empty before each test @@ -35,10 +36,15 @@ class FileWrapper { }); } // writing in a file is an async operation. we use this method to make sure logs have been written. - async isNotEmpty() { - const content = await this.read(); - const line = content[0]; - return line.length > 0; + async isWritten() { + // attempt at determinism - wait for the size of the file to stop changing. + await this.retry.waitForWithTimeout(`file '${this.path}' to be written`, 5000, async () => { + const sizeBefore = Fs.statSync(this.path).size; + await this.delay(500); + const sizeAfter = Fs.statSync(this.path).size; + return sizeAfter === sizeBefore; + }); + return Fs.statSync(this.path).size > 0; } } @@ -57,8 +63,7 @@ export default function ({ getService }: FtrProviderContext) { it('logs audit events when reading and writing saved objects', async () => { await supertest.get('/audit_log?query=param').set('kbn-xsrf', 'foo').expect(204); - await retry.waitFor('logs event in the dest file', async () => await logFile.isNotEmpty()); - + await logFile.isWritten(); const content = await logFile.readJSON(); const httpEvent = content.find((c) => c.event.action === 'http_request'); @@ -94,8 +99,7 @@ export default function ({ getService }: FtrProviderContext) { params: { username, password }, }) .expect(200); - await retry.waitFor('logs event in the dest file', async () => await logFile.isNotEmpty()); - + await logFile.isWritten(); const content = await logFile.readJSON(); const loginEvent = content.find((c) => c.event.action === 'user_login'); @@ -116,8 +120,7 @@ export default function ({ getService }: FtrProviderContext) { params: { username, password: 'invalid_password' }, }) .expect(401); - await retry.waitFor('logs event in the dest file', async () => await logFile.isNotEmpty()); - + await logFile.isWritten(); const content = await logFile.readJSON(); const loginEvent = content.find((c) => c.event.action === 'user_login');