Files
elasticsearch-js/test/behavior/sniff.test.js
Tomas Della Vedova fd8f02b9df Fix flaky test (#1158)
* Fix flaky test

* Fix #1154
2020-04-14 14:39:53 +02:00

282 lines
6.9 KiB
JavaScript

// Licensed to Elasticsearch B.V under one or more agreements.
// Elasticsearch B.V licenses this file to you under the Apache 2.0 License.
// See the LICENSE file in the project root for more information
'use strict'
const { test } = require('tap')
const { URL } = require('url')
const lolex = require('lolex')
const workq = require('workq')
const { buildCluster } = require('../utils')
const { Client, Connection, Transport, events, errors } = require('../../index')
/**
* The aim of this test is to verify how the sniffer behaves
* in a multi node situation.
* The `buildCluster` utility can boot an arbitrary number
* of nodes, that you can kill or spawn at your will.
* The sniffer component can be tested with its callback
* or by using the `sniff` event (to handle automatically
* triggered sniff).
*/
test('Should update the connection pool', t => {
t.plan(10)
buildCluster(({ nodes, shutdown }) => {
const client = new Client({
node: nodes[Object.keys(nodes)[0]].url
})
t.strictEqual(client.connectionPool.size, 1)
client.on(events.SNIFF, (err, request) => {
t.error(err)
t.strictEqual(
request.meta.sniff.reason,
Transport.sniffReasons.DEFAULT
)
})
// run the sniffer
client.transport.sniff((err, hosts) => {
t.error(err)
t.strictEqual(hosts.length, 4)
const ids = Object.keys(nodes)
for (var i = 0; i < hosts.length; i++) {
const id = ids[i]
// the first node will be an update of the existing one
if (id === 'node0') {
t.deepEqual(hosts[i], {
url: new URL(nodes[id].url),
id: id,
roles: {
master: true,
data: true,
ingest: true,
ml: false
}
})
} else {
t.deepEqual(hosts[i], {
url: new URL(nodes[id].url),
id: id,
roles: {
master: true,
data: true,
ingest: true,
ml: false
},
ssl: null,
agent: null
})
}
}
t.strictEqual(client.connectionPool.size, 4)
})
t.teardown(shutdown)
})
})
test('Should handle hostnames in publish_address', t => {
t.plan(10)
buildCluster({ hostPublishAddress: true }, ({ nodes, shutdown }) => {
const client = new Client({
node: nodes[Object.keys(nodes)[0]].url
})
t.strictEqual(client.connectionPool.size, 1)
client.on(events.SNIFF, (err, request) => {
t.error(err)
t.strictEqual(
request.meta.sniff.reason,
Transport.sniffReasons.DEFAULT
)
})
// run the sniffer
client.transport.sniff((err, hosts) => {
t.error(err)
t.strictEqual(hosts.length, 4)
for (var i = 0; i < hosts.length; i++) {
// the first node will be an update of the existing one
t.strictEqual(hosts[i].url.hostname, 'localhost')
}
t.strictEqual(client.connectionPool.size, 4)
})
t.teardown(shutdown)
})
})
test('Sniff interval', t => {
t.plan(11)
const clock = lolex.install({ toFake: ['Date'] })
const q = workq()
buildCluster(({ nodes, shutdown, kill }) => {
const client = new Client({
node: nodes[Object.keys(nodes)[0]].url,
sniffInterval: 50
})
// this event will be triggered by api calls
client.on(events.SNIFF, (err, request) => {
t.error(err)
const { hosts, reason } = request.meta.sniff
t.strictEqual(
client.connectionPool.size,
hosts.length
)
t.strictEqual(reason, Transport.sniffReasons.SNIFF_INTERVAL)
})
t.strictEqual(client.connectionPool.size, 1)
q.add((q, done) => {
clock.tick(51)
client.info(err => {
t.error(err)
waitSniffEnd(() => {
t.strictEqual(client.connectionPool.size, 4)
done()
})
})
})
q.add((q, done) => {
kill('node1', done)
})
q.add((q, done) => {
clock.tick(51)
client.info(err => {
t.error(err)
waitSniffEnd(() => {
t.strictEqual(client.connectionPool.size, 3)
done()
})
})
})
t.teardown(shutdown)
// it can happen that the sniff operation resolves
// after the API call that trioggered it, so to
// be sure that we are checking the connectionPool size
// at the right moment, we verify that the transport
// is no longer sniffing
function waitSniffEnd (callback) {
if (client.transport._isSniffing) {
setTimeout(waitSniffEnd, 500, callback)
} else {
callback()
}
}
})
})
test('Sniff on start', t => {
t.plan(4)
buildCluster(({ nodes, shutdown, kill }) => {
const client = new Client({
node: nodes[Object.keys(nodes)[0]].url,
sniffOnStart: true
})
client.on(events.SNIFF, (err, request) => {
t.error(err)
const { hosts, reason } = request.meta.sniff
t.strictEqual(
client.connectionPool.size,
hosts.length
)
t.strictEqual(reason, Transport.sniffReasons.SNIFF_ON_START)
})
t.strictEqual(client.connectionPool.size, 1)
t.teardown(shutdown)
})
})
test('Should not close living connections', t => {
t.plan(3)
buildCluster(({ nodes, shutdown, kill }) => {
class MyConnection extends Connection {
close () {
t.fail('Should not be called')
}
}
const client = new Client({
node: {
url: new URL(nodes[Object.keys(nodes)[0]].url),
id: 'node1'
},
Connection: MyConnection
})
t.strictEqual(client.connectionPool.size, 1)
client.transport.sniff((err, hosts) => {
t.error(err)
t.strictEqual(
client.connectionPool.size,
hosts.length
)
})
t.teardown(shutdown)
})
})
test('Sniff on connection fault', t => {
t.plan(5)
buildCluster(({ nodes, shutdown, kill }) => {
class MyConnection extends Connection {
request (params, callback) {
if (this.id === 'http://localhost:9200/') {
callback(new errors.ConnectionError('kaboom'), null)
return {}
} else {
return super.request(params, callback)
}
}
}
const client = new Client({
nodes: [
'http://localhost:9200',
nodes[Object.keys(nodes)[0]].url
],
maxRetries: 0,
sniffOnConnectionFault: true,
Connection: MyConnection
})
t.strictEqual(client.connectionPool.size, 2)
// this event will be triggered by the connection fault
client.on(events.SNIFF, (err, request) => {
t.error(err)
const { hosts, reason } = request.meta.sniff
t.strictEqual(
client.connectionPool.size,
hosts.length
)
t.strictEqual(reason, Transport.sniffReasons.SNIFF_ON_CONNECTION_FAULT)
})
client.info((err, result) => {
t.ok(err instanceof errors.ConnectionError)
})
t.teardown(shutdown)
})
})