Skip to content

Commit

Permalink
[8.x] [Observability] Use Fleet's default data output when onboarding…
Browse files Browse the repository at this point in the history
… integrations using auto-detect flow (elastic#201158) (elastic#201833)

# Backport

This will backport the following commits from `main` to `8.x`:
- [[Observability] Use Fleet's default data output when onboarding
integrations using auto-detect flow
(elastic#201158)](elastic#201158)

<!--- Backport version: 8.9.8 -->

### Questions ?
Please refer to the [Backport tool
documentation](https://github.com/sqren/backport)

<!--BACKPORT [{"author":{"name":"Thom
Heymann","email":"[email protected]"},"sourceCommit":{"committedDate":"2024-11-25T11:14:49Z","message":"[Observability]
Use Fleet's default data output when onboarding integrations using
auto-detect flow (elastic#201158)\n\nResolves elastic#199751\r\n\r\n##
Summary\r\n\r\nUse Fleet's default data output when onboarding
integrations using\r\nauto-detect flow.\r\n\r\n## Screenshot\r\n\r\n###
Fleet output settings\r\n\r\n<img width=\"1411\" alt=\"Screenshot
2024-11-21 at 15 10
24\"\r\nsrc=\"https://github.com/user-attachments/assets/ac193552-7f18-4566-a84b-061df45c13f3\">\r\n\r\n###
Generated Agent config\r\n\r\n```\r\n$ cat
/Library/Elastic/Agent/elastic-agent.yml\r\n\r\noutputs:\r\n
default:\r\n type: elasticsearch\r\n hosts:\r\n -
https://192.168.1.73:9200\r\n ssl.ca_trusted_fingerprint:
c48c98cdf7f85d7cc29f834704011c1002b5251d594fc0bb08e6177544fe304a\r\n
api_key:
b1BkR1Q1TUIyOUpfMWhaS2NCUXA6SS1Jb3dncGVReVNpcEdzOGpSVmlzQQ==\r\n preset:
balanced\r\n```\r\n\r\n## Testing\r\n\r\n1. Go to Fleet > Settings >
Outputs\r\n2. Edit the default output and enter dummy data into the
\"Elasticsearch\r\nCA trusted fingerprint\" field\r\n3. Go through the
auto-detect onboarding flow\r\n4. Inspect the `elastic-agent.yml` file
written to disk. It should\r\ncontain the default output configured in
Fleet including\r\n`ca_trusted_fingerprint`
setting","sha":"845aa8c47c858a6ccf83390bb38f332424ce2221","branchLabelMapping":{"^v9.0.0$":"main","^v8.18.0$":"8.x","^v(\\d+).(\\d+).\\d+$":"$1.$2"}},"sourcePullRequest":{"labels":["release_note:skip","backport
missing","Team:Fleet","v9.0.0","backport:prev-minor","ci:project-deploy-observability"],"number":201158,"url":"https://github.com/elastic/kibana/pull/201158","mergeCommit":{"message":"[Observability]
Use Fleet's default data output when onboarding integrations using
auto-detect flow (elastic#201158)\n\nResolves elastic#199751\r\n\r\n##
Summary\r\n\r\nUse Fleet's default data output when onboarding
integrations using\r\nauto-detect flow.\r\n\r\n## Screenshot\r\n\r\n###
Fleet output settings\r\n\r\n<img width=\"1411\" alt=\"Screenshot
2024-11-21 at 15 10
24\"\r\nsrc=\"https://github.com/user-attachments/assets/ac193552-7f18-4566-a84b-061df45c13f3\">\r\n\r\n###
Generated Agent config\r\n\r\n```\r\n$ cat
/Library/Elastic/Agent/elastic-agent.yml\r\n\r\noutputs:\r\n
default:\r\n type: elasticsearch\r\n hosts:\r\n -
https://192.168.1.73:9200\r\n ssl.ca_trusted_fingerprint:
c48c98cdf7f85d7cc29f834704011c1002b5251d594fc0bb08e6177544fe304a\r\n
api_key:
b1BkR1Q1TUIyOUpfMWhaS2NCUXA6SS1Jb3dncGVReVNpcEdzOGpSVmlzQQ==\r\n preset:
balanced\r\n```\r\n\r\n## Testing\r\n\r\n1. Go to Fleet > Settings >
Outputs\r\n2. Edit the default output and enter dummy data into the
\"Elasticsearch\r\nCA trusted fingerprint\" field\r\n3. Go through the
auto-detect onboarding flow\r\n4. Inspect the `elastic-agent.yml` file
written to disk. It should\r\ncontain the default output configured in
Fleet including\r\n`ca_trusted_fingerprint`
setting","sha":"845aa8c47c858a6ccf83390bb38f332424ce2221"}},"sourceBranch":"main","suggestedTargetBranches":[],"targetPullRequestStates":[{"branch":"main","label":"v9.0.0","labelRegex":"^v9.0.0$","isSourceBranch":true,"state":"MERGED","url":"https://github.com/elastic/kibana/pull/201158","number":201158,"mergeCommit":{"message":"[Observability]
Use Fleet's default data output when onboarding integrations using
auto-detect flow (elastic#201158)\n\nResolves elastic#199751\r\n\r\n##
Summary\r\n\r\nUse Fleet's default data output when onboarding
integrations using\r\nauto-detect flow.\r\n\r\n## Screenshot\r\n\r\n###
Fleet output settings\r\n\r\n<img width=\"1411\" alt=\"Screenshot
2024-11-21 at 15 10
24\"\r\nsrc=\"https://github.com/user-attachments/assets/ac193552-7f18-4566-a84b-061df45c13f3\">\r\n\r\n###
Generated Agent config\r\n\r\n```\r\n$ cat
/Library/Elastic/Agent/elastic-agent.yml\r\n\r\noutputs:\r\n
default:\r\n type: elasticsearch\r\n hosts:\r\n -
https://192.168.1.73:9200\r\n ssl.ca_trusted_fingerprint:
c48c98cdf7f85d7cc29f834704011c1002b5251d594fc0bb08e6177544fe304a\r\n
api_key:
b1BkR1Q1TUIyOUpfMWhaS2NCUXA6SS1Jb3dncGVReVNpcEdzOGpSVmlzQQ==\r\n preset:
balanced\r\n```\r\n\r\n## Testing\r\n\r\n1. Go to Fleet > Settings >
Outputs\r\n2. Edit the default output and enter dummy data into the
\"Elasticsearch\r\nCA trusted fingerprint\" field\r\n3. Go through the
auto-detect onboarding flow\r\n4. Inspect the `elastic-agent.yml` file
written to disk. It should\r\ncontain the default output configured in
Fleet including\r\n`ca_trusted_fingerprint`
setting","sha":"845aa8c47c858a6ccf83390bb38f332424ce2221"}}]}]
BACKPORT-->
  • Loading branch information
thomheymann authored Nov 27, 2024
1 parent 9a76b27 commit c7ecbec
Show file tree
Hide file tree
Showing 6 changed files with 161 additions and 20 deletions.
2 changes: 2 additions & 0 deletions x-pack/plugins/fleet/server/mocks/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ import { createFleetActionsClientMock } from '../services/actions/mocks';
import { createFleetFilesClientFactoryMock } from '../services/files/mocks';

import { createArtifactsClientMock } from '../services/artifacts/mocks';
import { createOutputClientMock } from '../services/output_client.mock';

import type { PackagePolicyClient } from '../services/package_policy_service';
import type { AgentPolicyServiceInterface } from '../services';
Expand Down Expand Up @@ -301,6 +302,7 @@ export const createFleetStartContractMock = (): DeeplyMockedKeys<FleetStartContr
uninstallTokenService: createUninstallTokenServiceMock(),
createFleetActionsClient: jest.fn((_) => fleetActionsClient),
getPackageSpecTagId: jest.fn(getPackageSpecTagId),
createOutputClient: jest.fn(async (_) => createOutputClientMock()),
};

return startContract;
Expand Down
13 changes: 13 additions & 0 deletions x-pack/plugins/fleet/server/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,8 @@ import {
MessageSigningService,
} from './services/security';

import { OutputClient, type OutputClientInterface } from './services/output_client';

import {
ASSETS_SAVED_OBJECT_TYPE,
DOWNLOAD_SOURCE_SAVED_OBJECT_TYPE,
Expand Down Expand Up @@ -262,6 +264,12 @@ export interface FleetStartContract {
Function exported to allow creating unique ids for saved object tags
*/
getPackageSpecTagId: (spaceId: string, pkgName: string, tagName: string) => string;

/**
* Create a Fleet Output Client instance
* @param packageName
*/
createOutputClient: (request: KibanaRequest) => Promise<OutputClientInterface>;
}

export class FleetPlugin
Expand Down Expand Up @@ -837,6 +845,11 @@ export class FleetPlugin
return new FleetActionsClient(core.elasticsearch.client.asInternalUser, packageName);
},
getPackageSpecTagId,
async createOutputClient(request: KibanaRequest) {
const soClient = appContextService.getSavedObjects().getScopedClient(request);
const authz = await getAuthzFromRequest(request);
return new OutputClient(soClient, authz);
},
};
}

Expand Down
21 changes: 21 additions & 0 deletions x-pack/plugins/fleet/server/services/output_client.mock.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
/*
* 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.
*/
/*
* 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 type { OutputClientInterface } from './output_client';

export const createOutputClientMock = (): jest.Mocked<OutputClientInterface> => {
return {
getDefaultDataOutputId: jest.fn(),
get: jest.fn(),
};
};
71 changes: 71 additions & 0 deletions x-pack/plugins/fleet/server/services/output_client.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
/*
* 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 { savedObjectsClientMock } from '@kbn/core/server/mocks';

import { createFleetAuthzMock } from '../../common/mocks';

import { OutputClient } from './output_client';
import { outputService } from './output';

jest.mock('./output');

const mockedOutputService = outputService as jest.Mocked<typeof outputService>;

describe('OutputClient', () => {
afterEach(() => {
jest.clearAllMocks();
});

describe('getDefaultDataOutputId()', () => {
it('should call output service `getDefaultDataOutputId()` method', async () => {
const soClient = savedObjectsClientMock.create();
const authz = createFleetAuthzMock();
const outputClient = new OutputClient(soClient, authz);
await outputClient.getDefaultDataOutputId();

expect(mockedOutputService.getDefaultDataOutputId).toHaveBeenCalledWith(soClient);
});

it('should throw error when no `fleet.readSettings` and no `fleet.readAgentPolicies` privileges', async () => {
const soClient = savedObjectsClientMock.create();
const authz = createFleetAuthzMock();
authz.fleet.readSettings = false;
authz.fleet.readAgentPolicies = false;
const outputClient = new OutputClient(soClient, authz);

await expect(outputClient.getDefaultDataOutputId()).rejects.toMatchInlineSnapshot(
`[OutputUnauthorizedError]`
);
expect(mockedOutputService.getDefaultDataOutputId).not.toHaveBeenCalled();
});
});

describe('get()', () => {
it('should call output service `get()` method', async () => {
const soClient = savedObjectsClientMock.create();
const authz = createFleetAuthzMock();
const outputClient = new OutputClient(soClient, authz);
await outputClient.get('default');

expect(mockedOutputService.get).toHaveBeenCalledWith(soClient, 'default');
});

it('should throw error when no `fleet.readSettings` and no `fleet.readAgentPolicies` privileges', async () => {
const soClient = savedObjectsClientMock.create();
const authz = createFleetAuthzMock();
authz.fleet.readSettings = false;
authz.fleet.readAgentPolicies = false;
const outputClient = new OutputClient(soClient, authz);

await expect(outputClient.get('default')).rejects.toMatchInlineSnapshot(
`[OutputUnauthorizedError]`
);
expect(mockedOutputService.get).not.toHaveBeenCalled();
});
});
});
40 changes: 40 additions & 0 deletions x-pack/plugins/fleet/server/services/output_client.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
/*
* 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 type { SavedObjectsClientContract } from '@kbn/core/server';

import type { FleetAuthz } from '../../common';

import { OutputUnauthorizedError } from '../errors';
import type { Output } from '../types';

import { outputService } from './output';

export { transformOutputToFullPolicyOutput } from './agent_policies/full_agent_policy';

export interface OutputClientInterface {
getDefaultDataOutputId(): Promise<string | null>;
get(outputId: string): Promise<Output>;
}

export class OutputClient implements OutputClientInterface {
constructor(private soClient: SavedObjectsClientContract, private authz: FleetAuthz) {}

async getDefaultDataOutputId() {
if (!this.authz.fleet.readSettings && !this.authz.fleet.readAgentPolicies) {
throw new OutputUnauthorizedError();
}
return outputService.getDefaultDataOutputId(this.soClient);
}

async get(outputId: string) {
if (!this.authz.fleet.readSettings && !this.authz.fleet.readAgentPolicies) {
throw new OutputUnauthorizedError();
}
return outputService.get(this.soClient, outputId);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,15 +13,15 @@ import {
type PackageClient,
} from '@kbn/fleet-plugin/server';
import { safeDump } from 'js-yaml';
import { PackageDataStreamTypes } from '@kbn/fleet-plugin/common/types';
import { PackageDataStreamTypes, Output } from '@kbn/fleet-plugin/common/types';
import { transformOutputToFullPolicyOutput } from '@kbn/fleet-plugin/server/services/output_client';
import { getObservabilityOnboardingFlow, saveObservabilityOnboardingFlow } from '../../lib/state';
import type { SavedObservabilityOnboardingFlow } from '../../saved_objects/observability_onboarding_status';
import { ObservabilityOnboardingFlow } from '../../saved_objects/observability_onboarding_status';
import { createObservabilityOnboardingServerRoute } from '../create_observability_onboarding_server_route';
import { getHasLogs } from './get_has_logs';
import { getKibanaUrl } from '../../lib/get_fallback_urls';
import { getAgentVersionInfo } from '../../lib/get_agent_version';
import { getFallbackESUrl } from '../../lib/get_fallback_urls';
import { ElasticAgentStepPayload, InstalledIntegration, StepProgressPayloadRT } from '../types';
import { createShipperApiKey } from '../../lib/api_key/create_shipper_api_key';
import { createInstallApiKey } from '../../lib/api_key/create_install_api_key';
Expand Down Expand Up @@ -329,6 +329,13 @@ const integrationsInstallRoute = createObservabilityOnboardingServerRoute({
throw Boom.notFound(`Onboarding session '${params.path.onboardingId}' not found.`);
}

const outputClient = await fleetStart.createOutputClient(request);
const defaultOutputId = await outputClient.getDefaultDataOutputId();
if (!defaultOutputId) {
throw Boom.notFound('Default data output not found');
}
const output = await outputClient.get(defaultOutputId);

const integrationsToInstall = parseIntegrationsTSV(params.body);
if (!integrationsToInstall.length) {
return response.badRequest({
Expand Down Expand Up @@ -381,15 +388,11 @@ const integrationsInstallRoute = createObservabilityOnboardingServerRoute({
},
});

const elasticsearchUrl = plugins.cloud?.setup?.elasticsearchUrl
? [plugins.cloud?.setup?.elasticsearchUrl]
: await getFallbackESUrl(services.esLegacyConfigService);

return response.ok({
headers: {
'content-type': 'application/x-tar',
},
body: generateAgentConfigTar({ elasticsearchUrl, installedIntegrations }),
body: generateAgentConfigTar(output, installedIntegrations),
});
},
});
Expand Down Expand Up @@ -565,14 +568,9 @@ function parseRegistryIntegrationMetadata(
}
}

const generateAgentConfigTar = ({
elasticsearchUrl,
installedIntegrations,
}: {
elasticsearchUrl: string[];
installedIntegrations: InstalledIntegration[];
}) => {
function generateAgentConfigTar(output: Output, installedIntegrations: InstalledIntegration[]) {
const now = new Date();

return makeTar([
{
type: 'File',
Expand All @@ -581,11 +579,7 @@ const generateAgentConfigTar = ({
mtime: now,
data: safeDump({
outputs: {
default: {
type: 'elasticsearch',
hosts: elasticsearchUrl,
api_key: '${API_KEY}', // Placeholder to be replaced by bash script with the actual API key
},
default: transformOutputToFullPolicyOutput(output, undefined, true),
},
}),
},
Expand All @@ -603,7 +597,7 @@ const generateAgentConfigTar = ({
data: integration.config,
})),
]);
};
}

export const flowRouteRepository = {
...createFlowRoute,
Expand Down

0 comments on commit c7ecbec

Please sign in to comment.