From 34b1d0a10d529da48f35106d6b82cf96498e6677 Mon Sep 17 00:00:00 2001
From: MadameSheema
Date: Fri, 10 Apr 2020 11:28:59 +0200
Subject: [PATCH 01/24] [SIEM] Updates cypress readme with documentation about
the test data. (#62747)
* updates test data section
* Update x-pack/legacy/plugins/siem/cypress/README.md
Co-Authored-By: Ryland Herrick
Co-authored-by: Ryland Herrick
---
x-pack/legacy/plugins/siem/cypress/README.md | 132 +++++++++++++++----
1 file changed, 107 insertions(+), 25 deletions(-)
diff --git a/x-pack/legacy/plugins/siem/cypress/README.md b/x-pack/legacy/plugins/siem/cypress/README.md
index 41137ce6d8a9d..a031fea172be5 100644
--- a/x-pack/legacy/plugins/siem/cypress/README.md
+++ b/x-pack/legacy/plugins/siem/cypress/README.md
@@ -111,6 +111,112 @@ elasticsearch:
hosts: ['https://:9200']
```
+## Running (Headless) Tests on the Command Line as a Jenkins execution (The preferred way)
+
+To run (headless) tests as a Jenkins execution.
+
+1. First bootstrap kibana changes from the Kibana root directory:
+
+```sh
+yarn kbn bootstrap
+```
+
+2. Launch Cypress command line test runner:
+
+```sh
+cd x-pack/legacy/plugins/siem
+yarn cypress:run-as-ci
+```
+
+Note that with this type of execution you don't need to have running a kibana and elasticsearch instance. This is because
+ the command, as it would happen in the CI, will launch the instances. The elasticsearch instance will be fed with the data
+ placed in: `x-pack/test/siem_cypress/es_archives`
+
+As in this case we want to mimic a CI execution we want to execute the tests with the same set of data, this is why
+in this case does not make sense to override Cypress environment variables.
+
+### Test data
+
+As said before when running the tests as Jenkins the tests are fed with the data placed in: `x-pack/test/siem_cypress/es_archives`.
+
+Currently there are two different ways of feeding data:
+1. By default
+2. Specifying a specific set of data for a specific test
+
+#### By default
+
+When a execution of the test is going to be done an empty kibana and a set of audibteat data are loaded (empty_kibana and auditbeat). With this data usually is enough to cover most of the scenarios that we are testing.
+
+#### Running tests with custom data
+
+Sometimes the default data is not enough and we need a specific set of data in order to being able to test the desired behaviour.
+
+In that case in the hooks of the test use the function `esArchiverLoad` to load the set of data neeed and `esArchiverUnload` to remove the changes done in the data.
+
+Example:
+
+```typescript
+import { esArchiverLoad, esArchiverUnload } from '../tasks/es_archiver';
+
+describe('This are going to be a set of tests', () => {
+ before(() => {
+ esArchiverLoad('name_of_the_data_set_you_want_to_load');
+ });
+
+ after(() => {
+ esArchiverUnload('name_of_the_data_set_you_want_to_unload');
+ });
+
+ it('Going to test something awesome', () => {
+ hereGoesYourAwesomeTestcode
+ });
+});
+
+```
+
+Note that loading and unloading data takes a signifcant amount of time so try to minimize the use of it when possible.
+
+### Current sets of data
+
+The current sets of data can be found in: `x-pack/test/siem_cypress/es_archives` folder.
+
+- auditbeat
+ - Auditbeat data generated in Sep, 2019 with the following hosts present:
+ - suricata-iowa
+ - siem-kibana
+ - siem-es
+ - jessie
+- closed_signals
+ - Set of data with 108 closed signals linked to "Signals test" custom rule.
+- custome_rules
+ - Set if data with just 4 custom activated rules.
+- empty_kibana
+ - Empty kibana board.
+- prebuilt_rules_loaded
+ - Elastic prebuilt loaded rules and deactivated.
+- signals
+ - Set of data with 108 opened signals linked to "Signals test" custom rule.
+
+### How to generate new test data
+
+We are using es_archiver in order to generate the data that our Cypress tests needs.
+
+1. Setup if possible a clean instance of kibana and elasticsearch (if not, possible please try to clean the data that you are going to generate).
+2. With the kibana and elasticsearch instance up and running, create the data that you need for your test.
+3. When you are sure that you have all the data you need run the following command from: `x-pack/legacy/plugins/siem`
+
+```sh
+node ../../../../scripts/es_archiver save --dir ../../../test/siem_cypress/es_archives --config ../../../../test/functional/config.js --es-url http://:@:
+```
+
+Example:
+```sh
+node ../../../../scripts/es_archiver save custom_rules ".kibana",".siem-signal*" --dir ../../../test/siem_cypress/es_archives --config ../../../../test/functional/config.js --es-url http://elastic:changeme@localhost:9220
+```
+
+Note that the command is going to create the folder if does not exist in the directory with the imported data.
+
+
## Running Tests Interactively
Use the Cypress interactive test runner to develop and debug specific tests
@@ -210,30 +316,6 @@ cd x-pack/legacy/plugins/siem
CYPRESS_baseUrl=http://localhost:5601 CYPRESS_ELASTICSEARCH_USERNAME=elastic CYPRESS_ELASTICSEARCH_PASSWORD= yarn cypress:run
```
-## Running (Headless) Tests on the Command Line as a Jenkins execution
-
-To run (headless) tests as a Jenkins execution.
-
-1. First bootstrap kibana changes from the Kibana root directory:
-
-```sh
-yarn kbn bootstrap
-```
-
-2. Launch Cypress command line test runner:
-
-```sh
-cd x-pack/legacy/plugins/siem
-yarn cypress:run-as-ci
-```
-
-Note that with this type of execution you don't need to have running a kibana and elasticsearch instance. This is because
- the command, as it would happen in the CI, will launch the instances. The elasticsearch instance will be fed with the data
- placed in: `x-pack/test/siem_cypress/es_archives`.
-
-As in this case we want to mimic a CI execution we want to execute the tests with the same set of data, this is why
-in this case does not make sense to override Cypress environment variables.
-
## Reporting
When Cypress tests are run on the command line via `yarn cypress:run`,
@@ -280,4 +362,4 @@ target/kibana-siem/cypress/videos
## Linting
-Optional linting rules for Cypress and linting setup can be found [here](https://github.com/cypress-io/eslint-plugin-cypress#usage)
\ No newline at end of file
+Optional linting rules for Cypress and linting setup can be found [here](https://github.com/cypress-io/eslint-plugin-cypress#usage)
From 93971dbfab74b39335115e8236741329486c4b79 Mon Sep 17 00:00:00 2001
From: Jean-Louis Leysens
Date: Fri, 10 Apr 2020 12:28:55 +0200
Subject: [PATCH 02/24] [Watcher] Preserve the watch active status after
updates (#61999)
* Preserve the watch active status after updates
* Use route validation params to set isActive
* Fix Jest test
* Implement PR feedback
- Make the isActive flag required on the save watch endpoint
- Move the isActive state to base_watch and serialize from
there.
* Fix Jest tests
Co-authored-by: Elastic Machine
---
.../watch_create_json.test.ts | 3 ++
.../watch_create_threshold.test.tsx | 8 +++++
.../client_integration/watch_edit.test.ts | 2 ++
.../application/models/watch/base_watch.js | 2 ++
.../server/lib/elasticsearch_js_plugin.ts | 4 +++
.../routes/api/watch/register_save_route.ts | 29 ++++++++-----------
6 files changed, 31 insertions(+), 17 deletions(-)
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts
index 2096c0dd61bd8..47de4548898cf 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts
@@ -108,6 +108,7 @@ describe(' create route', () => {
name: watch.name,
type: watch.type,
isNew: true,
+ isActive: true,
actions: [
{
id: DEFAULT_LOGGING_ACTION_ID,
@@ -185,6 +186,7 @@ describe(' create route', () => {
id,
type,
isNew: true,
+ isActive: true,
actions: [],
watch: defaultWatchJson,
};
@@ -246,6 +248,7 @@ describe(' create route', () => {
id,
type,
isNew: true,
+ isActive: true,
actions: [],
watch: defaultWatchJson,
};
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx
index 943233d3c14ed..e5bdcbbfb82cf 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx
@@ -244,6 +244,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'logging_1',
@@ -314,6 +315,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'index_1',
@@ -376,6 +378,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'slack_1',
@@ -448,6 +451,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'email_1',
@@ -540,6 +544,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'webhook_1',
@@ -629,6 +634,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'jira_1',
@@ -709,6 +715,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [
{
id: 'pagerduty_1',
@@ -772,6 +779,7 @@ describe(' create route', () => {
name: WATCH_NAME,
type: WATCH_TYPES.THRESHOLD,
isNew: true,
+ isActive: true,
actions: [],
index: MATCH_INDICES,
timeField: WATCH_TIME_FIELD,
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts b/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts
index 51285a5786b00..ced06562b1d20 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts
@@ -98,6 +98,7 @@ describe('', () => {
name: EDITED_WATCH_NAME,
type: watch.type,
isNew: false,
+ isActive: true,
actions: [
{
id: DEFAULT_LOGGING_ACTION_ID,
@@ -191,6 +192,7 @@ describe('', () => {
name: EDITED_WATCH_NAME,
type,
isNew: false,
+ isActive: true,
actions: [],
timeField,
triggerIntervalSize,
diff --git a/x-pack/plugins/watcher/public/application/models/watch/base_watch.js b/x-pack/plugins/watcher/public/application/models/watch/base_watch.js
index 3fe4fb006d241..af2e45b35501e 100644
--- a/x-pack/plugins/watcher/public/application/models/watch/base_watch.js
+++ b/x-pack/plugins/watcher/public/application/models/watch/base_watch.js
@@ -32,6 +32,7 @@ export class BaseWatch {
this.isSystemWatch = Boolean(get(props, 'isSystemWatch'));
this.watchStatus = WatchStatus.fromUpstreamJson(get(props, 'watchStatus'));
this.watchErrors = WatchErrors.fromUpstreamJson(get(props, 'watchErrors'));
+ this.isActive = this.watchStatus.isActive ?? true;
const actions = get(props, 'actions', []);
this.actions = actions.map(Action.fromUpstreamJson);
@@ -115,6 +116,7 @@ export class BaseWatch {
name: this.name,
type: this.type,
isNew: this.isNew,
+ isActive: this.isActive,
actions: map(this.actions, action => action.upstreamJson),
};
}
diff --git a/x-pack/plugins/watcher/server/lib/elasticsearch_js_plugin.ts b/x-pack/plugins/watcher/server/lib/elasticsearch_js_plugin.ts
index 240e93e160fe0..3cb8dfb623fac 100644
--- a/x-pack/plugins/watcher/server/lib/elasticsearch_js_plugin.ts
+++ b/x-pack/plugins/watcher/server/lib/elasticsearch_js_plugin.ts
@@ -174,6 +174,10 @@ export const elasticsearchJsPlugin = (Client: any, config: any, components: any)
name: 'master_timeout',
type: 'duration',
},
+ active: {
+ name: 'active',
+ type: 'boolean',
+ },
},
url: {
fmt: '/_watcher/watch/<%=id%>',
diff --git a/x-pack/plugins/watcher/server/routes/api/watch/register_save_route.ts b/x-pack/plugins/watcher/server/routes/api/watch/register_save_route.ts
index 61d167bb9bbcd..10ee0c4857862 100644
--- a/x-pack/plugins/watcher/server/routes/api/watch/register_save_route.ts
+++ b/x-pack/plugins/watcher/server/routes/api/watch/register_save_route.ts
@@ -5,7 +5,6 @@
*/
import { schema } from '@kbn/config-schema';
-import { IScopedClusterClient } from 'kibana/server';
import { i18n } from '@kbn/i18n';
import { WATCH_TYPES } from '../../../../common/constants';
import { serializeJsonWatch, serializeThresholdWatch } from '../../../../common/lib/serialization';
@@ -21,23 +20,11 @@ const bodySchema = schema.object(
{
type: schema.string(),
isNew: schema.boolean(),
+ isActive: schema.boolean(),
},
{ unknowns: 'allow' }
);
-function fetchWatch(dataClient: IScopedClusterClient, watchId: string) {
- return dataClient.callAsCurrentUser('watcher.getWatch', {
- id: watchId,
- });
-}
-
-function saveWatch(dataClient: IScopedClusterClient, id: string, body: any) {
- return dataClient.callAsCurrentUser('watcher.putWatch', {
- id,
- body,
- });
-}
-
export function registerSaveRoute(deps: RouteDependencies) {
deps.router.put(
{
@@ -49,12 +36,16 @@ export function registerSaveRoute(deps: RouteDependencies) {
},
licensePreRoutingFactory(deps, async (ctx, request, response) => {
const { id } = request.params;
- const { type, isNew, ...watchConfig } = request.body;
+ const { type, isNew, isActive, ...watchConfig } = request.body;
+
+ const dataClient = ctx.watcher!.client;
// For new watches, verify watch with the same ID doesn't already exist
if (isNew) {
try {
- const existingWatch = await fetchWatch(ctx.watcher!.client, id);
+ const existingWatch = await dataClient.callAsCurrentUser('watcher.getWatch', {
+ id,
+ });
if (existingWatch.found) {
return response.conflict({
body: {
@@ -92,7 +83,11 @@ export function registerSaveRoute(deps: RouteDependencies) {
try {
// Create new watch
return response.ok({
- body: await saveWatch(ctx.watcher!.client, id, serializedWatch),
+ body: await dataClient.callAsCurrentUser('watcher.putWatch', {
+ id,
+ active: isActive,
+ body: serializedWatch,
+ }),
});
} catch (e) {
// Case: Error from Elasticsearch JS client
From d662cb5f4525e3a808900403f54f3c4f2c2c7097 Mon Sep 17 00:00:00 2001
From: Jean-Louis Leysens
Date: Fri, 10 Apr 2020 12:58:18 +0200
Subject: [PATCH 03/24] [ESUI] Move XJson Mode to src/plugins/es_ui_shared
(#62758)
* Move XJson to x-pack/es_ui_shared
Also convert files to TS and clean up use of @ts-ignore
* Move console_lang to public
* Proper use of mocks
* Globally mock out .ace.worker.js modules
Having worker code exported in `public` of es_ui_shared poisoned
a lot of Jest tests with the need to mock out the ace worker.
Instead of adding mocks everywhere we handle the importing in
Jest by adding an entry to module mapper.
* Remove manual imports of mocks
* Delete es_ui_shared/public/mocks for now
* Put useXJson code in single place
* Import and instantiate xJsonMode
* Remove language mocks imports
Besides the fact that these paths are wrong these are no longer
needed because we mock at use Jest module mapper
Co-authored-by: Elastic Machine
---
src/dev/jest/config.js | 1 +
.../jest/mocks/ace_worker_module_mock.js} | 2 +-
.../legacy/console_editor/editor_output.tsx | 2 +-
.../send_request_to_es.ts | 2 +-
.../__tests__/output_tokenization.test.js | 1 +
.../mode/input_highlight_rules.js | 2 +-
.../mode/output_highlight_rules.js | 2 +-
.../models/legacy_core_editor/mode/script.js | 2 +-
.../__tests__/sense_editor.test.js | 2 +-
.../models/sense_editor/sense_editor.ts | 2 +-
src/plugins/console/public/lib/utils/index.ts | 5 +--
.../console_lang/ace/modes/index.ts} | 2 +-
.../elasticsearch_sql_highlight_rules.ts | 0
.../ace/modes/lexer_rules/index.ts} | 0
.../lexer_rules/script_highlight_rules.ts} | 4 +--
.../lexer_rules/x_json_highlight_rules.ts} | 12 +++----
.../console_lang/ace/modes/x_json/index.ts | 19 ++++++++++
.../ace/modes/x_json/worker/index.ts | 26 ++++++++++++++
.../ace/modes/x_json/worker/worker.d.ts | 25 +++++++++++++
.../modes/x_json/worker/x_json.ace.worker.js | 0
.../console_lang/ace/modes/x_json/x_json.ts} | 35 ++++++++++++++++---
.../{ => public}/console_lang/index.ts | 3 ++
.../{ => public}/console_lang/lib/index.ts | 0
.../json_xjson_translation_tools.test.ts | 0
.../__tests__/utils_string_collapsing.txt | 0
.../__tests__/utils_string_expanding.txt | 0
.../lib/json_xjson_translation_tools/index.ts | 0
src/plugins/es_ui_shared/public/index.ts | 11 ++++++
.../static/ace_x_json/hooks/index.ts | 20 +++++++++++
.../ace_x_json/hooks/use_x_json.ts} | 26 ++++++++------
x-pack/dev-tools/jest/create_jest_config.js | 1 +
.../console_lang/ace/modes/index.ts | 7 ----
.../console_lang/ace/modes/x_json/index.ts | 7 ----
.../ace/modes/x_json/worker/index.ts | 13 -------
.../ace/modes/x_json/worker/worker.d.ts | 12 -------
.../console_lang/ace/modes/x_json/x_json.ts | 34 ------------------
.../es_ui_shared/console_lang/index.ts | 7 ----
.../es_ui_shared/console_lang/mocks.ts | 9 -----
.../components/custom_hooks/index.ts | 1 -
.../custom_hooks/use_x_json_mode.ts | 26 --------------
.../create_analytics_advanced_editor.tsx | 5 ++-
.../use_create_analytics_form/reducer.ts | 2 +-
.../ml_job_editor/ml_job_editor.tsx | 5 ++-
x-pack/plugins/ml/shared_imports.ts | 5 ++-
.../public/application/editor/editor.test.tsx | 2 --
.../public/application/editor/init_editor.ts | 2 +-
.../utils/check_for_json_errors.ts | 2 +-
.../public/app/hooks/use_x_json_mode.ts | 20 -----------
.../step_define/step_define_form.tsx | 4 ++-
.../transform/public/shared_imports.ts | 4 +--
.../builtin_action_types/es_index.tsx | 2 +-
.../public/application/lib/use_x_json_mode.ts | 25 -------------
.../watch_create_json.test.ts | 2 --
.../watch_create_threshold.test.tsx | 2 --
.../client_integration/watch_edit.test.ts | 1 -
.../client_integration/watch_list.test.ts | 1 -
.../client_integration/watch_status.test.ts | 1 -
.../json_watch_edit/json_watch_edit_form.tsx | 3 +-
.../json_watch_edit_simulate.tsx | 2 +-
.../json_watch_edit/use_x_json_mode.ts | 24 -------------
.../public/application/shared_imports.ts | 2 ++
61 files changed, 193 insertions(+), 246 deletions(-)
rename src/{plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts => dev/jest/mocks/ace_worker_module_mock.js} (94%)
rename src/plugins/es_ui_shared/{console_lang/ace/modes/index.js => public/console_lang/ace/modes/index.ts} (94%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/ace/modes/lexer_rules/elasticsearch_sql_highlight_rules.ts (100%)
rename src/plugins/es_ui_shared/{console_lang/ace/modes/lexer_rules/index.js => public/console_lang/ace/modes/lexer_rules/index.ts} (100%)
rename src/plugins/es_ui_shared/{console_lang/ace/modes/lexer_rules/script_highlight_rules.js => public/console_lang/ace/modes/lexer_rules/script_highlight_rules.ts} (97%)
rename src/plugins/es_ui_shared/{console_lang/ace/modes/lexer_rules/x_json_highlight_rules.js => public/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.ts} (94%)
create mode 100644 src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/index.ts
create mode 100644 src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/index.ts
create mode 100644 src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/worker.d.ts
rename x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.js => src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/x_json.ace.worker.js (100%)
rename src/plugins/es_ui_shared/{console_lang/ace/modes/x_json/x_json_mode.ts => public/console_lang/ace/modes/x_json/x_json.ts} (61%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/index.ts (92%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/lib/index.ts (100%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/lib/json_xjson_translation_tools/__tests__/json_xjson_translation_tools.test.ts (100%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_collapsing.txt (100%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_expanding.txt (100%)
rename src/plugins/es_ui_shared/{ => public}/console_lang/lib/json_xjson_translation_tools/index.ts (100%)
create mode 100644 src/plugins/es_ui_shared/static/ace_x_json/hooks/index.ts
rename src/plugins/es_ui_shared/{console_lang/ace/modes/index.d.ts => static/ace_x_json/hooks/use_x_json.ts} (60%)
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/ace/modes/index.ts
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/index.ts
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.d.ts
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json.ts
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/index.ts
delete mode 100644 x-pack/plugins/es_ui_shared/console_lang/mocks.ts
delete mode 100644 x-pack/plugins/ml/public/application/components/custom_hooks/use_x_json_mode.ts
delete mode 100644 x-pack/plugins/transform/public/app/hooks/use_x_json_mode.ts
delete mode 100644 x-pack/plugins/triggers_actions_ui/public/application/lib/use_x_json_mode.ts
delete mode 100644 x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/use_x_json_mode.ts
diff --git a/src/dev/jest/config.js b/src/dev/jest/config.js
index a941735c7840e..7da14e0dfe51b 100644
--- a/src/dev/jest/config.js
+++ b/src/dev/jest/config.js
@@ -62,6 +62,7 @@ export default {
'\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$':
'/src/dev/jest/mocks/file_mock.js',
'\\.(css|less|scss)$': '/src/dev/jest/mocks/style_mock.js',
+ '\\.ace\\.worker.js$': '/src/dev/jest/mocks/ace_worker_module_mock.js',
},
setupFiles: [
'/src/dev/jest/setup/babel_polyfill.js',
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts b/src/dev/jest/mocks/ace_worker_module_mock.js
similarity index 94%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts
rename to src/dev/jest/mocks/ace_worker_module_mock.js
index caa7b518b8b66..9d267f494f8bf 100644
--- a/src/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts
+++ b/src/dev/jest/mocks/ace_worker_module_mock.js
@@ -17,4 +17,4 @@
* under the License.
*/
-export { XJsonMode } from './x_json_mode';
+module.exports = '';
diff --git a/src/plugins/console/public/application/containers/editor/legacy/console_editor/editor_output.tsx b/src/plugins/console/public/application/containers/editor/legacy/console_editor/editor_output.tsx
index 36d90bb6bff1a..8510aaebfaa1e 100644
--- a/src/plugins/console/public/application/containers/editor/legacy/console_editor/editor_output.tsx
+++ b/src/plugins/console/public/application/containers/editor/legacy/console_editor/editor_output.tsx
@@ -20,7 +20,7 @@
import { EuiScreenReaderOnly } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import React, { useEffect, useRef } from 'react';
-import { expandLiteralStrings } from '../../../../../../../es_ui_shared/console_lang/lib';
+import { expandLiteralStrings } from '../../../../../../../es_ui_shared/public';
import {
useEditorReadContext,
useRequestReadContext,
diff --git a/src/plugins/console/public/application/hooks/use_send_current_request_to_es/send_request_to_es.ts b/src/plugins/console/public/application/hooks/use_send_current_request_to_es/send_request_to_es.ts
index 102f90a9feb6f..cfbd5691bc22b 100644
--- a/src/plugins/console/public/application/hooks/use_send_current_request_to_es/send_request_to_es.ts
+++ b/src/plugins/console/public/application/hooks/use_send_current_request_to_es/send_request_to_es.ts
@@ -18,7 +18,7 @@
*/
import { extractDeprecationMessages } from '../../../lib/utils';
-import { collapseLiteralStrings } from '../../../../../es_ui_shared/console_lang/lib';
+import { collapseLiteralStrings } from '../../../../../es_ui_shared/public';
// @ts-ignore
import * as es from '../../../lib/es/es';
import { BaseResponseType } from '../../../types';
diff --git a/src/plugins/console/public/application/models/legacy_core_editor/__tests__/output_tokenization.test.js b/src/plugins/console/public/application/models/legacy_core_editor/__tests__/output_tokenization.test.js
index 6a93d007a6cdc..5c86b0a1d2092 100644
--- a/src/plugins/console/public/application/models/legacy_core_editor/__tests__/output_tokenization.test.js
+++ b/src/plugins/console/public/application/models/legacy_core_editor/__tests__/output_tokenization.test.js
@@ -16,6 +16,7 @@
* specific language governing permissions and limitations
* under the License.
*/
+import '../legacy_core_editor.test.mocks';
const $ = require('jquery');
import RowParser from '../../../../lib/row_parser';
import ace from 'brace';
diff --git a/src/plugins/console/public/application/models/legacy_core_editor/mode/input_highlight_rules.js b/src/plugins/console/public/application/models/legacy_core_editor/mode/input_highlight_rules.js
index 2c1b30f806f95..29f192f4ea858 100644
--- a/src/plugins/console/public/application/models/legacy_core_editor/mode/input_highlight_rules.js
+++ b/src/plugins/console/public/application/models/legacy_core_editor/mode/input_highlight_rules.js
@@ -18,7 +18,7 @@
*/
const ace = require('brace');
-import { addXJsonToRules } from '../../../../../../es_ui_shared/console_lang';
+import { addXJsonToRules } from '../../../../../../es_ui_shared/public';
export function addEOL(tokens, reg, nextIfEOL, normalNext) {
if (typeof reg === 'object') {
diff --git a/src/plugins/console/public/application/models/legacy_core_editor/mode/output_highlight_rules.js b/src/plugins/console/public/application/models/legacy_core_editor/mode/output_highlight_rules.js
index e27222ebd65e9..c9d538ab6ceb1 100644
--- a/src/plugins/console/public/application/models/legacy_core_editor/mode/output_highlight_rules.js
+++ b/src/plugins/console/public/application/models/legacy_core_editor/mode/output_highlight_rules.js
@@ -19,7 +19,7 @@
const ace = require('brace');
import 'brace/mode/json';
-import { addXJsonToRules } from '../../../../../../es_ui_shared/console_lang';
+import { addXJsonToRules } from '../../../../../../es_ui_shared/public';
const oop = ace.acequire('ace/lib/oop');
const JsonHighlightRules = ace.acequire('ace/mode/json_highlight_rules').JsonHighlightRules;
diff --git a/src/plugins/console/public/application/models/legacy_core_editor/mode/script.js b/src/plugins/console/public/application/models/legacy_core_editor/mode/script.js
index 13ae329380221..89adea8921c07 100644
--- a/src/plugins/console/public/application/models/legacy_core_editor/mode/script.js
+++ b/src/plugins/console/public/application/models/legacy_core_editor/mode/script.js
@@ -18,7 +18,7 @@
*/
import ace from 'brace';
-import { ScriptHighlightRules } from '../../../../../../es_ui_shared/console_lang';
+import { ScriptHighlightRules } from '../../../../../../es_ui_shared/public';
const oop = ace.acequire('ace/lib/oop');
const TextMode = ace.acequire('ace/mode/text').Mode;
diff --git a/src/plugins/console/public/application/models/sense_editor/__tests__/sense_editor.test.js b/src/plugins/console/public/application/models/sense_editor/__tests__/sense_editor.test.js
index 63f97345bc9ff..6afc03df13b4c 100644
--- a/src/plugins/console/public/application/models/sense_editor/__tests__/sense_editor.test.js
+++ b/src/plugins/console/public/application/models/sense_editor/__tests__/sense_editor.test.js
@@ -22,7 +22,7 @@ import $ from 'jquery';
import _ from 'lodash';
import { create } from '../create';
-import { collapseLiteralStrings } from '../../../../../../es_ui_shared/console_lang/lib';
+import { collapseLiteralStrings } from '../../../../../../es_ui_shared/public';
const editorInput1 = require('./editor_input1.txt');
describe('Editor', () => {
diff --git a/src/plugins/console/public/application/models/sense_editor/sense_editor.ts b/src/plugins/console/public/application/models/sense_editor/sense_editor.ts
index b1444bdf2bbab..9bcd3a6872196 100644
--- a/src/plugins/console/public/application/models/sense_editor/sense_editor.ts
+++ b/src/plugins/console/public/application/models/sense_editor/sense_editor.ts
@@ -19,7 +19,7 @@
import _ from 'lodash';
import RowParser from '../../../lib/row_parser';
-import { collapseLiteralStrings } from '../../../../../es_ui_shared/console_lang/lib';
+import { collapseLiteralStrings } from '../../../../../es_ui_shared/public';
import * as utils from '../../../lib/utils';
// @ts-ignore
diff --git a/src/plugins/console/public/lib/utils/index.ts b/src/plugins/console/public/lib/utils/index.ts
index f66c952dd3af7..0ebea0f9d4055 100644
--- a/src/plugins/console/public/lib/utils/index.ts
+++ b/src/plugins/console/public/lib/utils/index.ts
@@ -18,10 +18,7 @@
*/
import _ from 'lodash';
-import {
- expandLiteralStrings,
- collapseLiteralStrings,
-} from '../../../../es_ui_shared/console_lang/lib';
+import { expandLiteralStrings, collapseLiteralStrings } from '../../../../es_ui_shared/public';
export function textFromRequest(request: any) {
let data = request.data;
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/index.js b/src/plugins/es_ui_shared/public/console_lang/ace/modes/index.ts
similarity index 94%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/index.js
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/index.ts
index 955f23116f659..f9e016c5078af 100644
--- a/src/plugins/es_ui_shared/console_lang/ace/modes/index.js
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/index.ts
@@ -24,4 +24,4 @@ export {
addXJsonToRules,
} from './lexer_rules';
-export { XJsonMode, installXJsonMode } from './x_json';
+export { installXJsonMode, XJsonMode } from './x_json';
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/elasticsearch_sql_highlight_rules.ts b/src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/elasticsearch_sql_highlight_rules.ts
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/elasticsearch_sql_highlight_rules.ts
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/elasticsearch_sql_highlight_rules.ts
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/index.js b/src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/index.ts
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/index.js
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/index.ts
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/script_highlight_rules.js b/src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/script_highlight_rules.ts
similarity index 97%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/script_highlight_rules.js
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/script_highlight_rules.ts
index b3999c76493c0..4b2e965276a3c 100644
--- a/src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/script_highlight_rules.js
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/script_highlight_rules.ts
@@ -17,13 +17,13 @@
* under the License.
*/
-const ace = require('brace');
+import ace from 'brace';
const oop = ace.acequire('ace/lib/oop');
const { TextHighlightRules } = ace.acequire('ace/mode/text_highlight_rules');
const painlessKeywords =
'def|int|long|byte|String|float|double|char|null|if|else|while|do|for|continue|break|new|try|catch|throw|this|instanceof|return|ctx';
-export function ScriptHighlightRules() {
+export function ScriptHighlightRules(this: any) {
this.name = 'ScriptHighlightRules';
this.$rules = {
start: [
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.js b/src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.ts
similarity index 94%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.js
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.ts
index 14323b9320330..3663b92d92a6f 100644
--- a/src/plugins/es_ui_shared/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.js
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/lexer_rules/x_json_highlight_rules.ts
@@ -22,14 +22,14 @@ import ace from 'brace';
import 'brace/mode/json';
import { ElasticsearchSqlHighlightRules } from './elasticsearch_sql_highlight_rules';
-const { ScriptHighlightRules } = require('./script_highlight_rules');
+import { ScriptHighlightRules } from './script_highlight_rules';
const { JsonHighlightRules } = ace.acequire('ace/mode/json_highlight_rules');
const oop = ace.acequire('ace/lib/oop');
-const jsonRules = function(root) {
+const jsonRules = function(root: any) {
root = root ? root : 'json';
- const rules = {};
+ const rules: any = {};
const xJsonRules = [
{
token: [
@@ -135,7 +135,7 @@ const jsonRules = function(root) {
merge: false,
push: true,
},
- ].concat(xJsonRules);
+ ].concat(xJsonRules as any);
rules.string_literal = [
{
@@ -151,7 +151,7 @@ const jsonRules = function(root) {
return rules;
};
-export function XJsonHighlightRules() {
+export function XJsonHighlightRules(this: any) {
this.$rules = {
...jsonRules('start'),
};
@@ -175,7 +175,7 @@ export function XJsonHighlightRules() {
oop.inherits(XJsonHighlightRules, JsonHighlightRules);
-export function addToRules(otherRules, embedUnder) {
+export function addToRules(otherRules: any, embedUnder: any) {
otherRules.$rules = _.defaultsDeep(otherRules.$rules, jsonRules(embedUnder));
otherRules.embedRules(ScriptHighlightRules, 'script-', [
{
diff --git a/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/index.ts b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/index.ts
new file mode 100644
index 0000000000000..6fd819bb4e9eb
--- /dev/null
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/index.ts
@@ -0,0 +1,19 @@
+/*
+ * Licensed to Elasticsearch B.V. under one or more contributor
+ * license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright
+ * ownership. Elasticsearch B.V. licenses this file to you under
+ * the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+export { installXJsonMode, XJsonMode } from './x_json';
diff --git a/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/index.ts b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/index.ts
new file mode 100644
index 0000000000000..0f884765be05e
--- /dev/null
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/index.ts
@@ -0,0 +1,26 @@
+/*
+ * Licensed to Elasticsearch B.V. under one or more contributor
+ * license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright
+ * ownership. Elasticsearch B.V. licenses this file to you under
+ * the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+// @ts-ignore
+import src from '!!raw-loader!./x_json.ace.worker.js';
+
+export const workerModule = {
+ id: 'ace/mode/json_worker',
+ src,
+};
diff --git a/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/worker.d.ts b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/worker.d.ts
new file mode 100644
index 0000000000000..d2363a2610689
--- /dev/null
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/worker.d.ts
@@ -0,0 +1,25 @@
+/*
+ * Licensed to Elasticsearch B.V. under one or more contributor
+ * license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright
+ * ownership. Elasticsearch B.V. licenses this file to you under
+ * the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+// Satisfy TS's requirements that the module be declared per './index.ts'.
+declare module '!!raw-loader!./worker.js' {
+ const content: string;
+ // eslint-disable-next-line import/no-default-export
+ export default content;
+}
diff --git a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.js b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/x_json.ace.worker.js
similarity index 100%
rename from x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.js
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/worker/x_json.ace.worker.js
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json_mode.ts b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/x_json.ts
similarity index 61%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json_mode.ts
rename to src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/x_json.ts
index 9f804c29a5d27..9fbfedba1d23f 100644
--- a/src/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json_mode.ts
+++ b/src/plugins/es_ui_shared/public/console_lang/ace/modes/x_json/x_json.ts
@@ -18,23 +18,50 @@
*/
import ace from 'brace';
-
import { XJsonHighlightRules } from '../index';
+import { workerModule } from './worker';
+
+const { WorkerClient } = ace.acequire('ace/worker/worker_client');
const oop = ace.acequire('ace/lib/oop');
+
const { Mode: JSONMode } = ace.acequire('ace/mode/json');
const { Tokenizer: AceTokenizer } = ace.acequire('ace/tokenizer');
const { MatchingBraceOutdent } = ace.acequire('ace/mode/matching_brace_outdent');
const { CstyleBehaviour } = ace.acequire('ace/mode/behaviour/cstyle');
const { FoldMode: CStyleFoldMode } = ace.acequire('ace/mode/folding/cstyle');
-export function XJsonMode(this: any) {
- const ruleset: any = new XJsonHighlightRules();
+const XJsonMode: any = function XJsonMode(this: any) {
+ const ruleset: any = new (XJsonHighlightRules as any)();
ruleset.normalizeRules();
this.$tokenizer = new AceTokenizer(ruleset.getRules());
this.$outdent = new MatchingBraceOutdent();
this.$behaviour = new CstyleBehaviour();
this.foldingRules = new CStyleFoldMode();
-}
+};
oop.inherits(XJsonMode, JSONMode);
+
+// Then clobber `createWorker` method to install our worker source. Per ace's wiki: https://github.com/ajaxorg/ace/wiki/Syntax-validation
+(XJsonMode.prototype as any).createWorker = function(session: ace.IEditSession) {
+ const xJsonWorker = new WorkerClient(['ace'], workerModule, 'JsonWorker');
+
+ xJsonWorker.attachToDocument(session.getDocument());
+
+ xJsonWorker.on('annotate', function(e: { data: any }) {
+ session.setAnnotations(e.data);
+ });
+
+ xJsonWorker.on('terminate', function() {
+ session.clearAnnotations();
+ });
+
+ return xJsonWorker;
+};
+
+export { XJsonMode };
+
+export function installXJsonMode(editor: ace.Editor) {
+ const session = editor.getSession();
+ session.setMode(new XJsonMode());
+}
diff --git a/src/plugins/es_ui_shared/console_lang/index.ts b/src/plugins/es_ui_shared/public/console_lang/index.ts
similarity index 92%
rename from src/plugins/es_ui_shared/console_lang/index.ts
rename to src/plugins/es_ui_shared/public/console_lang/index.ts
index a4958911af412..7d83191569622 100644
--- a/src/plugins/es_ui_shared/console_lang/index.ts
+++ b/src/plugins/es_ui_shared/public/console_lang/index.ts
@@ -26,4 +26,7 @@ export {
XJsonHighlightRules,
addXJsonToRules,
XJsonMode,
+ installXJsonMode,
} from './ace/modes';
+
+export { expandLiteralStrings, collapseLiteralStrings } from './lib';
diff --git a/src/plugins/es_ui_shared/console_lang/lib/index.ts b/src/plugins/es_ui_shared/public/console_lang/lib/index.ts
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/lib/index.ts
rename to src/plugins/es_ui_shared/public/console_lang/lib/index.ts
diff --git a/src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/__tests__/json_xjson_translation_tools.test.ts b/src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/__tests__/json_xjson_translation_tools.test.ts
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/__tests__/json_xjson_translation_tools.test.ts
rename to src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/__tests__/json_xjson_translation_tools.test.ts
diff --git a/src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_collapsing.txt b/src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_collapsing.txt
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_collapsing.txt
rename to src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_collapsing.txt
diff --git a/src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_expanding.txt b/src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_expanding.txt
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_expanding.txt
rename to src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/__tests__/utils_string_expanding.txt
diff --git a/src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/index.ts b/src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/index.ts
similarity index 100%
rename from src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools/index.ts
rename to src/plugins/es_ui_shared/public/console_lang/lib/json_xjson_translation_tools/index.ts
diff --git a/src/plugins/es_ui_shared/public/index.ts b/src/plugins/es_ui_shared/public/index.ts
index 6db6248f4c68f..a0371bf351193 100644
--- a/src/plugins/es_ui_shared/public/index.ts
+++ b/src/plugins/es_ui_shared/public/index.ts
@@ -36,6 +36,17 @@ export { indices } from './indices';
export { useUIAceKeyboardMode } from './use_ui_ace_keyboard_mode';
+export {
+ installXJsonMode,
+ XJsonMode,
+ ElasticsearchSqlHighlightRules,
+ addXJsonToRules,
+ ScriptHighlightRules,
+ XJsonHighlightRules,
+ collapseLiteralStrings,
+ expandLiteralStrings,
+} from './console_lang';
+
/** dummy plugin, we just want esUiShared to have its own bundle */
export function plugin() {
return new (class EsUiSharedPlugin {
diff --git a/src/plugins/es_ui_shared/static/ace_x_json/hooks/index.ts b/src/plugins/es_ui_shared/static/ace_x_json/hooks/index.ts
new file mode 100644
index 0000000000000..1d2c33a9f0f47
--- /dev/null
+++ b/src/plugins/es_ui_shared/static/ace_x_json/hooks/index.ts
@@ -0,0 +1,20 @@
+/*
+ * Licensed to Elasticsearch B.V. under one or more contributor
+ * license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright
+ * ownership. Elasticsearch B.V. licenses this file to you under
+ * the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+export { useXJsonMode } from './use_x_json';
diff --git a/src/plugins/es_ui_shared/console_lang/ace/modes/index.d.ts b/src/plugins/es_ui_shared/static/ace_x_json/hooks/use_x_json.ts
similarity index 60%
rename from src/plugins/es_ui_shared/console_lang/ace/modes/index.d.ts
rename to src/plugins/es_ui_shared/static/ace_x_json/hooks/use_x_json.ts
index 06c9f9a51ea68..2b0bf0c8a3a7c 100644
--- a/src/plugins/es_ui_shared/console_lang/ace/modes/index.d.ts
+++ b/src/plugins/es_ui_shared/static/ace_x_json/hooks/use_x_json.ts
@@ -17,16 +17,22 @@
* under the License.
*/
-import { Editor } from 'brace';
+import { useState } from 'react';
+import { XJsonMode, collapseLiteralStrings, expandLiteralStrings } from '../../../public';
-export declare const ElasticsearchSqlHighlightRules: FunctionConstructor;
-export declare const ScriptHighlightRules: FunctionConstructor;
-export declare const XJsonHighlightRules: FunctionConstructor;
+const xJsonMode = new XJsonMode();
-export declare const XJsonMode: FunctionConstructor;
+export const useXJsonMode = (json: Record | string | null) => {
+ const [xJson, setXJson] = useState(() =>
+ json === null
+ ? ''
+ : expandLiteralStrings(typeof json === 'string' ? json : JSON.stringify(json, null, 2))
+ );
-/**
- * @param otherRules Another Ace ruleset
- * @param embedUnder The state name under which the rules will be embedded. Defaults to "json".
- */
-export declare const addXJsonToRules: (otherRules: any, embedUnder?: string) => void;
+ return {
+ xJson,
+ setXJson,
+ xJsonMode,
+ convertToJson: collapseLiteralStrings,
+ };
+};
diff --git a/x-pack/dev-tools/jest/create_jest_config.js b/x-pack/dev-tools/jest/create_jest_config.js
index 1f4311ae9bcf6..3068cdd0daa5b 100644
--- a/x-pack/dev-tools/jest/create_jest_config.js
+++ b/x-pack/dev-tools/jest/create_jest_config.js
@@ -28,6 +28,7 @@ export function createJestConfig({ kibanaDirectory, xPackKibanaDirectory }) {
'\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$': fileMockPath,
'\\.module.(css|scss)$': `${kibanaDirectory}/src/dev/jest/mocks/css_module_mock.js`,
'\\.(css|less|scss)$': `${kibanaDirectory}/src/dev/jest/mocks/style_mock.js`,
+ '\\.ace\\.worker.js$': `${kibanaDirectory}/src/dev/jest/mocks/ace_worker_module_mock.js`,
'^test_utils/enzyme_helpers': `${xPackKibanaDirectory}/test_utils/enzyme_helpers.tsx`,
'^test_utils/find_test_subject': `${xPackKibanaDirectory}/test_utils/find_test_subject.ts`,
'^test_utils/stub_web_worker': `${xPackKibanaDirectory}/test_utils/stub_web_worker.ts`,
diff --git a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/index.ts b/x-pack/plugins/es_ui_shared/console_lang/ace/modes/index.ts
deleted file mode 100644
index ae3c9962ecadb..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/index.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-export { installXJsonMode, XJsonMode } from './x_json';
diff --git a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts b/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts
deleted file mode 100644
index ae3c9962ecadb..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/index.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-export { installXJsonMode, XJsonMode } from './x_json';
diff --git a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/index.ts b/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/index.ts
deleted file mode 100644
index 0e40fd335dd31..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/index.ts
+++ /dev/null
@@ -1,13 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-// @ts-ignore
-import src from '!!raw-loader!./worker.js';
-
-export const workerModule = {
- id: 'ace/mode/json_worker',
- src,
-};
diff --git a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.d.ts b/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.d.ts
deleted file mode 100644
index 4ebad4f2ef91c..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/worker/worker.d.ts
+++ /dev/null
@@ -1,12 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-// Satisfy TS's requirements that the module be declared per './index.ts'.
-declare module '!!raw-loader!./worker.js' {
- const content: string;
- // eslint-disable-next-line import/no-default-export
- export default content;
-}
diff --git a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json.ts b/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json.ts
deleted file mode 100644
index bfeca045bea02..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/ace/modes/x_json/x_json.ts
+++ /dev/null
@@ -1,34 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-import ace from 'brace';
-import { XJsonMode } from '../../../../../../../src/plugins/es_ui_shared/console_lang';
-import { workerModule } from './worker';
-const { WorkerClient } = ace.acequire('ace/worker/worker_client');
-
-// Then clobber `createWorker` method to install our worker source. Per ace's wiki: https://github.com/ajaxorg/ace/wiki/Syntax-validation
-(XJsonMode.prototype as any).createWorker = function(session: ace.IEditSession) {
- const xJsonWorker = new WorkerClient(['ace'], workerModule, 'JsonWorker');
-
- xJsonWorker.attachToDocument(session.getDocument());
-
- xJsonWorker.on('annotate', function(e: { data: any }) {
- session.setAnnotations(e.data);
- });
-
- xJsonWorker.on('terminate', function() {
- session.clearAnnotations();
- });
-
- return xJsonWorker;
-};
-
-export { XJsonMode };
-
-export function installXJsonMode(editor: ace.Editor) {
- const session = editor.getSession();
- session.setMode(new (XJsonMode as any)());
-}
diff --git a/x-pack/plugins/es_ui_shared/console_lang/index.ts b/x-pack/plugins/es_ui_shared/console_lang/index.ts
deleted file mode 100644
index b5fe3a554e34d..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/index.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-export { XJsonMode, installXJsonMode } from './ace/modes';
diff --git a/x-pack/plugins/es_ui_shared/console_lang/mocks.ts b/x-pack/plugins/es_ui_shared/console_lang/mocks.ts
deleted file mode 100644
index 68480282ddc03..0000000000000
--- a/x-pack/plugins/es_ui_shared/console_lang/mocks.ts
+++ /dev/null
@@ -1,9 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-jest.mock('./ace/modes/x_json/worker', () => ({
- workerModule: { id: 'ace/mode/json_worker', src: '' },
-}));
diff --git a/x-pack/plugins/ml/public/application/components/custom_hooks/index.ts b/x-pack/plugins/ml/public/application/components/custom_hooks/index.ts
index dfd74d8970cb4..ffead802bd6f9 100644
--- a/x-pack/plugins/ml/public/application/components/custom_hooks/index.ts
+++ b/x-pack/plugins/ml/public/application/components/custom_hooks/index.ts
@@ -5,4 +5,3 @@
*/
export { usePartialState } from './use_partial_state';
-export { useXJsonMode, xJsonMode } from './use_x_json_mode';
diff --git a/x-pack/plugins/ml/public/application/components/custom_hooks/use_x_json_mode.ts b/x-pack/plugins/ml/public/application/components/custom_hooks/use_x_json_mode.ts
deleted file mode 100644
index c979632db54d6..0000000000000
--- a/x-pack/plugins/ml/public/application/components/custom_hooks/use_x_json_mode.ts
+++ /dev/null
@@ -1,26 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-import { useState } from 'react';
-import {
- collapseLiteralStrings,
- expandLiteralStrings,
- XJsonMode,
-} from '../../../../shared_imports';
-
-// @ts-ignore
-export const xJsonMode = new XJsonMode();
-
-export const useXJsonMode = (json: string) => {
- const [xJson, setXJson] = useState(expandLiteralStrings(json));
-
- return {
- xJson,
- setXJson,
- xJsonMode,
- convertToJson: collapseLiteralStrings,
- };
-};
diff --git a/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/components/create_analytics_advanced_editor/create_analytics_advanced_editor.tsx b/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/components/create_analytics_advanced_editor/create_analytics_advanced_editor.tsx
index a3e5da5e2d039..cef03cc0d0c76 100644
--- a/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/components/create_analytics_advanced_editor/create_analytics_advanced_editor.tsx
+++ b/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/components/create_analytics_advanced_editor/create_analytics_advanced_editor.tsx
@@ -18,8 +18,11 @@ import {
import { i18n } from '@kbn/i18n';
+import { XJsonMode } from '../../../../../../../shared_imports';
+
+const xJsonMode = new XJsonMode();
+
import { CreateAnalyticsFormProps } from '../../hooks/use_create_analytics_form';
-import { xJsonMode } from '../../../../../components/custom_hooks';
export const CreateAnalyticsAdvancedEditor: FC = ({ actions, state }) => {
const { setAdvancedEditorRawString, setFormState } = actions;
diff --git a/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/hooks/use_create_analytics_form/reducer.ts b/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/hooks/use_create_analytics_form/reducer.ts
index ded6e50947035..d55eb14a20e29 100644
--- a/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/hooks/use_create_analytics_form/reducer.ts
+++ b/x-pack/plugins/ml/public/application/data_frame_analytics/pages/analytics_management/hooks/use_create_analytics_form/reducer.ts
@@ -11,7 +11,7 @@ import numeral from '@elastic/numeral';
import { isEmpty } from 'lodash';
import { isValidIndexName } from '../../../../../../../common/util/es_utils';
-import { collapseLiteralStrings } from '../../../../../../../../../../src/plugins/es_ui_shared/console_lang/lib/json_xjson_translation_tools';
+import { collapseLiteralStrings } from '../../../../../../../../../../src/plugins/es_ui_shared/public';
import { Action, ACTION } from './actions';
import { getInitialState, getJobConfigFromFormState, State } from './state';
diff --git a/x-pack/plugins/ml/public/application/jobs/jobs_list/components/ml_job_editor/ml_job_editor.tsx b/x-pack/plugins/ml/public/application/jobs/jobs_list/components/ml_job_editor/ml_job_editor.tsx
index 0633c62f754e0..c10212ee31564 100644
--- a/x-pack/plugins/ml/public/application/jobs/jobs_list/components/ml_job_editor/ml_job_editor.tsx
+++ b/x-pack/plugins/ml/public/application/jobs/jobs_list/components/ml_job_editor/ml_job_editor.tsx
@@ -7,10 +7,9 @@
import React, { FC } from 'react';
import { EuiCodeEditor, EuiCodeEditorProps } from '@elastic/eui';
-import { expandLiteralStrings } from '../../../../../../shared_imports';
-import { xJsonMode } from '../../../../components/custom_hooks';
+import { expandLiteralStrings, XJsonMode } from '../../../../../../shared_imports';
-export const ML_EDITOR_MODE = { TEXT: 'text', JSON: 'json', XJSON: xJsonMode };
+export const ML_EDITOR_MODE = { TEXT: 'text', JSON: 'json', XJSON: new XJsonMode() };
interface MlJobEditorProps {
value: string;
diff --git a/x-pack/plugins/ml/shared_imports.ts b/x-pack/plugins/ml/shared_imports.ts
index 94ce8c82f1d95..a82ed5387818d 100644
--- a/x-pack/plugins/ml/shared_imports.ts
+++ b/x-pack/plugins/ml/shared_imports.ts
@@ -4,9 +4,8 @@
* you may not use this file except in compliance with the Elastic License.
*/
-export { XJsonMode } from '../../../src/plugins/es_ui_shared/console_lang/ace/modes/x_json';
-
export {
+ XJsonMode,
collapseLiteralStrings,
expandLiteralStrings,
-} from '../../../src/plugins/es_ui_shared/console_lang/lib';
+} from '../../../src/plugins/es_ui_shared/public';
diff --git a/x-pack/plugins/searchprofiler/public/application/editor/editor.test.tsx b/x-pack/plugins/searchprofiler/public/application/editor/editor.test.tsx
index ad56b3957eb74..e56fc79156666 100644
--- a/x-pack/plugins/searchprofiler/public/application/editor/editor.test.tsx
+++ b/x-pack/plugins/searchprofiler/public/application/editor/editor.test.tsx
@@ -6,8 +6,6 @@
import 'brace';
import 'brace/mode/json';
-import '../../../../es_ui_shared/console_lang/mocks';
-
import { registerTestBed } from '../../../../../test_utils';
import { Editor, Props } from '.';
diff --git a/x-pack/plugins/searchprofiler/public/application/editor/init_editor.ts b/x-pack/plugins/searchprofiler/public/application/editor/init_editor.ts
index 6f19ce12eb639..3ad92531e4367 100644
--- a/x-pack/plugins/searchprofiler/public/application/editor/init_editor.ts
+++ b/x-pack/plugins/searchprofiler/public/application/editor/init_editor.ts
@@ -5,7 +5,7 @@
*/
import ace from 'brace';
-import { installXJsonMode } from '../../../../es_ui_shared/console_lang';
+import { installXJsonMode } from '../../../../../../src/plugins/es_ui_shared/public';
export function initializeEditor({
el,
diff --git a/x-pack/plugins/searchprofiler/public/application/utils/check_for_json_errors.ts b/x-pack/plugins/searchprofiler/public/application/utils/check_for_json_errors.ts
index 99687de0f1440..58a62c4636c25 100644
--- a/x-pack/plugins/searchprofiler/public/application/utils/check_for_json_errors.ts
+++ b/x-pack/plugins/searchprofiler/public/application/utils/check_for_json_errors.ts
@@ -4,7 +4,7 @@
* you may not use this file except in compliance with the Elastic License.
*/
-import { collapseLiteralStrings } from '../../../../../../src/plugins/es_ui_shared/console_lang/lib';
+import { collapseLiteralStrings } from '../../../../../../src/plugins/es_ui_shared/public';
export function checkForParseErrors(json: string) {
const sanitizedJson = collapseLiteralStrings(json);
diff --git a/x-pack/plugins/transform/public/app/hooks/use_x_json_mode.ts b/x-pack/plugins/transform/public/app/hooks/use_x_json_mode.ts
deleted file mode 100644
index 1017ce198ff29..0000000000000
--- a/x-pack/plugins/transform/public/app/hooks/use_x_json_mode.ts
+++ /dev/null
@@ -1,20 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-import { useState } from 'react';
-import { collapseLiteralStrings, expandLiteralStrings, XJsonMode } from '../../shared_imports';
-
-export const xJsonMode = new XJsonMode();
-
-export const useXJsonMode = (json: string) => {
- const [xJson, setXJson] = useState(expandLiteralStrings(json));
-
- return {
- xJson,
- setXJson,
- xJsonMode,
- convertToJson: collapseLiteralStrings,
- };
-};
diff --git a/x-pack/plugins/transform/public/app/sections/create_transform/components/step_define/step_define_form.tsx b/x-pack/plugins/transform/public/app/sections/create_transform/components/step_define/step_define_form.tsx
index 5e0eb7ee08361..320e405b5d437 100644
--- a/x-pack/plugins/transform/public/app/sections/create_transform/components/step_define/step_define_form.tsx
+++ b/x-pack/plugins/transform/public/app/sections/create_transform/components/step_define/step_define_form.tsx
@@ -33,11 +33,12 @@ import {
QueryStringInput,
} from '../../../../../../../../../src/plugins/data/public';
+import { useXJsonMode } from '../../../../../../../../../src/plugins/es_ui_shared/static/ace_x_json/hooks';
+
import { PivotPreview } from '../../../../components/pivot_preview';
import { useDocumentationLinks } from '../../../../hooks/use_documentation_links';
import { SavedSearchQuery, SearchItems } from '../../../../hooks/use_search_items';
-import { useXJsonMode, xJsonMode } from '../../../../hooks/use_x_json_mode';
import { useToastNotifications } from '../../../../app_dependencies';
import { TransformPivotConfig } from '../../../../common';
import { dictionaryToArray, Dictionary } from '../../../../../../common/types/common';
@@ -432,6 +433,7 @@ export const StepDefineForm: FC = React.memo(({ overrides = {}, onChange,
convertToJson,
setXJson: setAdvancedEditorConfig,
xJson: advancedEditorConfig,
+ xJsonMode,
} = useXJsonMode(stringifiedPivotConfig);
useEffect(() => {
diff --git a/x-pack/plugins/transform/public/shared_imports.ts b/x-pack/plugins/transform/public/shared_imports.ts
index 8eb42ad677c0f..494b6db6aafe0 100644
--- a/x-pack/plugins/transform/public/shared_imports.ts
+++ b/x-pack/plugins/transform/public/shared_imports.ts
@@ -5,11 +5,11 @@
*/
export { createSavedSearchesLoader } from '../../../../src/plugins/discover/public';
-export { XJsonMode } from '../../es_ui_shared/console_lang/ace/modes/x_json';
export {
+ XJsonMode,
collapseLiteralStrings,
expandLiteralStrings,
-} from '../../../../src/plugins/es_ui_shared/console_lang/lib';
+} from '../../../../src/plugins/es_ui_shared/public';
export {
UseRequestConfig,
diff --git a/x-pack/plugins/triggers_actions_ui/public/application/components/builtin_action_types/es_index.tsx b/x-pack/plugins/triggers_actions_ui/public/application/components/builtin_action_types/es_index.tsx
index 9bd6a39d216e3..15f68e6a9f441 100644
--- a/x-pack/plugins/triggers_actions_ui/public/application/components/builtin_action_types/es_index.tsx
+++ b/x-pack/plugins/triggers_actions_ui/public/application/components/builtin_action_types/es_index.tsx
@@ -17,6 +17,7 @@ import {
} from '@elastic/eui';
import { FormattedMessage } from '@kbn/i18n/react';
import { i18n } from '@kbn/i18n';
+import { useXJsonMode } from '../../../../../../../src/plugins/es_ui_shared/static/ace_x_json/hooks';
import {
ActionTypeModel,
ActionConnectorFieldsProps,
@@ -31,7 +32,6 @@ import {
getIndexOptions,
getIndexPatterns,
} from '../../../common/index_controls';
-import { useXJsonMode } from '../../lib/use_x_json_mode';
import { AddMessageVariables } from '../add_message_variables';
export function getActionType(): ActionTypeModel {
diff --git a/x-pack/plugins/triggers_actions_ui/public/application/lib/use_x_json_mode.ts b/x-pack/plugins/triggers_actions_ui/public/application/lib/use_x_json_mode.ts
deleted file mode 100644
index b677924724ffe..0000000000000
--- a/x-pack/plugins/triggers_actions_ui/public/application/lib/use_x_json_mode.ts
+++ /dev/null
@@ -1,25 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-import { useState } from 'react';
-import { XJsonMode } from '../../../../../../src/plugins/es_ui_shared/console_lang/ace/modes/x_json';
-import {
- collapseLiteralStrings,
- expandLiteralStrings,
-} from '../../../../../../src/plugins/es_ui_shared/console_lang/lib';
-// @ts-ignore
-export const xJsonMode = new XJsonMode();
-export const useXJsonMode = (json: Record | null) => {
- const [xJson, setXJson] = useState(
- json === null ? '' : expandLiteralStrings(JSON.stringify(json, null, 2))
- );
-
- return {
- xJson,
- setXJson,
- xJsonMode,
- convertToJson: collapseLiteralStrings,
- };
-};
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts
index 47de4548898cf..a35dd1346dc90 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_json.test.ts
@@ -4,8 +4,6 @@
* you may not use this file except in compliance with the Elastic License.
*/
-import '../../../es_ui_shared/console_lang/mocks';
-
import { act } from 'react-dom/test-utils';
import { setupEnvironment, pageHelpers, nextTick, wrapBodyResponse } from './helpers';
import { WatchCreateJsonTestBed } from './helpers/watch_create_json.helpers';
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx
index e5bdcbbfb82cf..89cd5207fe250 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_create_threshold.test.tsx
@@ -4,8 +4,6 @@
* you may not use this file except in compliance with the Elastic License.
*/
-import '../../../es_ui_shared/console_lang/mocks';
-
import React from 'react';
import { act } from 'react-dom/test-utils';
import axiosXhrAdapter from 'axios/lib/adapters/xhr';
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts b/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts
index ced06562b1d20..18acf7339580c 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_edit.test.ts
@@ -3,7 +3,6 @@
* or more contributor license agreements. Licensed under the Elastic License;
* you may not use this file except in compliance with the Elastic License.
*/
-import '../../../es_ui_shared/console_lang/mocks';
import { act } from 'react-dom/test-utils';
import axiosXhrAdapter from 'axios/lib/adapters/xhr';
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_list.test.ts b/x-pack/plugins/watcher/__jest__/client_integration/watch_list.test.ts
index 3370e42b2417f..a0327c6dfa1db 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_list.test.ts
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_list.test.ts
@@ -3,7 +3,6 @@
* or more contributor license agreements. Licensed under the Elastic License;
* you may not use this file except in compliance with the Elastic License.
*/
-import '../../../es_ui_shared/console_lang/mocks';
import { act } from 'react-dom/test-utils';
import * as fixtures from '../../test/fixtures';
diff --git a/x-pack/plugins/watcher/__jest__/client_integration/watch_status.test.ts b/x-pack/plugins/watcher/__jest__/client_integration/watch_status.test.ts
index 20b7b526705c0..e9cbb2c92491a 100644
--- a/x-pack/plugins/watcher/__jest__/client_integration/watch_status.test.ts
+++ b/x-pack/plugins/watcher/__jest__/client_integration/watch_status.test.ts
@@ -3,7 +3,6 @@
* or more contributor license agreements. Licensed under the Elastic License;
* you may not use this file except in compliance with the Elastic License.
*/
-import '../../../es_ui_shared/console_lang/mocks';
import { act } from 'react-dom/test-utils';
import { setupEnvironment, pageHelpers, nextTick } from './helpers';
diff --git a/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_form.tsx b/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_form.tsx
index b3a09d3bc0e65..b5835d862000f 100644
--- a/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_form.tsx
+++ b/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_form.tsx
@@ -23,14 +23,13 @@ import { FormattedMessage } from '@kbn/i18n/react';
import { serializeJsonWatch } from '../../../../../../common/lib/serialization';
import { ErrableFormRow, SectionError, Error as ServerError } from '../../../../components';
+import { useXJsonMode } from '../../../../shared_imports';
import { onWatchSave } from '../../watch_edit_actions';
import { WatchContext } from '../../watch_context';
import { goToWatchList } from '../../../../lib/navigation';
import { RequestFlyout } from '../request_flyout';
import { useAppContext } from '../../../../app_context';
-import { useXJsonMode } from './use_x_json_mode';
-
export const JsonWatchEditForm = () => {
const {
links: { putWatchApiUrl },
diff --git a/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_simulate.tsx b/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_simulate.tsx
index b9fce52b480ef..fa0b9b0a2566f 100644
--- a/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_simulate.tsx
+++ b/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/json_watch_edit_simulate.tsx
@@ -40,7 +40,7 @@ import { JsonWatchEditSimulateResults } from './json_watch_edit_simulate_results
import { getTimeUnitLabel } from '../../../../lib/get_time_unit_label';
import { useAppContext } from '../../../../app_context';
-import { useXJsonMode } from './use_x_json_mode';
+import { useXJsonMode } from '../../../../shared_imports';
const actionModeOptions = Object.keys(ACTION_MODES).map(mode => ({
text: ACTION_MODES[mode],
diff --git a/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/use_x_json_mode.ts b/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/use_x_json_mode.ts
deleted file mode 100644
index 7aefb0554e0e8..0000000000000
--- a/x-pack/plugins/watcher/public/application/sections/watch_edit/components/json_watch_edit/use_x_json_mode.ts
+++ /dev/null
@@ -1,24 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-import { useState } from 'react';
-import { XJsonMode } from '../../../../../../../es_ui_shared/console_lang';
-import {
- collapseLiteralStrings,
- expandLiteralStrings,
-} from '../../../../../../../../../src/plugins/es_ui_shared/console_lang/lib';
-
-export const xJsonMode = new XJsonMode();
-
-export const useXJsonMode = (json: string) => {
- const [xJson, setXJson] = useState(expandLiteralStrings(json));
-
- return {
- xJson,
- setXJson,
- xJsonMode,
- convertToJson: collapseLiteralStrings,
- };
-};
diff --git a/x-pack/plugins/watcher/public/application/shared_imports.ts b/x-pack/plugins/watcher/public/application/shared_imports.ts
index cbc4dde7448ff..94ef7af1c28d1 100644
--- a/x-pack/plugins/watcher/public/application/shared_imports.ts
+++ b/x-pack/plugins/watcher/public/application/shared_imports.ts
@@ -11,3 +11,5 @@ export {
sendRequest,
useRequest,
} from '../../../../../src/plugins/es_ui_shared/public/request/np_ready_request';
+
+export { useXJsonMode } from '../../../../../src/plugins/es_ui_shared/static/ace_x_json/hooks';
From 77d22f55d9d952ed0d3b6ed6da25def23fcea4f0 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Cau=C3=AA=20Marcondes?=
<55978943+cauemarcondes@users.noreply.github.com>
Date: Fri, 10 Apr 2020 13:54:48 +0100
Subject: [PATCH 04/24] [APM] Agent config select box doesn't work on IE
(#63236)
* adding value property to select options
* fixing test
---
.../SettingsPage/SettingFormRow.tsx | 7 +++++--
.../setting_definitions/__snapshots__/index.test.ts.snap | 4 ++++
.../setting_definitions/general_settings.ts | 8 ++++----
.../agent_configuration/setting_definitions/types.d.ts | 2 +-
4 files changed, 14 insertions(+), 7 deletions(-)
diff --git a/x-pack/legacy/plugins/apm/public/components/app/Settings/AgentConfigurations/AgentConfigurationCreateEdit/SettingsPage/SettingFormRow.tsx b/x-pack/legacy/plugins/apm/public/components/app/Settings/AgentConfigurations/AgentConfigurationCreateEdit/SettingsPage/SettingFormRow.tsx
index 30c772bf5f634..baab600145b81 100644
--- a/x-pack/legacy/plugins/apm/public/components/app/Settings/AgentConfigurations/AgentConfigurationCreateEdit/SettingsPage/SettingFormRow.tsx
+++ b/x-pack/legacy/plugins/apm/public/components/app/Settings/AgentConfigurations/AgentConfigurationCreateEdit/SettingsPage/SettingFormRow.tsx
@@ -73,7 +73,10 @@ function FormRow({
return (
onChange(setting.key, e.target.value)}
/>
@@ -105,7 +108,7 @@ function FormRow({
defaultMessage: 'Select unit'
})}
value={unit}
- options={setting.units?.map(text => ({ text }))}
+ options={setting.units?.map(text => ({ text, value: text }))}
onChange={e =>
onChange(
setting.key,
diff --git a/x-pack/plugins/apm/common/agent_configuration/setting_definitions/__snapshots__/index.test.ts.snap b/x-pack/plugins/apm/common/agent_configuration/setting_definitions/__snapshots__/index.test.ts.snap
index 49840d2157af7..ea706be9f584a 100644
--- a/x-pack/plugins/apm/common/agent_configuration/setting_definitions/__snapshots__/index.test.ts.snap
+++ b/x-pack/plugins/apm/common/agent_configuration/setting_definitions/__snapshots__/index.test.ts.snap
@@ -29,15 +29,19 @@ Array [
"options": Array [
Object {
"text": "off",
+ "value": "off",
},
Object {
"text": "errors",
+ "value": "errors",
},
Object {
"text": "transactions",
+ "value": "transactions",
},
Object {
"text": "all",
+ "value": "all",
},
],
"type": "select",
diff --git a/x-pack/plugins/apm/common/agent_configuration/setting_definitions/general_settings.ts b/x-pack/plugins/apm/common/agent_configuration/setting_definitions/general_settings.ts
index e73aed35e87f9..7477238ba79ae 100644
--- a/x-pack/plugins/apm/common/agent_configuration/setting_definitions/general_settings.ts
+++ b/x-pack/plugins/apm/common/agent_configuration/setting_definitions/general_settings.ts
@@ -67,10 +67,10 @@ export const generalSettings: RawSettingDefinition[] = [
}
),
options: [
- { text: 'off' },
- { text: 'errors' },
- { text: 'transactions' },
- { text: 'all' }
+ { text: 'off', value: 'off' },
+ { text: 'errors', value: 'errors' },
+ { text: 'transactions', value: 'transactions' },
+ { text: 'all', value: 'all' }
],
excludeAgents: ['js-base', 'rum-js']
},
diff --git a/x-pack/plugins/apm/common/agent_configuration/setting_definitions/types.d.ts b/x-pack/plugins/apm/common/agent_configuration/setting_definitions/types.d.ts
index 282ced346dda0..815b8cb3d4e83 100644
--- a/x-pack/plugins/apm/common/agent_configuration/setting_definitions/types.d.ts
+++ b/x-pack/plugins/apm/common/agent_configuration/setting_definitions/types.d.ts
@@ -76,7 +76,7 @@ interface FloatSetting extends BaseSetting {
interface SelectSetting extends BaseSetting {
type: 'select';
- options: Array<{ text: string }>;
+ options: Array<{ text: string; value: string }>;
}
interface BooleanSetting extends BaseSetting {
From e8491adbab145590e877e5af9b1efa009400bf46 Mon Sep 17 00:00:00 2001
From: Nathan L Smith
Date: Fri, 10 Apr 2020 09:04:17 -0500
Subject: [PATCH 05/24] Use globe icon for "ext" span type on service map
(#63205)
Both "external" and "ext" can be returned and should have the same icon.
---
.../apm/public/components/app/ServiceMap/Cytoscape.stories.tsx | 1 +
.../legacy/plugins/apm/public/components/app/ServiceMap/icons.ts | 1 +
2 files changed, 2 insertions(+)
diff --git a/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/Cytoscape.stories.tsx b/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/Cytoscape.stories.tsx
index a8d3b843a1f3d..31c227d8bbcab 100644
--- a/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/Cytoscape.stories.tsx
+++ b/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/Cytoscape.stories.tsx
@@ -85,6 +85,7 @@ storiesOf('app/ServiceMap/Cytoscape', module)
}
},
{ data: { id: 'external', 'span.type': 'external' } },
+ { data: { id: 'ext', 'span.type': 'ext' } },
{ data: { id: 'messaging', 'span.type': 'messaging' } },
{
data: {
diff --git a/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/icons.ts b/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/icons.ts
index dd9b48d312725..095c2d9250e27 100644
--- a/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/icons.ts
+++ b/x-pack/legacy/plugins/apm/public/components/app/ServiceMap/icons.ts
@@ -32,6 +32,7 @@ export const defaultIcon = defaultIconImport;
const icons: { [key: string]: string } = {
cache: databaseIcon,
db: databaseIcon,
+ ext: globeIcon,
external: globeIcon,
messaging: documentsIcon,
resource: globeIcon
From ed975d0169e6728db9d8c5cf98ca655293217192 Mon Sep 17 00:00:00 2001
From: DianaDerevyankina <54894989+DianaDerevyankina@users.noreply.github.com>
Date: Fri, 10 Apr 2020 17:59:50 +0300
Subject: [PATCH 06/24] Move shared vislib components into Charts plugin
(#62957)
* Closes #56310
Move shared vislib components into Charts plugin
* Fixed imports in tests
* Changed i18n IDs to match charts namespace
* Renamed ColorSchemaVislibParams to ColorSchemaParams, added enums and got rid of useValidation function
* Renamed ColorSchemaVislibParams to ColorSchemaParams and got rid of useValidation function
* Fixed merge conflict
* Replaced enums with objects again
---
.../public/components/region_map_options.tsx | 6 ++-
.../public/components/tile_map_options.tsx | 2 +-
.../components/wms_internal_options.tsx | 2 +-
.../public/components/wms_options.tsx | 2 +-
.../public/settings_options.tsx | 2 +-
.../public/components/metric_vis_options.tsx | 4 +-
.../vis_type_metric/public/metric_vis_fn.ts | 3 +-
.../vis_type_metric/public/metric_vis_type.ts | 3 +-
.../vis_type_metric/public/types.ts | 3 +-
.../public/components/table_vis_options.tsx | 6 ++-
.../public/components/tag_cloud_options.tsx | 2 +-
.../vis_type_vislib/public/area.ts | 2 +-
.../public/components/common/index.ts | 8 ----
.../components/options/gauge/labels_panel.tsx | 2 +-
.../components/options/gauge/ranges_panel.tsx | 14 +++---
.../components/options/gauge/style_panel.tsx | 2 +-
.../components/options/heatmap/index.tsx | 6 +--
.../options/heatmap/labels_panel.tsx | 2 +-
.../metrics_axes/category_axis_panel.tsx | 2 +-
.../options/metrics_axes/chart_options.tsx | 2 +-
.../metrics_axes/custom_extents_options.tsx | 2 +-
.../options/metrics_axes/label_options.tsx | 3 +-
.../metrics_axes/line_options.test.tsx | 2 +-
.../options/metrics_axes/line_options.tsx | 6 ++-
.../components/options/metrics_axes/mocks.ts | 3 +-
.../metrics_axes/value_axis_options.test.tsx | 2 +-
.../metrics_axes/value_axis_options.tsx | 6 ++-
.../options/metrics_axes/y_extents.test.tsx | 2 +-
.../options/metrics_axes/y_extents.tsx | 2 +-
.../public/components/options/pie.tsx | 3 +-
.../options/point_series/grid_panel.tsx | 2 +-
.../options/point_series/point_series.tsx | 3 +-
.../options/point_series/threshold_panel.tsx | 12 ++++--
.../vis_type_vislib/public/gauge.ts | 13 ++++--
.../vis_type_vislib/public/goal.ts | 4 +-
.../vis_type_vislib/public/heatmap.ts | 6 +--
.../vis_type_vislib/public/histogram.ts | 2 +-
.../vis_type_vislib/public/horizontal_bar.ts | 2 +-
.../vis_type_vislib/public/index.ts | 14 ------
.../vis_type_vislib/public/line.ts | 2 +-
.../vis_type_vislib/public/types.ts | 25 +----------
.../public/utils/collections.ts | 16 +------
.../static/components}/basic_options.tsx | 4 +-
.../public/static/components/collections.ts} | 26 +++++------
.../static/components}/color_ranges.tsx | 7 +--
.../static/components}/color_schema.tsx | 18 ++++----
.../charts/public/static/components/index.ts | 30 +++++++++++++
.../static/components}/number_input.tsx | 0
.../public/static/components}/range.tsx | 2 +-
.../components}/required_number_input.tsx | 14 +++---
.../public/static/components}/select.tsx | 0
.../public/static/components}/switch.tsx | 0
.../public/static/components}/text_input.tsx | 0
.../charts/public/static/components/types.ts | 43 +++++++++++++++++++
src/plugins/charts/public/static/index.ts | 1 +
.../translations/translations/ja-JP.json | 16 +++----
.../translations/translations/zh-CN.json | 16 +++----
57 files changed, 215 insertions(+), 169 deletions(-)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/basic_options.tsx (90%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common/utils.ts => plugins/charts/public/static/components/collections.ts} (67%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/color_ranges.tsx (93%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/color_schema.tsx (82%)
create mode 100644 src/plugins/charts/public/static/components/index.ts
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/number_input.tsx (100%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/range.tsx (96%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/required_number_input.tsx (87%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/select.tsx (100%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/switch.tsx (100%)
rename src/{legacy/core_plugins/vis_type_vislib/public/components/common => plugins/charts/public/static/components}/text_input.tsx (100%)
create mode 100644 src/plugins/charts/public/static/components/types.ts
diff --git a/src/legacy/core_plugins/region_map/public/components/region_map_options.tsx b/src/legacy/core_plugins/region_map/public/components/region_map_options.tsx
index 61cfbf00ded9e..187b27953830d 100644
--- a/src/legacy/core_plugins/region_map/public/components/region_map_options.tsx
+++ b/src/legacy/core_plugins/region_map/public/components/region_map_options.tsx
@@ -24,7 +24,11 @@ import { FormattedMessage } from '@kbn/i18n/react';
import { FileLayerField, VectorLayer, ServiceSettings } from 'ui/vis/map/service_settings';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
-import { NumberInputOption, SelectOption, SwitchOption } from '../../../vis_type_vislib/public';
+import {
+ NumberInputOption,
+ SelectOption,
+ SwitchOption,
+} from '../../../../../plugins/charts/public';
import { WmsOptions } from '../../../tile_map/public/components/wms_options';
import { RegionMapVisParams } from '../types';
diff --git a/src/legacy/core_plugins/tile_map/public/components/tile_map_options.tsx b/src/legacy/core_plugins/tile_map/public/components/tile_map_options.tsx
index 9169647aa2aef..9ca42fe3e4074 100644
--- a/src/legacy/core_plugins/tile_map/public/components/tile_map_options.tsx
+++ b/src/legacy/core_plugins/tile_map/public/components/tile_map_options.tsx
@@ -27,7 +27,7 @@ import {
RangeOption,
SelectOption,
SwitchOption,
-} from '../../../vis_type_vislib/public';
+} from '../../../../../plugins/charts/public';
import { WmsOptions } from './wms_options';
import { TileMapVisParams } from '../types';
import { MapTypes } from '../map_types';
diff --git a/src/legacy/core_plugins/tile_map/public/components/wms_internal_options.tsx b/src/legacy/core_plugins/tile_map/public/components/wms_internal_options.tsx
index b81667400303d..47f5b8f31e62b 100644
--- a/src/legacy/core_plugins/tile_map/public/components/wms_internal_options.tsx
+++ b/src/legacy/core_plugins/tile_map/public/components/wms_internal_options.tsx
@@ -21,7 +21,7 @@ import React from 'react';
import { EuiLink, EuiSpacer, EuiText, EuiScreenReaderOnly } from '@elastic/eui';
import { FormattedMessage } from '@kbn/i18n/react';
-import { TextInputOption } from '../../../vis_type_vislib/public';
+import { TextInputOption } from '../../../../../plugins/charts/public';
import { WMSOptions } from '../types';
interface WmsInternalOptions {
diff --git a/src/legacy/core_plugins/tile_map/public/components/wms_options.tsx b/src/legacy/core_plugins/tile_map/public/components/wms_options.tsx
index 27127b781cd4d..b8535e72e8818 100644
--- a/src/legacy/core_plugins/tile_map/public/components/wms_options.tsx
+++ b/src/legacy/core_plugins/tile_map/public/components/wms_options.tsx
@@ -25,7 +25,7 @@ import { FormattedMessage } from '@kbn/i18n/react';
import { TmsLayer } from 'ui/vis/map/service_settings';
import { Vis } from '../../../../../plugins/visualizations/public';
import { RegionMapVisParams } from '../../../region_map/public/types';
-import { SelectOption, SwitchOption } from '../../../vis_type_vislib/public';
+import { SelectOption, SwitchOption } from '../../../../../plugins/charts/public';
import { WmsInternalOptions } from './wms_internal_options';
import { WMSOptions, TileMapVisParams } from '../types';
diff --git a/src/legacy/core_plugins/vis_type_markdown/public/settings_options.tsx b/src/legacy/core_plugins/vis_type_markdown/public/settings_options.tsx
index 552fd63373554..16b2749c34e10 100644
--- a/src/legacy/core_plugins/vis_type_markdown/public/settings_options.tsx
+++ b/src/legacy/core_plugins/vis_type_markdown/public/settings_options.tsx
@@ -22,7 +22,7 @@ import { EuiPanel } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
-import { RangeOption, SwitchOption } from '../../vis_type_vislib/public';
+import { RangeOption, SwitchOption } from '../../../../plugins/charts/public';
import { MarkdownVisParams } from './types';
function SettingsOptions({ stateParams, setValue }: VisOptionsProps) {
diff --git a/src/legacy/core_plugins/vis_type_metric/public/components/metric_vis_options.tsx b/src/legacy/core_plugins/vis_type_metric/public/components/metric_vis_options.tsx
index 5c3032511f09a..56af0ee91878f 100644
--- a/src/legacy/core_plugins/vis_type_metric/public/components/metric_vis_options.tsx
+++ b/src/legacy/core_plugins/vis_type_metric/public/components/metric_vis_options.tsx
@@ -37,9 +37,9 @@ import {
SwitchOption,
RangeOption,
SetColorSchemaOptionsValue,
-} from '../../../vis_type_vislib/public';
+ SetColorRangeValue,
+} from '../../../../../plugins/charts/public';
import { MetricVisParam, VisParams } from '../types';
-import { SetColorRangeValue } from '../../../vis_type_vislib/public/components/common/color_ranges';
function MetricVisOptions({
stateParams,
diff --git a/src/legacy/core_plugins/vis_type_metric/public/metric_vis_fn.ts b/src/legacy/core_plugins/vis_type_metric/public/metric_vis_fn.ts
index 03b412c6fff15..52784da1bd73b 100644
--- a/src/legacy/core_plugins/vis_type_metric/public/metric_vis_fn.ts
+++ b/src/legacy/core_plugins/vis_type_metric/public/metric_vis_fn.ts
@@ -26,9 +26,8 @@ import {
Render,
Style,
} from '../../../../plugins/expressions/public';
-import { ColorModes } from '../../vis_type_vislib/public';
import { visType, DimensionsVisParam, VisParams } from './types';
-import { ColorSchemas, vislibColorMaps } from '../../../../plugins/charts/public';
+import { ColorSchemas, vislibColorMaps, ColorModes } from '../../../../plugins/charts/public';
export type Input = KibanaDatatable;
diff --git a/src/legacy/core_plugins/vis_type_metric/public/metric_vis_type.ts b/src/legacy/core_plugins/vis_type_metric/public/metric_vis_type.ts
index 3bbb8964122e5..ab31c56921412 100644
--- a/src/legacy/core_plugins/vis_type_metric/public/metric_vis_type.ts
+++ b/src/legacy/core_plugins/vis_type_metric/public/metric_vis_type.ts
@@ -21,8 +21,7 @@ import { i18n } from '@kbn/i18n';
import { MetricVisComponent } from './components/metric_vis_component';
import { MetricVisOptions } from './components/metric_vis_options';
-import { ColorModes } from '../../vis_type_vislib/public';
-import { ColorSchemas, colorSchemas } from '../../../../plugins/charts/public';
+import { ColorSchemas, colorSchemas, ColorModes } from '../../../../plugins/charts/public';
import { AggGroupNames } from '../../../../plugins/data/public';
import { Schemas } from '../../../../plugins/vis_default_editor/public';
diff --git a/src/legacy/core_plugins/vis_type_metric/public/types.ts b/src/legacy/core_plugins/vis_type_metric/public/types.ts
index cae18dd8a2ab1..cdf62cba934d7 100644
--- a/src/legacy/core_plugins/vis_type_metric/public/types.ts
+++ b/src/legacy/core_plugins/vis_type_metric/public/types.ts
@@ -19,8 +19,7 @@
import { Range } from '../../../../plugins/expressions/public';
import { SchemaConfig } from '../../../../plugins/visualizations/public';
-import { ColorModes, Labels, Style } from '../../vis_type_vislib/public';
-import { ColorSchemas } from '../../../../plugins/charts/public';
+import { ColorModes, ColorSchemas, Labels, Style } from '../../../../plugins/charts/public';
export const visType = 'metric';
diff --git a/src/legacy/core_plugins/vis_type_table/public/components/table_vis_options.tsx b/src/legacy/core_plugins/vis_type_table/public/components/table_vis_options.tsx
index d01ab31e0a843..265528f33f9cd 100644
--- a/src/legacy/core_plugins/vis_type_table/public/components/table_vis_options.tsx
+++ b/src/legacy/core_plugins/vis_type_table/public/components/table_vis_options.tsx
@@ -25,7 +25,11 @@ import { FormattedMessage } from '@kbn/i18n/react';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
import { search } from '../../../../../plugins/data/public';
-import { NumberInputOption, SwitchOption, SelectOption } from '../../../vis_type_vislib/public';
+import {
+ SwitchOption,
+ SelectOption,
+ NumberInputOption,
+} from '../../../../../plugins/charts/public';
import { TableVisParams } from '../types';
import { totalAggregations } from './utils';
diff --git a/src/legacy/core_plugins/vis_type_tagcloud/public/components/tag_cloud_options.tsx b/src/legacy/core_plugins/vis_type_tagcloud/public/components/tag_cloud_options.tsx
index 80e4e1de7ddab..7a64549edd747 100644
--- a/src/legacy/core_plugins/vis_type_tagcloud/public/components/tag_cloud_options.tsx
+++ b/src/legacy/core_plugins/vis_type_tagcloud/public/components/tag_cloud_options.tsx
@@ -22,7 +22,7 @@ import { EuiPanel } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
import { ValidatedDualRange } from '../../../../../../src/plugins/kibana_react/public';
-import { SelectOption, SwitchOption } from '../../../vis_type_vislib/public';
+import { SelectOption, SwitchOption } from '../../../../../plugins/charts/public';
import { TagCloudVisParams } from '../types';
function TagCloudOptions({ stateParams, setValue, vis }: VisOptionsProps) {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/area.ts b/src/legacy/core_plugins/vis_type_vislib/public/area.ts
index 68decacaaa040..8a196da64fc4b 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/area.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/area.ts
@@ -33,13 +33,13 @@ import {
AxisTypes,
ScaleTypes,
AxisModes,
- Rotates,
ThresholdLineStyles,
getConfigCollections,
} from './utils/collections';
import { getAreaOptionTabs, countLabel } from './utils/common_config';
import { createVislibVisController } from './vis_controller';
import { VisTypeVislibDependencies } from './plugin';
+import { Rotates } from '../../../../plugins/charts/public';
export const createAreaVisTypeDefinition = (deps: VisTypeVislibDependencies) => ({
name: 'area',
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/index.ts b/src/legacy/core_plugins/vis_type_vislib/public/components/common/index.ts
index 05972d101f576..f0bec3167cb7c 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/common/index.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/common/index.ts
@@ -17,13 +17,5 @@
* under the License.
*/
-export { BasicOptions } from './basic_options';
-export { ColorRanges } from './color_ranges';
-export { ColorSchemaOptions, SetColorSchemaOptionsValue } from './color_schema';
-export { NumberInputOption } from './number_input';
-export { RangeOption } from './range';
-export { SelectOption } from './select';
-export { SwitchOption } from './switch';
-export { TextInputOption } from './text_input';
export { TruncateLabelsOption } from './truncate_labels';
export { ValidationWrapper, ValidationVisOptionsProps } from './validation_wrapper';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/labels_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/labels_panel.tsx
index b9bae1cad35e7..3fca9dc8adc08 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/labels_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/labels_panel.tsx
@@ -22,7 +22,7 @@ import { EuiPanel, EuiSpacer, EuiTitle } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
-import { SwitchOption, TextInputOption } from '../../common';
+import { SwitchOption, TextInputOption } from '../../../../../../../plugins/charts/public';
import { GaugeOptionsInternalProps } from '.';
function LabelsPanel({ stateParams, setValue, setGaugeValue }: GaugeOptionsInternalProps) {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/ranges_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/ranges_panel.tsx
index 7de64e5888096..433cc4edeb47b 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/ranges_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/ranges_panel.tsx
@@ -22,12 +22,16 @@ import { EuiPanel, EuiSpacer, EuiTitle } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
-import { ColorRanges, ColorSchemaOptions, SwitchOption } from '../../common';
+import {
+ ColorRanges,
+ ColorSchemaOptions,
+ ColorSchemaParams,
+ SetColorRangeValue,
+ SwitchOption,
+ ColorSchemas,
+} from '../../../../../../../plugins/charts/public';
import { GaugeOptionsInternalProps } from '.';
-import { ColorSchemaVislibParams } from '../../../types';
import { Gauge } from '../../../gauge';
-import { SetColorRangeValue } from '../../common/color_ranges';
-import { ColorSchemas } from '../../../../../../../plugins/charts/public';
function RangesPanel({
setGaugeValue,
@@ -39,7 +43,7 @@ function RangesPanel({
vis,
}: GaugeOptionsInternalProps) {
const setColorSchemaOptions = useCallback(
- (paramName: T, value: ColorSchemaVislibParams[T]) => {
+ (paramName: T, value: ColorSchemaParams[T]) => {
setGaugeValue(paramName, value as Gauge[T]);
// set outline if color schema is changed to greys
// if outline wasn't set explicitly yet
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/style_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/style_panel.tsx
index 9254c3c18347c..48711de7d171a 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/style_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/gauge/style_panel.tsx
@@ -22,7 +22,7 @@ import { EuiPanel, EuiSpacer, EuiTitle } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
-import { SelectOption } from '../../common';
+import { SelectOption } from '../../../../../../../plugins/charts/public';
import { GaugeOptionsInternalProps } from '.';
import { AggGroupNames } from '../../../../../../../plugins/data/public';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/index.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/index.tsx
index 715b5902b69da..dc207ad89286f 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/index.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/index.tsx
@@ -31,12 +31,12 @@ import {
NumberInputOption,
SelectOption,
SwitchOption,
-} from '../../common';
-import { SetColorSchemaOptionsValue } from '../../common/color_schema';
+ SetColorSchemaOptionsValue,
+ SetColorRangeValue,
+} from '../../../../../../../plugins/charts/public';
import { HeatmapVisParams } from '../../../heatmap';
import { ValueAxis } from '../../../types';
import { LabelsPanel } from './labels_panel';
-import { SetColorRangeValue } from '../../common/color_ranges';
function HeatmapOptions(props: VisOptionsProps) {
const { stateParams, vis, uiState, setValue, setValidity, setTouched } = props;
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/labels_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/labels_panel.tsx
index 38811bd836459..3d1629740df2c 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/labels_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/heatmap/labels_panel.tsx
@@ -26,7 +26,7 @@ import { FormattedMessage } from '@kbn/i18n/react';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
import { ValueAxis } from '../../../types';
import { HeatmapVisParams } from '../../../heatmap';
-import { SwitchOption } from '../../common';
+import { SwitchOption } from '../../../../../../../plugins/charts/public';
const VERTICAL_ROTATION = 270;
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/category_axis_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/category_axis_panel.tsx
index 915885388640c..246c20a14807c 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/category_axis_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/category_axis_panel.tsx
@@ -25,7 +25,7 @@ import { FormattedMessage } from '@kbn/i18n/react';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
import { Axis } from '../../../types';
-import { SelectOption, SwitchOption } from '../../common';
+import { SelectOption, SwitchOption } from '../../../../../../../plugins/charts/public';
import { LabelOptions, SetAxisLabel } from './label_options';
import { Positions } from '../../../utils/collections';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/chart_options.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/chart_options.tsx
index ec7a325ba43d1..89aab3a19c589 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/chart_options.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/chart_options.tsx
@@ -25,7 +25,7 @@ import { EuiFlexGroup, EuiFlexItem, EuiSpacer } from '@elastic/eui';
import { Vis } from '../../../../../../../plugins/visualizations/public';
import { SeriesParam, ValueAxis } from '../../../types';
import { ChartTypes } from '../../../utils/collections';
-import { SelectOption } from '../../common';
+import { SelectOption } from '../../../../../../../plugins/charts/public';
import { LineOptions } from './line_options';
import { SetParamByIndex, ChangeValueAxis } from './';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/custom_extents_options.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/custom_extents_options.tsx
index 53b2ffa55a941..a3a97df9e35ae 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/custom_extents_options.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/custom_extents_options.tsx
@@ -21,7 +21,7 @@ import React, { useCallback, useEffect } from 'react';
import { i18n } from '@kbn/i18n';
import { ValueAxis } from '../../../types';
-import { NumberInputOption, SwitchOption } from '../../common';
+import { NumberInputOption, SwitchOption } from '../../../../../../../plugins/charts/public';
import { YExtents } from './y_extents';
import { SetScale } from './value_axis_options';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/label_options.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/label_options.tsx
index b6b54193e9f4a..bc687e56646f6 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/label_options.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/label_options.tsx
@@ -24,8 +24,9 @@ import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
import { Axis } from '../../../types';
-import { SelectOption, SwitchOption, TruncateLabelsOption } from '../../common';
+import { TruncateLabelsOption } from '../../common';
import { getRotateOptions } from '../../../utils/collections';
+import { SelectOption, SwitchOption } from '../../../../../../../plugins/charts/public';
export type SetAxisLabel = (
paramName: T,
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.test.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.test.tsx
index 1d29d39bfcb7f..5354bc9c033e6 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.test.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.test.tsx
@@ -20,7 +20,7 @@
import React from 'react';
import { shallow } from 'enzyme';
import { LineOptions, LineOptionsParams } from './line_options';
-import { NumberInputOption } from '../../common';
+import { NumberInputOption } from '../../../../../../../plugins/charts/public';
import { seriesParam, vis } from './mocks';
jest.mock('ui/new_platform');
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.tsx
index 01a69a6fac70b..76f95bd93caf8 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/line_options.tsx
@@ -24,7 +24,11 @@ import { EuiFlexGroup, EuiFlexItem, EuiSpacer } from '@elastic/eui';
import { Vis } from '../../../../../../../plugins/visualizations/public';
import { SeriesParam } from '../../../types';
-import { NumberInputOption, SelectOption, SwitchOption } from '../../common';
+import {
+ NumberInputOption,
+ SelectOption,
+ SwitchOption,
+} from '../../../../../../../plugins/charts/public';
import { SetChart } from './chart_options';
export interface LineOptionsParams {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/mocks.ts b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/mocks.ts
index 0d9fa8c25a4f7..a296281375dac 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/mocks.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/mocks.ts
@@ -18,7 +18,7 @@
*/
import { Vis } from '../../../../../../../plugins/visualizations/public';
-import { Axis, ValueAxis, SeriesParam, Style } from '../../../types';
+import { Axis, ValueAxis, SeriesParam } from '../../../types';
import {
ChartTypes,
ChartModes,
@@ -31,6 +31,7 @@ import {
getPositions,
getInterpolationModes,
} from '../../../utils/collections';
+import { Style } from '../../../../../../../plugins/charts/public';
const defaultValueAxisId = 'ValueAxis-1';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.test.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.test.tsx
index 955867e66d09f..876a6917ee0b4 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.test.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.test.tsx
@@ -21,7 +21,7 @@ import React from 'react';
import { shallow } from 'enzyme';
import { ValueAxisOptions, ValueAxisOptionsParams } from './value_axis_options';
import { ValueAxis } from '../../../types';
-import { TextInputOption } from '../../common';
+import { TextInputOption } from '../../../../../../../plugins/charts/public';
import { LabelOptions } from './label_options';
import { ScaleTypes, Positions } from '../../../utils/collections';
import { valueAxis, vis } from './mocks';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.tsx
index 8f0327e78c7ab..1b89a766d0591 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/value_axis_options.tsx
@@ -24,7 +24,11 @@ import { EuiSpacer, EuiAccordion, EuiHorizontalRule } from '@elastic/eui';
import { Vis } from '../../../../../../../plugins/visualizations/public';
import { ValueAxis } from '../../../types';
import { Positions } from '../../../utils/collections';
-import { SelectOption, SwitchOption, TextInputOption } from '../../common';
+import {
+ SelectOption,
+ SwitchOption,
+ TextInputOption,
+} from '../../../../../../../plugins/charts/public';
import { LabelOptions, SetAxisLabel } from './label_options';
import { CustomExtentsOptions } from './custom_extents_options';
import { isAxisHorizontal } from './utils';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.test.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.test.tsx
index 17c47b35b20dc..b5ed475ca8e31 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.test.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.test.tsx
@@ -21,7 +21,7 @@ import React from 'react';
import { mount, shallow } from 'enzyme';
import { YExtents, YExtentsProps } from './y_extents';
import { ScaleTypes } from '../../../utils/collections';
-import { NumberInputOption } from '../../common';
+import { NumberInputOption } from '../../../../../../../plugins/charts/public';
jest.mock('ui/new_platform');
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.tsx
index c0db58a612e71..faeb6069b5126 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/metrics_axes/y_extents.tsx
@@ -23,7 +23,7 @@ import { i18n } from '@kbn/i18n';
import { Scale } from '../../../types';
import { ScaleTypes } from '../../../utils/collections';
-import { NumberInputOption } from '../../common';
+import { NumberInputOption } from '../../../../../../../plugins/charts/public';
import { SetScale } from './value_axis_options';
const rangeError = i18n.translate('visTypeVislib.controls.pointSeries.valueAxes.minErrorMessage', {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/pie.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/pie.tsx
index 4c0be456aad64..f6be9cd0bd8fe 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/pie.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/pie.tsx
@@ -23,7 +23,8 @@ import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
-import { BasicOptions, TruncateLabelsOption, SwitchOption } from '../common';
+import { TruncateLabelsOption } from '../common';
+import { BasicOptions, SwitchOption } from '../../../../../../plugins/charts/public';
import { PieVisParams } from '../../pie';
function PieOptions(props: VisOptionsProps) {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/grid_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/grid_panel.tsx
index bb2b3f8fddb49..392d180d2c5f2 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/grid_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/grid_panel.tsx
@@ -23,7 +23,7 @@ import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
import { VisOptionsProps } from 'src/plugins/vis_default_editor/public';
-import { SelectOption, SwitchOption } from '../../common';
+import { SelectOption, SwitchOption } from '../../../../../../../plugins/charts/public';
import { BasicVislibParams, ValueAxis } from '../../../types';
function GridPanel({ stateParams, setValue, hasHistogramAgg }: VisOptionsProps) {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/point_series.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/point_series.tsx
index b9872ab94bd0b..903c1917751d9 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/point_series.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/point_series.tsx
@@ -21,7 +21,8 @@ import { EuiPanel, EuiTitle, EuiSpacer } from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
-import { BasicOptions, SwitchOption, ValidationVisOptionsProps } from '../../common';
+import { ValidationVisOptionsProps } from '../../common';
+import { BasicOptions, SwitchOption } from '../../../../../../../plugins/charts/public';
import { GridPanel } from './grid_panel';
import { ThresholdPanel } from './threshold_panel';
import { BasicVislibParams } from '../../../types';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/threshold_panel.tsx b/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/threshold_panel.tsx
index 7866ad74ede7f..12f058ec7dd1f 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/threshold_panel.tsx
+++ b/src/legacy/core_plugins/vis_type_vislib/public/components/options/point_series/threshold_panel.tsx
@@ -21,8 +21,12 @@ import { EuiPanel, EuiTitle, EuiColorPicker, EuiFormRow, EuiSpacer } from '@elas
import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
-import { SelectOption, SwitchOption, ValidationVisOptionsProps } from '../../common';
-import { NumberInputOption } from '../../common/required_number_input';
+import { ValidationVisOptionsProps } from '../../common';
+import {
+ SelectOption,
+ SwitchOption,
+ RequiredNumberInputOption,
+} from '../../../../../../../plugins/charts/public';
import { BasicVislibParams } from '../../../types';
function ThresholdPanel({
@@ -73,7 +77,7 @@ function ThresholdPanel({
{stateParams.thresholdLine.show && (
<>
-
- ({
name: 'histogram',
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/horizontal_bar.ts b/src/legacy/core_plugins/vis_type_vislib/public/horizontal_bar.ts
index dc47252ccd44f..6f73271726660 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/horizontal_bar.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/horizontal_bar.ts
@@ -32,13 +32,13 @@ import {
AxisTypes,
ScaleTypes,
AxisModes,
- Rotates,
ThresholdLineStyles,
getConfigCollections,
} from './utils/collections';
import { getAreaOptionTabs, countLabel } from './utils/common_config';
import { createVislibVisController } from './vis_controller';
import { VisTypeVislibDependencies } from './plugin';
+import { Rotates } from '../../../../plugins/charts/public';
export const createHorizontalBarVisTypeDefinition = (deps: VisTypeVislibDependencies) => ({
name: 'horizontal_bar',
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/index.ts b/src/legacy/core_plugins/vis_type_vislib/public/index.ts
index 1f773c4efcb02..4d7091ffb204b 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/index.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/index.ts
@@ -24,18 +24,4 @@ export function plugin(initializerContext: PluginInitializerContext) {
return new Plugin(initializerContext);
}
-export {
- BasicOptions,
- RangeOption,
- ColorRanges,
- SelectOption,
- SetColorSchemaOptionsValue,
- ColorSchemaOptions,
- NumberInputOption,
- SwitchOption,
- TextInputOption,
-} from './components';
-
-export { ColorModes } from './utils/collections';
-
export * from './types';
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/line.ts b/src/legacy/core_plugins/vis_type_vislib/public/line.ts
index 885ab295d11e1..1f9a8d77398e6 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/line.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/line.ts
@@ -32,7 +32,6 @@ import {
AxisTypes,
ScaleTypes,
AxisModes,
- Rotates,
ThresholdLineStyles,
InterpolationModes,
getConfigCollections,
@@ -40,6 +39,7 @@ import {
import { getAreaOptionTabs, countLabel } from './utils/common_config';
import { createVislibVisController } from './vis_controller';
import { VisTypeVislibDependencies } from './plugin';
+import { Rotates } from '../../../../plugins/charts/public';
export const createLineVisTypeDefinition = (deps: VisTypeVislibDependencies) => ({
name: 'line',
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/types.ts b/src/legacy/core_plugins/vis_type_vislib/public/types.ts
index f33b42483c53e..25c6ae5439fe8 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/types.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/types.ts
@@ -25,39 +25,16 @@ import {
AxisModes,
AxisTypes,
InterpolationModes,
- Rotates,
ScaleTypes,
ThresholdLineStyles,
} from './utils/collections';
-import { ColorSchemas } from '../../../../plugins/charts/public';
+import { Labels, Style } from '../../../../plugins/charts/public';
export interface CommonVislibParams {
addTooltip: boolean;
legendPosition: Positions;
}
-export interface ColorSchemaVislibParams {
- colorSchema: ColorSchemas;
- invertColors: boolean;
-}
-
-export interface Labels {
- color?: string;
- filter?: boolean;
- overwriteColor?: boolean;
- rotate?: Rotates;
- show: boolean;
- truncate?: number | null;
-}
-
-export interface Style {
- bgFill: string;
- bgColor: boolean;
- labelColor: boolean;
- subText: string;
- fontSize: number;
-}
-
export interface Scale {
boundsMargin?: number | '';
defaultYExtents?: boolean;
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/utils/collections.ts b/src/legacy/core_plugins/vis_type_vislib/public/utils/collections.ts
index f32b765cd6e57..2024c43dd1c8b 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/utils/collections.ts
+++ b/src/legacy/core_plugins/vis_type_vislib/public/utils/collections.ts
@@ -20,7 +20,7 @@
import { i18n } from '@kbn/i18n';
import { $Values } from '@kbn/utility-types';
-import { colorSchemas } from '../../../../../plugins/charts/public';
+import { colorSchemas, Rotates } from '../../../../../plugins/charts/public';
export const Positions = Object.freeze({
RIGHT: 'right' as 'right',
@@ -203,13 +203,6 @@ const getAxisModes = () => [
},
];
-export const Rotates = Object.freeze({
- HORIZONTAL: 0,
- VERTICAL: 90,
- ANGLED: 75,
-});
-export type Rotates = $Values;
-
export const ThresholdLineStyles = Object.freeze({
FULL: 'full' as 'full',
DASHED: 'dashed' as 'dashed',
@@ -265,13 +258,6 @@ export const GaugeTypes = Object.freeze({
});
export type GaugeTypes = $Values;
-export const ColorModes = Object.freeze({
- BACKGROUND: 'Background' as 'Background',
- LABELS: 'Labels' as 'Labels',
- NONE: 'None' as 'None',
-});
-export type ColorModes = $Values;
-
const getGaugeTypes = () => [
{
text: i18n.translate('visTypeVislib.gauge.gaugeTypes.arcText', {
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/basic_options.tsx b/src/plugins/charts/public/static/components/basic_options.tsx
similarity index 90%
rename from src/legacy/core_plugins/vis_type_vislib/public/components/common/basic_options.tsx
rename to src/plugins/charts/public/static/components/basic_options.tsx
index baf3e8ecd1b28..cac4c8d70d796 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/common/basic_options.tsx
+++ b/src/plugins/charts/public/static/components/basic_options.tsx
@@ -37,7 +37,7 @@ function BasicOptions({
return (
<>
({
setValue={setValue}
/>
(
- setValidity: (paramName: ParamName, isValid: boolean) => void,
- paramName: ParamName,
- isValid: boolean
-) {
- useEffect(() => {
- setValidity(paramName, isValid);
+export const ColorModes = Object.freeze({
+ BACKGROUND: 'Background' as 'Background',
+ LABELS: 'Labels' as 'Labels',
+ NONE: 'None' as 'None',
+});
+export type ColorModes = $Values;
- return () => setValidity(paramName, true);
- }, [isValid, paramName, setValidity]);
-}
-
-export { useValidation };
+export const Rotates = Object.freeze({
+ HORIZONTAL: 0,
+ VERTICAL: 90,
+ ANGLED: 75,
+});
+export type Rotates = $Values;
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/color_ranges.tsx b/src/plugins/charts/public/static/components/color_ranges.tsx
similarity index 93%
rename from src/legacy/core_plugins/vis_type_vislib/public/components/common/color_ranges.tsx
rename to src/plugins/charts/public/static/components/color_ranges.tsx
index 84c70f10b12da..a9b05d7d91c7c 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/common/color_ranges.tsx
+++ b/src/plugins/charts/public/static/components/color_ranges.tsx
@@ -22,10 +22,7 @@ import { last } from 'lodash';
import { i18n } from '@kbn/i18n';
-import {
- RangeValues,
- RangesParamEditor,
-} from '../../../../../../plugins/vis_default_editor/public';
+import { RangeValues, RangesParamEditor } from '../../../../vis_default_editor/public';
export type SetColorRangeValue = (paramName: string, value: RangeValues[]) => void;
@@ -74,7 +71,7 @@ function ColorRanges({
return (
(
+export type SetColorSchemaOptionsValue = (
paramName: T,
- value: ColorSchemaVislibParams[T]
+ value: ColorSchemaParams[T]
) => void;
-interface ColorSchemaOptionsProps extends ColorSchemaVislibParams {
+interface ColorSchemaOptionsProps extends ColorSchemaParams {
disabled?: boolean;
colorSchemas: ColorSchema[];
uiState: VisOptionsProps['uiState'];
@@ -67,7 +67,7 @@ function ColorSchemaOptions({
}}
>
@@ -80,11 +80,11 @@ function ColorSchemaOptions({
disabled={disabled}
helpText={
showHelpText &&
- i18n.translate('visTypeVislib.controls.colorSchema.howToChangeColorsDescription', {
+ i18n.translate('charts.controls.colorSchema.howToChangeColorsDescription', {
defaultMessage: 'Individual colors can be changed in the legend.',
})
}
- label={i18n.translate('visTypeVislib.controls.colorSchema.colorSchemaLabel', {
+ label={i18n.translate('charts.controls.colorSchema.colorSchemaLabel', {
defaultMessage: 'Color schema',
})}
labelAppend={isCustomColors && resetColorsButton}
@@ -96,7 +96,7 @@ function ColorSchemaOptions({
({
const [stateValue, setStateValue] = useState(value);
const [isValidState, setIsValidState] = useState(true);
- const error = i18n.translate('visTypeVislib.controls.rangeErrorMessage', {
+ const error = i18n.translate('charts.controls.rangeErrorMessage', {
defaultMessage: 'Values must be on or between {min} and {max}',
values: { min, max },
});
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/required_number_input.tsx b/src/plugins/charts/public/static/components/required_number_input.tsx
similarity index 87%
rename from src/legacy/core_plugins/vis_type_vislib/public/components/common/required_number_input.tsx
rename to src/plugins/charts/public/static/components/required_number_input.tsx
index 7b62016c4e502..7594c775b07ad 100644
--- a/src/legacy/core_plugins/vis_type_vislib/public/components/common/required_number_input.tsx
+++ b/src/plugins/charts/public/static/components/required_number_input.tsx
@@ -17,9 +17,8 @@
* under the License.
*/
-import React, { ReactNode, useCallback, ChangeEvent } from 'react';
+import React, { ReactNode, useCallback, ChangeEvent, useEffect } from 'react';
import { EuiFormRow, EuiFieldNumber } from '@elastic/eui';
-import { useValidation } from './utils';
interface NumberInputOptionProps {
disabled?: boolean;
@@ -42,7 +41,7 @@ interface NumberInputOptionProps {
*
* @param {number} props.value Should be numeric only
*/
-function NumberInputOption({
+function RequiredNumberInputOption({
disabled,
error,
isInvalid,
@@ -57,7 +56,12 @@ function NumberInputOption({
'data-test-subj': dataTestSubj,
}: NumberInputOptionProps) {
const isValid = value !== null;
- useValidation(setValidity, paramName, isValid);
+
+ useEffect(() => {
+ setValidity(paramName, isValid);
+
+ return () => setValidity(paramName, true);
+ }, [isValid, paramName, setValidity]);
const onChange = useCallback(
(ev: ChangeEvent) =>
@@ -84,4 +88,4 @@ function NumberInputOption({
);
}
-export { NumberInputOption };
+export { RequiredNumberInputOption };
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/select.tsx b/src/plugins/charts/public/static/components/select.tsx
similarity index 100%
rename from src/legacy/core_plugins/vis_type_vislib/public/components/common/select.tsx
rename to src/plugins/charts/public/static/components/select.tsx
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/switch.tsx b/src/plugins/charts/public/static/components/switch.tsx
similarity index 100%
rename from src/legacy/core_plugins/vis_type_vislib/public/components/common/switch.tsx
rename to src/plugins/charts/public/static/components/switch.tsx
diff --git a/src/legacy/core_plugins/vis_type_vislib/public/components/common/text_input.tsx b/src/plugins/charts/public/static/components/text_input.tsx
similarity index 100%
rename from src/legacy/core_plugins/vis_type_vislib/public/components/common/text_input.tsx
rename to src/plugins/charts/public/static/components/text_input.tsx
diff --git a/src/plugins/charts/public/static/components/types.ts b/src/plugins/charts/public/static/components/types.ts
new file mode 100644
index 0000000000000..196eb60b06aec
--- /dev/null
+++ b/src/plugins/charts/public/static/components/types.ts
@@ -0,0 +1,43 @@
+/*
+ * Licensed to Elasticsearch B.V. under one or more contributor
+ * license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright
+ * ownership. Elasticsearch B.V. licenses this file to you under
+ * the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+import { ColorSchemas } from '../color_maps';
+import { Rotates } from './collections';
+
+export interface ColorSchemaParams {
+ colorSchema: ColorSchemas;
+ invertColors: boolean;
+}
+
+export interface Labels {
+ color?: string;
+ filter?: boolean;
+ overwriteColor?: boolean;
+ rotate?: Rotates;
+ show: boolean;
+ truncate?: number | null;
+}
+
+export interface Style {
+ bgFill: string;
+ bgColor: boolean;
+ labelColor: boolean;
+ subText: string;
+ fontSize: number;
+}
diff --git a/src/plugins/charts/public/static/index.ts b/src/plugins/charts/public/static/index.ts
index bee58e4f1e3e1..6fc097d05467f 100644
--- a/src/plugins/charts/public/static/index.ts
+++ b/src/plugins/charts/public/static/index.ts
@@ -18,3 +18,4 @@
*/
export * from './color_maps';
+export * from './components';
diff --git a/x-pack/plugins/translations/translations/ja-JP.json b/x-pack/plugins/translations/translations/ja-JP.json
index c07ec68e99b4f..09903c34e2e5e 100644
--- a/x-pack/plugins/translations/translations/ja-JP.json
+++ b/x-pack/plugins/translations/translations/ja-JP.json
@@ -130,6 +130,14 @@
"charts.colormaps.greysText": "グレー",
"charts.colormaps.redsText": "赤",
"charts.colormaps.yellowToRedText": "黄色から赤",
+ "charts.controls.colorRanges.errorText": "各範囲は前の範囲よりも大きくなければなりません。",
+ "charts.controls.colorSchema.colorSchemaLabel": "配色",
+ "charts.controls.colorSchema.howToChangeColorsDescription": "それぞれの色は凡例で変更できます。",
+ "charts.controls.colorSchema.resetColorsButtonLabel": "色をリセット",
+ "charts.controls.colorSchema.reverseColorSchemaLabel": "図表を反転",
+ "charts.controls.rangeErrorMessage": "値は {min} と {max} の間でなければなりません",
+ "charts.controls.vislibBasicOptions.legendPositionLabel": "凡例位置",
+ "charts.controls.vislibBasicOptions.showTooltipLabel": "ツールヒントを表示",
"common.ui.errorAutoCreateIndex.breadcrumbs.errorText": "エラー",
"common.ui.errorAutoCreateIndex.errorDescription": "Elasticsearch クラスターの {autoCreateIndexActionConfig} 設定が原因で、Kibana が保存されたオブジェクトを格納するインデックスを自動的に作成できないようです。Kibana は、保存されたオブジェクトインデックスが適切なマッピング/スキーマを使用し Kibana から Elasticsearch へのポーリングの回数を減らすための最適な手段であるため、この Elasticsearch の機能を使用します。",
"common.ui.errorAutoCreateIndex.errorDisclaimer": "申し訳ございませんが、この問題が解決されるまで Kibana で何も保存することができません。",
@@ -3831,11 +3839,6 @@
"visTypeVislib.chartTypes.areaText": "エリア",
"visTypeVislib.chartTypes.barText": "バー",
"visTypeVislib.chartTypes.lineText": "折れ線",
- "visTypeVislib.controls.colorRanges.errorText": "各範囲は前の範囲よりも大きくなければなりません。",
- "visTypeVislib.controls.colorSchema.colorSchemaLabel": "配色",
- "visTypeVislib.controls.colorSchema.howToChangeColorsDescription": "それぞれの色は凡例で変更できます。",
- "visTypeVislib.controls.colorSchema.resetColorsButtonLabel": "色をリセット",
- "visTypeVislib.controls.colorSchema.reverseColorSchemaLabel": "図表を反転",
"visTypeVislib.controls.gaugeOptions.alignmentLabel": "アラインメント",
"visTypeVislib.controls.gaugeOptions.autoExtendRangeLabel": "範囲を自動拡張",
"visTypeVislib.controls.gaugeOptions.displayWarningsLabel": "警告を表示",
@@ -3902,10 +3905,7 @@
"visTypeVislib.controls.pointSeries.valueAxes.toggleCustomExtendsAriaLabel": "カスタム範囲を切り替える",
"visTypeVislib.controls.pointSeries.valueAxes.toggleOptionsAriaLabel": "{axisName} オプションを切り替える",
"visTypeVislib.controls.pointSeries.valueAxes.yAxisTitle": "Y 軸",
- "visTypeVislib.controls.rangeErrorMessage": "値は {min} と {max} の間でなければなりません",
"visTypeVislib.controls.truncateLabel": "切り捨て",
- "visTypeVislib.controls.vislibBasicOptions.legendPositionLabel": "凡例位置",
- "visTypeVislib.controls.vislibBasicOptions.showTooltipLabel": "ツールヒントを表示",
"visTypeVislib.editors.heatmap.basicSettingsTitle": "基本設定",
"visTypeVislib.editors.heatmap.heatmapSettingsTitle": "ヒートマップ設定",
"visTypeVislib.editors.heatmap.highlightLabel": "ハイライト範囲",
diff --git a/x-pack/plugins/translations/translations/zh-CN.json b/x-pack/plugins/translations/translations/zh-CN.json
index de8aaa75632ee..cc1b7d7980a0b 100644
--- a/x-pack/plugins/translations/translations/zh-CN.json
+++ b/x-pack/plugins/translations/translations/zh-CN.json
@@ -130,6 +130,14 @@
"charts.colormaps.greysText": "灰色",
"charts.colormaps.redsText": "红色",
"charts.colormaps.yellowToRedText": "黄到红",
+ "charts.controls.colorRanges.errorText": "每个范围应大于前一范围。",
+ "charts.controls.colorSchema.colorSchemaLabel": "颜色模式",
+ "charts.controls.colorSchema.howToChangeColorsDescription": "可以更改图例中的各个颜色。",
+ "charts.controls.colorSchema.resetColorsButtonLabel": "重置颜色",
+ "charts.controls.colorSchema.reverseColorSchemaLabel": "反转模式",
+ "charts.controls.rangeErrorMessage": "值必须是在 {min} 到 {max} 的范围内",
+ "charts.controls.vislibBasicOptions.legendPositionLabel": "图例位置",
+ "charts.controls.vislibBasicOptions.showTooltipLabel": "显示工具提示",
"common.ui.errorAutoCreateIndex.breadcrumbs.errorText": "错误",
"common.ui.errorAutoCreateIndex.errorDescription": "似乎 Elasticsearch 集群的 {autoCreateIndexActionConfig} 设置使 Kibana 无法自动创建用于存储已保存对象的索引。Kibana 将使用此 Elasticsearch 功能,因为这是确保已保存对象索引使用正确映射/架构的最好方式,而且其允许 Kibana 较少地轮询 Elasticsearch。",
"common.ui.errorAutoCreateIndex.errorDisclaimer": "但是,只有解决了此问题后,您才能在 Kibana 保存内容。",
@@ -3832,11 +3840,6 @@
"visTypeVislib.chartTypes.areaText": "面积图",
"visTypeVislib.chartTypes.barText": "条形图",
"visTypeVislib.chartTypes.lineText": "折线图",
- "visTypeVislib.controls.colorRanges.errorText": "每个范围应大于前一范围。",
- "visTypeVislib.controls.colorSchema.colorSchemaLabel": "颜色模式",
- "visTypeVislib.controls.colorSchema.howToChangeColorsDescription": "可以更改图例中的各个颜色。",
- "visTypeVislib.controls.colorSchema.resetColorsButtonLabel": "重置颜色",
- "visTypeVislib.controls.colorSchema.reverseColorSchemaLabel": "反转模式",
"visTypeVislib.controls.gaugeOptions.alignmentLabel": "对齐方式",
"visTypeVislib.controls.gaugeOptions.autoExtendRangeLabel": "自动扩展范围",
"visTypeVislib.controls.gaugeOptions.displayWarningsLabel": "显示警告",
@@ -3903,10 +3906,7 @@
"visTypeVislib.controls.pointSeries.valueAxes.toggleCustomExtendsAriaLabel": "切换定制范围",
"visTypeVislib.controls.pointSeries.valueAxes.toggleOptionsAriaLabel": "切换 {axisName} 选项",
"visTypeVislib.controls.pointSeries.valueAxes.yAxisTitle": "Y 轴",
- "visTypeVislib.controls.rangeErrorMessage": "值必须是在 {min} 到 {max} 的范围内",
"visTypeVislib.controls.truncateLabel": "截断",
- "visTypeVislib.controls.vislibBasicOptions.legendPositionLabel": "图例位置",
- "visTypeVislib.controls.vislibBasicOptions.showTooltipLabel": "显示工具提示",
"visTypeVislib.editors.heatmap.basicSettingsTitle": "基本设置",
"visTypeVislib.editors.heatmap.heatmapSettingsTitle": "热图设置",
"visTypeVislib.editors.heatmap.highlightLabel": "高亮范围",
From 36acb373876f3c6ea5077caae81d3a1cad239dfd Mon Sep 17 00:00:00 2001
From: Brian Seeders
Date: Fri, 10 Apr 2020 11:18:12 -0400
Subject: [PATCH 07/24] Make uptime alert flyout test a little more resilient
(#62702)
---
.../apps/uptime/alert_flyout.ts | 40 ++++++++++---------
1 file changed, 21 insertions(+), 19 deletions(-)
diff --git a/x-pack/test/functional_with_es_ssl/apps/uptime/alert_flyout.ts b/x-pack/test/functional_with_es_ssl/apps/uptime/alert_flyout.ts
index 2a0358160da51..3e5a8c57c4c7e 100644
--- a/x-pack/test/functional_with_es_ssl/apps/uptime/alert_flyout.ts
+++ b/x-pack/test/functional_with_es_ssl/apps/uptime/alert_flyout.ts
@@ -33,7 +33,7 @@ export default ({ getPageObjects, getService }: FtrProviderContext) => {
// put the fetch code in a retry block with a timeout.
let alert: any;
await retry.tryForTime(15000, async () => {
- const apiResponse = await supertest.get('/api/alert/_find');
+ const apiResponse = await supertest.get('/api/alert/_find?search=uptime-test');
const alertsFromThisTest = apiResponse.body.data.filter(
({ name }: { name: string }) => name === 'uptime-test'
);
@@ -54,25 +54,27 @@ export default ({ getPageObjects, getService }: FtrProviderContext) => {
tags,
} = alert;
- // we're not testing the flyout's ability to associate alerts with action connectors
- expect(actions).to.eql([]);
+ try {
+ // we're not testing the flyout's ability to associate alerts with action connectors
+ expect(actions).to.eql([]);
- expect(alertTypeId).to.eql('xpack.uptime.alerts.monitorStatus');
- expect(consumer).to.eql('uptime');
- expect(interval).to.eql('11m');
- expect(tags).to.eql(['uptime', 'another']);
- expect(numTimes).to.be(3);
- expect(timerange.from).to.be('now-1h');
- expect(timerange.to).to.be('now');
- expect(locations).to.eql(['mpls']);
- expect(filters).to.eql(
- '{"bool":{"should":[{"match_phrase":{"monitor.id":"0001-up"}}],"minimum_should_match":1}}'
- );
-
- await supertest
- .delete(`/api/alert/${id}`)
- .set('kbn-xsrf', 'true')
- .expect(204);
+ expect(alertTypeId).to.eql('xpack.uptime.alerts.monitorStatus');
+ expect(consumer).to.eql('uptime');
+ expect(interval).to.eql('11m');
+ expect(tags).to.eql(['uptime', 'another']);
+ expect(numTimes).to.be(3);
+ expect(timerange.from).to.be('now-1h');
+ expect(timerange.to).to.be('now');
+ expect(locations).to.eql(['mpls']);
+ expect(filters).to.eql(
+ '{"bool":{"should":[{"match_phrase":{"monitor.id":"0001-up"}}],"minimum_should_match":1}}'
+ );
+ } finally {
+ await supertest
+ .delete(`/api/alert/${id}`)
+ .set('kbn-xsrf', 'true')
+ .expect(204);
+ }
});
});
};
From 55a3cc45835ccf267c06c2d0d62000027d2bf006 Mon Sep 17 00:00:00 2001
From: Steph Milovic
Date: Fri, 10 Apr 2020 09:55:38 -0600
Subject: [PATCH 08/24] [SIEM] [Cases] Unit tests for case UI components
(#63005)
---
.../components/filter_popover/index.tsx | 1 +
.../components/header_page/editable_title.tsx | 9 +-
.../siem/public/containers/case/api.ts | 4 +-
.../case/configure/use_configure.tsx | 4 +-
.../plugins/siem/public/pages/case/case.tsx | 9 +-
.../siem/public/pages/case/case_details.tsx | 6 +-
.../case/components/__mock__/case_data.tsx | 226 ++++++++++++
.../pages/case/components/__mock__/form.ts | 37 ++
.../pages/case/components/__mock__/router.ts | 39 +++
.../components/add_comment/index.test.tsx | 144 ++++++++
.../case/components/add_comment/index.tsx | 6 +-
.../components/all_cases/__mock__/index.tsx | 115 ------
.../components/all_cases/columns.test.tsx | 48 +++
.../case/components/all_cases/columns.tsx | 14 +-
.../case/components/all_cases/index.test.tsx | 66 +++-
.../all_cases/table_filters.test.tsx | 121 +++++++
.../components/all_cases/table_filters.tsx | 7 +-
.../case/components/all_cases/translations.ts | 4 +
.../pages/case/components/callout/helpers.tsx | 4 +-
.../case/components/callout/index.test.tsx | 71 ++++
.../pages/case/components/callout/index.tsx | 10 +-
.../case/components/case_status/index.tsx | 2 +-
.../components/case_view/__mock__/index.tsx | 93 -----
.../components/case_view/actions.test.tsx | 10 +-
.../case/components/case_view/actions.tsx | 1 -
.../case/components/case_view/index.test.tsx | 320 +++++++++++++----
.../pages/case/components/case_view/index.tsx | 8 +-
.../case/components/create/index.test.tsx | 121 +++++++
.../pages/case/components/create/index.tsx | 7 +-
.../case/components/tag_list/index.test.tsx | 138 ++++++++
.../pages/case/components/tag_list/index.tsx | 17 +-
.../use_push_to_service/index.test.tsx | 192 ++++++++++
.../components/use_push_to_service/index.tsx | 5 +-
.../user_action_tree/helpers.test.tsx | 143 ++++++++
.../components/user_action_tree/helpers.tsx | 12 +-
.../user_action_tree/index.test.tsx | 331 ++++++++++++++++++
.../components/user_action_tree/index.tsx | 10 +-
.../user_action_tree/user_action_item.tsx | 15 +-
.../user_action_tree/user_action_markdown.tsx | 20 +-
.../user_action_title.test.tsx | 57 +++
.../user_action_tree/user_action_title.tsx | 16 +-
.../siem/public/pages/case/translations.ts | 4 +
.../scripts/generate_case_and_comment_data.sh | 6 +-
.../case/server/scripts/generate_case_data.sh | 4 +-
44 files changed, 2117 insertions(+), 360 deletions(-)
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/case_data.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/form.ts
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/router.ts
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.test.tsx
delete mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/__mock__/index.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/columns.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.test.tsx
delete mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/case_view/__mock__/index.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/create/index.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/tag_list/index.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/use_push_to_service/index.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/user_action_tree/helpers.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/user_action_tree/index.test.tsx
create mode 100644 x-pack/legacy/plugins/siem/public/pages/case/components/user_action_tree/user_action_title.test.tsx
diff --git a/x-pack/legacy/plugins/siem/public/components/filter_popover/index.tsx b/x-pack/legacy/plugins/siem/public/components/filter_popover/index.tsx
index 3c01ec18a879f..fca6396a53745 100644
--- a/x-pack/legacy/plugins/siem/public/components/filter_popover/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/components/filter_popover/index.tsx
@@ -89,6 +89,7 @@ export const FilterPopoverComponent = ({
{options.map((option, index) => (
diff --git a/x-pack/legacy/plugins/siem/public/components/header_page/editable_title.tsx b/x-pack/legacy/plugins/siem/public/components/header_page/editable_title.tsx
index 165be00384779..0c6f7258d09dc 100644
--- a/x-pack/legacy/plugins/siem/public/components/header_page/editable_title.tsx
+++ b/x-pack/legacy/plugins/siem/public/components/header_page/editable_title.tsx
@@ -60,12 +60,9 @@ const EditableTitleComponent: React.FC = ({
}, [changedTitle, title]);
const handleOnChange = useCallback(
- (e: ChangeEvent) => {
- onTitleChange(e.target.value);
- },
- [onTitleChange]
+ (e: ChangeEvent) => onTitleChange(e.target.value),
+ []
);
-
return editMode ? (
@@ -107,7 +104,7 @@ const EditableTitleComponent: React.FC = ({
- {isLoading && }
+ {isLoading && }
{!isLoading && (
(decodeCaseResponse(response));
};
-export const deleteCases = async (caseIds: string[], signal: AbortSignal): Promise => {
+export const deleteCases = async (caseIds: string[], signal: AbortSignal): Promise => {
const response = await KibanaServices.get().http.fetch(CASES_URL, {
method: 'DELETE',
query: { ids: JSON.stringify(caseIds) },
signal,
});
- return response === 'true' ? true : false;
+ return response;
};
export const pushCase = async (
diff --git a/x-pack/legacy/plugins/siem/public/containers/case/configure/use_configure.tsx b/x-pack/legacy/plugins/siem/public/containers/case/configure/use_configure.tsx
index 7f57149d4e56d..1c03a09a8c2ea 100644
--- a/x-pack/legacy/plugins/siem/public/containers/case/configure/use_configure.tsx
+++ b/x-pack/legacy/plugins/siem/public/containers/case/configure/use_configure.tsx
@@ -55,7 +55,6 @@ export const useCaseConfigure = ({
setLoading(true);
const res = await getCaseConfigure({ signal: abortCtrl.signal });
if (!didCancel) {
- setLoading(false);
if (res != null) {
setConnector(res.connectorId, res.connectorName);
if (setClosureType != null) {
@@ -73,6 +72,7 @@ export const useCaseConfigure = ({
}
}
}
+ setLoading(false);
}
} catch (error) {
if (!didCancel) {
@@ -117,7 +117,6 @@ export const useCaseConfigure = ({
abortCtrl.signal
);
if (!didCancel) {
- setPersistLoading(false);
setConnector(res.connectorId);
if (setClosureType) {
setClosureType(res.closureType);
@@ -131,6 +130,7 @@ export const useCaseConfigure = ({
}
displaySuccessToast(i18n.SUCCESS_CONFIGURE, dispatchToaster);
+ setPersistLoading(false);
}
} catch (error) {
if (!didCancel) {
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/case.tsx b/x-pack/legacy/plugins/siem/public/pages/case/case.tsx
index 2ae35796387b8..aefb0a93366b8 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/case.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/case.tsx
@@ -11,11 +11,9 @@ import { useGetUserSavedObjectPermissions } from '../../lib/kibana';
import { SpyRoute } from '../../utils/route/spy_routes';
import { AllCases } from './components/all_cases';
-import { getSavedObjectReadOnly, CaseCallOut } from './components/callout';
+import { savedObjectReadOnly, CaseCallOut } from './components/callout';
import { CaseSavedObjectNoPermissions } from './saved_object_no_permissions';
-const infoReadSavedObject = getSavedObjectReadOnly();
-
export const CasesPage = React.memo(() => {
const userPermissions = useGetUserSavedObjectPermissions();
@@ -24,10 +22,11 @@ export const CasesPage = React.memo(() => {
{userPermissions != null && !userPermissions?.crud && userPermissions?.read && (
)}
+
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/case_details.tsx b/x-pack/legacy/plugins/siem/public/pages/case/case_details.tsx
index cbc7bbc62fbf9..4bb8afa7f8d42 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/case_details.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/case_details.tsx
@@ -13,9 +13,7 @@ import { SpyRoute } from '../../utils/route/spy_routes';
import { getCaseUrl } from '../../components/link_to';
import { navTabs } from '../home/home_navigations';
import { CaseView } from './components/case_view';
-import { getSavedObjectReadOnly, CaseCallOut } from './components/callout';
-
-const infoReadSavedObject = getSavedObjectReadOnly();
+import { savedObjectReadOnly, CaseCallOut } from './components/callout';
export const CaseDetailsPage = React.memo(() => {
const userPermissions = useGetUserSavedObjectPermissions();
@@ -29,7 +27,7 @@ export const CaseDetailsPage = React.memo(() => {
return caseId != null ? (
<>
{userPermissions != null && !userPermissions?.crud && userPermissions?.read && (
-
+
)}
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/case_data.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/case_data.tsx
new file mode 100644
index 0000000000000..64c6276fc1be2
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/case_data.tsx
@@ -0,0 +1,226 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+
+import { CaseProps } from '../case_view';
+import { Case, Comment, SortFieldCase } from '../../../../containers/case/types';
+import { UseGetCasesState } from '../../../../containers/case/use_get_cases';
+import { UserAction, UserActionField } from '../../../../../../../../plugins/case/common/api/cases';
+
+const updateCase = jest.fn();
+const fetchCase = jest.fn();
+
+const basicCaseId = 'basic-case-id';
+const basicCommentId = 'basic-comment-id';
+const basicCreatedAt = '2020-02-20T23:06:33.798Z';
+const elasticUser = {
+ fullName: 'Leslie Knope',
+ username: 'lknope',
+ email: 'leslie.knope@elastic.co',
+};
+
+export const basicComment: Comment = {
+ comment: 'Solve this fast!',
+ id: basicCommentId,
+ createdAt: basicCreatedAt,
+ createdBy: elasticUser,
+ pushedAt: null,
+ pushedBy: null,
+ updatedAt: '2020-02-20T23:06:33.798Z',
+ updatedBy: {
+ username: 'elastic',
+ },
+ version: 'WzQ3LDFc',
+};
+
+export const basicCase: Case = {
+ closedAt: null,
+ closedBy: null,
+ id: basicCaseId,
+ comments: [basicComment],
+ createdAt: '2020-02-13T19:44:23.627Z',
+ createdBy: elasticUser,
+ description: 'Security banana Issue',
+ externalService: null,
+ status: 'open',
+ tags: ['defacement'],
+ title: 'Another horrible breach!!',
+ totalComment: 1,
+ updatedAt: '2020-02-19T15:02:57.995Z',
+ updatedBy: {
+ username: 'elastic',
+ },
+ version: 'WzQ3LDFd',
+};
+
+export const caseProps: CaseProps = {
+ caseId: basicCaseId,
+ userCanCrud: true,
+ caseData: basicCase,
+ fetchCase,
+ updateCase,
+};
+
+export const caseClosedProps: CaseProps = {
+ ...caseProps,
+ caseData: {
+ ...caseProps.caseData,
+ closedAt: '2020-02-20T23:06:33.798Z',
+ closedBy: {
+ username: 'elastic',
+ },
+ status: 'closed',
+ },
+};
+
+export const basicCaseClosed: Case = {
+ ...caseClosedProps.caseData,
+};
+
+const basicAction = {
+ actionAt: basicCreatedAt,
+ actionBy: elasticUser,
+ oldValue: null,
+ newValue: 'what a cool value',
+ caseId: basicCaseId,
+ commentId: null,
+};
+export const caseUserActions = [
+ {
+ ...basicAction,
+ actionBy: elasticUser,
+ actionField: ['comment'],
+ action: 'create',
+ actionId: 'tt',
+ },
+];
+
+export const useGetCasesMockState: UseGetCasesState = {
+ data: {
+ countClosedCases: 0,
+ countOpenCases: 5,
+ cases: [
+ basicCase,
+ {
+ closedAt: null,
+ closedBy: null,
+ id: '362a5c10-4e99-11ea-9290-35d05cb55c15',
+ createdAt: '2020-02-13T19:44:13.328Z',
+ createdBy: { username: 'elastic' },
+ comments: [],
+ description: 'Security banana Issue',
+ externalService: {
+ pushedAt: '2020-02-13T19:45:01.901Z',
+ pushedBy: 'elastic',
+ connectorId: 'string',
+ connectorName: 'string',
+ externalId: 'string',
+ externalTitle: 'string',
+ externalUrl: 'string',
+ },
+ status: 'open',
+ tags: ['phishing'],
+ title: 'Bad email',
+ totalComment: 0,
+ updatedAt: '2020-02-13T15:45:01.901Z',
+ updatedBy: { username: 'elastic' },
+ version: 'WzQ3LDFd',
+ },
+ {
+ closedAt: null,
+ closedBy: null,
+ id: '34f8b9e0-4e99-11ea-9290-35d05cb55c15',
+ createdAt: '2020-02-13T19:44:11.328Z',
+ createdBy: { username: 'elastic' },
+ comments: [],
+ description: 'Security banana Issue',
+ externalService: {
+ pushedAt: '2020-02-13T19:45:01.901Z',
+ pushedBy: 'elastic',
+ connectorId: 'string',
+ connectorName: 'string',
+ externalId: 'string',
+ externalTitle: 'string',
+ externalUrl: 'string',
+ },
+ status: 'open',
+ tags: ['phishing'],
+ title: 'Bad email',
+ totalComment: 0,
+ updatedAt: '2020-02-14T19:45:01.901Z',
+ updatedBy: { username: 'elastic' },
+ version: 'WzQ3LDFd',
+ },
+ {
+ closedAt: '2020-02-13T19:44:13.328Z',
+ closedBy: { username: 'elastic' },
+ id: '31890e90-4e99-11ea-9290-35d05cb55c15',
+ createdAt: '2020-02-13T19:44:05.563Z',
+ createdBy: { username: 'elastic' },
+ comments: [],
+ description: 'Security banana Issue',
+ externalService: null,
+ status: 'closed',
+ tags: ['phishing'],
+ title: 'Uh oh',
+ totalComment: 0,
+ updatedAt: null,
+ updatedBy: null,
+ version: 'WzQ3LDFd',
+ },
+ {
+ closedAt: null,
+ closedBy: null,
+ id: '2f5b3210-4e99-11ea-9290-35d05cb55c15',
+ createdAt: '2020-02-13T19:44:01.901Z',
+ createdBy: { username: 'elastic' },
+ comments: [],
+ description: 'Security banana Issue',
+ externalService: null,
+ status: 'open',
+ tags: ['phishing'],
+ title: 'Uh oh',
+ totalComment: 0,
+ updatedAt: null,
+ updatedBy: null,
+ version: 'WzQ3LDFd',
+ },
+ ],
+ page: 1,
+ perPage: 5,
+ total: 10,
+ },
+ loading: [],
+ selectedCases: [],
+ isError: false,
+ queryParams: {
+ page: 1,
+ perPage: 5,
+ sortField: SortFieldCase.createdAt,
+ sortOrder: 'desc',
+ },
+ filterOptions: { search: '', reporters: [], tags: [], status: 'open' },
+};
+
+const basicPush = {
+ connector_id: 'connector_id',
+ connector_name: 'connector name',
+ external_id: 'external_id',
+ external_title: 'external title',
+ external_url: 'basicPush.com',
+ pushed_at: basicCreatedAt,
+ pushed_by: elasticUser,
+};
+export const getUserAction = (af: UserActionField, a: UserAction) => ({
+ ...basicAction,
+ actionId: `${af[0]}-${a}`,
+ actionField: af,
+ action: a,
+ commentId: af[0] === 'comment' ? basicCommentId : null,
+ newValue:
+ a === 'push-to-service' && af[0] === 'pushed'
+ ? JSON.stringify(basicPush)
+ : basicAction.newValue,
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/form.ts b/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/form.ts
new file mode 100644
index 0000000000000..9d2ac29bc47d7
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/form.ts
@@ -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;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+export const mockFormHook = {
+ isSubmitted: false,
+ isSubmitting: false,
+ isValid: true,
+ submit: jest.fn(),
+ subscribe: jest.fn(),
+ setFieldValue: jest.fn(),
+ setFieldErrors: jest.fn(),
+ getFields: jest.fn(),
+ getFormData: jest.fn(),
+ getFieldDefaultValue: jest.fn(),
+ /* Returns a list of all errors in the form */
+ getErrors: jest.fn(),
+ reset: jest.fn(),
+ __options: {},
+ __formData$: {},
+ __addField: jest.fn(),
+ __removeField: jest.fn(),
+ __validateFields: jest.fn(),
+ __updateFormDataAt: jest.fn(),
+ __readFieldConfigFromSchema: jest.fn(),
+};
+// eslint-disable-next-line @typescript-eslint/no-explicit-any
+export const getFormMock = (sampleData: any) => ({
+ ...mockFormHook,
+ submit: () =>
+ Promise.resolve({
+ data: sampleData,
+ isValid: true,
+ }),
+ getFormData: () => sampleData,
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/router.ts b/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/router.ts
new file mode 100644
index 0000000000000..a20ab00852a36
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/__mock__/router.ts
@@ -0,0 +1,39 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+
+import { Router } from 'react-router-dom';
+// eslint-disable-next-line @kbn/eslint/module_migration
+import routeData from 'react-router';
+type Action = 'PUSH' | 'POP' | 'REPLACE';
+const pop: Action = 'POP';
+const location = {
+ pathname: '/network',
+ search: '',
+ state: '',
+ hash: '',
+};
+export const mockHistory = {
+ length: 2,
+ location,
+ action: pop,
+ push: jest.fn(),
+ replace: jest.fn(),
+ go: jest.fn(),
+ goBack: jest.fn(),
+ goForward: jest.fn(),
+ block: jest.fn(),
+ createHref: jest.fn(),
+ listen: jest.fn(),
+};
+
+export const mockLocation = {
+ pathname: '/welcome',
+ hash: '',
+ search: '',
+ state: '',
+};
+
+export { Router, routeData };
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.test.tsx
new file mode 100644
index 0000000000000..74f6411f17fa0
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.test.tsx
@@ -0,0 +1,144 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { AddComment } from './';
+import { TestProviders } from '../../../../mock';
+import { getFormMock } from '../__mock__/form';
+import { Router, routeData, mockHistory, mockLocation } from '../__mock__/router';
+
+import { useInsertTimeline } from '../../../../components/timeline/insert_timeline_popover/use_insert_timeline';
+import { usePostComment } from '../../../../containers/case/use_post_comment';
+import { useForm } from '../../../../../../../../../src/plugins/es_ui_shared/static/forms/hook_form_lib/hooks/use_form';
+import { wait } from '../../../../lib/helpers';
+jest.mock(
+ '../../../../../../../../../src/plugins/es_ui_shared/static/forms/hook_form_lib/hooks/use_form'
+);
+jest.mock('../../../../components/timeline/insert_timeline_popover/use_insert_timeline');
+jest.mock('../../../../containers/case/use_post_comment');
+
+export const useFormMock = useForm as jest.Mock;
+
+const useInsertTimelineMock = useInsertTimeline as jest.Mock;
+const usePostCommentMock = usePostComment as jest.Mock;
+
+const onCommentSaving = jest.fn();
+const onCommentPosted = jest.fn();
+const postComment = jest.fn();
+const handleCursorChange = jest.fn();
+const handleOnTimelineChange = jest.fn();
+
+const addCommentProps = {
+ caseId: '1234',
+ disabled: false,
+ insertQuote: null,
+ onCommentSaving,
+ onCommentPosted,
+ showLoading: false,
+};
+
+const defaultInsertTimeline = {
+ cursorPosition: {
+ start: 0,
+ end: 0,
+ },
+ handleCursorChange,
+ handleOnTimelineChange,
+};
+
+const defaultPostCommment = {
+ isLoading: false,
+ isError: false,
+ postComment,
+};
+const sampleData = {
+ comment: 'what a cool comment',
+};
+describe('AddComment ', () => {
+ const formHookMock = getFormMock(sampleData);
+
+ beforeEach(() => {
+ jest.resetAllMocks();
+ useInsertTimelineMock.mockImplementation(() => defaultInsertTimeline);
+ usePostCommentMock.mockImplementation(() => defaultPostCommment);
+ useFormMock.mockImplementation(() => ({ form: formHookMock }));
+ jest.spyOn(routeData, 'useLocation').mockReturnValue(mockLocation);
+ });
+
+ it('should post comment on submit click', async () => {
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(wrapper.find(`[data-test-subj="add-comment"]`).exists()).toBeTruthy();
+ expect(wrapper.find(`[data-test-subj="loading-spinner"]`).exists()).toBeFalsy();
+
+ wrapper
+ .find(`[data-test-subj="submit-comment"]`)
+ .first()
+ .simulate('click');
+ await wait();
+ expect(onCommentSaving).toBeCalled();
+ expect(postComment).toBeCalledWith(sampleData, onCommentPosted);
+ expect(formHookMock.reset).toBeCalled();
+ });
+
+ it('should render spinner and disable submit when loading', () => {
+ usePostCommentMock.mockImplementation(() => ({ ...defaultPostCommment, isLoading: true }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(wrapper.find(`[data-test-subj="loading-spinner"]`).exists()).toBeTruthy();
+ expect(
+ wrapper
+ .find(`[data-test-subj="submit-comment"]`)
+ .first()
+ .prop('isDisabled')
+ ).toBeTruthy();
+ });
+
+ it('should disable submit button when disabled prop passed', () => {
+ usePostCommentMock.mockImplementation(() => ({ ...defaultPostCommment, isLoading: true }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(
+ wrapper
+ .find(`[data-test-subj="submit-comment"]`)
+ .first()
+ .prop('isDisabled')
+ ).toBeTruthy();
+ });
+
+ it('should insert a quote if one is available', () => {
+ const sampleQuote = 'what a cool quote';
+ mount(
+
+
+
+
+
+ );
+
+ expect(formHookMock.setFieldValue).toBeCalledWith(
+ 'comment',
+ `${sampleData.comment}\n\n${sampleQuote}`
+ );
+ });
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.tsx
index ecc57c50e28eb..eaba708948a99 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/add_comment/index.tsx
@@ -71,10 +71,9 @@ export const AddComment = React.memo(
form.reset();
}
}, [form, onCommentPosted, onCommentSaving]);
-
return (
);
}, [theCase]);
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/index.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/index.test.tsx
index a6da45a8c5bb1..f65736e7cd109 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/index.test.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/index.test.tsx
@@ -9,11 +9,15 @@ import { mount } from 'enzyme';
import moment from 'moment-timezone';
import { AllCases } from './';
import { TestProviders } from '../../../../mock';
-import { useGetCasesMockState } from './__mock__';
+import { useGetCasesMockState } from '../__mock__/case_data';
+import * as i18n from './translations';
+
+import { getEmptyTagValue } from '../../../../components/empty_value';
import { useDeleteCases } from '../../../../containers/case/use_delete_cases';
import { useGetCases } from '../../../../containers/case/use_get_cases';
import { useGetCasesStatus } from '../../../../containers/case/use_get_cases_status';
import { useUpdateCases } from '../../../../containers/case/use_bulk_update_case';
+import { getCasesColumns } from './columns';
jest.mock('../../../../containers/case/use_bulk_update_case');
jest.mock('../../../../containers/case/use_delete_cases');
jest.mock('../../../../containers/case/use_get_cases');
@@ -35,6 +39,7 @@ describe('AllCases', () => {
const setSelectedCases = jest.fn();
const updateBulkStatus = jest.fn();
const fetchCasesStatus = jest.fn();
+ const emptyTag = getEmptyTagValue().props.children;
const defaultGetCases = {
...useGetCasesMockState,
@@ -115,7 +120,7 @@ describe('AllCases', () => {
.find(`[data-test-subj="case-table-column-createdBy"]`)
.first()
.text()
- ).toEqual(useGetCasesMockState.data.cases[0].createdBy.username);
+ ).toEqual(useGetCasesMockState.data.cases[0].createdBy.fullName);
expect(
wrapper
.find(`[data-test-subj="case-table-column-createdAt"]`)
@@ -129,6 +134,39 @@ describe('AllCases', () => {
.text()
).toEqual('Showing 10 cases');
});
+ it('should render empty fields', () => {
+ useGetCasesMock.mockImplementation(() => ({
+ ...defaultGetCases,
+ data: {
+ ...defaultGetCases.data,
+ cases: [
+ {
+ ...defaultGetCases.data.cases[0],
+ id: null,
+ createdAt: null,
+ createdBy: null,
+ tags: null,
+ title: null,
+ totalComment: null,
+ },
+ ],
+ },
+ }));
+ const wrapper = mount(
+
+
+
+ );
+ const checkIt = (columnName: string, key: number) => {
+ const column = wrapper.find('[data-test-subj="cases-table"] tbody .euiTableRowCell').at(key);
+ if (columnName === i18n.ACTIONS) {
+ return;
+ }
+ expect(column.find('.euiTableRowCell--hideForDesktop').text()).toEqual(columnName);
+ expect(column.find('span').text()).toEqual(emptyTag);
+ };
+ getCasesColumns([], 'open').map((i, key) => i.name != null && checkIt(`${i.name}`, key));
+ });
it('should tableHeaderSortButton AllCases', () => {
const wrapper = mount(
@@ -165,6 +203,30 @@ describe('AllCases', () => {
version: firstCase.version,
});
});
+ it('opens case when row action icon clicked', () => {
+ useGetCasesMock.mockImplementation(() => ({
+ ...defaultGetCases,
+ filterOptions: { ...defaultGetCases.filterOptions, status: 'closed' },
+ }));
+
+ const wrapper = mount(
+
+
+
+ );
+ wrapper
+ .find('[data-test-subj="action-open"]')
+ .first()
+ .simulate('click');
+ const firstCase = useGetCasesMockState.data.cases[0];
+ expect(dispatchUpdateCaseProperty).toBeCalledWith({
+ caseId: firstCase.id,
+ updateKey: 'status',
+ updateValue: 'open',
+ refetchCasesStatus: fetchCasesStatus,
+ version: firstCase.version,
+ });
+ });
it('Bulk delete', () => {
useGetCasesMock.mockImplementation(() => ({
...defaultGetCases,
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.test.tsx
new file mode 100644
index 0000000000000..615d052347203
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.test.tsx
@@ -0,0 +1,121 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { CasesTableFilters } from './table_filters';
+import { TestProviders } from '../../../../mock';
+
+import { useGetTags } from '../../../../containers/case/use_get_tags';
+import { useGetReporters } from '../../../../containers/case/use_get_reporters';
+import { DEFAULT_FILTER_OPTIONS } from '../../../../containers/case/use_get_cases';
+jest.mock('../../../../components/timeline/insert_timeline_popover/use_insert_timeline');
+jest.mock('../../../../containers/case/use_get_reporters');
+jest.mock('../../../../containers/case/use_get_tags');
+
+const onFilterChanged = jest.fn();
+const fetchReporters = jest.fn();
+
+const props = {
+ countClosedCases: 1234,
+ countOpenCases: 99,
+ onFilterChanged,
+ initial: DEFAULT_FILTER_OPTIONS,
+};
+describe('CasesTableFilters ', () => {
+ beforeEach(() => {
+ jest.resetAllMocks();
+ (useGetTags as jest.Mock).mockReturnValue({ tags: ['coke', 'pepsi'] });
+ (useGetReporters as jest.Mock).mockReturnValue({
+ reporters: ['casetester'],
+ respReporters: [{ username: 'casetester' }],
+ isLoading: true,
+ isError: false,
+ fetchReporters,
+ });
+ });
+ it('should render the initial case count', () => {
+ const wrapper = mount(
+
+
+
+ );
+ expect(
+ wrapper
+ .find(`[data-test-subj="open-case-count"]`)
+ .last()
+ .text()
+ ).toEqual('Open cases (99)');
+ expect(
+ wrapper
+ .find(`[data-test-subj="closed-case-count"]`)
+ .last()
+ .text()
+ ).toEqual('Closed cases (1234)');
+ });
+ it('should call onFilterChange when tags change', () => {
+ const wrapper = mount(
+
+
+
+ );
+ wrapper
+ .find(`[data-test-subj="options-filter-popover-button-Tags"]`)
+ .last()
+ .simulate('click');
+ wrapper
+ .find(`[data-test-subj="options-filter-popover-item-0"]`)
+ .last()
+ .simulate('click');
+
+ expect(onFilterChanged).toBeCalledWith({ tags: ['coke'] });
+ });
+ it('should call onFilterChange when reporters change', () => {
+ const wrapper = mount(
+
+
+
+ );
+ wrapper
+ .find(`[data-test-subj="options-filter-popover-button-Reporter"]`)
+ .last()
+ .simulate('click');
+
+ wrapper
+ .find(`[data-test-subj="options-filter-popover-item-0"]`)
+ .last()
+ .simulate('click');
+
+ expect(onFilterChanged).toBeCalledWith({ reporters: [{ username: 'casetester' }] });
+ });
+ it('should call onFilterChange when search changes', () => {
+ const wrapper = mount(
+
+
+
+ );
+
+ wrapper
+ .find(`[data-test-subj="search-cases"]`)
+ .last()
+ .simulate('keyup', { keyCode: 13, target: { value: 'My search' } });
+ expect(onFilterChanged).toBeCalledWith({ search: 'My search' });
+ });
+ it('should call onFilterChange when status toggled', () => {
+ const wrapper = mount(
+
+
+
+ );
+ wrapper
+ .find(`[data-test-subj="closed-case-count"]`)
+ .last()
+ .simulate('click');
+
+ expect(onFilterChanged).toBeCalledWith({ status: 'closed' });
+ });
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.tsx
index a344dd7891010..da477a56c0a22 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/table_filters.tsx
@@ -42,7 +42,7 @@ const CasesTableFiltersComponent = ({
onFilterChanged,
initial = defaultInitial,
}: CasesTableFiltersProps) => {
- const [selectedReporters, setselectedReporters] = useState(
+ const [selectedReporters, setSelectedReporters] = useState(
initial.reporters.map(r => r.full_name ?? r.username ?? '')
);
const [search, setSearch] = useState(initial.search);
@@ -54,7 +54,7 @@ const CasesTableFiltersComponent = ({
const handleSelectedReporters = useCallback(
newReporters => {
if (!isEqual(newReporters, selectedReporters)) {
- setselectedReporters(newReporters);
+ setSelectedReporters(newReporters);
const reportersObj = respReporters.filter(
r => newReporters.includes(r.username) || newReporters.includes(r.full_name)
);
@@ -97,6 +97,7 @@ const CasesTableFiltersComponent = ({
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/translations.ts b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/translations.ts
index 1bee96bc23fff..d3dcfa50ecfa5 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/translations.ts
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/all_cases/translations.ts
@@ -46,6 +46,10 @@ export const BULK_ACTIONS = i18n.translate('xpack.siem.case.caseTable.bulkAction
defaultMessage: 'Bulk actions',
});
+export const SERVICENOW_INCIDENT = i18n.translate('xpack.siem.case.caseTable.snIncident', {
+ defaultMessage: 'ServiceNow Incident',
+});
+
export const SEARCH_PLACEHOLDER = i18n.translate('xpack.siem.case.caseTable.searchPlaceholder', {
defaultMessage: 'e.g. case name',
});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/callout/helpers.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/callout/helpers.tsx
index 929e8640dceb6..3237104274473 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/callout/helpers.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/callout/helpers.tsx
@@ -6,7 +6,7 @@
import * as i18n from './translations';
-export const getSavedObjectReadOnly = () => ({
+export const savedObjectReadOnly = {
title: i18n.READ_ONLY_SAVED_OBJECT_TITLE,
description: i18n.READ_ONLY_SAVED_OBJECT_MSG,
-});
+};
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.test.tsx
new file mode 100644
index 0000000000000..126ea13e96af6
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.test.tsx
@@ -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;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { CaseCallOut } from './';
+
+const defaultProps = {
+ title: 'hey title',
+};
+
+describe('CaseCallOut ', () => {
+ it('Renders single message callout', () => {
+ const props = {
+ ...defaultProps,
+ message: 'we have one message',
+ };
+ const wrapper = mount();
+ expect(
+ wrapper
+ .find(`[data-test-subj="callout-message"]`)
+ .last()
+ .exists()
+ ).toBeTruthy();
+ expect(
+ wrapper
+ .find(`[data-test-subj="callout-messages"]`)
+ .last()
+ .exists()
+ ).toBeFalsy();
+ });
+ it('Renders multi message callout', () => {
+ const props = {
+ ...defaultProps,
+ messages: [
+ { ...defaultProps, description: {'we have two messages'}
},
+ { ...defaultProps, description: {'for real'}
},
+ ],
+ };
+ const wrapper = mount();
+ expect(
+ wrapper
+ .find(`[data-test-subj="callout-message"]`)
+ .last()
+ .exists()
+ ).toBeFalsy();
+ expect(
+ wrapper
+ .find(`[data-test-subj="callout-messages"]`)
+ .last()
+ .exists()
+ ).toBeTruthy();
+ });
+ it('Dismisses callout', () => {
+ const props = {
+ ...defaultProps,
+ message: 'we have one message',
+ };
+ const wrapper = mount();
+ expect(wrapper.find(`[data-test-subj="case-call-out"]`).exists()).toBeTruthy();
+ wrapper
+ .find(`[data-test-subj="callout-dismiss"]`)
+ .last()
+ .simulate('click');
+ expect(wrapper.find(`[data-test-subj="case-call-out"]`).exists()).toBeFalsy();
+ });
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.tsx
index 30a95db2d82a5..0fc93af7f318d 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/callout/index.tsx
@@ -24,10 +24,12 @@ const CaseCallOutComponent = ({ title, message, messages }: CaseCallOutProps) =>
return showCallOut ? (
<>
-
- {!isEmpty(messages) && }
- {!isEmpty(message) && {message}
}
-
+
+ {!isEmpty(messages) && (
+
+ )}
+ {!isEmpty(message) && {message}
}
+
{i18n.DISMISS_CALLOUT}
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/case_status/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/case_status/index.tsx
index 2b16dfa150d61..718eb95767f2e 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/case_status/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/case_status/index.tsx
@@ -84,7 +84,7 @@ const CaseStatusComp: React.FC = ({
-
+
{i18n.CASE_REFRESH}
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/__mock__/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/__mock__/index.tsx
deleted file mode 100644
index 0e57326707e97..0000000000000
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/__mock__/index.tsx
+++ /dev/null
@@ -1,93 +0,0 @@
-/*
- * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
- * or more contributor license agreements. Licensed under the Elastic License;
- * you may not use this file except in compliance with the Elastic License.
- */
-
-import { CaseProps } from '../index';
-import { Case } from '../../../../../containers/case/types';
-
-const updateCase = jest.fn();
-const fetchCase = jest.fn();
-
-export const caseProps: CaseProps = {
- caseId: '3c4ddcc0-4e99-11ea-9290-35d05cb55c15',
- userCanCrud: true,
- caseData: {
- closedAt: null,
- closedBy: null,
- id: '3c4ddcc0-4e99-11ea-9290-35d05cb55c15',
- comments: [
- {
- comment: 'Solve this fast!',
- id: 'a357c6a0-5435-11ea-b427-fb51a1fcb7b8',
- createdAt: '2020-02-20T23:06:33.798Z',
- createdBy: {
- fullName: 'Steph Milovic',
- username: 'smilovic',
- email: 'notmyrealemailfool@elastic.co',
- },
- pushedAt: null,
- pushedBy: null,
- updatedAt: '2020-02-20T23:06:33.798Z',
- updatedBy: {
- username: 'elastic',
- },
- version: 'WzQ3LDFd',
- },
- ],
- createdAt: '2020-02-13T19:44:23.627Z',
- createdBy: { fullName: null, email: 'testemail@elastic.co', username: 'elastic' },
- description: 'Security banana Issue',
- externalService: null,
- status: 'open',
- tags: ['defacement'],
- title: 'Another horrible breach!!',
- totalComment: 1,
- updatedAt: '2020-02-19T15:02:57.995Z',
- updatedBy: {
- username: 'elastic',
- },
- version: 'WzQ3LDFd',
- },
- fetchCase,
- updateCase,
-};
-
-export const caseClosedProps: CaseProps = {
- ...caseProps,
- caseData: {
- ...caseProps.caseData,
- closedAt: '2020-02-20T23:06:33.798Z',
- closedBy: {
- username: 'elastic',
- },
- status: 'closed',
- },
-};
-
-export const data: Case = {
- ...caseProps.caseData,
-};
-
-export const dataClosed: Case = {
- ...caseClosedProps.caseData,
-};
-
-export const caseUserActions = [
- {
- actionField: ['comment'],
- action: 'create',
- actionAt: '2020-03-20T17:10:09.814Z',
- actionBy: {
- fullName: 'Steph Milovic',
- username: 'smilovic',
- email: 'notmyrealemailfool@elastic.co',
- },
- newValue: 'Solve this fast!',
- oldValue: null,
- actionId: '3c4ddcc0-4e99-11ea-9290-35d05cb55c15',
- caseId: '9b833a50-6acd-11ea-8fad-af86b1071bd9',
- commentId: 'a357c6a0-5435-11ea-b427-fb51a1fcb7b8',
- },
-];
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.test.tsx
index 49f5f44cba271..8a25a2121104d 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.test.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.test.tsx
@@ -9,7 +9,7 @@ import { mount } from 'enzyme';
import { useDeleteCases } from '../../../../containers/case/use_delete_cases';
import { TestProviders } from '../../../../mock';
-import { data } from './__mock__';
+import { basicCase } from '../__mock__/case_data';
import { CaseViewActions } from './actions';
jest.mock('../../../../containers/case/use_delete_cases');
const useDeleteCasesMock = useDeleteCases as jest.Mock;
@@ -34,7 +34,7 @@ describe('CaseView actions', () => {
it('clicking trash toggles modal', () => {
const wrapper = mount(
-
+
);
@@ -54,12 +54,14 @@ describe('CaseView actions', () => {
}));
const wrapper = mount(
-
+
);
expect(wrapper.find('[data-test-subj="confirm-delete-case-modal"]').exists()).toBeTruthy();
wrapper.find('button[data-test-subj="confirmModalConfirmButton"]').simulate('click');
- expect(handleOnDeleteConfirm.mock.calls[0][0]).toEqual([{ id: data.id, title: data.title }]);
+ expect(handleOnDeleteConfirm.mock.calls[0][0]).toEqual([
+ { id: basicCase.id, title: basicCase.title },
+ ]);
});
});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.tsx
index 0b08b866df964..216180eb2cf0a 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/actions.tsx
@@ -40,7 +40,6 @@ const CaseViewActionsComponent: React.FC = ({ caseData, disable
),
[isDisplayConfirmDeleteModal, caseData]
);
- // TO DO refactor each of these const's into their own components
const propertyActions = useMemo(
() => [
{
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.test.tsx
index 3f5b3a3127177..3721a5a727ca5 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.test.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.test.tsx
@@ -5,56 +5,43 @@
*/
import React from 'react';
-import { Router } from 'react-router-dom';
import { mount } from 'enzyme';
-/* eslint-disable @kbn/eslint/module_migration */
-import routeData from 'react-router';
-/* eslint-enable @kbn/eslint/module_migration */
-import { CaseComponent } from './';
-import { caseProps, caseClosedProps, data, dataClosed, caseUserActions } from './__mock__';
+
+import { Router, routeData, mockHistory, mockLocation } from '../__mock__/router';
+import { CaseComponent, CaseView } from './';
+import {
+ basicCaseClosed,
+ caseClosedProps,
+ caseProps,
+ caseUserActions,
+} from '../__mock__/case_data';
import { TestProviders } from '../../../../mock';
import { useUpdateCase } from '../../../../containers/case/use_update_case';
+import { useGetCase } from '../../../../containers/case/use_get_case';
import { useGetCaseUserActions } from '../../../../containers/case/use_get_case_user_actions';
import { wait } from '../../../../lib/helpers';
import { usePushToService } from '../use_push_to_service';
jest.mock('../../../../containers/case/use_update_case');
jest.mock('../../../../containers/case/use_get_case_user_actions');
+jest.mock('../../../../containers/case/use_get_case');
jest.mock('../use_push_to_service');
const useUpdateCaseMock = useUpdateCase as jest.Mock;
const useGetCaseUserActionsMock = useGetCaseUserActions as jest.Mock;
const usePushToServiceMock = usePushToService as jest.Mock;
-type Action = 'PUSH' | 'POP' | 'REPLACE';
-const pop: Action = 'POP';
-const location = {
- pathname: '/network',
- search: '',
- state: '',
- hash: '',
-};
-const mockHistory = {
- length: 2,
- location,
- action: pop,
- push: jest.fn(),
- replace: jest.fn(),
- go: jest.fn(),
- goBack: jest.fn(),
- goForward: jest.fn(),
- block: jest.fn(),
- createHref: jest.fn(),
- listen: jest.fn(),
-};
-
-const mockLocation = {
- pathname: '/welcome',
- hash: '',
- search: '',
- state: '',
-};
describe('CaseView ', () => {
const updateCaseProperty = jest.fn();
const fetchCaseUserActions = jest.fn();
+ const fetchCase = jest.fn();
+ const updateCase = jest.fn();
+ const data = caseProps.caseData;
+ const defaultGetCase = {
+ isLoading: false,
+ isError: false,
+ data,
+ updateCase,
+ fetchCase,
+ };
/* eslint-disable no-console */
// Silence until enzyme fixed to use ReactTestUtils.act()
const originalError = console.error;
@@ -84,17 +71,23 @@ describe('CaseView ', () => {
participants: [data.createdBy],
};
- const defaultUsePushToServiceMock = {
- pushButton: <>{'Hello Button'}>,
- pushCallouts: null,
- };
-
beforeEach(() => {
jest.resetAllMocks();
useUpdateCaseMock.mockImplementation(() => defaultUpdateCaseState);
jest.spyOn(routeData, 'useLocation').mockReturnValue(mockLocation);
useGetCaseUserActionsMock.mockImplementation(() => defaultUseGetCaseUserActions);
- usePushToServiceMock.mockImplementation(() => defaultUsePushToServiceMock);
+ usePushToServiceMock.mockImplementation(({ updateCase: updateCaseMockCall }) => ({
+ pushButton: (
+
+ ),
+ pushCallouts: null,
+ }));
});
it('should render CaseComponent', async () => {
@@ -120,7 +113,7 @@ describe('CaseView ', () => {
).toEqual(data.status);
expect(
wrapper
- .find(`[data-test-subj="case-view-tag-list"] .euiBadge__text`)
+ .find(`[data-test-subj="case-view-tag-list"] [data-test-subj="case-tag"]`)
.first()
.text()
).toEqual(data.tags[0]);
@@ -139,7 +132,7 @@ describe('CaseView ', () => {
).toEqual(data.createdAt);
expect(
wrapper
- .find(`[data-test-subj="case-view-description"]`)
+ .find(`[data-test-subj="description-action"] [data-test-subj="user-action-markdown"]`)
.first()
.prop('raw')
).toEqual(data.description);
@@ -148,7 +141,7 @@ describe('CaseView ', () => {
it('should show closed indicators in header when case is closed', async () => {
useUpdateCaseMock.mockImplementation(() => ({
...defaultUpdateCaseState,
- caseData: dataClosed,
+ caseData: basicCaseClosed,
}));
const wrapper = mount(
@@ -164,13 +157,13 @@ describe('CaseView ', () => {
.find(`[data-test-subj="case-view-closedAt"]`)
.first()
.prop('value')
- ).toEqual(dataClosed.closedAt);
+ ).toEqual(basicCaseClosed.closedAt);
expect(
wrapper
.find(`[data-test-subj="case-view-status"]`)
.first()
.text()
- ).toEqual(dataClosed.status);
+ ).toEqual(basicCaseClosed.status);
});
it('should dispatch update state when button is toggled', async () => {
@@ -188,7 +181,12 @@ describe('CaseView ', () => {
expect(updateCaseProperty).toHaveBeenCalled();
});
- it('should render comments', async () => {
+ it('should display EditableTitle isLoading', () => {
+ useUpdateCaseMock.mockImplementation(() => ({
+ ...defaultUpdateCaseState,
+ isLoading: true,
+ updateKey: 'title',
+ }));
const wrapper = mount(
@@ -196,32 +194,230 @@ describe('CaseView ', () => {
);
- await wait();
expect(
wrapper
- .find(
- `div[data-test-subj="user-action-${data.comments[0].id}-avatar"] [data-test-subj="user-action-avatar"]`
- )
+ .find('[data-test-subj="editable-title-loading"]')
+ .first()
+ .exists()
+ ).toBeTruthy();
+ expect(
+ wrapper
+ .find('[data-test-subj="editable-title-edit-icon"]')
.first()
- .prop('name')
- ).toEqual(data.comments[0].createdBy.fullName);
+ .exists()
+ ).toBeFalsy();
+ });
+ it('should display Toggle Status isLoading', () => {
+ useUpdateCaseMock.mockImplementation(() => ({
+ ...defaultUpdateCaseState,
+ isLoading: true,
+ updateKey: 'status',
+ }));
+ const wrapper = mount(
+
+
+
+
+
+ );
expect(
wrapper
- .find(
- `div[data-test-subj="user-action-${data.comments[0].id}"] [data-test-subj="user-action-title"] strong`
- )
+ .find('[data-test-subj="toggle-case-status"]')
.first()
- .text()
- ).toEqual(data.comments[0].createdBy.username);
+ .prop('isLoading')
+ ).toBeTruthy();
+ });
+ it('should display description isLoading', () => {
+ useUpdateCaseMock.mockImplementation(() => ({
+ ...defaultUpdateCaseState,
+ isLoading: true,
+ updateKey: 'description',
+ }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(
+ wrapper
+ .find('[data-test-subj="description-action"] [data-test-subj="user-action-title-loading"]')
+ .first()
+ .exists()
+ ).toBeTruthy();
+ expect(
+ wrapper
+ .find('[data-test-subj="description-action"] [data-test-subj="property-actions"]')
+ .first()
+ .exists()
+ ).toBeFalsy();
+ });
+
+ it('should display tags isLoading', () => {
+ useUpdateCaseMock.mockImplementation(() => ({
+ ...defaultUpdateCaseState,
+ isLoading: true,
+ updateKey: 'tags',
+ }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(
+ wrapper
+ .find('[data-test-subj="case-view-tag-list"] [data-test-subj="tag-list-loading"]')
+ .first()
+ .exists()
+ ).toBeTruthy();
expect(
wrapper
- .find(
- `div[data-test-subj="user-action-${data.comments[0].id}"] [data-test-subj="markdown"]`
- )
+ .find('[data-test-subj="tag-list-edit"]')
.first()
- .prop('source')
- ).toEqual(data.comments[0].comment);
+ .exists()
+ ).toBeFalsy();
+ });
+
+ it('should update title', () => {
+ const wrapper = mount(
+
+
+
+
+
+ );
+ const newTitle = 'The new title';
+ wrapper
+ .find(`[data-test-subj="editable-title-edit-icon"]`)
+ .first()
+ .simulate('click');
+ wrapper.update();
+ wrapper
+ .find(`[data-test-subj="editable-title-input-field"]`)
+ .last()
+ .simulate('change', { target: { value: newTitle } });
+
+ wrapper.update();
+ wrapper
+ .find(`[data-test-subj="editable-title-submit-btn"]`)
+ .first()
+ .simulate('click');
+
+ wrapper.update();
+ const updateObject = updateCaseProperty.mock.calls[0][0];
+ expect(updateObject.updateKey).toEqual('title');
+ expect(updateObject.updateValue).toEqual(newTitle);
+ });
+
+ it('should push updates on button click', async () => {
+ useGetCaseUserActionsMock.mockImplementation(() => ({
+ ...defaultUseGetCaseUserActions,
+ hasDataToPush: true,
+ }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(
+ wrapper
+ .find('[data-test-subj="has-data-to-push-button"]')
+ .first()
+ .exists()
+ ).toBeTruthy();
+ wrapper
+ .find('[data-test-subj="mock-button"]')
+ .first()
+ .simulate('click');
+ wrapper.update();
+ await wait();
+ expect(updateCase).toBeCalledWith(caseProps.caseData);
+ expect(fetchCaseUserActions).toBeCalledWith(caseProps.caseData.id);
+ });
+
+ it('should return null if error', () => {
+ (useGetCase as jest.Mock).mockImplementation(() => ({
+ ...defaultGetCase,
+ isError: true,
+ }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(wrapper).toEqual({});
+ });
+
+ it('should return spinner if loading', () => {
+ (useGetCase as jest.Mock).mockImplementation(() => ({
+ ...defaultGetCase,
+ isLoading: true,
+ }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(wrapper.find('[data-test-subj="case-view-loading"]').exists()).toBeTruthy();
+ });
+
+ it('should return case view when data is there', () => {
+ (useGetCase as jest.Mock).mockImplementation(() => defaultGetCase);
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(wrapper.find('[data-test-subj="case-view-title"]').exists()).toBeTruthy();
+ });
+
+ it('should refresh data on refresh', () => {
+ (useGetCase as jest.Mock).mockImplementation(() => defaultGetCase);
+ const wrapper = mount(
+
+
+
+
+
+ );
+ wrapper
+ .find('[data-test-subj="case-refresh"]')
+ .first()
+ .simulate('click');
+ expect(fetchCaseUserActions).toBeCalledWith(caseProps.caseData.id);
+ expect(fetchCase).toBeCalled();
});
});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.tsx
index 947da51365d66..3cf0405f40637 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/case_view/index.tsx
@@ -271,7 +271,11 @@ export const CaseComponent = React.memo(
onChange={toggleStatusCase}
/>
- {hasDataToPush && {pushButton}}
+ {hasDataToPush && (
+
+ {pushButton}
+
+ )}
>
)}
@@ -316,7 +320,7 @@ export const CaseView = React.memo(({ caseId, userCanCrud }: Props) => {
return (
-
+
);
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.test.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.test.tsx
new file mode 100644
index 0000000000000..d480744fc932a
--- /dev/null
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.test.tsx
@@ -0,0 +1,121 @@
+/*
+ * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
+ * or more contributor license agreements. Licensed under the Elastic License;
+ * you may not use this file except in compliance with the Elastic License.
+ */
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { Create } from './';
+import { TestProviders } from '../../../../mock';
+import { getFormMock } from '../__mock__/form';
+import { Router, routeData, mockHistory, mockLocation } from '../__mock__/router';
+
+import { useInsertTimeline } from '../../../../components/timeline/insert_timeline_popover/use_insert_timeline';
+import { usePostCase } from '../../../../containers/case/use_post_case';
+jest.mock('../../../../components/timeline/insert_timeline_popover/use_insert_timeline');
+jest.mock('../../../../containers/case/use_post_case');
+import { useForm } from '../../../../../../../../../src/plugins/es_ui_shared/static/forms/hook_form_lib/hooks';
+import { wait } from '../../../../lib/helpers';
+import { SiemPageName } from '../../../home/types';
+jest.mock(
+ '../../../../../../../../../src/plugins/es_ui_shared/static/forms/hook_form_lib/hooks/use_form'
+);
+
+export const useFormMock = useForm as jest.Mock;
+
+const useInsertTimelineMock = useInsertTimeline as jest.Mock;
+const usePostCaseMock = usePostCase as jest.Mock;
+
+const postCase = jest.fn();
+const handleCursorChange = jest.fn();
+const handleOnTimelineChange = jest.fn();
+
+const defaultInsertTimeline = {
+ cursorPosition: {
+ start: 0,
+ end: 0,
+ },
+ handleCursorChange,
+ handleOnTimelineChange,
+};
+const sampleData = {
+ description: 'what a great description',
+ tags: ['coke', 'pepsi'],
+ title: 'what a cool title',
+};
+const defaultPostCase = {
+ isLoading: false,
+ isError: false,
+ caseData: null,
+ postCase,
+};
+describe('Create case', () => {
+ const formHookMock = getFormMock(sampleData);
+
+ beforeEach(() => {
+ jest.resetAllMocks();
+ useInsertTimelineMock.mockImplementation(() => defaultInsertTimeline);
+ usePostCaseMock.mockImplementation(() => defaultPostCase);
+ useFormMock.mockImplementation(() => ({ form: formHookMock }));
+ jest.spyOn(routeData, 'useLocation').mockReturnValue(mockLocation);
+ });
+
+ it('should post case on submit click', async () => {
+ const wrapper = mount(
+
+
+
+
+
+ );
+ wrapper
+ .find(`[data-test-subj="create-case-submit"]`)
+ .first()
+ .simulate('click');
+ await wait();
+ expect(postCase).toBeCalledWith(sampleData);
+ });
+
+ it('should redirect to all cases on cancel click', () => {
+ const wrapper = mount(
+
+
+
+
+
+ );
+ wrapper
+ .find(`[data-test-subj="create-case-cancel"]`)
+ .first()
+ .simulate('click');
+ expect(mockHistory.replace.mock.calls[0][0].pathname).toEqual(`/${SiemPageName.case}`);
+ });
+ it('should redirect to new case when caseData is there', () => {
+ const sampleId = '777777';
+ usePostCaseMock.mockImplementation(() => ({ ...defaultPostCase, caseData: { id: sampleId } }));
+ mount(
+
+
+
+
+
+ );
+ expect(mockHistory.replace.mock.calls[0][0].pathname).toEqual(
+ `/${SiemPageName.case}/${sampleId}`
+ );
+ });
+
+ it('should render spinner when loading', () => {
+ usePostCaseMock.mockImplementation(() => ({ ...defaultPostCase, isLoading: true }));
+ const wrapper = mount(
+
+
+
+
+
+ );
+ expect(wrapper.find(`[data-test-subj="create-case-loading-spinner"]`).exists()).toBeTruthy();
+ });
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.tsx
index 740909db408ec..53b792bb9b5eb 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/create/index.tsx
@@ -73,7 +73,7 @@ export const Create = React.memo(() => {
const handleSetIsCancel = useCallback(() => {
setIsCancel(true);
- }, [isCancel]);
+ }, []);
if (caseData != null && caseData.id) {
return ;
@@ -85,7 +85,7 @@ export const Create = React.memo(() => {
return (
- {isLoading && }
+ {isLoading && }
{
+ const sampleTags = ['coke', 'pepsi'];
+ const formHookMock = getFormMock({ tags: sampleTags });
+ beforeEach(() => {
+ jest.resetAllMocks();
+ (useForm as jest.Mock).mockImplementation(() => ({ form: formHookMock }));
+ });
+ it('Renders no tags, and then edit', () => {
+ const wrapper = mount(
+
+
+
+ );
+ expect(
+ wrapper
+ .find(`[data-test-subj="no-tags"]`)
+ .last()
+ .exists()
+ ).toBeTruthy();
+ wrapper
+ .find(`[data-test-subj="tag-list-edit-button"]`)
+ .last()
+ .simulate('click');
+ expect(
+ wrapper
+ .find(`[data-test-subj="no-tags"]`)
+ .last()
+ .exists()
+ ).toBeFalsy();
+ expect(
+ wrapper
+ .find(`[data-test-subj="edit-tags"]`)
+ .last()
+ .exists()
+ ).toBeTruthy();
+ });
+ it('Edit tag on submit', async () => {
+ const wrapper = mount(
+
+
+
+ );
+ wrapper
+ .find(`[data-test-subj="tag-list-edit-button"]`)
+ .last()
+ .simulate('click');
+ await act(async () => {
+ wrapper
+ .find(`[data-test-subj="edit-tags-submit"]`)
+ .last()
+ .simulate('click');
+ await wait();
+ expect(onSubmit).toBeCalledWith(sampleTags);
+ });
+ });
+ it('Cancels on cancel', async () => {
+ const props = {
+ ...defaultProps,
+ tags: ['pepsi'],
+ };
+ const wrapper = mount(
+
+
+
+ );
+ expect(
+ wrapper
+ .find(`[data-test-subj="case-tag"]`)
+ .last()
+ .exists()
+ ).toBeTruthy();
+ wrapper
+ .find(`[data-test-subj="tag-list-edit-button"]`)
+ .last()
+ .simulate('click');
+ await act(async () => {
+ expect(
+ wrapper
+ .find(`[data-test-subj="case-tag"]`)
+ .last()
+ .exists()
+ ).toBeFalsy();
+ wrapper
+ .find(`[data-test-subj="edit-tags-cancel"]`)
+ .last()
+ .simulate('click');
+ await wait();
+ wrapper.update();
+ expect(
+ wrapper
+ .find(`[data-test-subj="case-tag"]`)
+ .last()
+ .exists()
+ ).toBeTruthy();
+ });
+ });
+ it('Renders disabled button', () => {
+ const props = { ...defaultProps, disabled: true };
+ const wrapper = mount(
+
+
+
+ );
+ expect(
+ wrapper
+ .find(`[data-test-subj="tag-list-edit-button"]`)
+ .last()
+ .prop('disabled')
+ ).toBeTruthy();
+ });
+});
diff --git a/x-pack/legacy/plugins/siem/public/pages/case/components/tag_list/index.tsx b/x-pack/legacy/plugins/siem/public/pages/case/components/tag_list/index.tsx
index f7d890ca60b16..9bac000b93235 100644
--- a/x-pack/legacy/plugins/siem/public/pages/case/components/tag_list/index.tsx
+++ b/x-pack/legacy/plugins/siem/public/pages/case/components/tag_list/index.tsx
@@ -61,10 +61,11 @@ export const TagList = React.memo(
{i18n.TAGS}
- {isLoading && }
+ {isLoading && }
{!isLoading && (
-
+
-
- {tags.length === 0 && !isEditTags && {i18n.NO_TAGS}
}
+
+ {tags.length === 0 && !isEditTags && {i18n.NO_TAGS}
}
{tags.length > 0 &&
!isEditTags &&
tags.map((tag, key) => (
- {tag}
+
+ {tag}
+
))}
{isEditTags && (
-
+
-
+ {!connector.isPreconfigured ? (
+
+ ) : (
+
+
+ {i18n.translate(
+ 'xpack.triggersActionsUI.sections.editConnectorForm.descriptionText',
+ {
+ defaultMessage: 'This connector is readonly.',
+ }
+ )}
+
+
+
+
+
+ )}
@@ -148,7 +215,7 @@ export const ConnectorEditFlyout = ({
)}
- {canSave && actionTypeModel ? (
+ {canSave && actionTypeModel && !connector.isPreconfigured ? (
{
id: '1',
actionTypeId: 'test',
description: 'My test',
+ isPreconfigured: false,
referencedByCount: 1,
config: {},
},
@@ -119,6 +120,15 @@ describe('actions_connectors_list component with items', () => {
actionTypeId: 'test2',
description: 'My test 2',
referencedByCount: 1,
+ isPreconfigured: false,
+ config: {},
+ },
+ {
+ id: '3',
+ actionTypeId: 'test2',
+ description: 'My preconfigured test 2',
+ referencedByCount: 1,
+ isPreconfigured: true,
config: {},
},
]);
@@ -185,7 +195,11 @@ describe('actions_connectors_list component with items', () => {
it('renders table of connectors', () => {
expect(wrapper.find('EuiInMemoryTable')).toHaveLength(1);
- expect(wrapper.find('EuiTableRow')).toHaveLength(2);
+ expect(wrapper.find('EuiTableRow')).toHaveLength(3);
+ });
+
+ it('renders table with preconfigured connectors', () => {
+ expect(wrapper.find('[data-test-subj="preConfiguredTitleMessage"]')).toHaveLength(2);
});
test('if select item for edit should render ConnectorEditFlyout', () => {
diff --git a/x-pack/plugins/triggers_actions_ui/public/application/sections/actions_connectors_list/components/actions_connectors_list.tsx b/x-pack/plugins/triggers_actions_ui/public/application/sections/actions_connectors_list/components/actions_connectors_list.tsx
index 47e058f473946..043a644489d82 100644
--- a/x-pack/plugins/triggers_actions_ui/public/application/sections/actions_connectors_list/components/actions_connectors_list.tsx
+++ b/x-pack/plugins/triggers_actions_ui/public/application/sections/actions_connectors_list/components/actions_connectors_list.tsx
@@ -15,6 +15,9 @@ import {
EuiIconTip,
EuiFlexGroup,
EuiFlexItem,
+ EuiBetaBadge,
+ EuiToolTip,
+ EuiButtonIcon,
} from '@elastic/eui';
import { i18n } from '@kbn/i18n';
import { FormattedMessage } from '@kbn/i18n/react';
@@ -200,31 +203,58 @@ export const ActionsConnectorsList: React.FunctionComponent = () => {
},
},
{
- field: '',
+ field: 'isPreconfigured',
name: '',
- actions: [
- {
- enabled: () => canDelete,
- 'data-test-subj': 'deleteConnector',
- name: i18n.translate(
- 'xpack.triggersActionsUI.sections.actionsConnectorsList.connectorsListTable.columns.actions.deleteActionName',
- { defaultMessage: 'Delete' }
- ),
- description: canDelete
- ? i18n.translate(
- 'xpack.triggersActionsUI.sections.actionsConnectorsList.connectorsListTable.columns.actions.deleteActionDescription',
- { defaultMessage: 'Delete this connector' }
- )
- : i18n.translate(
- 'xpack.triggersActionsUI.sections.actionsConnectorsList.connectorsListTable.columns.actions.deleteActionDisabledDescription',
- { defaultMessage: 'Unable to delete connectors' }
- ),
- type: 'icon',
- icon: 'trash',
- color: 'danger',
- onClick: (item: ActionConnectorTableItem) => setConnectorsToDelete([item.id]),
- },
- ],
+ render: (value: number, item: ActionConnectorTableItem) => {
+ if (item.isPreconfigured) {
+ return (
+
+
+
+
+
+ );
+ }
+ return (
+
+
+
+ setConnectorsToDelete([item.id])}
+ iconType={'trash'}
+ />
+
+
+
+ );
+ },
},
];
diff --git a/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/alerts.ts b/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/alerts.ts
index 029af1ea06e4f..c94e7116c5cea 100644
--- a/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/alerts.ts
+++ b/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/alerts.ts
@@ -65,23 +65,21 @@ export default ({ getPageObjects, getService }: FtrProviderContext) => {
// need this two out of popup clicks to close them
await nameInput.click();
+ // test for normal connector
+ await testSubjects.click('.webhook-ActionTypeSelectOption');
+ const webhookBodyInput = await find.byCssSelector('.ace_text-input');
+ await webhookBodyInput.focus();
+ await webhookBodyInput.type('{\\"test\\":1}');
+
+ await testSubjects.click('addAlertActionButton');
+ // pre-configured connector is loaded an displayed correctly
await testSubjects.click('.slack-ActionTypeSelectOption');
- await testSubjects.click('createActionConnectorButton');
- const connectorNameInput = await testSubjects.find('nameInput');
- await connectorNameInput.click();
- await connectorNameInput.clearValue();
- const connectorName = generateUniqueKey();
- await connectorNameInput.type(connectorName);
- const slackWebhookUrlInput = await testSubjects.find('slackWebhookUrlInput');
- await slackWebhookUrlInput.click();
- await slackWebhookUrlInput.clearValue();
- await slackWebhookUrlInput.type('https://test');
- await find.clickByCssSelector('[data-test-subj="saveActionButtonModal"]:not(disabled)');
+ expect(await (await find.byCssSelector('#my-slack1')).isDisplayed()).to.be(true);
const loggingMessageInput = await testSubjects.find('slackMessageTextArea');
await loggingMessageInput.click();
await loggingMessageInput.clearValue();
await loggingMessageInput.type('test message');
- await testSubjects.click('slackAddVariableButton');
+ await testSubjects.click('messageAddVariableButton');
const variableMenuButton = await testSubjects.find('variableMenuButton-0');
await variableMenuButton.click();
await testSubjects.click('saveAlertButton');
diff --git a/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/connectors.ts b/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/connectors.ts
index b5bcd33c3b9ab..0e6f991be24d0 100644
--- a/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/connectors.ts
+++ b/x-pack/test/functional_with_es_ssl/apps/triggers_actions_ui/connectors.ts
@@ -184,5 +184,30 @@ export default ({ getPageObjects, getService }: FtrProviderContext) => {
const searchResultsAfterDelete = await pageObjects.triggersActionsUI.getConnectorsList();
expect(searchResultsAfterDelete.length).to.eql(0);
});
+
+ it('should not be able to delete a pre-configured connector', async () => {
+ const preconfiguredConnectorName = 'xyz';
+ await pageObjects.triggersActionsUI.searchConnectors(preconfiguredConnectorName);
+
+ const searchResults = await pageObjects.triggersActionsUI.getConnectorsList();
+ expect(searchResults.length).to.eql(1);
+
+ expect(await testSubjects.exists('deleteConnector')).to.be(false);
+ expect(await testSubjects.exists('preConfiguredTitleMessage')).to.be(true);
+ });
+
+ it('should not be able to edit a pre-configured connector', async () => {
+ const preconfiguredConnectorName = 'xyz';
+
+ await pageObjects.triggersActionsUI.searchConnectors(preconfiguredConnectorName);
+
+ const searchResultsBeforeEdit = await pageObjects.triggersActionsUI.getConnectorsList();
+ expect(searchResultsBeforeEdit.length).to.eql(1);
+
+ await find.clickByCssSelector('[data-test-subj="connectorsTableCell-name"] button');
+
+ expect(await testSubjects.exists('preconfiguredBadge')).to.be(true);
+ expect(await testSubjects.exists('saveEditedActionButton')).to.be(false);
+ });
});
};
diff --git a/x-pack/test/functional_with_es_ssl/config.ts b/x-pack/test/functional_with_es_ssl/config.ts
index 538817bd9d14c..a620b1d953376 100644
--- a/x-pack/test/functional_with_es_ssl/config.ts
+++ b/x-pack/test/functional_with_es_ssl/config.ts
@@ -52,6 +52,16 @@ export default async function({ readConfigFile }: FtrConfigProviderContext) {
`--plugin-path=${join(__dirname, 'fixtures', 'plugins', 'alerts')}`,
'--xpack.actions.enabled=true',
'--xpack.alerting.enabled=true',
+ `--xpack.actions.preconfigured=${JSON.stringify([
+ {
+ id: 'my-slack1',
+ actionTypeId: '.slack',
+ name: 'Slack#xyz',
+ config: {
+ webhookUrl: 'https://hooks.slack.com/services/abcd/efgh/ijklmnopqrstuvwxyz',
+ },
+ },
+ ])}`,
],
},
};
From cc85573c8adec5ad5f3828688136c190b8d79e8b Mon Sep 17 00:00:00 2001
From: Brandon Kobel
Date: Fri, 10 Apr 2020 16:01:30 -0700
Subject: [PATCH 20/24] TaskManager tasks scheduled without attempting to run
(#62078)
* TaskManager tasks scheduled without attempting to run
* Removing unused import
Co-authored-by: Elastic Machine
---
x-pack/plugins/task_manager/server/task_manager.ts | 12 +++++-------
1 file changed, 5 insertions(+), 7 deletions(-)
diff --git a/x-pack/plugins/task_manager/server/task_manager.ts b/x-pack/plugins/task_manager/server/task_manager.ts
index c3f24a4aae88a..a7c67d190e72e 100644
--- a/x-pack/plugins/task_manager/server/task_manager.ts
+++ b/x-pack/plugins/task_manager/server/task_manager.ts
@@ -9,7 +9,7 @@ import { filter } from 'rxjs/operators';
import { performance } from 'perf_hooks';
import { pipe } from 'fp-ts/lib/pipeable';
-import { Option, none, some, map as mapOptional } from 'fp-ts/lib/Option';
+import { Option, some, map as mapOptional } from 'fp-ts/lib/Option';
import {
SavedObjectsSerializer,
IScopedClusterClient,
@@ -156,8 +156,8 @@ export class TaskManager {
this.events$.next(event);
};
- private attemptToRun(task: Option = none) {
- this.claimRequests$.next(task);
+ private attemptToRun(task: string) {
+ this.claimRequests$.next(some(task));
}
private createTaskRunnerForTask = (instance: ConcreteTaskInstance) => {
@@ -280,9 +280,7 @@ export class TaskManager {
...options,
taskInstance: ensureDeprecatedFieldsAreCorrected(taskInstance, this.logger),
});
- const result = await this.store.schedule(modifiedTask);
- this.attemptToRun();
- return result;
+ return await this.store.schedule(modifiedTask);
}
/**
@@ -298,7 +296,7 @@ export class TaskManager {
.then(resolve)
.catch(reject);
- this.attemptToRun(some(taskId));
+ this.attemptToRun(taskId);
});
}
From 53ee20b306556b9cecc7f94533481c43eed25b35 Mon Sep 17 00:00:00 2001
From: Yuliia Naumenko
Date: Fri, 10 Apr 2020 19:18:19 -0700
Subject: [PATCH 21/24] Changed alerting wrong param name for help
xpack.encrypted_saved_objects.encryptionKey to
xpack.encryptedSavedObjects.encryptionKey (#63307)
---
docs/settings/alert-action-settings.asciidoc | 4 ++--
docs/user/alerting/index.asciidoc | 2 +-
rfcs/text/0002_encrypted_attributes.md | 2 +-
.../public/application/components/health_check.test.tsx | 2 +-
.../public/application/components/health_check.tsx | 2 +-
5 files changed, 6 insertions(+), 6 deletions(-)
diff --git a/docs/settings/alert-action-settings.asciidoc b/docs/settings/alert-action-settings.asciidoc
index d7f1ec637d1df..d4dbe9407b7a9 100644
--- a/docs/settings/alert-action-settings.asciidoc
+++ b/docs/settings/alert-action-settings.asciidoc
@@ -9,7 +9,7 @@ Alerts and actions are enabled by default in {kib}, but require you configure th
. <>.
. <>.
-. <>.
+. <>.
You can configure the following settings in the `kibana.yml` file.
@@ -18,7 +18,7 @@ You can configure the following settings in the `kibana.yml` file.
[[general-alert-action-settings]]
==== General settings
-`xpack.encrypted_saved_objects.encryptionKey`::
+`xpack.encryptedSavedObjects.encryptionKey`::
A string of 32 or more characters used to encrypt sensitive properties on alerts and actions before they're stored in {es}. Third party credentials — such as the username and password used to connect to an SMTP service — are an example of encrypted properties.
+
diff --git a/docs/user/alerting/index.asciidoc b/docs/user/alerting/index.asciidoc
index c7cf1186a44be..f556cf71bf06c 100644
--- a/docs/user/alerting/index.asciidoc
+++ b/docs/user/alerting/index.asciidoc
@@ -157,7 +157,7 @@ Pre-packaged *alert types* simplify setup, hide the details complex domain-speci
If you are using an *on-premises* Elastic Stack deployment with <>:
* TLS must be configured for communication <>. {kib} alerting uses <> to secure background alert checks and actions, and API keys require {ref}/configuring-tls.html#tls-http[TLS on the HTTP interface].
-* In the kibana.yml configuration file, add the <>
+* In the kibana.yml configuration file, add the <>
[float]
[[alerting-security]]
diff --git a/rfcs/text/0002_encrypted_attributes.md b/rfcs/text/0002_encrypted_attributes.md
index aa7307edb66bd..c6553c177d995 100644
--- a/rfcs/text/0002_encrypted_attributes.md
+++ b/rfcs/text/0002_encrypted_attributes.md
@@ -166,7 +166,7 @@ take a look at the source code of this library to know how encryption is perform
parameters are used, but in short it's AES Encryption with AES-256-GCM that uses random initialization vector and salt.
As with encryption key for Kibana's session cookie, master encryption key used by `encrypted_saved_objects` plugin can be
-defined as a configuration value (`xpack.encrypted_saved_objects.encryptionKey`) via `kibana.yml`, but it's **highly
+defined as a configuration value (`xpack.encryptedSavedObjects.encryptionKey`) via `kibana.yml`, but it's **highly
recommended** to define this key in the [Kibana Keystore](https://www.elastic.co/guide/en/kibana/current/secure-settings.html)
instead. The master key should be cryptographically safe and be equal or greater than 32 bytes.
diff --git a/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.test.tsx b/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.test.tsx
index 9c51139993b3f..3fbcd13e98f5d 100644
--- a/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.test.tsx
+++ b/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.test.tsx
@@ -92,7 +92,7 @@ describe('health check', () => {
const description = queryByRole(/banner/i);
expect(description!.textContent).toMatchInlineSnapshot(
- `"To create an alert, set a value for xpack.encrypted_saved_objects.encryptionKey in your kibana.yml file. Learn how."`
+ `"To create an alert, set a value for xpack.encryptedSavedObjects.encryptionKey in your kibana.yml file. Learn how."`
);
const action = queryByText(/Learn/i);
diff --git a/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.tsx b/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.tsx
index c967cf5de0771..afd5e08f52f25 100644
--- a/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.tsx
+++ b/x-pack/plugins/triggers_actions_ui/public/application/components/health_check.tsx
@@ -132,7 +132,7 @@ const EncryptionError = ({
defaultMessage: 'To create an alert, set a value for ',
}
)}
- {'xpack.encrypted_saved_objects.encryptionKey'}
+ {'xpack.encryptedSavedObjects.encryptionKey'}
{i18n.translate(
'xpack.triggersActionsUI.components.healthCheck.encryptionErrorAfterKey',
{
From bd159c7d59979d66c09c94402daf2c30373ee9f9 Mon Sep 17 00:00:00 2001
From: Pierre Gayvallet
Date: Sat, 11 Apr 2020 09:27:45 +0200
Subject: [PATCH 22/24] fix ScopedHistory.createHref to prepend location with
scoped history basePath (#62407)
* fix createHref to prepend with scoped history basePath + add option to exclude it.
* fix prependBasePath behavior
* fix test plugins urls
* add pathname to endpoint url builder methods
* Revert "add pathname to endpoint url builder methods"
This reverts commit 7604932b
* adapt createHref instead of prependBasePath
* use object options for createHref
* update generated doc
---
...in-core-public.scopedhistory.createhref.md | 6 +++--
...kibana-plugin-core-public.scopedhistory.md | 2 +-
.../public/application/scoped_history.test.ts | 25 +++++++++++++++++--
src/core/public/application/scoped_history.ts | 20 ++++++++++++---
src/core/public/public.api.md | 4 ++-
.../core_plugin_a/public/application.tsx | 2 +-
.../core_plugin_b/public/application.tsx | 2 +-
7 files changed, 50 insertions(+), 11 deletions(-)
diff --git a/docs/development/core/public/kibana-plugin-core-public.scopedhistory.createhref.md b/docs/development/core/public/kibana-plugin-core-public.scopedhistory.createhref.md
index 7058656d09947..6bbab43ff6ffc 100644
--- a/docs/development/core/public/kibana-plugin-core-public.scopedhistory.createhref.md
+++ b/docs/development/core/public/kibana-plugin-core-public.scopedhistory.createhref.md
@@ -4,10 +4,12 @@
## ScopedHistory.createHref property
-Creates an href (string) to the location.
+Creates an href (string) to the location. If `prependBasePath` is true (default), it will prepend the location's path with the scoped history basePath.
Signature:
```typescript
-createHref: (location: LocationDescriptorObject) => string;
+createHref: (location: LocationDescriptorObject, { prependBasePath }?: {
+ prependBasePath?: boolean | undefined;
+ }) => string;
```
diff --git a/docs/development/core/public/kibana-plugin-core-public.scopedhistory.md b/docs/development/core/public/kibana-plugin-core-public.scopedhistory.md
index 5ea47d2090d71..fa29b32c0bafc 100644
--- a/docs/development/core/public/kibana-plugin-core-public.scopedhistory.md
+++ b/docs/development/core/public/kibana-plugin-core-public.scopedhistory.md
@@ -28,7 +28,7 @@ export declare class ScopedHistory implements Hi
| --- | --- | --- | --- |
| [action](./kibana-plugin-core-public.scopedhistory.action.md) | | Action
| The last action dispatched on the history stack. |
| [block](./kibana-plugin-core-public.scopedhistory.block.md) | | (prompt?: string | boolean | History.TransitionPromptHook<HistoryLocationState> | undefined) => UnregisterCallback
| Not supported. Use [AppMountParameters.onAppLeave](./kibana-plugin-core-public.appmountparameters.onappleave.md). |
-| [createHref](./kibana-plugin-core-public.scopedhistory.createhref.md) | | (location: LocationDescriptorObject<HistoryLocationState>) => string
| Creates an href (string) to the location. |
+| [createHref](./kibana-plugin-core-public.scopedhistory.createhref.md) | | (location: LocationDescriptorObject<HistoryLocationState>, { prependBasePath }?: {
prependBasePath?: boolean | undefined;
}) => string
| Creates an href (string) to the location. If prependBasePath
is true (default), it will prepend the location's path with the scoped history basePath. |
| [createSubHistory](./kibana-plugin-core-public.scopedhistory.createsubhistory.md) | | <SubHistoryLocationState = unknown>(basePath: string) => ScopedHistory<SubHistoryLocationState>
| Creates a ScopedHistory
for a subpath of this ScopedHistory
. Useful for applications that may have sub-apps that do not need access to the containing application's history. |
| [go](./kibana-plugin-core-public.scopedhistory.go.md) | | (n: number) => void
| Send the user forward or backwards in the history stack. |
| [goBack](./kibana-plugin-core-public.scopedhistory.goback.md) | | () => void
| Send the user one location back in the history stack. Equivalent to calling [ScopedHistory.go(-1)](./kibana-plugin-core-public.scopedhistory.go.md). If no more entries are available backwards, this is a no-op. |
diff --git a/src/core/public/application/scoped_history.test.ts b/src/core/public/application/scoped_history.test.ts
index c01eb50830516..a56cffef1e2f2 100644
--- a/src/core/public/application/scoped_history.test.ts
+++ b/src/core/public/application/scoped_history.test.ts
@@ -268,11 +268,32 @@ describe('ScopedHistory', () => {
const gh = createMemoryHistory();
gh.push('/app/wow');
const h = new ScopedHistory(gh, '/app/wow');
- expect(h.createHref({ pathname: '' })).toEqual(`/`);
+ expect(h.createHref({ pathname: '' })).toEqual(`/app/wow`);
+ expect(h.createHref({})).toEqual(`/app/wow`);
expect(h.createHref({ pathname: '/new-page', search: '?alpha=true' })).toEqual(
- `/new-page?alpha=true`
+ `/app/wow/new-page?alpha=true`
);
});
+
+ it('behave correctly with slash-ending basePath', () => {
+ const gh = createMemoryHistory();
+ gh.push('/app/wow/');
+ const h = new ScopedHistory(gh, '/app/wow/');
+ expect(h.createHref({ pathname: '' })).toEqual(`/app/wow/`);
+ expect(h.createHref({ pathname: '/new-page', search: '?alpha=true' })).toEqual(
+ `/app/wow/new-page?alpha=true`
+ );
+ });
+
+ it('skips the scoped history path when `prependBasePath` is false', () => {
+ const gh = createMemoryHistory();
+ gh.push('/app/wow');
+ const h = new ScopedHistory(gh, '/app/wow');
+ expect(h.createHref({ pathname: '' }, { prependBasePath: false })).toEqual(`/`);
+ expect(
+ h.createHref({ pathname: '/new-page', search: '?alpha=true' }, { prependBasePath: false })
+ ).toEqual(`/new-page?alpha=true`);
+ });
});
describe('action', () => {
diff --git a/src/core/public/application/scoped_history.ts b/src/core/public/application/scoped_history.ts
index c5febc7604feb..9fa8f0b7f8148 100644
--- a/src/core/public/application/scoped_history.ts
+++ b/src/core/public/application/scoped_history.ts
@@ -219,11 +219,26 @@ export class ScopedHistory
/**
* Creates an href (string) to the location.
+ * If `prependBasePath` is true (default), it will prepend the location's path with the scoped history basePath.
*
* @param location
+ * @param prependBasePath
*/
- public createHref = (location: LocationDescriptorObject): Href => {
+ public createHref = (
+ location: LocationDescriptorObject,
+ { prependBasePath = true }: { prependBasePath?: boolean } = {}
+ ): Href => {
this.verifyActive();
+ if (prependBasePath) {
+ location = this.prependBasePath(location);
+ if (location.pathname === undefined) {
+ // we always want to create an url relative to the basePath
+ // so if pathname is not present, we use the history's basePath as default
+ // we are doing that here because `prependBasePath` should not
+ // alter pathname for other method calls
+ location.pathname = this.basePath;
+ }
+ }
return this.parentHistory.createHref(location);
};
@@ -254,8 +269,7 @@ export class ScopedHistory
* Prepends the base path to string.
*/
private prependBasePathToString(path: string): string {
- path = path.startsWith('/') ? path.slice(1) : path;
- return path.length ? `${this.basePath}/${path}` : this.basePath;
+ return path.length ? `${this.basePath}/${path}`.replace(/\/{2,}/g, '/') : this.basePath;
}
/**
diff --git a/src/core/public/public.api.md b/src/core/public/public.api.md
index a5aa37becabc2..6d95d1bc7069c 100644
--- a/src/core/public/public.api.md
+++ b/src/core/public/public.api.md
@@ -1196,7 +1196,9 @@ export class ScopedHistory implements History | undefined) => UnregisterCallback;
- createHref: (location: LocationDescriptorObject) => string;
+ createHref: (location: LocationDescriptorObject, { prependBasePath }?: {
+ prependBasePath?: boolean | undefined;
+ }) => string;
createSubHistory: (basePath: string) => ScopedHistory;
go: (n: number) => void;
goBack: () => void;
diff --git a/test/plugin_functional/plugins/core_plugin_a/public/application.tsx b/test/plugin_functional/plugins/core_plugin_a/public/application.tsx
index abea970749cbc..159bb54f50903 100644
--- a/test/plugin_functional/plugins/core_plugin_a/public/application.tsx
+++ b/test/plugin_functional/plugins/core_plugin_a/public/application.tsx
@@ -95,7 +95,7 @@ const Nav = withRouter(({ history, navigateToApp }: NavProps) => (
{
id: 'home',
name: 'Home',
- onClick: () => history.push('/'),
+ onClick: () => history.push(''),
'data-test-subj': 'fooNavHome',
},
{
diff --git a/test/plugin_functional/plugins/core_plugin_b/public/application.tsx b/test/plugin_functional/plugins/core_plugin_b/public/application.tsx
index 447307920c04c..01a63f9782563 100644
--- a/test/plugin_functional/plugins/core_plugin_b/public/application.tsx
+++ b/test/plugin_functional/plugins/core_plugin_b/public/application.tsx
@@ -102,7 +102,7 @@ const Nav = withRouter(({ history, navigateToApp }: NavProps) => (
{
id: 'home',
name: 'Home',
- onClick: () => navigateToApp('bar', { path: '/' }),
+ onClick: () => navigateToApp('bar', { path: '' }),
'data-test-subj': 'barNavHome',
},
{
From 2be6b7fdcecbe2848cc1da23eb98af756f513d6b Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Cau=C3=AA=20Marcondes?=
<55978943+cauemarcondes@users.noreply.github.com>
Date: Sat, 11 Apr 2020 10:06:39 +0100
Subject: [PATCH 23/24] fixing custom link popover size and hiding scroll
(#63240)
---
.../TransactionActionMenu/CustomLink/CustomLinkPopover.tsx | 1 +
.../shared/TransactionActionMenu/TransactionActionMenu.tsx | 2 +-
2 files changed, 2 insertions(+), 1 deletion(-)
diff --git a/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/CustomLink/CustomLinkPopover.tsx b/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/CustomLink/CustomLinkPopover.tsx
index a20bc7e21cfc5..3aed1b7ac2953 100644
--- a/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/CustomLink/CustomLinkPopover.tsx
+++ b/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/CustomLink/CustomLinkPopover.tsx
@@ -19,6 +19,7 @@ import { ManageCustomLink } from './ManageCustomLink';
import { px } from '../../../../style/variables';
const ScrollableContainer = styled.div`
+ -ms-overflow-style: none;
max-height: ${px(535)};
overflow: scroll;
`;
diff --git a/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/TransactionActionMenu.tsx b/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/TransactionActionMenu.tsx
index e3fbcf8485d54..7ebfe26b83630 100644
--- a/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/TransactionActionMenu.tsx
+++ b/x-pack/legacy/plugins/apm/public/components/shared/TransactionActionMenu/TransactionActionMenu.tsx
@@ -124,7 +124,7 @@ export const TransactionActionMenu: FunctionComponent = ({
setIsActionPopoverOpen(true)} />
}
>
-
+
{isCustomLinksPopoverOpen ? (
Date: Sun, 12 Apr 2020 17:06:03 +0200
Subject: [PATCH 24/24] Bugfix dashboard unpins filters (#62301)
Fixes following cases:
Saving dashboard with pinned filter unpins it. Do not save pinned filters with dashboard see #62301 (comment)
When navigating with global filter to dashboard with same saved filter, filter becomes unpinned
When navigating from listing to dashboard with saved filter, back button didn't work
Co-authored-by: Elastic Machine
---
.../application/dashboard_app_controller.tsx | 14 +++++--
.../application/dashboard_state_manager.ts | 7 ++--
.../public/application/lib/filter_utils.ts | 2 +-
.../application/lib/update_saved_dashboard.ts | 7 ++++
.../filter_manager/filter_manager.test.ts | 11 +++---
.../query/filter_manager/filter_manager.ts | 17 ++-------
.../data/public/ui/filter_bar/filter_item.tsx | 3 +-
.../apps/dashboard/dashboard_filter_bar.js | 38 +++++++++++++++++++
test/functional/services/filter_bar.ts | 15 +++++++-
9 files changed, 85 insertions(+), 29 deletions(-)
diff --git a/src/plugins/dashboard/public/application/dashboard_app_controller.tsx b/src/plugins/dashboard/public/application/dashboard_app_controller.tsx
index d1c7d2d9eba3e..babd1fac274eb 100644
--- a/src/plugins/dashboard/public/application/dashboard_app_controller.tsx
+++ b/src/plugins/dashboard/public/application/dashboard_app_controller.tsx
@@ -143,6 +143,7 @@ export class DashboardAppController {
});
// sync initial app filters from state to filterManager
+ // if there is an existing similar global filter, then leave it as global
filterManager.setAppFilters(_.cloneDeep(dashboardStateManager.appState.filters));
// setup syncing of app filters between appState and filterManager
const stopSyncingAppFilters = connectToQueryState(
@@ -150,7 +151,11 @@ export class DashboardAppController {
{
set: ({ filters }) => dashboardStateManager.setFilters(filters || []),
get: () => ({ filters: dashboardStateManager.appState.filters }),
- state$: dashboardStateManager.appState$.pipe(map(state => ({ filters: state.filters }))),
+ state$: dashboardStateManager.appState$.pipe(
+ map(state => ({
+ filters: state.filters,
+ }))
+ ),
},
{
filters: esFilters.FilterStateStore.APP_STATE,
@@ -170,13 +175,16 @@ export class DashboardAppController {
}
// starts syncing `_g` portion of url with query services
- // note: dashboard_state_manager.ts syncs `_a` portion of url
// it is important to start this syncing after `dashboardStateManager.syncTimefilterWithDashboard(timefilter);` above is run,
- // otherwise it will case redundant browser history record
+ // otherwise it will case redundant browser history records
const { stop: stopSyncingQueryServiceStateWithUrl } = syncQueryStateWithUrl(
queryService,
kbnUrlStateStorage
);
+
+ // starts syncing `_a` portion of url
+ dashboardStateManager.startStateSyncing();
+
$scope.showSaveQuery = dashboardCapabilities.saveQuery as boolean;
const getShouldShowEditHelp = () =>
diff --git a/src/plugins/dashboard/public/application/dashboard_state_manager.ts b/src/plugins/dashboard/public/application/dashboard_state_manager.ts
index 6025f535ae761..13ba3c6d0b60d 100644
--- a/src/plugins/dashboard/public/application/dashboard_state_manager.ts
+++ b/src/plugins/dashboard/public/application/dashboard_state_manager.ts
@@ -164,9 +164,6 @@ export class DashboardStateManager {
this.changeListeners.forEach(listener => listener({ dirty: this.isDirty }));
});
- // make sure url ('_a') matches initial state
- this.kbnUrlStateStorage.set(this.STATE_STORAGE_KEY, initialState, { replace: true });
-
// setup state syncing utils. state container will be synced with url into `this.STATE_STORAGE_KEY` query param
this.stateSyncRef = syncState({
storageKey: this.STATE_STORAGE_KEY,
@@ -201,8 +198,10 @@ export class DashboardStateManager {
},
stateStorage: this.kbnUrlStateStorage,
});
+ }
- // actually start syncing state with container
+ public startStateSyncing() {
+ this.saveState({ replace: true });
this.stateSyncRef.start();
}
diff --git a/src/plugins/dashboard/public/application/lib/filter_utils.ts b/src/plugins/dashboard/public/application/lib/filter_utils.ts
index 1ec231db0c3d2..775bf90643a21 100644
--- a/src/plugins/dashboard/public/application/lib/filter_utils.ts
+++ b/src/plugins/dashboard/public/application/lib/filter_utils.ts
@@ -19,7 +19,7 @@
import _ from 'lodash';
import moment, { Moment } from 'moment';
-import { Filter } from 'src/plugins/data/public';
+import { Filter } from '../../../../data/public';
/**
* @typedef {Object} QueryFilter
diff --git a/src/plugins/dashboard/public/application/lib/update_saved_dashboard.ts b/src/plugins/dashboard/public/application/lib/update_saved_dashboard.ts
index 53dc7d9b460de..fc519327b41ee 100644
--- a/src/plugins/dashboard/public/application/lib/update_saved_dashboard.ts
+++ b/src/plugins/dashboard/public/application/lib/update_saved_dashboard.ts
@@ -22,6 +22,7 @@ import { RefreshInterval, TimefilterContract } from 'src/plugins/data/public';
import { FilterUtils } from './filter_utils';
import { SavedObjectDashboard } from '../../saved_dashboards';
import { DashboardAppState } from '../../types';
+import { esFilters } from '../../../../data/public';
export function updateSavedDashboard(
savedDashboard: SavedObjectDashboard,
@@ -48,4 +49,10 @@ export function updateSavedDashboard(
'value',
]);
savedDashboard.refreshInterval = savedDashboard.timeRestore ? timeRestoreObj : undefined;
+
+ // save only unpinned filters
+ const unpinnedFilters = savedDashboard
+ .getFilters()
+ .filter(filter => !esFilters.isFilterPinned(filter));
+ savedDashboard.searchSource.setField('filter', unpinnedFilters);
}
diff --git a/src/plugins/data/public/query/filter_manager/filter_manager.test.ts b/src/plugins/data/public/query/filter_manager/filter_manager.test.ts
index ce574ace45705..3b4ca08cbbf14 100644
--- a/src/plugins/data/public/query/filter_manager/filter_manager.test.ts
+++ b/src/plugins/data/public/query/filter_manager/filter_manager.test.ts
@@ -204,18 +204,19 @@ describe('filter_manager', () => {
).toBe(3);
});
- test('should set app filters and remove any duplicated global filters', async function() {
- filterManager.addFilters(readyFilters, true);
+ test('should set app filters and merge them with duplicate global filters', async function() {
+ const [filter, ...otherFilters] = readyFilters;
+ filterManager.addFilters(otherFilters, true);
const appFilter1 = _.cloneDeep(readyFilters[1]);
const appFilter2 = _.cloneDeep(readyFilters[2]);
- filterManager.setAppFilters([appFilter1, appFilter2]);
+ filterManager.setAppFilters([filter, appFilter1, appFilter2]);
const newGlobalFilters = filterManager.getGlobalFilters();
const newAppFilters = filterManager.getAppFilters();
- expect(newGlobalFilters).toHaveLength(1);
- expect(newAppFilters).toHaveLength(2);
+ expect(newGlobalFilters).toHaveLength(2);
+ expect(newAppFilters).toHaveLength(1);
});
test('should set global filters and remove any duplicated app filters', async function() {
diff --git a/src/plugins/data/public/query/filter_manager/filter_manager.ts b/src/plugins/data/public/query/filter_manager/filter_manager.ts
index fba1866ebd615..e206286bce147 100644
--- a/src/plugins/data/public/query/filter_manager/filter_manager.ts
+++ b/src/plugins/data/public/query/filter_manager/filter_manager.ts
@@ -177,13 +177,9 @@ export class FilterManager {
public setGlobalFilters(newGlobalFilters: Filter[]) {
newGlobalFilters = mapAndFlattenFilters(newGlobalFilters);
FilterManager.setFiltersStore(newGlobalFilters, FilterStateStore.GLOBAL_STATE, true);
- const { appFilters: currentAppFilters } = this.getPartitionedFilters();
- // remove duplicates from current app filters, to make sure global will take precedence
- const filteredAppFilters = currentAppFilters.filter(
- appFilter => !newGlobalFilters.find(globalFilter => compareFilters(globalFilter, appFilter))
- );
+ const { appFilters } = this.getPartitionedFilters();
const newFilters = this.mergeIncomingFilters({
- appFilters: filteredAppFilters,
+ appFilters,
globalFilters: newGlobalFilters,
});
@@ -198,14 +194,9 @@ export class FilterManager {
public setAppFilters(newAppFilters: Filter[]) {
newAppFilters = mapAndFlattenFilters(newAppFilters);
FilterManager.setFiltersStore(newAppFilters, FilterStateStore.APP_STATE, true);
- const { globalFilters: currentGlobalFilters } = this.getPartitionedFilters();
- // remove duplicates from current global filters, to make sure app will take precedence
- const filteredGlobalFilters = currentGlobalFilters.filter(
- globalFilter => !newAppFilters.find(appFilter => compareFilters(appFilter, globalFilter))
- );
-
+ const { globalFilters } = this.getPartitionedFilters();
const newFilters = this.mergeIncomingFilters({
- globalFilters: filteredGlobalFilters,
+ globalFilters,
appFilters: newAppFilters,
});
this.handleStateUpdate(newFilters);
diff --git a/src/plugins/data/public/ui/filter_bar/filter_item.tsx b/src/plugins/data/public/ui/filter_bar/filter_item.tsx
index 6b5fd41dc06ea..528ec4800e7b9 100644
--- a/src/plugins/data/public/ui/filter_bar/filter_item.tsx
+++ b/src/plugins/data/public/ui/filter_bar/filter_item.tsx
@@ -88,6 +88,7 @@ class FilterItemUI extends Component {
const dataTestSubjDisabled = `filter-${
this.props.filter.meta.disabled ? 'disabled' : 'enabled'
}`;
+ const dataTestSubjPinned = `filter-${isFilterPinned(filter) ? 'pinned' : 'unpinned'}`;
const classes = classNames(
'globalFilterItem',
@@ -107,7 +108,7 @@ class FilterItemUI extends Component {
className={classes}
iconOnClick={() => this.props.onRemove()}
onClick={this.handleBadgeClick}
- data-test-subj={`filter ${dataTestSubjDisabled} ${dataTestSubjKey} ${dataTestSubjValue}`}
+ data-test-subj={`filter ${dataTestSubjDisabled} ${dataTestSubjKey} ${dataTestSubjValue} ${dataTestSubjPinned}`}
/>
);
diff --git a/test/functional/apps/dashboard/dashboard_filter_bar.js b/test/functional/apps/dashboard/dashboard_filter_bar.js
index 6d2a30fa85325..f6089871010c3 100644
--- a/test/functional/apps/dashboard/dashboard_filter_bar.js
+++ b/test/functional/apps/dashboard/dashboard_filter_bar.js
@@ -27,6 +27,7 @@ export default function({ getService, getPageObjects }) {
const pieChart = getService('pieChart');
const esArchiver = getService('esArchiver');
const kibanaServer = getService('kibanaServer');
+ const browser = getService('browser');
const PageObjects = getPageObjects(['common', 'dashboard', 'header', 'visualize', 'timePicker']);
describe('dashboard filter bar', () => {
@@ -126,9 +127,46 @@ export default function({ getService, getPageObjects }) {
const filterCount = await filterBar.getFilterCount();
expect(filterCount).to.equal(1);
+ await pieChart.expectPieSliceCount(1);
+ });
+ it("restoring filters doesn't break back button", async () => {
+ await browser.goBack();
+ await PageObjects.dashboard.expectExistsDashboardLandingPage();
+ await browser.goForward();
+ await PageObjects.header.waitUntilLoadingHasFinished();
+ await PageObjects.dashboard.waitForRenderComplete();
await pieChart.expectPieSliceCount(1);
});
+
+ it("saving with pinned filter doesn't unpin them", async () => {
+ const filterKey = 'bytes';
+ await filterBar.toggleFilterPinned(filterKey);
+ await PageObjects.dashboard.switchToEditMode();
+ await PageObjects.dashboard.saveDashboard('saved with pinned filters', {
+ saveAsNew: true,
+ });
+ expect(await filterBar.isFilterPinned(filterKey)).to.be(true);
+ await pieChart.expectPieSliceCount(1);
+ });
+
+ it("navigating to a dashboard with global filter doesn't unpin it if same filter is saved with dashboard", async () => {
+ await PageObjects.dashboard.preserveCrossAppState();
+ await PageObjects.dashboard.gotoDashboardLandingPage();
+ await PageObjects.dashboard.loadSavedDashboard('with filters');
+ await PageObjects.header.waitUntilLoadingHasFinished();
+ expect(await filterBar.isFilterPinned('bytes')).to.be(true);
+ await pieChart.expectPieSliceCount(1);
+ });
+
+ it("pinned filters aren't saved", async () => {
+ await filterBar.removeFilter('bytes');
+ await PageObjects.dashboard.gotoDashboardLandingPage();
+ await PageObjects.dashboard.loadSavedDashboard('saved with pinned filters');
+ await PageObjects.header.waitUntilLoadingHasFinished();
+ expect(await filterBar.getFilterCount()).to.be(0);
+ await pieChart.expectPieSliceCount(5);
+ });
});
describe('saved search filtering', function() {
diff --git a/test/functional/services/filter_bar.ts b/test/functional/services/filter_bar.ts
index 9d494b1e6d950..a463a593e9e04 100644
--- a/test/functional/services/filter_bar.ts
+++ b/test/functional/services/filter_bar.ts
@@ -32,10 +32,16 @@ export function FilterBarProvider({ getService, getPageObjects }: FtrProviderCon
* @param value filter value
* @param enabled filter status
*/
- public async hasFilter(key: string, value: string, enabled: boolean = true): Promise {
+ public async hasFilter(
+ key: string,
+ value: string,
+ enabled: boolean = true,
+ pinned: boolean = false
+ ): Promise {
const filterActivationState = enabled ? 'enabled' : 'disabled';
+ const filterPinnedState = pinned ? 'pinned' : 'unpinned';
return testSubjects.exists(
- `filter filter-${filterActivationState} filter-key-${key} filter-value-${value}`,
+ `filter filter-${filterActivationState} filter-key-${key} filter-value-${value} filter-${filterPinnedState}`,
{
allowHidden: true,
}
@@ -80,6 +86,11 @@ export function FilterBarProvider({ getService, getPageObjects }: FtrProviderCon
await PageObjects.header.awaitGlobalLoadingIndicatorHidden();
}
+ public async isFilterPinned(key: string): Promise {
+ const filter = await testSubjects.find(`~filter & ~filter-key-${key}`);
+ return (await filter.getAttribute('data-test-subj')).includes('filter-pinned');
+ }
+
public async getFilterCount(): Promise {
const filters = await testSubjects.findAll('~filter');
return filters.length;