mirror of
https://github.com/elastic/kibana.git
synced 2025-04-24 09:48:58 -04:00
Use Joi schemas for validation instead of manual checks, separate server concerns from frontend code, and start enforcing snake case in simulate api
This commit is contained in:
parent
fb7804431d
commit
a6bf99a58f
13 changed files with 188 additions and 218 deletions
|
@ -0,0 +1,87 @@
|
|||
import expect from 'expect.js';
|
||||
import _ from 'lodash';
|
||||
import ingestSimulateApiToEsConverter from '../../converters/ingest_simulate_api_to_es_converter';
|
||||
|
||||
describe('ingestSimulateApiToEsConverter', function () {
|
||||
|
||||
it('populates the docs._source section and converts known processors', function () {
|
||||
|
||||
function buildSamplePipeline(input) {
|
||||
return {
|
||||
processors: [ { processor_id: 'processor1', type_id: 'set', target_field: 'bar', value: 'foo' } ],
|
||||
input: input
|
||||
};
|
||||
}
|
||||
|
||||
function buildExpected(input) {
|
||||
return {
|
||||
pipeline : {
|
||||
processors: [{
|
||||
set: {
|
||||
field: 'bar',
|
||||
tag: 'processor1',
|
||||
value: 'foo'
|
||||
}
|
||||
}]
|
||||
},
|
||||
'docs' : [
|
||||
{ '_source': input }
|
||||
]
|
||||
};
|
||||
}
|
||||
|
||||
let expected;
|
||||
let actual;
|
||||
|
||||
expected = buildExpected(undefined);
|
||||
actual = ingestSimulateApiToEsConverter(buildSamplePipeline(undefined));
|
||||
expect(actual).to.eql(expected);
|
||||
|
||||
expected = buildExpected('foo');
|
||||
actual = ingestSimulateApiToEsConverter(buildSamplePipeline('foo'));
|
||||
expect(actual).to.eql(expected);
|
||||
|
||||
expected = buildExpected({ foo: 'bar' });
|
||||
actual = ingestSimulateApiToEsConverter(buildSamplePipeline({ foo: 'bar' }));
|
||||
expect(actual).to.eql(expected);
|
||||
});
|
||||
|
||||
it('handles multiple processors', function () {
|
||||
const pipeline = {
|
||||
processors: [
|
||||
{ processor_id: 'processor1', type_id: 'set', target_field: 'bar', value: 'foo' },
|
||||
{ processor_id: 'processor2', type_id: 'set', target_field: 'bar', value: 'foo' },
|
||||
],
|
||||
input: {}
|
||||
};
|
||||
const expected = {
|
||||
'pipeline': {
|
||||
'processors': [
|
||||
{
|
||||
set: {
|
||||
field: 'bar',
|
||||
tag: 'processor1',
|
||||
value: 'foo'
|
||||
}
|
||||
},
|
||||
{
|
||||
set: {
|
||||
field: 'bar',
|
||||
tag: 'processor2',
|
||||
value: 'foo'
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
'docs': [
|
||||
{'_source': {}}
|
||||
]
|
||||
};
|
||||
|
||||
const actual = ingestSimulateApiToEsConverter(pipeline);
|
||||
|
||||
expect(actual).to.eql(expected);
|
||||
});
|
||||
|
||||
|
||||
});
|
|
@ -1,158 +0,0 @@
|
|||
const expect = require('expect.js');
|
||||
const _ = require('lodash');
|
||||
import { buildRequest } from '../ingest_simulate';
|
||||
|
||||
describe('buildRequest', function () {
|
||||
|
||||
const processorTypes = [
|
||||
{
|
||||
typeId: 'simple1',
|
||||
getDefinition: function (processor) {
|
||||
return {
|
||||
'modified_value': `modified_${processor.value}`
|
||||
};
|
||||
}
|
||||
},
|
||||
{
|
||||
typeId: 'simple2',
|
||||
getDefinition: function (processor) {
|
||||
return {
|
||||
'value1': processor.value,
|
||||
'value2': `${processor.typeId}-${processor.value}`
|
||||
};
|
||||
}
|
||||
}
|
||||
];
|
||||
|
||||
it('should throw an error if no processorTypes argument is passed or the argument is not a plain object', function () {
|
||||
expect(buildRequest).to.throwException(/requires a processorTypes object array argument/);
|
||||
expect(buildRequest).withArgs('').to.throwException(/requires a processorTypes object array argument/);
|
||||
expect(buildRequest).withArgs({}).to.throwException(/requires a processorTypes object array argument/);
|
||||
expect(buildRequest).withArgs([]).to.throwException(/requires a processorTypes object array argument/);
|
||||
});
|
||||
|
||||
it('should throw an error if no pipeline argument is passed or the argument is not a plain object', function () {
|
||||
expect(buildRequest).withArgs([{}], []).to.throwException(/requires a pipeline object argument/);
|
||||
});
|
||||
|
||||
it('should throw an error if pipeline contains no processors', function () {
|
||||
expect(buildRequest).withArgs([{}], {}).to.throwException(/pipeline contains no processors/);
|
||||
expect(buildRequest).withArgs([{}], { processors: 'foo' }).to.throwException(/pipeline contains no processors/);
|
||||
expect(buildRequest).withArgs([{}], { processors: {} }).to.throwException(/pipeline contains no processors/);
|
||||
expect(buildRequest).withArgs([{}], { processors: [] }).to.throwException(/pipeline contains no processors/);
|
||||
});
|
||||
|
||||
it('populates the docs._source section', function () {
|
||||
|
||||
function buildSamplePipeline(input) {
|
||||
return {
|
||||
processors: [ { typeId: 'simple1', value: 'foo' } ],
|
||||
input: input
|
||||
};
|
||||
}
|
||||
|
||||
function buildExpected(input) {
|
||||
return {
|
||||
'pipeline' : {
|
||||
'processors': [ { modified_value: 'modified_foo' } ]
|
||||
},
|
||||
'docs' : [
|
||||
{ '_source': input }
|
||||
]
|
||||
};
|
||||
}
|
||||
|
||||
let expected;
|
||||
let actual;
|
||||
|
||||
expected = buildExpected(undefined);
|
||||
actual = buildRequest(processorTypes, buildSamplePipeline(undefined));
|
||||
expect(actual).to.eql(expected);
|
||||
|
||||
expected = buildExpected('foo');
|
||||
actual = buildRequest(processorTypes, buildSamplePipeline('foo'));
|
||||
expect(actual).to.eql(expected);
|
||||
|
||||
expected = buildExpected({ foo: 'bar' });
|
||||
actual = buildRequest(processorTypes, buildSamplePipeline({ foo: 'bar' }));
|
||||
expect(actual).to.eql(expected);
|
||||
});
|
||||
|
||||
describe('populates the pipeline.processors section with type.getDefinition()', function () {
|
||||
|
||||
it(' - single processor type', function () {
|
||||
const pipeline = {
|
||||
processors: [ { typeId: 'simple1', value: 'foo' } ],
|
||||
input: {}
|
||||
};
|
||||
const expected = {
|
||||
'pipeline' : {
|
||||
'processors': [ { modified_value: 'modified_foo' } ]
|
||||
},
|
||||
'docs' : [
|
||||
{ '_source': {} }
|
||||
]
|
||||
};
|
||||
|
||||
const actual = buildRequest(processorTypes, pipeline);
|
||||
|
||||
expect(actual).to.eql(expected);
|
||||
});
|
||||
|
||||
it(' - multiple of same type of processor type', function () {
|
||||
const pipeline = {
|
||||
processors: [
|
||||
{ typeId: 'simple1', value: 'foo' },
|
||||
{ typeId: 'simple1', value: 'bar' },
|
||||
{ typeId: 'simple1', value: 'baz' }
|
||||
],
|
||||
input: {}
|
||||
};
|
||||
const expected = {
|
||||
'pipeline' : {
|
||||
'processors': [
|
||||
{ modified_value: 'modified_foo' },
|
||||
{ modified_value: 'modified_bar' },
|
||||
{ modified_value: 'modified_baz' }
|
||||
]
|
||||
},
|
||||
'docs' : [
|
||||
{ '_source': {} }
|
||||
]
|
||||
};
|
||||
|
||||
const actual = buildRequest(processorTypes, pipeline);
|
||||
|
||||
expect(actual).to.eql(expected);
|
||||
});
|
||||
|
||||
it(' - multiple processor types', function () {
|
||||
const pipeline = {
|
||||
processors: [
|
||||
{ typeId: 'simple1', value: 'foo' },
|
||||
{ typeId: 'simple2', value: 'bar' },
|
||||
{ typeId: 'simple1', value: 'baz' }
|
||||
],
|
||||
input: {}
|
||||
};
|
||||
const expected = {
|
||||
'pipeline' : {
|
||||
'processors': [
|
||||
{ modified_value: 'modified_foo' },
|
||||
{ value1: 'bar', value2: 'simple2-bar' },
|
||||
{ modified_value: 'modified_baz' }
|
||||
]
|
||||
},
|
||||
'docs' : [
|
||||
{ '_source': {} }
|
||||
]
|
||||
};
|
||||
|
||||
const actual = buildRequest(processorTypes, pipeline);
|
||||
|
||||
expect(actual).to.eql(expected);
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
});
|
|
@ -0,0 +1,9 @@
|
|||
export function set(processorApiDocument) {
|
||||
return {
|
||||
set: {
|
||||
tag: processorApiDocument.processor_id,
|
||||
field: processorApiDocument.target_field,
|
||||
value: processorApiDocument.value
|
||||
}
|
||||
};
|
||||
}
|
|
@ -0,0 +1,17 @@
|
|||
import _ from 'lodash';
|
||||
import * as ingestProcessorApiToEsConverters from './ingest_processor_api_to_es_converters';
|
||||
|
||||
export default function ingestSimulateApiToEsConverter(simulateApiDocument) {
|
||||
return {
|
||||
pipeline: {
|
||||
processors: _.map(simulateApiDocument.processors, (processor) => {
|
||||
return ingestProcessorApiToEsConverters[processor.type_id](processor);
|
||||
})
|
||||
},
|
||||
docs: [
|
||||
{
|
||||
_source: simulateApiDocument.input
|
||||
}
|
||||
]
|
||||
};
|
||||
}
|
|
@ -6,49 +6,11 @@ function translateError(esError) {
|
|||
return _.get(rootCause, 'reason') || _.get(rootCause, 'type');
|
||||
}
|
||||
|
||||
export function buildRequest(processorTypes, pipeline) {
|
||||
if (processorTypes === undefined ||
|
||||
!_.isArray(processorTypes) ||
|
||||
processorTypes.length === 0) {
|
||||
throw new Error('requires a processorTypes object array argument');
|
||||
}
|
||||
|
||||
if (pipeline === undefined || !_.isPlainObject(pipeline)) {
|
||||
throw new Error('requires a pipeline object argument');
|
||||
}
|
||||
|
||||
if (pipeline.processors === undefined ||
|
||||
!_.isArray(pipeline.processors) ||
|
||||
pipeline.processors.length === 0) {
|
||||
throw new Error('pipeline contains no processors');
|
||||
}
|
||||
|
||||
const processors = pipeline.processors;
|
||||
const body = {
|
||||
'pipeline': {
|
||||
'processors': []
|
||||
},
|
||||
'docs': [
|
||||
{
|
||||
_source: pipeline.input
|
||||
}
|
||||
]
|
||||
};
|
||||
|
||||
processors.forEach((processor) => {
|
||||
const processorType = _.find(processorTypes, { 'typeId': processor.typeId });
|
||||
const definition = processorType.getDefinition(processor);
|
||||
body.pipeline.processors.push(definition);
|
||||
});
|
||||
|
||||
return body;
|
||||
};
|
||||
|
||||
export function processResponse(pipeline, err, resp) {
|
||||
const results = pipeline.processors.map((processor) => {
|
||||
export function processResponse(simulateApiDocument, err, resp) {
|
||||
const results = simulateApiDocument.processors.map((processor) => {
|
||||
return {
|
||||
processorId: processor.processorId,
|
||||
output: processor.outputObject,
|
||||
processorId: processor.processor_id,
|
||||
output: processor.output_object,
|
||||
error: undefined
|
||||
};
|
||||
});
|
||||
|
|
|
@ -1,6 +0,0 @@
|
|||
import Joi from 'joi';
|
||||
|
||||
export default Joi.object({
|
||||
processorId: Joi.string().required(),
|
||||
typeId: Joi.string().required()
|
||||
}).unknown();
|
|
@ -0,0 +1,11 @@
|
|||
import Joi from 'joi';
|
||||
|
||||
const base = Joi.object({
|
||||
processor_id: Joi.string().required()
|
||||
}).unknown();
|
||||
|
||||
export const set = base.keys({
|
||||
type_id: Joi.string().only('set').required(),
|
||||
target_field: Joi.string().required(),
|
||||
value: Joi.any().required()
|
||||
});
|
|
@ -1,7 +1,8 @@
|
|||
import Joi from 'joi';
|
||||
import ingestProcessorSchema from './resources/ingest_processor_schema';
|
||||
import * as ingestProcessorSchemas from './resources/ingest_processor_schemas';
|
||||
import _ from 'lodash';
|
||||
|
||||
export default Joi.object({
|
||||
processors: Joi.array().items(ingestProcessorSchema).required().min(1),
|
||||
processors: Joi.array().items(_.values(ingestProcessorSchemas)).required().min(1),
|
||||
input: Joi.object().required()
|
||||
});
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
const _ = require('lodash');
|
||||
import { buildRequest, processResponse } from '../../../lib/ingest_simulate';
|
||||
import processorTypes from '../../../../common/ingest_processor_types';
|
||||
import _ from 'lodash';
|
||||
import { processResponse } from '../../../lib/ingest_simulate';
|
||||
import simulateRequestSchema from '../../../lib/schemas/simulate_request_schema';
|
||||
import ingestSimulateApiToEsConverter from '../../../lib/converters/ingest_simulate_api_to_es_converter';
|
||||
|
||||
module.exports = function registerSimulate(server) {
|
||||
server.route({
|
||||
|
@ -14,8 +14,8 @@ module.exports = function registerSimulate(server) {
|
|||
},
|
||||
handler: function (request, reply) {
|
||||
const client = server.plugins.elasticsearch.client;
|
||||
const pipeline = request.payload;
|
||||
const body = buildRequest(processorTypes, pipeline);
|
||||
const simulateApiDocument = request.payload;
|
||||
const body = ingestSimulateApiToEsConverter(simulateApiDocument);
|
||||
|
||||
client.transport.request({
|
||||
path: '_ingest/pipeline/_simulate',
|
||||
|
@ -24,7 +24,7 @@ module.exports = function registerSimulate(server) {
|
|||
body: body
|
||||
},
|
||||
function (err, resp) {
|
||||
reply(processResponse(pipeline, err, resp));
|
||||
reply(processResponse(simulateApiDocument, err, resp));
|
||||
});
|
||||
}
|
||||
});
|
||||
|
|
|
@ -6,9 +6,9 @@ define(function (require) {
|
|||
|
||||
const testPipeline = {
|
||||
processors: [{
|
||||
processorId: 'processor1',
|
||||
typeId: 'set',
|
||||
targetField: 'foo',
|
||||
processor_id: 'processor1',
|
||||
type_id: 'set',
|
||||
target_field: 'foo',
|
||||
value: 'bar'
|
||||
}],
|
||||
input: {}
|
||||
|
@ -44,7 +44,6 @@ define(function (require) {
|
|||
.expect(200);
|
||||
});
|
||||
|
||||
// test for 400 when required fields for particular processor are missing, in separate files for each processor type
|
||||
});
|
||||
};
|
||||
});
|
||||
|
|
|
@ -9,6 +9,7 @@ define(function (require) {
|
|||
var post = require('./_post');
|
||||
var del = require('./_del');
|
||||
var simulate = require('./_simulate');
|
||||
var processors = require('./processors/index');
|
||||
|
||||
bdd.describe('ingest API', function () {
|
||||
var scenarioManager = new ScenarioManager(url.format(serverConfig.servers.elasticsearch));
|
||||
|
@ -25,5 +26,6 @@ define(function (require) {
|
|||
post(bdd, scenarioManager, request);
|
||||
del(bdd, scenarioManager, request);
|
||||
simulate(bdd, scenarioManager, request);
|
||||
processors(bdd, scenarioManager, request);
|
||||
});
|
||||
});
|
||||
|
|
37
test/unit/api/ingest/processors/_set.js
Normal file
37
test/unit/api/ingest/processors/_set.js
Normal file
|
@ -0,0 +1,37 @@
|
|||
define(function (require) {
|
||||
var Promise = require('bluebird');
|
||||
var _ = require('intern/dojo/node!lodash');
|
||||
var expect = require('intern/dojo/node!expect.js');
|
||||
|
||||
const testPipeline = {
|
||||
processors: [{
|
||||
processorId: 'processor1',
|
||||
typeId: 'set',
|
||||
targetField: 'foo',
|
||||
value: 'bar'
|
||||
}],
|
||||
input: {}
|
||||
};
|
||||
|
||||
return function (bdd, scenarioManager, request) {
|
||||
bdd.describe('simulate', function simulatePipeline() {
|
||||
|
||||
bdd.it('should return 400 for an invalid payload', function invalidPayload() {
|
||||
return Promise.all([
|
||||
// Set processor requires targetField property
|
||||
request.post('/kibana/ingest/simulate')
|
||||
.send({
|
||||
input: {},
|
||||
processors: [{
|
||||
processorId: 'processor1',
|
||||
typeId: 'set',
|
||||
value: 'bar'
|
||||
}]
|
||||
})
|
||||
.expect(400)
|
||||
]);
|
||||
});
|
||||
|
||||
});
|
||||
};
|
||||
});
|
9
test/unit/api/ingest/processors/index.js
Normal file
9
test/unit/api/ingest/processors/index.js
Normal file
|
@ -0,0 +1,9 @@
|
|||
define(function (require) {
|
||||
var set = require('./_set');
|
||||
|
||||
return function processors(bdd, scenarioManager, request) {
|
||||
set(bdd, scenarioManager, request);
|
||||
};
|
||||
|
||||
});
|
||||
|
Loading…
Add table
Add a link
Reference in a new issue