Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Synthetics] Template string syntax breaks inline monitors when running against the service #176094

Open
wants to merge 79 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
79 commits
Select commit Hold shift + click to select a range
be2a4e8
PoC stage of fix.
justinkambic Feb 1, 2024
9993d23
Include expect in list of imports.
justinkambic Feb 2, 2024
1bc597c
Add writable stream and zip for add/edit/sync inline monitors.
justinkambic Feb 9, 2024
643ab7a
Fix run_once.
justinkambic Feb 9, 2024
c1dadd4
Encapsulate.
justinkambic Feb 9, 2024
15cdea6
PR cleanup.
justinkambic Feb 12, 2024
45246c2
PR feedback.
justinkambic Feb 14, 2024
48255f4
PR feedback.
justinkambic Feb 14, 2024
9de233d
Simplify per PR advice.
justinkambic Feb 19, 2024
ce574a9
Improve tests, reorganize code.
justinkambic Feb 19, 2024
8a599ca
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 18, 2024
b43ab54
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 20, 2024
130d1d8
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 23, 2024
4473510
Working to clean up tests. WIP.
justinkambic Sep 23, 2024
27b5060
Fix tests.
justinkambic Sep 24, 2024
6b9669f
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 24, 2024
771b15a
Delete unneeded file.
justinkambic Sep 24, 2024
5d796af
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 25, 2024
5cbe70e
Fix project field mapping issue where monitor type is not browser.
justinkambic Sep 25, 2024
f4bc77d
Default to whatever the inline script is if zipping should fail.
justinkambic Sep 25, 2024
c5ace2c
Fix types.
justinkambic Sep 25, 2024
a5eb787
Simplify.
justinkambic Sep 25, 2024
64d7bb8
Add test.
justinkambic Sep 26, 2024
c11b121
Touch up promise.
justinkambic Sep 26, 2024
1967d7d
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 26, 2024
abecb34
Remove regression from monitor GET API.
justinkambic Sep 26, 2024
02c81ef
Modify utility API to use object param instead of individual args.
justinkambic Sep 26, 2024
bdfb632
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 26, 2024
bc6be09
Fix types.
justinkambic Sep 30, 2024
b74a7af
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Sep 30, 2024
f996e4c
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Oct 1, 2024
f530675
Add a test.
justinkambic Oct 1, 2024
816ea89
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Oct 2, 2024
3aaf679
Fix run_once.
justinkambic Oct 2, 2024
ca237b2
Test run_once.
justinkambic Oct 2, 2024
841f17f
Unroll changes to `formatHeartbeatRequest`.
justinkambic Oct 2, 2024
caff1d0
Rename `AddEditMonitorAPI` to `UpsertMonitorAPI`.
justinkambic Oct 2, 2024
210387a
Fix all usages of `UpsertMonitorAPI`.
justinkambic Oct 2, 2024
5904329
Unbreak formatter tests.
justinkambic Oct 2, 2024
c93625f
Rename upsert api file.
justinkambic Oct 2, 2024
c96dde9
Rename test file.
justinkambic Oct 2, 2024
cc09e3f
Remove inline zip procedure from hydrate function.
justinkambic Oct 2, 2024
7c4af54
Make hydrate function sync again.
justinkambic Oct 2, 2024
743de32
Modify edit procedure to zip project and omit inline script field.
justinkambic Oct 2, 2024
1a7ff07
Rename a var.
justinkambic Oct 2, 2024
1dd2482
Remove unneeded async/await.
justinkambic Oct 2, 2024
0224832
Remove unneeded async/await.
justinkambic Oct 2, 2024
f8d8e8f
Simplify.
justinkambic Oct 2, 2024
ac78c00
Fix ping heatmap payload.
justinkambic Oct 4, 2024
b316cc9
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Oct 4, 2024
a86d955
Fix zip for persist.
justinkambic Oct 4, 2024
a4dee3b
Fix types.
justinkambic Oct 4, 2024
2df2554
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Oct 9, 2024
abbcd79
Merge branch 'main' of github.com:elastic/kibana into 169963/inline-s…
justinkambic Oct 14, 2024
0cc010c
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Oct 16, 2024
13c6907
Do not drop inline script when persisting a monitor edit.
justinkambic Oct 16, 2024
3e8bf39
Fix broken edit unit test.
justinkambic Oct 16, 2024
440015a
Simplify project mapping API.
justinkambic Oct 16, 2024
32bf01a
Include `mfa` keyword in script wrapper.
justinkambic Oct 16, 2024
823143d
Add fixed date to archiver call.
justinkambic Oct 16, 2024
09782ef
Fix types.
justinkambic Oct 16, 2024
6e12216
Fix broken unit tests.
justinkambic Oct 16, 2024
716c2b2
TEMP
justinkambic Oct 30, 2024
7d6ae2b
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Oct 30, 2024
53f5ff1
Fix delete crud api.
justinkambic Oct 30, 2024
84294c7
Revert "TEMP"
justinkambic Oct 30, 2024
8c77e40
TEMP
justinkambic Oct 30, 2024
43f7c7b
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Nov 14, 2024
4975ab4
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Nov 20, 2024
c82b865
Add code to skip inline source field for private locations.
justinkambic Nov 20, 2024
cb6ae0f
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Nov 21, 2024
94bcd61
Omit inline script from policy for private location monitors.
justinkambic Nov 22, 2024
6cf85b2
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
justinkambic Nov 22, 2024
ba824a5
Fix types.
justinkambic Nov 22, 2024
b0deda0
Remove unwanted logs.
justinkambic Nov 22, 2024
2c08084
Fix API tests.
justinkambic Nov 22, 2024
a277078
[CI] Auto-commit changed files from 'node scripts/eslint --no-cache -…
kibanamachine Nov 22, 2024
6c1a256
Merge branch 'main' into 169963/inline-scripts-with-template-syntax-f…
shahzad31 Nov 25, 2024
e604c5d
revert changes
shahzad31 Nov 25, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -259,6 +259,8 @@ export const BrowserSensitiveSimpleFieldsCodec = t.intersection([
CommonFieldsCodec,
]);

