Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[SIEM] Adds Connections (Pewpew) Map to Network Page #43965

Merged
merged 41 commits into from
Aug 29, 2019
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
9a624f3
Initial pass at working with new embeddables API
spong Aug 12, 2019
efddfa8
Merge branch 'master' of github.com:elastic/kibana into pewpew
XavierM Aug 14, 2019
1f8ae3d
wip
XavierM Aug 14, 2019
13ddb74
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 16, 2019
453fb71
Render and style + configuration fixes to embedded map
spong Aug 16, 2019
933e79a
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 19, 2019
1cf7d63
Adding support for KQL and timerange filtering
spong Aug 19, 2019
ffeeea1
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 21, 2019
7574604
Adding support for loading from saved object and surrounding error sc…
spong Aug 22, 2019
af3026b
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 22, 2019
f35c347
pew pew source
nreese Jul 18, 2019
d88908d
refetch data on zoom level change
nreese Jul 18, 2019
982400e
add metric aggs to request
nreese Aug 16, 2019
f52146c
fix bug where initial draw did not have styles set up
nreese Aug 16, 2019
789b855
make tooltips work
nreese Aug 16, 2019
d145054
fix import broken with merging master
nreese Aug 19, 2019
f7d8c87
use custom labels in tooltips
nreese Aug 19, 2019
bb5624e
Provide ability to create MapEmbeddable directly from a map configura…
nreese Aug 23, 2019
f557b2d
Now loading map from dynamic config based on configured index patterns
spong Aug 24, 2019
d54f843
Cleanup and consolidation
spong Aug 26, 2019
365c3df
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 26, 2019
6ad2406
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 26, 2019
226fdc7
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 26, 2019
6755b74
Updating error message, fixing missing-index loading bug, and hiding …
spong Aug 27, 2019
e969d8a
Updating test snapshot
spong Aug 27, 2019
2e5d7e9
Wired up to global refresh, fixed stale filter on property bug, and c…
spong Aug 27, 2019
fdf9cf3
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 27, 2019
9f641ba
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 28, 2019
0ff47e1
Increasing test coverage
spong Aug 28, 2019
bec1aa2
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 28, 2019
76f283b
Reverting changes from previous maps + bootstrap merge
spong Aug 28, 2019
b8bbba4
Changes from PR comments
spong Aug 28, 2019
a041dc2
Remaining comments from PR
spong Aug 29, 2019
c3772b0
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 29, 2019
785c389
Fixing error scenario flag
spong Aug 29, 2019
30bae0e
Merge branch 'master' of github.com:elastic/kibana into pewpew
Aug 29, 2019
df9c3ed
Merge branch 'master' of github.com:elastic/kibana into pewpew
Aug 29, 2019
e5ea5bd
Merge branch 'master' of github.com:elastic/kibana into pewpew
tsg Aug 29, 2019
3fcf73b
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 29, 2019
85dab2c
Resolving merge conflict
spong Aug 29, 2019
a60e76d
Merge branch 'master' of github.com:elastic/kibana into pewpew
spong Aug 29, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -59,11 +59,11 @@ export const dashboardInput: DashboardContainerInput = {
y: 0,
i: '822cd0f0-ce7c-419d-aeaa-1171cf452745',
},
type: 'visualization',
type: 'map',
explicitInput: {
id: '822cd0f0-ce7c-419d-aeaa-1171cf452745',
},
savedObjectId: '3fe22200-3dcb-11e8-8660-4d65aa086b3c',
savedObjectId: '53006120-b56c-11e9-bfa8-fd22ee6bf542',
},
'66f0a265-7b06-4974-accd-d05f74f7aa82': {
gridData: {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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';
Expand All @@ -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,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
/*
* 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];

// flatten metrics
Object.keys(rest).forEach(key => {
if (_.has(rest[key], 'value')) {
rest[key] = rest[key].value;
}
});

lineFeatures.push({
type: 'Feature',
geometry: {
type: 'LineString',
coordinates: [[sourceCentroid.location.lon, sourceCentroid.location.lat], dest]
},
properties: {
...rest
}
});
}
}

return {
featureCollection: {
type: 'FeatureCollection',
features: lineFeatures
}
};
}
Original file line number Diff line number Diff line change
@@ -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';
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 (
<Fragment>
<EuiFormRow label={i18n.translate('xpack.maps.source.pewPew.sourceGeoFieldLabel', {
defaultMessage: 'Source'
})}
>
<SingleFieldSelect
placeholder={i18n.translate('xpack.maps.source.pewPew.sourceGeoFieldPlaceholder', {
defaultMessage: 'Select source geo field'
})}
value={this.state.sourceGeoField}
onChange={this._onSourceGeoSelect}
filterField={filterGeoField}
fields={this.state.indexPattern ? this.state.indexPattern.fields : undefined}
/>
</EuiFormRow>

<EuiFormRow label={i18n.translate('xpack.maps.source.pewPew.destGeoFieldLabel', {
defaultMessage: 'Destination'
})}
>
<SingleFieldSelect
placeholder={i18n.translate('xpack.maps.source.pewPew.destGeoFieldPlaceholder', {
defaultMessage: 'Select destination geo field'
})}
value={this.state.destGeoField}
onChange={this._onDestGeoSelect}
filterField={filterGeoField}
fields={this.state.indexPattern ? this.state.indexPattern.fields : undefined}
/>
</EuiFormRow>
</Fragment>
);
}

_renderIndexPatternSelect() {
return (
<EuiFormRow label={i18n.translate('xpack.maps.source.pewPew.indexPatternLabel', {
defaultMessage: 'Index pattern'
})}
>
<IndexPatternSelect
indexPatternId={this.state.indexPatternId}
onChange={this.onIndexPatternSelect}
placeholder={i18n.translate('xpack.maps.source.pewPew.indexPatternPlaceholder', {
defaultMessage: 'Select index pattern'
})}
fieldTypes={GEO_FIELD_TYPES}
/>
</EuiFormRow>
);
}

render() {
let callout;
if (this.state.indexPattern && !this.state.indexPatternHasMultipleGeoFields) {
callout = (
<EuiCallOut
color="warning"
>
<p>
<FormattedMessage
id="xpack.maps.source.pewPew.noSourceAndDestDetails"
defaultMessage="Selected index pattern does not contain source and destination fields."
/>
</p>
</EuiCallOut>
);
}

return (
<Fragment>
{callout}
{this._renderIndexPatternSelect()}
{this._renderGeoSelects()}
</Fragment>
);
}
}
Loading