From 01f87adec630ea4fad92ae3c5d93f5b9ce24a2d3 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Thu, 18 Jul 2019 13:14:50 -0600 Subject: [PATCH 01/16] [Maps][POC] pew pew source --- .../maps/public/layers/sources/all_sources.js | 4 +- .../es_pew_pew_source/convert_to_lines.js | 54 +++++ .../es_pew_pew_source/create_source_editor.js | 208 ++++++++++++++++++ .../es_pew_pew_source/es_pew_pew_source.js | 205 +++++++++++++++++ 4 files changed, 469 insertions(+), 2 deletions(-) create mode 100644 x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js create mode 100644 x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js create mode 100644 x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/all_sources.js b/x-pack/legacy/plugins/maps/public/layers/sources/all_sources.js index d81d218910ccb..6a518609dd77f 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/all_sources.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/all_sources.js @@ -4,7 +4,6 @@ * you may not use this file except in compliance with the Elastic License. */ - import { EMSFileSource } from './ems_file_source'; import { GeojsonFileSource } from './client_file_source'; import { KibanaRegionmapSource } from './kibana_regionmap_source'; @@ -14,12 +13,13 @@ import { WMSSource } from './wms_source'; import { KibanaTilemapSource } from './kibana_tilemap_source'; import { ESGeoGridSource } from './es_geo_grid_source'; import { ESSearchSource } from './es_search_source'; - +import { ESPewPewSource } from './es_pew_pew_source/es_pew_pew_source'; export const ALL_SOURCES = [ GeojsonFileSource, ESSearchSource, ESGeoGridSource, + ESPewPewSource, EMSFileSource, EMSTMSSource, KibanaRegionmapSource, diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js new file mode 100644 index 0000000000000..89cf0b8bc5f67 --- /dev/null +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js @@ -0,0 +1,54 @@ +/* + * 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 _ from 'lodash'; + +const LAT_INDEX = 0; +const LON_INDEX = 1; + +function parsePointFromKey(key) { + const split = key.split(','); + const lat = parseFloat(split[LAT_INDEX]); + const lon = parseFloat(split[LON_INDEX]); + return [lon, lat]; +} + +export function convertToLines(esResonse) { + + const lineFeatures = []; + + const destBuckets = _.get(esResonse, 'aggregations.destSplit.buckets', []); + for (let i = 0; i < destBuckets.length; i++) { + const destBucket = destBuckets[i]; + const dest = parsePointFromKey(destBucket.key); + const sourceBuckets = _.get(destBucket, 'sourceGrid.buckets', []); + for (let j = 0; j < sourceBuckets.length; j++) { + const { + key, // eslint-disable-line no-unused-vars + sourceCentroid, + ...rest + } = sourceBuckets[j]; + + lineFeatures.push({ + type: 'Feature', + geometry: { + type: 'LineString', + coordinates: [[sourceCentroid.location.lon, sourceCentroid.location.lat], dest] + }, + properties: { + ...rest + } + }); + } + } + + return { + featureCollection: { + type: 'FeatureCollection', + features: lineFeatures + } + }; +} diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js new file mode 100644 index 0000000000000..d728d444e7914 --- /dev/null +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js @@ -0,0 +1,208 @@ +/* + * 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 _ from 'lodash'; +import React, { Fragment, Component } from 'react'; +import PropTypes from 'prop-types'; + +import { IndexPatternSelect } from 'ui/index_patterns/components/index_pattern_select'; +import { SingleFieldSelect } from '../../../components/single_field_select'; +import { indexPatternService } from '../../../kibana_services'; +import { i18n } from '@kbn/i18n'; +import { FormattedMessage } from '@kbn/i18n/react'; + +import { + EuiFormRow, + EuiCallOut, +} from '@elastic/eui'; +import { ES_GEO_FIELD_TYPE } from '../../../../common/constants'; + +const GEO_FIELD_TYPES = [ES_GEO_FIELD_TYPE.GEO_POINT]; + +function filterGeoField({ type }) { + return [ES_GEO_FIELD_TYPE.GEO_POINT].includes(type); +} + +export class CreateSourceEditor extends Component { + + static propTypes = { + onSourceConfigChange: PropTypes.func.isRequired, + }; + + state = { + isLoadingIndexPattern: false, + indexPattern: undefined, + indexPatternId: undefined, + sourceGeoField: undefined, + destGeoField: undefined, + indexPatternHasMultipleGeoFields: false, + } + + componentWillUnmount() { + this._isMounted = false; + } + + componentDidMount() { + this._isMounted = true; + } + + onIndexPatternSelect = (indexPatternId) => { + this.setState({ + indexPatternId, + }, this.loadIndexPattern.bind(null, indexPatternId)); + }; + + loadIndexPattern = (indexPatternId) => { + this.setState({ + isLoadingIndexPattern: true, + indexPattern: undefined, + sourceGeoField: undefined, + destGeoField: undefined, + indexPatternHasMultipleGeoFields: false, + }, this.debouncedLoad.bind(null, indexPatternId)); + }; + + debouncedLoad = _.debounce(async (indexPatternId) => { + if (!indexPatternId || indexPatternId.length === 0) { + return; + } + + let indexPattern; + try { + indexPattern = await indexPatternService.get(indexPatternId); + } catch (err) { + // index pattern no longer exists + return; + } + + if (!this._isMounted) { + return; + } + + // props.indexPatternId may be updated before getIndexPattern returns + // ignore response when fetched index pattern does not match active index pattern + if (this.state.indexPatternId !== indexPatternId) { + return; + } + + const geoFields = indexPattern.fields.filter(filterGeoField); + + this.setState({ + isLoadingIndexPattern: false, + indexPattern: indexPattern, + indexPatternHasMultipleGeoFields: geoFields.length >= 2, + }); + }, 300); + + _onSourceGeoSelect = (sourceGeoField) => { + this.setState({ + sourceGeoField + }, this.previewLayer); + }; + + _onDestGeoSelect = (destGeoField) => { + this.setState({ + destGeoField + }, this.previewLayer); + }; + + previewLayer = () => { + const { + indexPatternId, + sourceGeoField, + destGeoField, + } = this.state; + + const sourceConfig = (indexPatternId && sourceGeoField && destGeoField) + ? { indexPatternId, sourceGeoField, destGeoField } + : null; + this.props.onSourceConfigChange(sourceConfig); + }; + + _renderGeoSelects() { + if (!this.state.indexPattern || !this.state.indexPatternHasMultipleGeoFields) { + return null; + } + + return ( + + + + + + + + + + ); + } + + _renderIndexPatternSelect() { + return ( + + + + ); + } + + render() { + let callout; + if (this.state.indexPattern && !this.state.indexPatternHasMultipleGeoFields) { + callout = ( + +

+ +

+
+ ); + } + + return ( + + {callout} + {this._renderIndexPatternSelect()} + {this._renderGeoSelects()} + + ); + } +} diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js new file mode 100644 index 0000000000000..553aa5a7370c5 --- /dev/null +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -0,0 +1,205 @@ +/* + * 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 uuid from 'uuid/v4'; + +import { VECTOR_SHAPE_TYPES } from '../vector_feature_types'; +import { AbstractESSource } from '../es_source'; +import { VectorLayer } from '../../vector_layer'; +import { CreateSourceEditor } from './create_source_editor'; +import { VectorStyle } from '../../styles/vector_style'; +import { i18n } from '@kbn/i18n'; +import { SOURCE_DATA_ID_ORIGIN } from '../../../../common/constants'; +import { getDataSourceLabel } from '../../../../common/i18n_getters'; +import { convertToLines } from './convert_to_lines'; + +const COUNT_PROP_LABEL = 'count'; +const COUNT_PROP_NAME = 'doc_count'; +const MAX_GEOTILE_LEVEL = 29; + +export class ESPewPewSource extends AbstractESSource { + + static type = 'ES_PEW_PEW'; + static title = i18n.translate('xpack.maps.source.pewPew', { + defaultMessage: 'pew-pew' + }); + static description = i18n.translate('xpack.maps.source.pewPew', { + defaultMessage: 'Graph connections between sources and destinations' + }); + + static createDescriptor({ indexPatternId, sourceGeoField, destGeoField }) { + return { + type: ESPewPewSource.type, + id: uuid(), + indexPatternId: indexPatternId, + sourceGeoField, + destGeoField + }; + } + + static renderEditor({ onPreviewSource, inspectorAdapters }) { + const onSourceConfigChange = (sourceConfig) => { + if (!sourceConfig) { + onPreviewSource(null); + return; + } + + const sourceDescriptor = ESPewPewSource.createDescriptor(sourceConfig); + const source = new ESPewPewSource(sourceDescriptor, inspectorAdapters); + onPreviewSource(source); + }; + + return (); + } + + isFilterByMapBounds() { + return true; + } + + isJoinable() { + return false; + } + + async getSupportedShapeTypes() { + return [VECTOR_SHAPE_TYPES.LINE]; + } + + async getImmutableProperties() { + let indexPatternTitle = this._descriptor.indexPatternId; + try { + const indexPattern = await this._getIndexPattern(); + indexPatternTitle = indexPattern.title; + } catch (error) { + // ignore error, title will just default to id + } + + return [ + { + label: getDataSourceLabel(), + value: ESPewPewSource.title + }, + { + label: i18n.translate('xpack.maps.source.pewPew.indexPatternLabel', { + defaultMessage: 'Index pattern' + }), + value: indexPatternTitle }, + { + label: i18n.translate('xpack.maps.source.pewPew.sourceGeoFieldLabel', { + defaultMessage: 'Source' + }), + value: this._descriptor.sourceGeoField + }, + { + label: i18n.translate('xpack.maps.source.pewPew.destGeoFieldLabel', { + defaultMessage: 'Destination' + }), + value: this._descriptor.destGeoField + }, + ]; + } + + createDefaultLayer(options) { + const styleDescriptor = VectorStyle.createDescriptor({ + lineColor: { + type: VectorStyle.STYLE_TYPE.DYNAMIC, + options: { + field: { + label: COUNT_PROP_LABEL, + name: COUNT_PROP_NAME, + origin: SOURCE_DATA_ID_ORIGIN + }, + color: 'Blues' + } + }, + lineWidth: { + type: VectorStyle.STYLE_TYPE.DYNAMIC, + options: { + field: { + label: COUNT_PROP_LABEL, + name: COUNT_PROP_NAME, + origin: SOURCE_DATA_ID_ORIGIN + }, + minSize: 4, + maxSize: 32, + } + } + }); + + return new VectorLayer({ + layerDescriptor: VectorLayer.createDescriptor({ + ...options, + sourceDescriptor: this._descriptor, + style: styleDescriptor + }), + source: this, + style: new VectorStyle(styleDescriptor, this) + }); + } + + getGeoGridPrecision(zoom) { + const targetGeotileLevel = Math.ceil(zoom) + 2; + return Math.min(targetGeotileLevel, MAX_GEOTILE_LEVEL); + } + + async getGeoJsonWithMeta(layerName, searchFilters) { + const searchSource = await this._makeSearchSource(searchFilters, 0); + searchSource.setField('aggs', { + destSplit: { + terms: { + script: { + source: `doc['${this._descriptor.destGeoField}'].value.toString()`, + lang: 'painless' + }, + order: { + _count: 'desc' + }, + size: 100 + }, + aggs: { + sourceGrid: { + geotile_grid: { + field: this._descriptor.sourceGeoField, + precision: searchFilters.geogridPrecision, + }, + aggs: { + sourceCentroid: { + geo_centroid: { + field: this._descriptor.sourceGeoField + } + } + } + } + } + } + }); + + const esResponse = await this._runEsQuery(layerName, searchSource, i18n.translate('xpack.maps.source.esGrid.inspectorDescription', { + defaultMessage: 'Elasticsearch geo grid aggregation request' + })); + + const { featureCollection } = convertToLines(esResponse); + + return { + data: featureCollection, + meta: { + areResultsTrimmed: false + } + }; + } + + async _getGeoField() { + const indexPattern = await this._getIndexPattern(); + const geoField = indexPattern.fields.byName[this._descriptor.destGeoField]; + if (!geoField) { + throw new Error(i18n.translate('xpack.maps.source.esSource.noGeoFieldErrorMessage', { + defaultMessage: `Index pattern {indexPatternTitle} no longer contains the geo field {geoField}`, + values: { indexPatternTitle: indexPattern.title, geoField: this._descriptor.geoField } + })); + } + return geoField; + } +} From 3887d7ef9e94b7e1cdaeaec1082833aa2615d357 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Thu, 18 Jul 2019 16:00:45 -0600 Subject: [PATCH 02/16] refetch data on zoom level change --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index 553aa5a7370c5..cac4ae6f77a1b 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -64,6 +64,10 @@ export class ESPewPewSource extends AbstractESSource { return false; } + isGeoGridPrecisionAware() { + return true; + } + async getSupportedShapeTypes() { return [VECTOR_SHAPE_TYPES.LINE]; } From f730139ab123e184b9b7a2ad46cff8d7b9234e26 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Fri, 16 Aug 2019 15:44:41 -0600 Subject: [PATCH 03/16] add metric aggs to request --- .../es_geo_grid_source/es_geo_grid_source.js | 12 +-- .../es_pew_pew_source/convert_to_lines.js | 7 ++ .../es_pew_pew_source/es_pew_pew_source.js | 56 ++++++++++++- .../es_pew_pew_source/update_source_editor.js | 84 +++++++++++++++++++ .../maps/public/layers/sources/es_source.js | 14 ++-- 5 files changed, 151 insertions(+), 22 deletions(-) create mode 100644 x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js index 0b8b9aceae05a..31ddd799cce2f 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js @@ -8,7 +8,7 @@ import React from 'react'; import uuid from 'uuid/v4'; import { VECTOR_SHAPE_TYPES } from '../vector_feature_types'; -import { AbstractESSource } from '../es_source'; +import { AbstractESSource, COUNT_PROP_LABEL, COUNT_PROP_NAME } from '../es_source'; import { HeatmapLayer } from '../../heatmap_layer'; import { VectorLayer } from '../../vector_layer'; import { Schemas } from 'ui/vis/editors/default/schemas'; @@ -24,8 +24,6 @@ import { SOURCE_DATA_ID_ORIGIN, ES_GEO_GRID } from '../../../../common/constants import { i18n } from '@kbn/i18n'; import { getDataSourceLabel } from '../../../../common/i18n_getters'; -const COUNT_PROP_LABEL = 'count'; -const COUNT_PROP_NAME = 'doc_count'; const MAX_GEOTILE_LEVEL = 29; const aggSchemas = new Schemas([ @@ -209,14 +207,6 @@ export class ESGeoGridSource extends AbstractESSource { return true; } - _formatMetricKey(metric) { - return metric.type !== 'count' ? `${metric.type}_of_${metric.field}` : COUNT_PROP_NAME; - } - - _formatMetricLabel(metric) { - return metric.type !== 'count' ? `${metric.type} of ${metric.field}` : COUNT_PROP_LABEL; - } - _makeAggConfigs(precision) { const metricAggConfigs = this.getMetricFields().map(metric => { const metricAggConfig = { diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js index 89cf0b8bc5f67..39ea3f4989e38 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js @@ -32,6 +32,13 @@ export function convertToLines(esResonse) { ...rest } = sourceBuckets[j]; + // flatten metrics + Object.keys(rest).forEach(key => { + if (_.has(rest[key], 'value')) { + rest[key] = rest[key].value; + } + }); + lineFeatures.push({ type: 'Feature', geometry: { diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index cac4ae6f77a1b..418cf11c53117 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -8,19 +8,34 @@ import React from 'react'; import uuid from 'uuid/v4'; import { VECTOR_SHAPE_TYPES } from '../vector_feature_types'; -import { AbstractESSource } from '../es_source'; +import { AbstractESSource, COUNT_PROP_LABEL, COUNT_PROP_NAME } from '../es_source'; import { VectorLayer } from '../../vector_layer'; import { CreateSourceEditor } from './create_source_editor'; +import { UpdateSourceEditor } from './update_source_editor'; import { VectorStyle } from '../../styles/vector_style'; import { i18n } from '@kbn/i18n'; import { SOURCE_DATA_ID_ORIGIN } from '../../../../common/constants'; import { getDataSourceLabel } from '../../../../common/i18n_getters'; import { convertToLines } from './convert_to_lines'; +import { Schemas } from 'ui/vis/editors/default/schemas'; +import { AggConfigs } from 'ui/vis/agg_configs'; -const COUNT_PROP_LABEL = 'count'; -const COUNT_PROP_NAME = 'doc_count'; const MAX_GEOTILE_LEVEL = 29; +const aggSchemas = new Schemas([ + { + group: 'metrics', + name: 'metric', + title: 'Value', + min: 1, + max: Infinity, + aggFilter: ['avg', 'count', 'max', 'min', 'sum'], + defaults: [ + { schema: 'metric', type: 'count' } + ] + } +]); + export class ESPewPewSource extends AbstractESSource { static type = 'ES_PEW_PEW'; @@ -56,6 +71,16 @@ export class ESPewPewSource extends AbstractESSource { return (); } + renderSourceSettingsEditor({ onChange }) { + return ( + + ); + } + isFilterByMapBounds() { return true; } @@ -68,6 +93,12 @@ export class ESPewPewSource extends AbstractESSource { return true; } + async getNumberFields() { + return this.getMetricFields().map(({ propertyKey: name, propertyLabel: label }) => { + return { label, name }; + }); + } + async getSupportedShapeTypes() { return [VECTOR_SHAPE_TYPES.LINE]; } @@ -150,6 +181,22 @@ export class ESPewPewSource extends AbstractESSource { } async getGeoJsonWithMeta(layerName, searchFilters) { + const indexPattern = await this._getIndexPattern(); + const metricAggConfigs = this.getMetricFields().map(metric => { + const metricAggConfig = { + id: metric.propertyKey, + enabled: true, + type: metric.type, + schema: 'metric', + params: {} + }; + if (metric.type !== 'count') { + metricAggConfig.params = { field: metric.field }; + } + return metricAggConfig; + }); + const aggConfigs = new AggConfigs(indexPattern, metricAggConfigs, aggSchemas.all); + const searchSource = await this._makeSearchSource(searchFilters, 0); searchSource.setField('aggs', { destSplit: { @@ -174,7 +221,8 @@ export class ESPewPewSource extends AbstractESSource { geo_centroid: { field: this._descriptor.sourceGeoField } - } + }, + ...aggConfigs.toDsl() } } } diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js new file mode 100644 index 0000000000000..0ddce40c68a91 --- /dev/null +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js @@ -0,0 +1,84 @@ +/* + * 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, { Fragment, Component } from 'react'; + +import { MetricsEditor } from '../../../components/metrics_editor'; +import { indexPatternService } from '../../../kibana_services'; +import { i18n } from '@kbn/i18n'; +import { EuiFormRow } from '@elastic/eui'; + +export class UpdateSourceEditor extends Component { + + state = { + fields: null, + }; + + componentDidMount() { + this._isMounted = true; + this._loadFields(); + } + + componentWillUnmount() { + this._isMounted = false; + } + + async _loadFields() { + let indexPattern; + try { + indexPattern = await indexPatternService.get(this.props.indexPatternId); + } catch (err) { + if (this._isMounted) { + this.setState({ + loadError: i18n.translate('xpack.maps.source.pewPew.noIndexPatternErrorMessage', { + defaultMessage: `Unable to find Index pattern {id}`, + values: { + id: this.props.indexPatternId + } + }) + }); + } + return; + } + + if (!this._isMounted) { + return; + } + + this.setState({ fields: indexPattern.fields }); + } + + _onMetricsChange = (metrics) => { + this.props.onChange({ propName: 'metrics', value: metrics }); + }; + + _renderMetricsEditor() { + return ( + +
+ +
+
+ ); + } + + render() { + return ( + + {this._renderMetricsEditor()} + + ); + } +} diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js index 6d64b468d14f7..f7a5a7f071f54 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js @@ -21,6 +21,9 @@ import uuid from 'uuid/v4'; import { copyPersistentState } from '../../reducers/util'; import { ES_GEO_FIELD_TYPE } from '../../../common/constants'; +const COUNT_PROP_LABEL = 'count'; +const COUNT_PROP_NAME = 'doc_count'; + export class AbstractESSource extends AbstractVectorSource { static icon = 'logoElasticsearch'; @@ -73,12 +76,12 @@ export class AbstractESSource extends AbstractVectorSource { return metrics; } - _formatMetricKey() { - throw new Error('should implement'); + _formatMetricKey(metric) { + return metric.type !== 'count' ? `${metric.type}_of_${metric.field}` : COUNT_PROP_NAME; } - _formatMetricLabel() { - throw new Error('should implement'); + _formatMetricLabel(metric) { + return metric.type !== 'count' ? `${metric.type} of ${metric.field}` : COUNT_PROP_LABEL; } getMetricFields() { @@ -104,7 +107,6 @@ export class AbstractESSource extends AbstractVectorSource { return properties; } - const metricFields = this.getMetricFields(); const tooltipProperties = []; metricFields.forEach((metricField) => { @@ -130,7 +132,6 @@ export class AbstractESSource extends AbstractVectorSource { } - async _runEsQuery(requestName, searchSource, requestDescription) { try { return await fetchSearchSourceAndRecordWithInspector({ @@ -290,5 +291,4 @@ export class AbstractESSource extends AbstractVectorSource { getId() { return this._descriptor.id; } - } From f8d4a27482bc2090501b3b7b33917ea6205482b2 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Fri, 16 Aug 2019 16:02:58 -0600 Subject: [PATCH 04/16] fix bug where initial draw did not have styles set up --- .../es_geo_grid_source/es_geo_grid_source.js | 12 +++++++++++- .../sources/es_pew_pew_source/es_pew_pew_source.js | 12 +++++++++++- .../maps/public/layers/sources/es_source.js | 14 +++++++------- 3 files changed, 29 insertions(+), 9 deletions(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js index 31ddd799cce2f..0b8b9aceae05a 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_geo_grid_source/es_geo_grid_source.js @@ -8,7 +8,7 @@ import React from 'react'; import uuid from 'uuid/v4'; import { VECTOR_SHAPE_TYPES } from '../vector_feature_types'; -import { AbstractESSource, COUNT_PROP_LABEL, COUNT_PROP_NAME } from '../es_source'; +import { AbstractESSource } from '../es_source'; import { HeatmapLayer } from '../../heatmap_layer'; import { VectorLayer } from '../../vector_layer'; import { Schemas } from 'ui/vis/editors/default/schemas'; @@ -24,6 +24,8 @@ import { SOURCE_DATA_ID_ORIGIN, ES_GEO_GRID } from '../../../../common/constants import { i18n } from '@kbn/i18n'; import { getDataSourceLabel } from '../../../../common/i18n_getters'; +const COUNT_PROP_LABEL = 'count'; +const COUNT_PROP_NAME = 'doc_count'; const MAX_GEOTILE_LEVEL = 29; const aggSchemas = new Schemas([ @@ -207,6 +209,14 @@ export class ESGeoGridSource extends AbstractESSource { return true; } + _formatMetricKey(metric) { + return metric.type !== 'count' ? `${metric.type}_of_${metric.field}` : COUNT_PROP_NAME; + } + + _formatMetricLabel(metric) { + return metric.type !== 'count' ? `${metric.type} of ${metric.field}` : COUNT_PROP_LABEL; + } + _makeAggConfigs(precision) { const metricAggConfigs = this.getMetricFields().map(metric => { const metricAggConfig = { diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index 418cf11c53117..a27bd35484938 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -8,7 +8,7 @@ import React from 'react'; import uuid from 'uuid/v4'; import { VECTOR_SHAPE_TYPES } from '../vector_feature_types'; -import { AbstractESSource, COUNT_PROP_LABEL, COUNT_PROP_NAME } from '../es_source'; +import { AbstractESSource } from '../es_source'; import { VectorLayer } from '../../vector_layer'; import { CreateSourceEditor } from './create_source_editor'; import { UpdateSourceEditor } from './update_source_editor'; @@ -20,6 +20,8 @@ import { convertToLines } from './convert_to_lines'; import { Schemas } from 'ui/vis/editors/default/schemas'; import { AggConfigs } from 'ui/vis/agg_configs'; +const COUNT_PROP_LABEL = 'count'; +const COUNT_PROP_NAME = 'doc_count'; const MAX_GEOTILE_LEVEL = 29; const aggSchemas = new Schemas([ @@ -243,6 +245,14 @@ export class ESPewPewSource extends AbstractESSource { }; } + _formatMetricKey(metric) { + return metric.type !== 'count' ? `${metric.type}_of_${metric.field}` : COUNT_PROP_NAME; + } + + _formatMetricLabel(metric) { + return metric.type !== 'count' ? `${metric.type} of ${metric.field}` : COUNT_PROP_LABEL; + } + async _getGeoField() { const indexPattern = await this._getIndexPattern(); const geoField = indexPattern.fields.byName[this._descriptor.destGeoField]; diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js index f7a5a7f071f54..6d64b468d14f7 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_source.js @@ -21,9 +21,6 @@ import uuid from 'uuid/v4'; import { copyPersistentState } from '../../reducers/util'; import { ES_GEO_FIELD_TYPE } from '../../../common/constants'; -const COUNT_PROP_LABEL = 'count'; -const COUNT_PROP_NAME = 'doc_count'; - export class AbstractESSource extends AbstractVectorSource { static icon = 'logoElasticsearch'; @@ -76,12 +73,12 @@ export class AbstractESSource extends AbstractVectorSource { return metrics; } - _formatMetricKey(metric) { - return metric.type !== 'count' ? `${metric.type}_of_${metric.field}` : COUNT_PROP_NAME; + _formatMetricKey() { + throw new Error('should implement'); } - _formatMetricLabel(metric) { - return metric.type !== 'count' ? `${metric.type} of ${metric.field}` : COUNT_PROP_LABEL; + _formatMetricLabel() { + throw new Error('should implement'); } getMetricFields() { @@ -107,6 +104,7 @@ export class AbstractESSource extends AbstractVectorSource { return properties; } + const metricFields = this.getMetricFields(); const tooltipProperties = []; metricFields.forEach((metricField) => { @@ -132,6 +130,7 @@ export class AbstractESSource extends AbstractVectorSource { } + async _runEsQuery(requestName, searchSource, requestDescription) { try { return await fetchSearchSourceAndRecordWithInspector({ @@ -291,4 +290,5 @@ export class AbstractESSource extends AbstractVectorSource { getId() { return this._descriptor.id; } + } From 771038a83aa94524e05e4271bc2633478476f601 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Fri, 16 Aug 2019 16:31:31 -0600 Subject: [PATCH 05/16] make tooltips work --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index a27bd35484938..08dd7f9171cd8 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -264,4 +264,8 @@ export class ESPewPewSource extends AbstractESSource { } return geoField; } + + canFormatFeatureProperties() { + return true; + } } From 96394f3dfc8b213c1e921ae1559f433e44dcac3d Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Mon, 19 Aug 2019 15:52:57 -0600 Subject: [PATCH 06/16] fix import broken with merging master --- .../layers/sources/es_pew_pew_source/create_source_editor.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js index d728d444e7914..1b58fb09097f2 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js @@ -8,7 +8,7 @@ import _ from 'lodash'; import React, { Fragment, Component } from 'react'; import PropTypes from 'prop-types'; -import { IndexPatternSelect } from 'ui/index_patterns/components/index_pattern_select'; +import { IndexPatternSelect } from 'ui/index_patterns'; import { SingleFieldSelect } from '../../../components/single_field_select'; import { indexPatternService } from '../../../kibana_services'; import { i18n } from '@kbn/i18n'; From f7e6b1df576091140d8b8eb3744b00a6f2fc6b71 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Mon, 19 Aug 2019 16:06:05 -0600 Subject: [PATCH 07/16] use custom labels in tooltips --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index 08dd7f9171cd8..fd1f671eca2cc 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -217,6 +217,7 @@ export class ESPewPewSource extends AbstractESSource { geotile_grid: { field: this._descriptor.sourceGeoField, precision: searchFilters.geogridPrecision, + size: 500, }, aggs: { sourceCentroid: { @@ -268,4 +269,8 @@ export class ESPewPewSource extends AbstractESSource { canFormatFeatureProperties() { return true; } + + async filterAndFormatPropertiesToHtml(properties) { + return await this.filterAndFormatPropertiesToHtmlForMetricFields(properties); + } } From b5b18b3db9ab71f8d43009a7b55adf5c84cba35f Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Tue, 27 Aug 2019 06:37:02 -0600 Subject: [PATCH 08/16] remove duplicate keys and clean up title and desc wording --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index fd1f671eca2cc..16215d5be9666 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -41,11 +41,11 @@ const aggSchemas = new Schemas([ export class ESPewPewSource extends AbstractESSource { static type = 'ES_PEW_PEW'; - static title = i18n.translate('xpack.maps.source.pewPew', { - defaultMessage: 'pew-pew' + static title = i18n.translate('xpack.maps.source.pewPewTitle', { + defaultMessage: 'Connections' }); - static description = i18n.translate('xpack.maps.source.pewPew', { - defaultMessage: 'Graph connections between sources and destinations' + static description = i18n.translate('xpack.maps.source.pewPewDescription', { + defaultMessage: 'Graph source-destination connections. Destinations grouped in grids.' }); static createDescriptor({ indexPatternId, sourceGeoField, destGeoField }) { From 2934f2b0c504b169e4aaf937c2276bf4d60963cc Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Tue, 27 Aug 2019 06:47:11 -0600 Subject: [PATCH 09/16] update source description --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index 16215d5be9666..73aadcb1dcaca 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -45,7 +45,7 @@ export class ESPewPewSource extends AbstractESSource { defaultMessage: 'Connections' }); static description = i18n.translate('xpack.maps.source.pewPewDescription', { - defaultMessage: 'Graph source-destination connections. Destinations grouped in grids.' + defaultMessage: 'Graph aggregated source-destination connections. Sources are grouped into grids for each destination.' }); static createDescriptor({ indexPatternId, sourceGeoField, destGeoField }) { From b2299d3b18d8fe1b3190702775d37c6b0cdf451c Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Tue, 27 Aug 2019 09:01:14 -0600 Subject: [PATCH 10/16] update references migration to extract references for pew pew source --- .../legacy/plugins/maps/common/constants.js | 1 + .../maps/common/migrations/references.js | 4 +- .../maps/common/migrations/references.test.js | 46 ++++++++++++++++++- .../es_pew_pew_source/es_pew_pew_source.js | 4 +- 4 files changed, 50 insertions(+), 5 deletions(-) diff --git a/x-pack/legacy/plugins/maps/common/constants.js b/x-pack/legacy/plugins/maps/common/constants.js index 65234bcad6695..5af3f8cfc0714 100644 --- a/x-pack/legacy/plugins/maps/common/constants.js +++ b/x-pack/legacy/plugins/maps/common/constants.js @@ -43,6 +43,7 @@ export const EMS_TMS = 'EMS_TMS'; export const EMS_FILE = 'EMS_FILE'; export const ES_GEO_GRID = 'ES_GEO_GRID'; export const ES_SEARCH = 'ES_SEARCH'; +export const ES_PEW_PEW = 'ES_PEW_PEW'; export const SOURCE_DATA_ID_ORIGIN = 'source'; export const GEOJSON_FILE = 'GEOJSON_FILE'; diff --git a/x-pack/legacy/plugins/maps/common/migrations/references.js b/x-pack/legacy/plugins/maps/common/migrations/references.js index a86abdf27a1f1..d235427f54e61 100644 --- a/x-pack/legacy/plugins/maps/common/migrations/references.js +++ b/x-pack/legacy/plugins/maps/common/migrations/references.js @@ -7,11 +7,11 @@ // Can not use public Layer classes to extract references since this logic must run in both client and server. import _ from 'lodash'; -import { ES_GEO_GRID, ES_SEARCH } from '../constants'; +import { ES_GEO_GRID, ES_SEARCH, ES_PEW_PEW } from '../constants'; function doesSourceUseIndexPattern(layerDescriptor) { const sourceType = _.get(layerDescriptor, 'sourceDescriptor.type'); - return sourceType === ES_GEO_GRID || sourceType === ES_SEARCH; + return sourceType === ES_GEO_GRID || sourceType === ES_SEARCH || sourceType === ES_PEW_PEW; } export function extractReferences({ attributes, references = [] }) { diff --git a/x-pack/legacy/plugins/maps/common/migrations/references.test.js b/x-pack/legacy/plugins/maps/common/migrations/references.test.js index 362d6ed718e20..cede1fd6e4da0 100644 --- a/x-pack/legacy/plugins/maps/common/migrations/references.test.js +++ b/x-pack/legacy/plugins/maps/common/migrations/references.test.js @@ -7,7 +7,7 @@ /* eslint max-len: 0 */ import { extractReferences, injectReferences } from './references'; -import { ES_GEO_GRID, ES_SEARCH } from '../constants'; +import { ES_GEO_GRID, ES_SEARCH, ES_PEW_PEW } from '../constants'; const layerListJSON = { esSearchSource: { @@ -21,6 +21,10 @@ const layerListJSON = { join: { withIndexPatternId: '[{\"joins\":[{\"right\":{\"indexPatternId\":\"e20b2a30-f735-11e8-8ce0-9723965e01e3\"}}]}]', withIndexPatternRef: '[{\"joins\":[{\"right\":{\"indexPatternRefName\":\"layer_0_join_0_index_pattern\"}}]}]', + }, + pewPewSource: { + withIndexPatternId: `[{\"sourceDescriptor\":{\"type\":\"${ES_PEW_PEW}\",\"indexPatternId\":\"c698b940-e149-11e8-a35a-370a8516603a\"}}]`, + withIndexPatternRef: `[{\"sourceDescriptor\":{\"type\":\"${ES_PEW_PEW}\",\"indexPatternRefName\":\"layer_0_source_index_pattern\"}}]`, } }; @@ -78,6 +82,26 @@ describe('extractReferences', () => { }); }); + test('Should extract index-pattern reference from ES pew pew source descriptor', () => { + const attributes = { + title: 'my map', + layerListJSON: layerListJSON.pewPewSource.withIndexPatternId, + }; + expect(extractReferences({ attributes })).toEqual({ + attributes: { + title: 'my map', + layerListJSON: layerListJSON.pewPewSource.withIndexPatternRef, + }, + references: [ + { + id: 'c698b940-e149-11e8-a35a-370a8516603a', + name: 'layer_0_source_index_pattern', + type: 'index-pattern', + } + ], + }); + }); + test('Should extract index-pattern reference from joins', () => { const attributes = { title: 'my map', @@ -151,6 +175,26 @@ describe('injectReferences', () => { }); }); + test('Should inject index-pattern reference into ES pew pew source descriptor', () => { + const attributes = { + title: 'my map', + layerListJSON: layerListJSON.pewPewSource.withIndexPatternRef, + }; + const references = [ + { + id: 'c698b940-e149-11e8-a35a-370a8516603a', + name: 'layer_0_source_index_pattern', + type: 'index-pattern', + } + ]; + expect(injectReferences({ attributes, references })).toEqual({ + attributes: { + title: 'my map', + layerListJSON: layerListJSON.pewPewSource.withIndexPatternId, + } + }); + }); + test('Should inject index-pattern reference into joins', () => { const attributes = { title: 'my map', diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index 73aadcb1dcaca..e412ca1227b14 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -14,7 +14,7 @@ import { CreateSourceEditor } from './create_source_editor'; import { UpdateSourceEditor } from './update_source_editor'; import { VectorStyle } from '../../styles/vector_style'; import { i18n } from '@kbn/i18n'; -import { SOURCE_DATA_ID_ORIGIN } from '../../../../common/constants'; +import { SOURCE_DATA_ID_ORIGIN, ES_PEW_PEW } from '../../../../common/constants'; import { getDataSourceLabel } from '../../../../common/i18n_getters'; import { convertToLines } from './convert_to_lines'; import { Schemas } from 'ui/vis/editors/default/schemas'; @@ -40,7 +40,7 @@ const aggSchemas = new Schemas([ export class ESPewPewSource extends AbstractESSource { - static type = 'ES_PEW_PEW'; + static type = ES_PEW_PEW; static title = i18n.translate('xpack.maps.source.pewPewTitle', { defaultMessage: 'Connections' }); From d2fd34fc0fd1f39d8a5aa4d6ec0c6ab2df596dbc Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Tue, 27 Aug 2019 09:14:12 -0600 Subject: [PATCH 11/16] add percy visual test to ensure pew pew map data is fetch, processed, and visualized as expected --- .../es_archives/maps/data/data.json | 48 ++++++++++++ .../es_archives/maps/data/mappings.json | 23 ++++++ .../es_archives/maps/kibana/data.json | 76 +++++++++++++++++++ .../tests/maps/vector_styling.js | 14 ++++ 4 files changed, 161 insertions(+) diff --git a/x-pack/test/functional/es_archives/maps/data/data.json b/x-pack/test/functional/es_archives/maps/data/data.json index 41f4dda80b5e5..e44b19504af38 100644 --- a/x-pack/test/functional/es_archives/maps/data/data.json +++ b/x-pack/test/functional/es_archives/maps/data/data.json @@ -388,3 +388,51 @@ } } } + +{ + "type": "doc", + "value": { + "id": "1", + "index": "connections", + "source": { + "source" : [-65, 40], + "destination" : "drm3btev3e86" + } + } +} + +{ + "type": "doc", + "value": { + "id": "2", + "index": "connections", + "source": { + "source" : [-75.00001, 41], + "destination" : { "lon": -71, "lat": 40 } + } + } +} + +{ + "type": "doc", + "value": { + "id": "3", + "index": "connections", + "source": { + "source" : [-75, 41.00003], + "destination" : [ -71.000000000000000, 40 ] + } + } +} + +{ + "type": "doc", + "value": { + "id": "4", + "index": "connections", + "source": { + "source" : [-75, 41.00004], + "destination" : "40.000,-71" + } + } +} diff --git a/x-pack/test/functional/es_archives/maps/data/mappings.json b/x-pack/test/functional/es_archives/maps/data/mappings.json index 2c394b8e3b75b..7e642ca49f3ae 100644 --- a/x-pack/test/functional/es_archives/maps/data/mappings.json +++ b/x-pack/test/functional/es_archives/maps/data/mappings.json @@ -109,3 +109,26 @@ } } } + +{ + "type": "index", + "value": { + "index": "connections", + "mappings": { + "properties": { + "source": { + "type": "geo_point" + }, + "destination": { + "type": "geo_point" + } + } + }, + "settings": { + "index": { + "number_of_replicas": "0", + "number_of_shards": "1" + } + } + } +} diff --git a/x-pack/test/functional/es_archives/maps/kibana/data.json b/x-pack/test/functional/es_archives/maps/kibana/data.json index 8dc4f08f59b82..d9baecf23d852 100644 --- a/x-pack/test/functional/es_archives/maps/kibana/data.json +++ b/x-pack/test/functional/es_archives/maps/kibana/data.json @@ -109,6 +109,26 @@ } } +{ + "type": "doc", + "value": { + "id": "index-pattern:dedd3180-c8d8-11e9-b36c-81f9f9da524f", + "index": ".kibana", + "source": { + "index-pattern" : { + "title" : "connections", + "fields" : "[{\"name\":\"_id\",\"type\":\"string\",\"esTypes\":[\"_id\"],\"count\":0,\"scripted\":false,\"searchable\":true,\"aggregatable\":true,\"readFromDocValues\":false},{\"name\":\"_index\",\"type\":\"string\",\"esTypes\":[\"_index\"],\"count\":0,\"scripted\":false,\"searchable\":true,\"aggregatable\":true,\"readFromDocValues\":false},{\"name\":\"_score\",\"type\":\"number\",\"count\":0,\"scripted\":false,\"searchable\":false,\"aggregatable\":false,\"readFromDocValues\":false},{\"name\":\"_source\",\"type\":\"_source\",\"esTypes\":[\"_source\"],\"count\":0,\"scripted\":false,\"searchable\":false,\"aggregatable\":false,\"readFromDocValues\":false},{\"name\":\"_type\",\"type\":\"string\",\"esTypes\":[\"_type\"],\"count\":0,\"scripted\":false,\"searchable\":true,\"aggregatable\":true,\"readFromDocValues\":false},{\"name\":\"destination\",\"type\":\"geo_point\",\"esTypes\":[\"geo_point\"],\"count\":0,\"scripted\":false,\"searchable\":true,\"aggregatable\":true,\"readFromDocValues\":true},{\"name\":\"source\",\"type\":\"geo_point\",\"esTypes\":[\"geo_point\"],\"count\":0,\"scripted\":false,\"searchable\":true,\"aggregatable\":true,\"readFromDocValues\":true}]" + }, + "type" : "index-pattern", + "references" : [ ], + "migrationVersion" : { + "index-pattern" : "6.5.0" + }, + "updated_at" : "2019-08-27T14:42:20.061Z" + } + } +} + { "type": "doc", "value": { @@ -617,6 +637,62 @@ } } +{ + "type": "doc", + "value": { + "id": "map:3c9949f0-c8dc-11e9-9ea1-8b2710d4a86b", + "index": ".kibana", + "source": { + "map" : { + "title" : "pew pew demo", + "description" : "", + "mapStateJSON" : "{\"zoom\":5.5,\"center\":{\"lon\":-71.34293,\"lat\":40.33097},\"timeFilters\":{\"from\":\"now-15m\",\"to\":\"now\"},\"refreshConfig\":{\"isPaused\":false,\"interval\":0},\"query\":{\"query\":\"\",\"language\":\"kuery\"},\"filters\":[]}", + "layerListJSON" : "[{\"style\":{\"type\":\"VECTOR\",\"properties\":{\"fillColor\":{\"type\":\"STATIC\",\"options\":{\"color\":\"#1EA593\"}},\"lineColor\":{\"type\":\"DYNAMIC\",\"options\":{\"field\":{\"label\":\"count\",\"name\":\"doc_count\",\"origin\":\"source\"},\"color\":\"Blues\"}},\"lineWidth\":{\"type\":\"DYNAMIC\",\"options\":{\"field\":{\"label\":\"count\",\"name\":\"doc_count\",\"origin\":\"source\"},\"minSize\":4,\"maxSize\":32}},\"iconSize\":{\"type\":\"STATIC\",\"options\":{\"size\":10}},\"iconOrientation\":{\"type\":\"STATIC\",\"options\":{\"orientation\":0}},\"symbol\":{\"options\":{\"symbolizeAs\":\"circle\",\"symbolId\":\"airfield\"}}}},\"sourceDescriptor\":{\"type\":\"ES_PEW_PEW\",\"id\":\"d7ca27a8-dba2-4f5b-857c-8f529511ad81\",\"sourceGeoField\":\"source\",\"destGeoField\":\"destination\",\"indexPatternRefName\":\"layer_0_source_index_pattern\"},\"id\":\"67c1de2c-2fc5-4425-8983-094b589afe61\",\"label\":null,\"minZoom\":0,\"maxZoom\":24,\"alpha\":0.75,\"visible\":true,\"applyGlobalQuery\":true,\"type\":\"VECTOR\"}]", + "uiStateJSON" : "{\"isLayerTOCOpen\":true,\"openTOCDetails\":[]}", + "bounds" : { + "type" : "Polygon", + "coordinates" : [ + [ + [ + -83.72374, + 44.30172 + ], + [ + -83.72374, + 36.11211 + ], + [ + -58.96211, + 36.11211 + ], + [ + -58.96211, + 44.30172 + ], + [ + -83.72374, + 44.30172 + ] + ] + ] + } + }, + "type" : "map", + "references" : [ + { + "name" : "layer_0_source_index_pattern", + "type" : "index-pattern", + "id" : "dedd3180-c8d8-11e9-b36c-81f9f9da524f" + } + ], + "migrationVersion" : { + "map" : "7.4.0" + }, + "updated_at" : "2019-08-27T15:06:24.654Z" + } + } +} + { "type": "doc", "value": { diff --git a/x-pack/test/visual_regression/tests/maps/vector_styling.js b/x-pack/test/visual_regression/tests/maps/vector_styling.js index 9b398fd56610e..25a9e794219a2 100644 --- a/x-pack/test/visual_regression/tests/maps/vector_styling.js +++ b/x-pack/test/visual_regression/tests/maps/vector_styling.js @@ -35,5 +35,19 @@ export default function ({ getPageObjects, getService }) { }); }); + + describe('dynamic line coloring', () => { + before(async () => { + await PageObjects.maps.loadSavedMap('pew pew demo'); + await PageObjects.maps.enterFullScreen(); + await PageObjects.maps.closeLegend(); + }); + + // eslint-disable-next-line max-len + it('should symbolize pew pew lines', async () => { + await visualTesting.snapshot(); + }); + + }); }); } From ad5bdcab1b37a8c9f8b4b38023b14ff52be66dd9 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Tue, 27 Aug 2019 13:18:25 -0600 Subject: [PATCH 12/16] update title and description --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index e412ca1227b14..612eae5b99ccb 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -42,10 +42,10 @@ export class ESPewPewSource extends AbstractESSource { static type = ES_PEW_PEW; static title = i18n.translate('xpack.maps.source.pewPewTitle', { - defaultMessage: 'Connections' + defaultMessage: 'Source-destination connections' }); static description = i18n.translate('xpack.maps.source.pewPewDescription', { - defaultMessage: 'Graph aggregated source-destination connections. Sources are grouped into grids for each destination.' + defaultMessage: 'Aggregated data paths between the origin and destinations.' }); static createDescriptor({ indexPatternId, sourceGeoField, destGeoField }) { From 556c7a8203684d1f84b754500b771fcdf555407f Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Wed, 28 Aug 2019 11:21:58 -0600 Subject: [PATCH 13/16] use GEO_FIELD_TYPES in filterGeoField function --- .../layers/sources/es_pew_pew_source/create_source_editor.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js index 1b58fb09097f2..9f9789374274a 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/create_source_editor.js @@ -23,7 +23,7 @@ import { ES_GEO_FIELD_TYPE } from '../../../../common/constants'; const GEO_FIELD_TYPES = [ES_GEO_FIELD_TYPE.GEO_POINT]; function filterGeoField({ type }) { - return [ES_GEO_FIELD_TYPE.GEO_POINT].includes(type); + return GEO_FIELD_TYPES.includes(type); } export class CreateSourceEditor extends Component { From fa5f8d21260ddde9eaf0820e216dd8326dc160c4 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Wed, 28 Aug 2019 12:27:05 -0600 Subject: [PATCH 14/16] remove unneeded Fragment wrapper --- .../es_pew_pew_source/update_source_editor.js | 12 ++---------- 1 file changed, 2 insertions(+), 10 deletions(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js index 0ddce40c68a91..573366b5ef844 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/update_source_editor.js @@ -4,7 +4,7 @@ * you may not use this file except in compliance with the Elastic License. */ -import React, { Fragment, Component } from 'react'; +import React, { Component } from 'react'; import { MetricsEditor } from '../../../components/metrics_editor'; import { indexPatternService } from '../../../kibana_services'; @@ -55,7 +55,7 @@ export class UpdateSourceEditor extends Component { this.props.onChange({ propName: 'metrics', value: metrics }); }; - _renderMetricsEditor() { + render() { return ( ); } - - render() { - return ( - - {this._renderMetricsEditor()} - - ); - } } From 6ee0e0d904f04793623eb1617c61112d48993671 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Wed, 28 Aug 2019 12:27:47 -0600 Subject: [PATCH 15/16] fix typo --- .../layers/sources/es_pew_pew_source/convert_to_lines.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js index 39ea3f4989e38..ce5efb3b44176 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/convert_to_lines.js @@ -16,11 +16,11 @@ function parsePointFromKey(key) { return [lon, lat]; } -export function convertToLines(esResonse) { +export function convertToLines(esResponse) { const lineFeatures = []; - const destBuckets = _.get(esResonse, 'aggregations.destSplit.buckets', []); + const destBuckets = _.get(esResponse, 'aggregations.destSplit.buckets', []); for (let i = 0; i < destBuckets.length; i++) { const destBucket = destBuckets[i]; const dest = parsePointFromKey(destBucket.key); From 67028c6d05dcf037e208884568fdfd06c864e5c8 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Wed, 28 Aug 2019 12:32:33 -0600 Subject: [PATCH 16/16] update inspector description id and message --- .../layers/sources/es_pew_pew_source/es_pew_pew_source.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js index 612eae5b99ccb..e1b930c8ab39b 100644 --- a/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js +++ b/x-pack/legacy/plugins/maps/public/layers/sources/es_pew_pew_source/es_pew_pew_source.js @@ -232,8 +232,8 @@ export class ESPewPewSource extends AbstractESSource { } }); - const esResponse = await this._runEsQuery(layerName, searchSource, i18n.translate('xpack.maps.source.esGrid.inspectorDescription', { - defaultMessage: 'Elasticsearch geo grid aggregation request' + const esResponse = await this._runEsQuery(layerName, searchSource, i18n.translate('xpack.maps.source.pewPew.inspectorDescription', { + defaultMessage: 'Source-destination connections request' })); const { featureCollection } = convertToLines(esResponse);