Skip to content

Commit

Permalink
Merge branch 'main' of github.com:elastic/kibana into chore/use-obser…
Browse files Browse the repository at this point in the history
…vability-page-template-from-observability-shared
  • Loading branch information
CoenWarmer committed Apr 18, 2023
2 parents 74d2dd6 + ad2d5ad commit f4156ac
Show file tree
Hide file tree
Showing 276 changed files with 8,088 additions and 2,221 deletions.
117 changes: 110 additions & 7 deletions dev_docs/tutorials/versioning_http_apis.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ router.get(
);
```

#### Why is this problemetic for versioning?
#### Why is this problematic for versioning?

Whenever we perform a data migration the body of this endpoint will change for all clients. This prevents us from being able to maintain past interfaces and gracefully introduce new ones.

Expand Down Expand Up @@ -119,7 +119,7 @@ router.post(
}
);
```
#### Why is this problemetic for versioning?
#### Why is this problematic for versioning?

This HTTP API currently accepts all numbers and strings as input which allows for unexpected inputs like negative numbers or non-URL friendly characters. This may break future migrations or integrations that assume your data will always be within certain parameters.

Expand All @@ -141,7 +141,7 @@ This HTTP API currently accepts all numbers and strings as input which allows fo

Adding this validation we negate the risk of unexpected values. It is not necessary to use `@kbn/config-schema`, as long as your validation mechanism provides finer grained controls than "number" or "string".

In summary: think about the acceptable paramaters for every input your HTTP API expects.
In summary: think about the acceptable parameters for every input your HTTP API expects.

### 3. Keep interfaces as "narrow" as possible