export type BrowserSensitiveSimpleFields = t.TypeOf<typeof BrowserSensitiveSimpleFieldsCodec>;

export const ThrottlingConfigValueCodec = t.interface({
download: t.string,
upload: t.string,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { inlineToProjectZip } from './inline_to_zip';
import { unzipFile } from './unzip_project_code';

describe('inlineToProjectZip', () => {
it('should return base64 encoded zip data', async () => {
const inlineJourney = `
step('goto', () => page.goto('https://elastic.co'));
step('throw error', () => { throw new Error('error'); });
`;
const result = await inlineToProjectZip(inlineJourney, 'testMonitorId', jest.fn() as any);

expect(result.length).toBeGreaterThan(0);
expect(await unzipFile(result)).toEqual(
`import { journey, step, expect, mfa } from '@elastic/synthetics';

journey('inline', ({ page, context, browser, params, request }) => {

step('goto', () => page.goto('https://elastic.co'));
step('throw error', () => { throw new Error('error'); });

});`
);
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { Logger } from '@kbn/logging';
import archiver from 'archiver';
import { MemWritable } from './mem_writable';

function wrapInlineInProject(inlineJourney: string) {
return `import { journey, step, expect, mfa } from '@elastic/synthetics';

journey('inline', ({ page, context, browser, params, request }) => {
${inlineJourney}
});`;
}

export async function inlineToProjectZip(
inlineJourney: string,
monitorId: string,
logger: Logger
): Promise<string> {
const mWriter = new MemWritable();
try {
await new Promise<void>((resolve, reject) => {
const archive = archiver('zip', {
zlib: { level: 9 },
});
archive.on('error', reject);
mWriter.on('close', resolve);
archive.pipe(mWriter);
vigneshshanmugam marked this conversation as resolved.
Show resolved Hide resolved
archive.append(wrapInlineInProject(inlineJourney), {
justinkambic marked this conversation as resolved.
Show resolved Hide resolved
name: 'inline.journey.ts',
justinkambic marked this conversation as resolved.
Show resolved Hide resolved
// Date is fixed to Unix epoch so the file metadata is
// not modified everytime when files are bundled
date: new Date('1970-01-01'),
});
return archive.finalize();
});
} catch (e) {
logger.error(`Failed to create zip for inline monitor ${monitorId}`);
throw e;
}
return mWriter.buffer.toString('base64');
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { MemWritable } from './mem_writable';

describe('MemWritable', () => {
it('should write chunks to the buffer', async () => {
const memWritable = new MemWritable();

const chunk1 = `step('goto', () => page.goto('https://elastic.co'));`;
const chunk2 = `step('throw error', () => { throw new Error('error'); });`;
const expectedBuffer = Buffer.from(chunk1 + chunk2);

await new Promise<void>((resolve, reject) => {
memWritable.write(Buffer.from(chunk1), (err) => {
if (err) {
reject(err);
}
expect(memWritable.buffer).toEqual(Buffer.from(chunk1));
resolve();
});
});
await new Promise<void>((resolve, reject) => {
memWritable.write(Buffer.from(chunk2), (err) => {
if (err) {
reject(err);
}
resolve();
});
});
expect(memWritable.buffer).toEqual(expectedBuffer);
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { Writable, WritableOptions } from 'node:stream';

export class MemWritable extends Writable {
private _queue: Buffer[];
constructor(opts?: WritableOptions) {
super(opts);
this._queue = [];
}

public get buffer(): Buffer {
return Buffer.concat(this._queue);
}

_write(
chunk: Buffer,
_encoding: BufferEncoding,
callback: (error?: Error | null | undefined) => void
): void {
this._queue.push(chunk);
callback();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,143 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { ConfigKey, SyntheticsMonitor } from '../../../common/runtime_types';
import { UpsertMonitorAPI } from './add_monitor/upsert_monitor_api';

describe('hydrateMonitorFields', () => {
it('does not add project field value for inline browser monitor', () => {
const normalizedMonitor: SyntheticsMonitor = {
// @ts-expect-error extra field
type: 'browser',
// @ts-expect-error extra field
form_monitor_type: 'multistep',
enabled: true,
alert: { status: { enabled: true }, tls: { enabled: true } },
// @ts-expect-error extra field
schedule: { unit: 'm', number: '10' },
'service.name': '',
config_id: '',
tags: [],
timeout: null,
name: 'test-once-more',
locations: [{ id: 'dev', label: 'Dev Service', isServiceManaged: true }],
namespace: 'default',
// @ts-expect-error extra field
origin: 'ui',
journey_id: '',
hash: '',
id: '',
params: '',
max_attempts: 2,
project_id: '',
playwright_options: '',
__ui: { script_source: { is_generated_script: false, file_name: '' } },
'url.port': null,
'source.inline.script': `step('goto', () => page.goto('https://elastic.co'))
step('fail', () => {
throw Error('fail');
})`,
playwright_text_assertion: '',
urls: '',
screenshots: 'on',
synthetics_args: [],
'filter_journeys.match': '',
'filter_journeys.tags': [],
ignore_https_errors: false,
throttling: {
value: { download: '5', upload: '3', latency: '20' },
id: 'default',
label: 'Default',
},
'ssl.certificate_authorities': '',
'ssl.certificate': '',
'ssl.key': '',
'ssl.key_passphrase': '',
'ssl.verification_mode': 'full',
'ssl.supported_protocols': ['TLSv1.1', 'TLSv1.2', 'TLSv1.3'],
};

const api = new UpsertMonitorAPI({
request: {
query: {
preserve_namespace: true,
},
},
server: {
logger: jest.fn(),
},
} as any);
const hydratedMonitor = api.hydrateMonitorFields({
normalizedMonitor,
newMonitorId: 'testMonitorId',
});

expect((hydratedMonitor as any)[ConfigKey.SOURCE_PROJECT_CONTENT]).toBeUndefined();
});

it('does not add b64 zip data to lightweight monitors', () => {
const newMonitorId = 'testMonitorId';
const routeContext = {
request: {
query: {
preserve_namespace: true,
},
},
server: {
logger: jest.fn(),
},
};
const normalizedMonitor: SyntheticsMonitor = {
// @ts-expect-error extra field
type: 'tcp',
// @ts-expect-error extra field
form_monitor_type: 'tcp',
enabled: true,
alert: { status: { enabled: true }, tls: { enabled: true } },
// @ts-expect-error extra field
schedule: { number: '3', unit: 'm' },
'service.name': '',
config_id: '',
tags: [],
timeout: '16',
name: 'tcp://google.com:80',
locations: [{ id: 'dev', label: 'Dev Service', isServiceManaged: true }],
namespace: 'default',
// @ts-expect-error extra field
origin: 'ui',
journey_id: '',
hash: '',
id: '',
params: '',
max_attempts: 2,
__ui: { is_tls_enabled: false },
hosts: 'tcp://google.com:80',
urls: '',
'url.port': null,
proxy_url: '',
proxy_use_local_resolver: false,
'check.receive': '',
'check.send': '',
mode: 'any',
ipv4: true,
ipv6: true,
'ssl.certificate_authorities': '',
'ssl.certificate': '',
'ssl.key': '',
'ssl.key_passphrase': '',
'ssl.verification_mode': 'full',
'ssl.supported_protocols': ['TLSv1.1', 'TLSv1.2', 'TLSv1.3'],
};
const api = new UpsertMonitorAPI(routeContext as any);
const hydratedMonitor = api.hydrateMonitorFields({
normalizedMonitor,
newMonitorId,
});

expect((hydratedMonitor as any)[ConfigKey.SOURCE_PROJECT_CONTENT]).toBeFalsy();
});
});
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,12 @@ import { SavedObjectsErrorHelpers } from '@kbn/core/server';
import { i18n } from '@kbn/i18n';
import { validatePermissions } from './edit_monitor';
import { InvalidLocationError } from '../../synthetics_service/project_monitor/normalizers/common_fields';
import { AddEditMonitorAPI, CreateMonitorPayLoad } from './add_monitor/add_monitor_api';
import { UpsertMonitorAPI, CreateMonitorPayLoad } from './add_monitor/upsert_monitor_api';
import { SyntheticsRestApiRouteFactory } from '../types';
import { SYNTHETICS_API_URLS } from '../../../common/constants';
import { normalizeAPIConfig, validateMonitor } from './monitor_validation';
import { mapSavedObjectToMonitor } from './formatters/saved_object_to_monitor';
import { mapInlineToProjectFields } from '../../synthetics_service/utils/map_inline_to_project_fields';

export const addSyntheticsMonitorRoute: SyntheticsRestApiRouteFactory = () => ({
method: 'POST',
Expand All @@ -39,7 +40,7 @@ export const addSyntheticsMonitorRoute: SyntheticsRestApiRouteFactory = () => ({
// usually id is auto generated, but this is useful for testing
const { id, internal } = request.query;

const addMonitorAPI = new AddEditMonitorAPI(routeContext);
const addMonitorAPI = new UpsertMonitorAPI(routeContext);

const {
locations,
Expand Down Expand Up @@ -86,7 +87,15 @@ export const addSyntheticsMonitorRoute: SyntheticsRestApiRouteFactory = () => ({
});
}

const normalizedMonitor = validationResult.decodedMonitor;
const normalizedMonitor = Object.assign(
{},
validationResult.decodedMonitor,
await mapInlineToProjectFields({
monitorType: validationResult.decodedMonitor?.type,
monitor: validationResult?.decodedMonitor,
logger: server.logger,
})
);

const err = await validatePermissions(routeContext, normalizedMonitor.locations);
if (err) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
* 2.0.
*/

import { AddEditMonitorAPI } from './add_monitor_api';
import { UpsertMonitorAPI } from './upsert_monitor_api';
import { SyntheticsMonitorClient } from '../../../synthetics_service/synthetics_monitor/synthetics_monitor_client';
import { SyntheticsService } from '../../../synthetics_service/synthetics_service';

Expand All @@ -16,7 +16,7 @@ describe('AddNewMonitorsPublicAPI', () => {
enabled: true,
},
} as any);
const api = new AddEditMonitorAPI({
const api = new UpsertMonitorAPI({
syntheticsMonitorClient: new SyntheticsMonitorClient(syntheticsService, {} as any),
request: {
body: {},
Expand Down Expand Up @@ -55,7 +55,7 @@ describe('AddNewMonitorsPublicAPI', () => {
const syntheticsService = new SyntheticsService({
config: {},
} as any);
const api = new AddEditMonitorAPI({
const api = new UpsertMonitorAPI({
syntheticsMonitorClient: new SyntheticsMonitorClient(syntheticsService, {} as any),
request: {
body: {},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,12 +49,9 @@ export type CreateMonitorPayLoad = MonitorFields & {
schedule?: number | MonitorFields['schedule'];
};

export class AddEditMonitorAPI {
routeContext: RouteContext;
export class UpsertMonitorAPI {
allPrivateLocations?: PrivateLocationAttributes[];
constructor(routeContext: RouteContext) {
this.routeContext = routeContext;
}
constructor(private readonly routeContext: RouteContext) {}

async syncNewMonitor({
id,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
*/

import { SavedObjectsClientContract } from '@kbn/core-saved-objects-api-server';
import { CreateMonitorPayLoad } from './add_monitor_api';
import { CreateMonitorPayLoad } from './upsert_monitor_api';
import { MonitorFields, SyntheticsMonitor } from '../../../../common/runtime_types';
import { getPrivateLocations } from '../../../synthetics_service/get_private_locations';

Expand Down
Loading