Skip to content

Commit

Permalink
[8.x] Add ESLINT constraints to detect inter-group dependencies (#194810
Browse files Browse the repository at this point in the history
) (#197670)

# Backport

This will backport the following commits from `main` to `8.x`:
- [Add ESLINT constraints to detect inter-group dependencies
(#194810)](#194810)

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

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

<!--BACKPORT [{"author":{"name":"Gerard
Soldevila","email":"[email protected]"},"sourceCommit":{"committedDate":"2024-10-22T11:34:19Z","message":"Add
ESLINT constraints to detect inter-group dependencies (#194810)\n\n##
Summary\r\n\r\nAddresses
https://github.com/elastic/kibana-team/issues/1175\r\n\r\nAs part of the
**Sustainable Kibana Architecture** initiative, this PR\r\nsets the
foundation to start classifying plugins in isolated groups,\r\nmatching
our current solutions / project types:\r\n\r\n* It adds support for the
following fields in the packages' manifests\r\n(kibana.jsonc):\r\n*
`group?: 'search' | 'security' | 'observability' | 'platform'
|\r\n'common'`\r\n * `visibility?: 'private' | 'shared'`\r\n\r\n* It
proposes a folder structure to automatically infer
groups:\r\n```javascript\r\n 'src/platform/plugins/shared': {\r\n group:
'platform',\r\n visibility: 'shared',\r\n },\r\n
'src/platform/plugins/internal': {\r\n group: 'platform',\r\n
visibility: 'private',\r\n },\r\n 'x-pack/platform/plugins/shared':
{\r\n group: 'platform',\r\n visibility: 'shared',\r\n },\r\n
'x-pack/platform/plugins/internal': {\r\n group: 'platform',\r\n
visibility: 'private',\r\n },\r\n
'x-pack/solutions/observability/plugins': {\r\n group:
'observability',\r\n visibility: 'private',\r\n },\r\n
'x-pack/solutions/security/plugins': {\r\n group: 'security',\r\n
visibility: 'private',\r\n },\r\n 'x-pack/solutions/search/plugins':
{\r\n group: 'search',\r\n visibility: 'private',\r\n },\r\n```\r\n\r\n*
If a plugin is moved to one of the specific locations above, the
group\r\nand visibility in the manifest (if specified) must match those
inferred\r\nfrom the path.\r\n* Plugins that are not relocated are
considered: `group: 'common',\r\nvisibility: 'shared'` by default. As
soon as we specify a custom\r\n`group`, the ESLINT rules will check
violations against dependencies /\r\ndependants.\r\n\r\nThe ESLINT rules
are pretty simple:\r\n* Plugins can only depend on:\r\n * Plugins in the
same group\r\n * OR plugins with `'shared'` visibility\r\n* Plugins in
`'observability', 'security', 'search'` groups are\r\nmandatorily
`'private'`.\r\n\r\n---------\r\n\r\nCo-authored-by: kibanamachine
<[email protected]>","sha":"2a085e103afe8c7bdfb626d0dc683fc8be0e6c05","branchLabelMapping":{"^v9.0.0$":"main","^v8.17.0$":"8.x","^v(\\d+).(\\d+).\\d+$":"$1.$2"}},"sourcePullRequest":{"labels":["Team:Core","backport
missing","v9.0.0","release_note:feature","backport:prev-minor"],"number":194810,"url":"https://github.com/elastic/kibana/pull/194810","mergeCommit":{"message":"Add
ESLINT constraints to detect inter-group dependencies (#194810)\n\n##
Summary\r\n\r\nAddresses
https://github.com/elastic/kibana-team/issues/1175\r\n\r\nAs part of the
**Sustainable Kibana Architecture** initiative, this PR\r\nsets the
foundation to start classifying plugins in isolated groups,\r\nmatching
our current solutions / project types:\r\n\r\n* It adds support for the
following fields in the packages' manifests\r\n(kibana.jsonc):\r\n*
`group?: 'search' | 'security' | 'observability' | 'platform'
|\r\n'common'`\r\n * `visibility?: 'private' | 'shared'`\r\n\r\n* It
proposes a folder structure to automatically infer
groups:\r\n```javascript\r\n 'src/platform/plugins/shared': {\r\n group:
'platform',\r\n visibility: 'shared',\r\n },\r\n
'src/platform/plugins/internal': {\r\n group: 'platform',\r\n
visibility: 'private',\r\n },\r\n 'x-pack/platform/plugins/shared':
{\r\n group: 'platform',\r\n visibility: 'shared',\r\n },\r\n
'x-pack/platform/plugins/internal': {\r\n group: 'platform',\r\n
visibility: 'private',\r\n },\r\n
'x-pack/solutions/observability/plugins': {\r\n group:
'observability',\r\n visibility: 'private',\r\n },\r\n
'x-pack/solutions/security/plugins': {\r\n group: 'security',\r\n
visibility: 'private',\r\n },\r\n 'x-pack/solutions/search/plugins':
{\r\n group: 'search',\r\n visibility: 'private',\r\n },\r\n```\r\n\r\n*
If a plugin is moved to one of the specific locations above, the
group\r\nand visibility in the manifest (if specified) must match those
inferred\r\nfrom the path.\r\n* Plugins that are not relocated are
considered: `group: 'common',\r\nvisibility: 'shared'` by default. As
soon as we specify a custom\r\n`group`, the ESLINT rules will check
violations against dependencies /\r\ndependants.\r\n\r\nThe ESLINT rules
are pretty simple:\r\n* Plugins can only depend on:\r\n * Plugins in the
same group\r\n * OR plugins with `'shared'` visibility\r\n* Plugins in
`'observability', 'security', 'search'` groups are\r\nmandatorily
`'private'`.\r\n\r\n---------\r\n\r\nCo-authored-by: kibanamachine
<[email protected]>","sha":"2a085e103afe8c7bdfb626d0dc683fc8be0e6c05"}},"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/194810","number":194810,"mergeCommit":{"message":"Add
ESLINT constraints to detect inter-group dependencies (#194810)\n\n##
Summary\r\n\r\nAddresses
https://github.com/elastic/kibana-team/issues/1175\r\n\r\nAs part of the
**Sustainable Kibana Architecture** initiative, this PR\r\nsets the
foundation to start classifying plugins in isolated groups,\r\nmatching
our current solutions / project types:\r\n\r\n* It adds support for the
following fields in the packages' manifests\r\n(kibana.jsonc):\r\n*
`group?: 'search' | 'security' | 'observability' | 'platform'
|\r\n'common'`\r\n * `visibility?: 'private' | 'shared'`\r\n\r\n* It
proposes a folder structure to automatically infer
groups:\r\n```javascript\r\n 'src/platform/plugins/shared': {\r\n group:
'platform',\r\n visibility: 'shared',\r\n },\r\n
'src/platform/plugins/internal': {\r\n group: 'platform',\r\n
visibility: 'private',\r\n },\r\n 'x-pack/platform/plugins/shared':
{\r\n group: 'platform',\r\n visibility: 'shared',\r\n },\r\n
'x-pack/platform/plugins/internal': {\r\n group: 'platform',\r\n
visibility: 'private',\r\n },\r\n
'x-pack/solutions/observability/plugins': {\r\n group:
'observability',\r\n visibility: 'private',\r\n },\r\n
'x-pack/solutions/security/plugins': {\r\n group: 'security',\r\n
visibility: 'private',\r\n },\r\n 'x-pack/solutions/search/plugins':
{\r\n group: 'search',\r\n visibility: 'private',\r\n },\r\n```\r\n\r\n*
If a plugin is moved to one of the specific locations above, the
group\r\nand visibility in the manifest (if specified) must match those
inferred\r\nfrom the path.\r\n* Plugins that are not relocated are
considered: `group: 'common',\r\nvisibility: 'shared'` by default. As
soon as we specify a custom\r\n`group`, the ESLINT rules will check
violations against dependencies /\r\ndependants.\r\n\r\nThe ESLINT rules
are pretty simple:\r\n* Plugins can only depend on:\r\n * Plugins in the
same group\r\n * OR plugins with `'shared'` visibility\r\n* Plugins in
`'observability', 'security', 'search'` groups are\r\nmandatorily
`'private'`.\r\n\r\n---------\r\n\r\nCo-authored-by: kibanamachine
<[email protected]>","sha":"2a085e103afe8c7bdfb626d0dc683fc8be0e6c05"}}]}]
BACKPORT-->
  • Loading branch information
gsoldevila authored Oct 24, 2024
1 parent 9d54a62 commit 7b82013
Show file tree
Hide file tree
Showing 36 changed files with 1,277 additions and 48 deletions.
7 changes: 4 additions & 3 deletions .github/CODEOWNERS
Validating CODEOWNERS rules …
Original file line number Diff line number Diff line change
Expand Up @@ -596,6 +596,7 @@ packages/kbn-management/settings/types @elastic/kibana-management
packages/kbn-management/settings/utilities @elastic/kibana-management
packages/kbn-management/storybook/config @elastic/kibana-management
test/plugin_functional/plugins/management_test_plugin @elastic/kibana-management
packages/kbn-manifest @elastic/kibana-core
packages/kbn-mapbox-gl @elastic/kibana-gis
x-pack/examples/third_party_maps_source_example @elastic/kibana-gis
src/plugins/maps_ems @elastic/kibana-gis
Expand Down Expand Up @@ -930,9 +931,9 @@ packages/kbn-test-eui-helpers @elastic/kibana-visualizations
x-pack/test/licensing_plugin/plugins/test_feature_usage @elastic/kibana-security
packages/kbn-test-jest-helpers @elastic/kibana-operations @elastic/appex-qa
packages/kbn-test-subj-selector @elastic/kibana-operations @elastic/appex-qa
x-pack/test_serverless
test
x-pack/test
x-pack/test_serverless
test
x-pack/test
x-pack/performance @elastic/appex-qa
x-pack/examples/testing_embedded_lens @elastic/kibana-visualizations
x-pack/examples/third_party_lens_navigation_prompt @elastic/kibana-visualizations
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -633,6 +633,7 @@
"@kbn/management-settings-types": "link:packages/kbn-management/settings/types",
"@kbn/management-settings-utilities": "link:packages/kbn-management/settings/utilities",
"@kbn/management-test-plugin": "link:test/plugin_functional/plugins/management_test_plugin",
"@kbn/manifest": "link:packages/kbn-manifest",
"@kbn/mapbox-gl": "link:packages/kbn-mapbox-gl",
"@kbn/maps-custom-raster-source-plugin": "link:x-pack/examples/third_party_maps_source_example",
"@kbn/maps-ems-plugin": "link:src/plugins/maps_ems",
Expand Down
35 changes: 30 additions & 5 deletions packages/kbn-dev-utils/src/plugin_list/run_plugin_list_cli.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,14 +20,39 @@ const OUTPUT_PATH = Path.resolve(REPO_ROOT, 'docs/developer/plugin-list.asciidoc
export function runPluginListCli() {
run(async ({ log }) => {
log.info('looking for oss plugins');
const ossPlugins = discoverPlugins('src/plugins');
log.success(`found ${ossPlugins.length} plugins`);
const ossLegacyPlugins = discoverPlugins('src/plugins');
const ossPlatformPlugins = discoverPlugins('src/platform/plugins');
log.success(`found ${ossLegacyPlugins.length + ossPlatformPlugins.length} plugins`);

log.info('looking for x-pack plugins');
const xpackPlugins = discoverPlugins('x-pack/plugins');
log.success(`found ${xpackPlugins.length} plugins`);
const xpackLegacyPlugins = discoverPlugins('x-pack/plugins');
const xpackPlatformPlugins = discoverPlugins('x-pack/platform/plugins');
const xpackSearchPlugins = discoverPlugins('x-pack/solutions/search/plugins');
const xpackSecurityPlugins = discoverPlugins('x-pack/solutions/security/plugins');
const xpackObservabilityPlugins = discoverPlugins('x-pack/solutions/observability/plugins');
log.success(
`found ${
xpackLegacyPlugins.length +
xpackPlatformPlugins.length +
xpackSearchPlugins.length +
xpackSecurityPlugins.length +
xpackObservabilityPlugins.length
} plugins`
);

log.info('writing plugin list to', OUTPUT_PATH);
Fs.writeFileSync(OUTPUT_PATH, generatePluginList(ossPlugins, xpackPlugins));
Fs.writeFileSync(
OUTPUT_PATH,
generatePluginList(
[...ossLegacyPlugins, ...ossPlatformPlugins],
[
...xpackLegacyPlugins,
...xpackPlatformPlugins,
...xpackSearchPlugins,
...xpackSecurityPlugins,
...xpackObservabilityPlugins,
]
)
);
});
}
3 changes: 2 additions & 1 deletion packages/kbn-eslint-config/.eslintrc.js
Original file line number Diff line number Diff line change
Expand Up @@ -326,7 +326,8 @@ module.exports = {
'@kbn/imports/uniform_imports': 'error',
'@kbn/imports/no_unused_imports': 'error',
'@kbn/imports/no_boundary_crossing': 'error',

'@kbn/imports/no_group_crossing_manifests': 'error',
'@kbn/imports/no_group_crossing_imports': 'error',
'no-new-func': 'error',
'no-implied-eval': 'error',
'no-prototype-builtins': 'error',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,6 @@ export const PROTECTED_RULES = new Set([
'@kbn/disable/no_protected_eslint_disable',
'@kbn/disable/no_naked_eslint_disable',
'@kbn/imports/no_unused_imports',
'@kbn/imports/no_group_crossing_imports',
'@kbn/imports/no_group_crossing_manifests',
]);
4 changes: 4 additions & 0 deletions packages/kbn-eslint-plugin-imports/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@ import { UniformImportsRule } from './src/rules/uniform_imports';
import { ExportsMovedPackagesRule } from './src/rules/exports_moved_packages';
import { NoUnusedImportsRule } from './src/rules/no_unused_imports';
import { NoBoundaryCrossingRule } from './src/rules/no_boundary_crossing';
import { NoGroupCrossingImportsRule } from './src/rules/no_group_crossing_imports';
import { NoGroupCrossingManifestsRule } from './src/rules/no_group_crossing_manifests';
import { RequireImportRule } from './src/rules/require_import';

/**
Expand All @@ -25,5 +27,7 @@ export const rules = {
exports_moved_packages: ExportsMovedPackagesRule,
no_unused_imports: NoUnusedImportsRule,
no_boundary_crossing: NoBoundaryCrossingRule,
no_group_crossing_imports: NoGroupCrossingImportsRule,
no_group_crossing_manifests: NoGroupCrossingManifestsRule,
require_import: RequireImportRule,
};
25 changes: 25 additions & 0 deletions packages/kbn-eslint-plugin-imports/src/helpers/groups.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
/*
* 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", the "GNU Affero General Public License v3.0 only", and the "Server Side
* Public License v 1"; you may not use this file except in compliance with, at
* your election, the "Elastic License 2.0", the "GNU Affero General Public
* License v3.0 only", or the "Server Side Public License, v 1".
*/

import type { ModuleGroup, ModuleVisibility } from '@kbn/repo-info/types';

/**
* Checks whether a given ModuleGroup can import from another one
* @param importerGroup The group of the module that we are checking
* @param importedGroup The group of the imported module
* @param importedVisibility The visibility of the imported module
* @returns true if importerGroup is allowed to import from importedGroup/Visibiliy
*/
export function isImportableFrom(
importerGroup: ModuleGroup,
importedGroup: ModuleGroup,
importedVisibility: ModuleVisibility
): boolean {
return importerGroup === importedGroup || importedVisibility === 'shared';
}
16 changes: 16 additions & 0 deletions packages/kbn-eslint-plugin-imports/src/helpers/report.ts
Original file line number Diff line number Diff line change
Expand Up @@ -30,3 +30,19 @@ export function report(context: Rule.RuleContext, options: ReportOptions) {
: null,
});
}

export const toList = (strings: string[]) => {
const items = strings.map((s) => `"${s}"`);
const list = items.slice(0, -1).join(', ');
const last = items.at(-1);
return !list.length ? last ?? '' : `${list} or ${last}`;
};

export const formatSuggestions = (suggestions: string[]) => {
const s = suggestions.map((l) => l.trim()).filter(Boolean);
if (!s.length) {
return '';
}

return ` \nSuggestions:\n - ${s.join('\n - ')}\n\n`;
};
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,9 @@

import { RuleTester } from 'eslint';
import { NoBoundaryCrossingRule } from './no_boundary_crossing';
import { ModuleType } from '@kbn/repo-source-classifier';
import type { ModuleType } from '@kbn/repo-source-classifier';
import dedent from 'dedent';
import { formatSuggestions } from '../helpers/report';

const make = (from: ModuleType, to: ModuleType, imp = 'import') => ({
filename: `${from}.ts`,
Expand Down Expand Up @@ -107,13 +108,12 @@ for (const [name, tester] of [tsTester, babelTester]) {
data: {
importedType: 'server package',
ownType: 'common package',
suggestion: ` ${dedent`
Suggestions:
- Remove the import statement.
- Limit your imports to "common package" or "static" code.
- Covert to a type-only import.
- Reach out to #kibana-operations for help.
`}`,
suggestion: formatSuggestions([
'Remove the import statement.',
'Limit your imports to "common package" or "static" code.',
'Covert to a type-only import.',
'Reach out to #kibana-operations for help.',
]),
},
},
],
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,13 +12,14 @@ import Path from 'path';
import { TSESTree } from '@typescript-eslint/typescript-estree';
import * as Bt from '@babel/types';
import type { Rule } from 'eslint';
import ESTree from 'estree';
import { ModuleType } from '@kbn/repo-source-classifier';
import type { Node } from 'estree';
import type { ModuleType } from '@kbn/repo-source-classifier';

import { visitAllImportStatements, Importer } from '../helpers/visit_all_import_statements';
import { getSourcePath } from '../helpers/source';
import { getRepoSourceClassifier } from '../helpers/repo_source_classifier';
import { getImportResolver } from '../get_import_resolver';
import { formatSuggestions, toList } from '../helpers/report';

const ANY = Symbol();

Expand All @@ -33,22 +34,6 @@ const IMPORTABLE_FROM: Record<ModuleType, ModuleType[] | typeof ANY> = {
tooling: ANY,
};

const toList = (strings: string[]) => {
const items = strings.map((s) => `"${s}"`);
const list = items.slice(0, -1).join(', ');
const last = items.at(-1);
return !list.length ? last ?? '' : `${list} or ${last}`;
};

const formatSuggestions = (suggestions: string[]) => {
const s = suggestions.map((l) => l.trim()).filter(Boolean);
if (!s.length) {
return '';
}

return ` Suggestions:\n - ${s.join('\n - ')}`;
};

const isTypeOnlyImport = (importer: Importer) => {
// handle babel nodes
if (Bt.isImportDeclaration(importer)) {
Expand Down Expand Up @@ -125,7 +110,7 @@ export const NoBoundaryCrossingRule: Rule.RuleModule = {

if (!importable.includes(imported.type)) {
context.report({
node: node as ESTree.Node,
node: node as Node,
messageId: 'TYPE_MISMATCH',
data: {
ownType: self.type,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,155 @@
/*
* 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", the "GNU Affero General Public License v3.0 only", and the "Server Side
* Public License v 1"; you may not use this file except in compliance with, at
* your election, the "Elastic License 2.0", the "GNU Affero General Public
* License v3.0 only", or the "Server Side Public License, v 1".
*/

import { RuleTester } from 'eslint';
import dedent from 'dedent';
import { NoGroupCrossingImportsRule } from './no_group_crossing_imports';
import { formatSuggestions } from '../helpers/report';
import { ModuleGroup, ModuleVisibility } from '@kbn/repo-info/types';

const make = (
fromGroup: ModuleGroup,
fromVisibility: ModuleVisibility,
toGroup: ModuleGroup,
toVisibility: ModuleVisibility,
imp = 'import'
) => ({
filename: `${fromGroup}.${fromVisibility}.ts`,
code: dedent`
${imp} '${toGroup}.${toVisibility}'
`,
});

jest.mock('../get_import_resolver', () => {
return {
getImportResolver() {
return {
resolve(req: string) {
return {
type: 'file',
absolute: req.split('.'),
};
},
};
},
};
});

jest.mock('../helpers/repo_source_classifier', () => {
return {
getRepoSourceClassifier() {
return {
classify(r: string | [string, string]) {
const [group, visibility] =
typeof r === 'string' ? (r.endsWith('.ts') ? r.slice(0, -3) : r).split('.') : r;
return {
pkgInfo: {
pkgId: 'aPackage',
},
group,
visibility,
};
},
};
},
};
});

const tsTester = [
'@typescript-eslint/parser',
new RuleTester({
parser: require.resolve('@typescript-eslint/parser'),
parserOptions: {
sourceType: 'module',
ecmaVersion: 2018,
ecmaFeatures: {
jsx: true,
},
},
}),
] as const;

const babelTester = [
'@babel/eslint-parser',
new RuleTester({
parser: require.resolve('@babel/eslint-parser'),
parserOptions: {
sourceType: 'module',
ecmaVersion: 2018,
requireConfigFile: false,
babelOptions: {
presets: ['@kbn/babel-preset/node_preset'],
},
},
}),
] as const;

for (const [name, tester] of [tsTester, babelTester]) {
describe(name, () => {
tester.run('@kbn/imports/no_group_crossing_imports', NoGroupCrossingImportsRule, {
valid: [
make('observability', 'private', 'observability', 'private'),
make('security', 'private', 'security', 'private'),
make('search', 'private', 'search', 'private'),
make('observability', 'private', 'platform', 'shared'),
make('security', 'private', 'common', 'shared'),
make('platform', 'shared', 'platform', 'shared'),
make('platform', 'shared', 'platform', 'private'),
make('common', 'shared', 'common', 'shared'),
],

invalid: [
{
...make('observability', 'private', 'security', 'private'),
errors: [
{
line: 1,
messageId: 'ILLEGAL_IMPORT',
data: {
importerPackage: 'aPackage',
importerGroup: 'observability',
importedPackage: 'aPackage',
importedGroup: 'security',
importedVisibility: 'private',
sourcePath: 'observability.private.ts',
suggestion: formatSuggestions([
`Please review the dependencies in your module's manifest (kibana.jsonc).`,
`Relocate this module to a different group, and/or make sure it has the right 'visibility'.`,
`Address the conflicting dependencies by refactoring the code`,
]),
},
},
],
},
{
...make('security', 'private', 'platform', 'private'),
errors: [
{
line: 1,
messageId: 'ILLEGAL_IMPORT',
data: {
importerPackage: 'aPackage',
importerGroup: 'security',
importedPackage: 'aPackage',
importedGroup: 'platform',
importedVisibility: 'private',
sourcePath: 'security.private.ts',
suggestion: formatSuggestions([
`Please review the dependencies in your module's manifest (kibana.jsonc).`,
`Relocate this module to a different group, and/or make sure it has the right 'visibility'.`,
`Address the conflicting dependencies by refactoring the code`,
]),
},
},
],
},
],
});
});
}
Loading

0 comments on commit 7b82013

Please sign in to comment.