Expand Down Expand Up @@ -170,7 +170,7 @@ router.get(

The above code follows guidelines from steps 1 and 2, but it allows clients to specify ANY string by which to sort. This is a far "wider" API than we need for this endpoint.

#### Why is this problemetic for versioning?
#### Why is this problematic for versioning?

Without telemetry it is impossible to know what values clients might be passing through — and what type of sort behaviour they are expecting.

Expand Down Expand Up @@ -207,9 +207,112 @@ router.get(

The changes are:

1. New input validation accepts a known set of values. This makes our HTTP API far _narrower_ and specific to our use case. It does not matter that our `sortSchema` has the same values as our persistence schema, what matters is that we created a **translation layer** between our HTTP API and our internal schema. This faclitates easily versioning this endpoint.
1. New input validation accepts a known set of values. This makes our HTTP API far _narrower_ and specific to our use case. It does not matter that our `sortSchema` has the same values as our persistence schema, what matters is that we created a **translation layer** between our HTTP API and our internal schema. This facilitates easily versioning this endpoint.
2. **Bonus point**: we use the `escapeKuery` utility to defend against KQL injection attacks.

### 4. Use the versioned API spec
### 4. Adhere to the HTTP versioning specification

#### Choosing the right version

##### Public endpoints
Public endpoints include any endpoint that is intended for users to directly integrate with via HTTP.

Choose a date string in the format `YYYY-MM-DD`. This date should be the date that a (group) of APIs was made available.

##### Internal endpoints
Internal endpoints are all non-public endpoints (see definition above).

If you need to maintain backwards-compatibility for an internal endpoint use a single, larger-than-zero number. Ex. `1`.


#### Use the versioned router

Core exposes a versioned router that ensures your endpoint's behaviour and formatting all conforms to the versioning specification.

```typescript
router.versioned.
.post({
access: 'public', // This endpoint is intended for a public audience
path: '/api/my-app/foo/{id?}',
options: { timeout: { payload: 60000 } },
})
.addVersion(
{
version: '2023-01-01', // The public version of this API
validate: {
request: {
query: schema.object({
name: schema.maybe(schema.string({ minLength: 2, maxLength: 50 })),
}),
params: schema.object({
id: schema.maybe(schema.string({ minLength: 10, maxLength: 13 })),
}),
body: schema.object({ foo: schema.string() }),
},
response: {
200: { // In development environments, this validation will run against 200 responses
body: schema.object({ foo: schema.string() }),
},
},
},
},
async (ctx, req, res) => {
await ctx.fooService.create(req.body.foo, req.params.id, req.query.name);
return res.ok({ body: { foo: req.body.foo } });
}
)
// BREAKING CHANGE: { foo: string } => { fooString: string } in response body
.addVersion(
{
version: '2023-02-01',
validate: {
request: {
query: schema.object({
name: schema.maybe(schema.string({ minLength: 2, maxLength: 50 })),
}),
params: schema.object({
id: schema.maybe(schema.string({ minLength: 10, maxLength: 13 })),
}),
body: schema.object({ fooString: schema.string() }),
},
response: {
200: {
body: schema.object({ fooName: schema.string() }),
},
},
},
},
async (ctx, req, res) => {
await ctx.fooService.create(req.body.fooString, req.params.id, req.query.name);
return res.ok({ body: { fooName: req.body.fooString } });
}
)
// BREAKING CHANGES: Enforce min/max length on fooString
.addVersion(
{
version: '2023-03-01',
validate: {
request: {
query: schema.object({
name: schema.maybe(schema.string({ minLength: 2, maxLength: 50 })),
}),
params: schema.object({
id: schema.maybe(schema.string({ minLength: 10, maxLength: 13 })),
}),
body: schema.object({ fooString: schema.string({ minLength: 0, maxLength: 1000 }) }),
},
response: {
200: {
body: schema.object({ fooName: schema.string() }),
},
},
},
},
async (ctx, req, res) => {
await ctx.fooService.create(req.body.fooString, req.params.id, req.query.name);
return res.ok({ body: { fooName: req.body.fooString } });
}
```
_Under construction, check back here soon!_
#### Additional reading
For a more details on the versioning specification see [this document](https://docs.google.com/document/d/1YpF6hXIHZaHvwNaQAxWFzexUF1nbqACTtH2IfDu0ldA/edit?usp=sharing).
6 changes: 2 additions & 4 deletions docs/management/connectors/action-types/email.asciidoc
Original file line number Diff line number Diff line change
Expand Up @@ -234,14 +234,12 @@ is considered `false`. Typically, `port: 465` uses `secure: true`, and
[float]
[[elasticcloud]]
==== Sending email from Elastic Cloud

IMPORTANT: To receive notifications, the email addresses must be added to an
link:{cloud}/ec-watcher.html#ec-watcher-allowlist[allowlist] in the
Elasticsearch Service Console.

Use the preconfigured email connector (`Elastic-Cloud-SMTP`) to send emails from
Elastic Cloud.

NOTE: For more information on the preconfigured email connector, see link:{cloud}/ec-watcher.html#ec-cloud-email-service-limits[Elastic Cloud email service limits].

[float]
[[gmail]]
==== Sending email from Gmail
Expand Down
2 changes: 1 addition & 1 deletion packages/kbn-optimizer/limits.yml
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ pageLoadAssetSize:
banners: 17946
bfetch: 22837
canvas: 1066647
cases: 144442
cases: 170000
charts: 55000
cloud: 21076
cloudChat: 19894
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ import { context } from './context';
/**
* An object representing an uploaded file
*/
interface UploadedFile<Meta = unknown> {
export interface UploadedFile<Meta = unknown> {
/**
* The ID that was generated for the uploaded file
*/
Expand Down
41 changes: 31 additions & 10 deletions src/cli/serve/serve.js
Original file line number Diff line number Diff line change
Expand Up @@ -78,25 +78,43 @@ const configPathCollector = pathCollector();
const pluginPathCollector = pathCollector();

/**
* @param {string} name
* @param {string[]} configs
* @param {'push' | 'unshift'} method
* @param {string} name The config file name
* @returns {boolean} Whether the file exists
*/
function maybeAddConfig(name, configs, method) {
function configFileExists(name) {
const path = resolve(getConfigDirectory(), name);
try {
if (statSync(path).isFile()) {
configs[method](path);
}
return statSync(path).isFile();
} catch (err) {
if (err.code === 'ENOENT') {
return;
return false;
}

throw err;
}
}

/**
* @returns {boolean} Whether the distribution can run in Serverless mode
*/
function isServerlessCapableDistribution() {
// For now, checking if the `serverless.yml` config file exists should be enough
// We could also check the following as well, but I don't think it's necessary:
// VALID_SERVERLESS_PROJECT_MODE.some((projectType) => configFileExists(`serverless.${projectType}.yml`))
return configFileExists('serverless.yml');
}

/**
* @param {string} name
* @param {string[]} configs
* @param {'push' | 'unshift'} method
*/
function maybeAddConfig(name, configs, method) {
if (configFileExists(name)) {
configs[method](resolve(getConfigDirectory(), name));
}
}

/**
* @returns {string[]}
*/
Expand Down Expand Up @@ -233,8 +251,11 @@ export default function (program) {
.option(
'--run-examples',
'Adds plugin paths for all the Kibana example plugins and runs with no base path'
)
.option('--serverless <oblt|security|es>', 'Start Kibana in a serverless project mode');
);
}

if (isServerlessCapableDistribution()) {
command.option('--serverless <oblt|security|es>', 'Start Kibana in a serverless project mode');
}

if (DEV_MODE_SUPPORTED) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -55,9 +55,9 @@ describe('checking migration metadata changes on all registered SO types', () =>

expect(hashMap).toMatchInlineSnapshot(`
Object {
"action": "6cfc277ed3211639e37546ac625f4a68f2494215",
"action_task_params": "5f419caba96dd8c77d0f94013e71d43890e3d5d6",
"alert": "7bec97d7775a025ecf36a33baf17386b9e7b4c3c",
"action": "12c6b25ef1fffb36d8de893318f8a2bc5d6a46a6",
"action_task_params": "c725c37de66135934150465f9b1e76fe349e8a23",
"alert": "0cd1f1921014004a9ff5c0a9333ca9bde14bf748",
"api_key_pending_invalidation": "16e7bcf8e78764102d7f525542d5b616809a21ee",
"apm-indices": "d19dd7fb51f2d2cbc1f8769481721e0953f9a6d2",
"apm-server-schema": "1d42f17eff9ec6c16d3a9324d9539e2d123d0a9a",
Expand All @@ -77,7 +77,7 @@ describe('checking migration metadata changes on all registered SO types', () =>
"cases-user-actions": "8ad74294b71edffa58fad7a40eea2388209477c9",
"config": "97e16b8f5dc10c404fd3b201ef36bc6c3c63dc80",
"config-global": "d9791e8f73edee884630e1cb6e4954ae513ce75e",
"connector_token": "fb05ff5afdcb6e2f20c9c6513ff7a1ab12b66f36",
"connector_token": "aff1aa0ebc0a6b44b570057972af25178b0bca42",
"core-usage-stats": "b3c04da317c957741ebcdedfea4524049fdc79ff",
"csp-rule-template": "099c229bf97578d9ca72b3a672d397559b84ee0b",
"dashboard": "71e3f8dfcffeb5fbd410dec81ce46f5691763c43",
Expand Down Expand Up @@ -124,7 +124,7 @@ describe('checking migration metadata changes on all registered SO types', () =>
"osquery-pack-asset": "18e08979d46ee7e5538f54c080aec4d8c58516ca",
"osquery-saved-query": "f5e4e303f65c7607248ea8b2672f1ee30e4fb15e",
"query": "cfc049e1f0574fb4fdb2d653d7c10bdc970a2610",
"rules-settings": "9854495c3b54b16a6625fb250c35e5504da72266",
"rules-settings": "eb8d40b7d60aeffe66831f7d5d83de28d85776d8",
"sample-data-telemetry": "c38daf1a49ed24f2a4fb091e6e1e833fccf19935",
"search": "ed3a9b1681b57d69560909d51933fdf17576ea68",
"search-session": "fae0dfc63274d6a3b90ca583802c48cab8760637",
Expand All @@ -142,7 +142,7 @@ describe('checking migration metadata changes on all registered SO types', () =>
"synthetics-param": "9776c9b571d35f0d0397e8915e035ea1dc026db7",
"synthetics-privates-locations": "7d032fc788905e32152029ae7ab3d6038c48ae44",
"tag": "87f21f07df9cc37001b15a26e413c18f50d1fbfe",
"task": "ff760534a44c4cfabcf4baf8cfe8283f717cab02",
"task": "533ee80c50c47f0505846bfac73fc10962c5bc45",
"telemetry": "561b329aaed3c15b91aaf2075645be3097247612",
"ui-metric": "410a8ad28e0f44b161c960ff0ce950c712b17c52",
"upgrade-assistant-ml-upgrade-operation": "d8816e5ce32649e7a3a43e2c406c632319ff84bb",
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

import React, { Component } from 'react';
import PropTypes from 'prop-types';

import { css } from '@emotion/react';
import {
EuiButtonIcon,
EuiFlexGroup,
Expand Down Expand Up @@ -192,12 +192,25 @@ export class DataTableFormat extends Component<DataTableFormatProps, DataTableFo

return (
<EuiInMemoryTable
className="insDataTableFormat__table"
tableLayout="auto"
className="insDataTableFormat__table eui-xScroll"
data-test-subj="inspectorTable"
columns={columns}
items={rows}
sorting={true}
pagination={pagination}
css={css`
// Set a min width on each column - you can use [data-test-subj] to target specific columns
.euiTableHeaderCell {
min-width: 100px;
}
// Make sure the pagination follows the scroll
> div:last-child {
position: sticky;
left: 0;
}
`}
/>
);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,9 @@ const currentDocumentSelector = (state: PreviewState) => state.documents[state.c
const currentDocumentIsLoadingSelector = (state: PreviewState) => state.isLoadingDocuments;

const ScriptFieldComponent = ({ existingConcreteFields, links, placeholder }: Props) => {
const {
validation: { setScriptEditorValidation },
} = useFieldPreviewContext();
const monacoEditor = useRef<monaco.editor.IStandaloneCodeEditor | null>(null);
const editorValidationSubscription = useRef<Subscription>();
const fieldCurrentValue = useRef<string>('');
Expand Down Expand Up @@ -143,15 +146,15 @@ const ScriptFieldComponent = ({ existingConcreteFields, links, placeholder }: Pr

editorValidationSubscription.current = PainlessLang.validation$().subscribe(
({ isValid, isValidating, errors }) => {
controller.setScriptEditorValidation({
setScriptEditorValidation({
isValid,
isValidating,
message: errors[0]?.message ?? null,
});
}
);
},
[controller]
[setScriptEditorValidation]
);

const updateMonacoMarkers = useCallback((markers: monaco.editor.IMarkerData[]) => {
Expand Down
Loading

0 comments on commit f4156ac

Please sign in to comment.