Files
elasticsearch-js/test/unit/connection.test.js
2021-05-20 16:18:01 +02:00

950 lines
20 KiB
JavaScript

/*
* 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.
*/
'use strict'
const { test } = require('tap')
const { inspect } = require('util')
const { URL } = require('url')
const { Agent } = require('http')
const { Readable } = require('stream')
const hpagent = require('hpagent')
const intoStream = require('into-stream')
const { buildServer } = require('../utils')
const Connection = require('../../lib/Connection')
const { TimeoutError, ConfigurationError, RequestAbortedError } = require('../../lib/errors')
test('Basic (http)', t => {
t.plan(4)
function handler (req, res) {
t.match(req.headers, {
'x-custom-test': 'true',
connection: 'keep-alive'
})
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'GET',
headers: {
'X-Custom-Test': true
}
}, (err, res) => {
t.error(err)
t.match(res.headers, {
connection: 'keep-alive'
})
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Basic (https)', t => {
t.plan(4)
function handler (req, res) {
t.match(req.headers, {
'x-custom-test': 'true',
connection: 'keep-alive'
})
res.end('ok')
}
buildServer(handler, { secure: true }, ({ port }, server) => {
const connection = new Connection({
url: new URL(`https://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'GET',
headers: {
'X-Custom-Test': true
}
}, (err, res) => {
t.error(err)
t.match(res.headers, {
connection: 'keep-alive'
})
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Basic (https with ssl agent)', t => {
t.plan(4)
function handler (req, res) {
t.match(req.headers, {
'x-custom-test': 'true',
connection: 'keep-alive'
})
res.end('ok')
}
buildServer(handler, { secure: true }, ({ port, key, cert }, server) => {
const connection = new Connection({
url: new URL(`https://localhost:${port}`),
ssl: { key, cert }
})
connection.request({
path: '/hello',
method: 'GET',
headers: {
'X-Custom-Test': true
}
}, (err, res) => {
t.error(err)
t.match(res.headers, {
connection: 'keep-alive'
})
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Custom http agent', t => {
t.plan(6)
function handler (req, res) {
t.match(req.headers, {
'x-custom-test': 'true',
connection: 'keep-alive'
})
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const agent = new Agent({
keepAlive: true,
keepAliveMsecs: 1000,
maxSockets: 256,
maxFreeSockets: 256
})
agent.custom = true
const connection = new Connection({
url: new URL(`http://localhost:${port}`),
agent: opts => {
t.match(opts, {
url: new URL(`http://localhost:${port}`)
})
return agent
}
})
t.ok(connection.agent.custom)
connection.request({
path: '/hello',
method: 'GET',
headers: {
'X-Custom-Test': true
}
}, (err, res) => {
t.error(err)
t.match(res.headers, {
connection: 'keep-alive'
})
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Disable keep alive', t => {
t.plan(3)
function handler (req, res) {
t.match(req.headers, {
'x-custom-test': 'true',
connection: 'close'
})
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`),
agent: false
})
connection.request({
path: '/hello',
method: 'GET',
headers: {
'X-Custom-Test': true
}
}, (err, res) => {
t.error(err)
t.match(res.headers, {
connection: 'close'
})
server.stop()
})
})
})
test('Timeout support', t => {
t.plan(1)
function handler (req, res) {
setTimeout(
() => res.end('ok'),
1000
)
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'GET',
timeout: 500
}, (err, res) => {
t.ok(err instanceof TimeoutError)
server.stop()
})
})
})
test('querystring', t => {
t.test('Should concatenate the querystring', t => {
t.plan(2)
function handler (req, res) {
t.equal(req.url, '/hello?hello=world&you_know=for%20search')
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'GET',
querystring: 'hello=world&you_know=for%20search'
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
t.test('If the querystring is null should not do anything', t => {
t.plan(2)
function handler (req, res) {
t.equal(req.url, '/hello')
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'GET',
querystring: null
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
t.end()
})
test('Body request', t => {
t.plan(2)
function handler (req, res) {
let payload = ''
req.setEncoding('utf8')
req.on('data', chunk => { payload += chunk })
req.on('error', err => t.fail(err))
req.on('end', () => {
t.equal(payload, 'hello')
res.end('ok')
})
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'POST',
body: 'hello'
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
test('Send body as buffer', t => {
t.plan(2)
function handler (req, res) {
let payload = ''
req.setEncoding('utf8')
req.on('data', chunk => { payload += chunk })
req.on('error', err => t.fail(err))
req.on('end', () => {
t.equal(payload, 'hello')
res.end('ok')
})
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'POST',
body: Buffer.from('hello')
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
test('Send body as stream', t => {
t.plan(2)
function handler (req, res) {
let payload = ''
req.setEncoding('utf8')
req.on('data', chunk => { payload += chunk })
req.on('error', err => t.fail(err))
req.on('end', () => {
t.equal(payload, 'hello')
res.end('ok')
})
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'POST',
body: intoStream('hello')
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
test('Should not close a connection if there are open requests', t => {
t.plan(4)
function handler (req, res) {
setTimeout(() => res.end('ok'), 1000)
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
setTimeout(() => {
t.equal(connection._openRequests, 1)
connection.close()
}, 500)
connection.request({
path: '/hello',
method: 'GET'
}, (err, res) => {
t.error(err)
t.equal(connection._openRequests, 0)
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Should not close a connection if there are open requests (with agent disabled)', t => {
t.plan(4)
function handler (req, res) {
setTimeout(() => res.end('ok'), 1000)
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`),
agent: false
})
setTimeout(() => {
t.equal(connection._openRequests, 1)
connection.close()
}, 500)
connection.request({
path: '/hello',
method: 'GET'
}, (err, res) => {
t.error(err)
t.equal(connection._openRequests, 0)
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Url with auth', t => {
t.plan(2)
function handler (req, res) {
t.match(req.headers, {
authorization: 'Basic Zm9vOmJhcg=='
})
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://foo:bar@localhost:${port}`),
auth: { username: 'foo', password: 'bar' }
})
connection.request({
path: '/hello',
method: 'GET'
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
test('Url with querystring', t => {
t.plan(2)
function handler (req, res) {
t.equal(req.url, '/hello?foo=bar&baz=faz')
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}?foo=bar`)
})
connection.request({
path: '/hello',
method: 'GET',
querystring: 'baz=faz'
}, (err, res) => {
t.error(err)
server.stop()
})
})
})
test('Custom headers for connection', t => {
t.plan(3)
function handler (req, res) {
t.match(req.headers, {
'x-custom-test': 'true',
'x-foo': 'bar'
})
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`),
headers: { 'x-foo': 'bar' }
})
connection.request({
path: '/hello',
method: 'GET',
headers: {
'X-Custom-Test': true
}
}, (err, res) => {
t.error(err)
// should not update the default
t.same(connection.headers, { 'x-foo': 'bar' })
server.stop()
})
})
})
// TODO: add a check that the response is not decompressed
test('asStream set to true', t => {
t.plan(2)
function handler (req, res) {
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
connection.request({
path: '/hello',
method: 'GET',
asStream: true
}, (err, res) => {
t.error(err)
let payload = ''
res.setEncoding('utf8')
res.on('data', chunk => { payload += chunk })
res.on('error', err => t.fail(err))
res.on('end', () => {
t.equal(payload, 'ok')
server.stop()
})
})
})
})
test('Connection id should not contain credentials', t => {
const connection = new Connection({
url: new URL('http://user:password@localhost:9200')
})
t.equal(connection.id, 'http://localhost:9200/')
t.end()
})
test('Ipv6 support', t => {
const connection = new Connection({
url: new URL('http://[::1]:9200')
})
t.equal(connection.buildRequestObject({}).hostname, '::1')
t.end()
})
test('Should throw if the protocol is not http or https', t => {
try {
new Connection({ // eslint-disable-line
url: new URL('nope://nope')
})
t.fail('Should throw')
} catch (err) {
t.ok(err instanceof ConfigurationError)
t.equal(err.message, 'Invalid protocol: \'nope:\'')
}
t.end()
})
// https://github.com/nodejs/node/commit/b961d9fd83
test('Should disallow two-byte characters in URL path', t => {
t.plan(1)
const connection = new Connection({
url: new URL('http://localhost:9200')
})
connection.request({
path: '/thisisinvalid\uffe2',
method: 'GET'
}, (err, res) => {
t.equal(
err.message,
'ERR_UNESCAPED_CHARACTERS: /thisisinvalid\uffe2'
)
})
})
test('setRole', t => {
t.test('Update the value of a role', t => {
t.plan(2)
const connection = new Connection({
url: new URL('http://localhost:9200')
})
t.same(connection.roles, {
master: true,
data: true,
ingest: true,
ml: false
})
connection.setRole('master', false)
t.same(connection.roles, {
master: false,
data: true,
ingest: true,
ml: false
})
})
t.test('Invalid role', t => {
t.plan(2)
const connection = new Connection({
url: new URL('http://localhost:9200')
})
try {
connection.setRole('car', true)
t.fail('Shoud throw')
} catch (err) {
t.ok(err instanceof ConfigurationError)
t.equal(err.message, 'Unsupported role: \'car\'')
}
})
t.test('Invalid value', t => {
t.plan(2)
const connection = new Connection({
url: new URL('http://localhost:9200')
})
try {
connection.setRole('master', 1)
t.fail('Shoud throw')
} catch (err) {
t.ok(err instanceof ConfigurationError)
t.equal(err.message, 'enabled should be a boolean')
}
})
t.end()
})
test('Util.inspect Connection class should hide agent, ssl and auth', t => {
t.plan(1)
const connection = new Connection({
url: new URL('http://user:password@localhost:9200'),
id: 'node-id',
headers: { foo: 'bar' }
})
// Removes spaces and new lines because
// utils.inspect is handled differently
// between major versions of Node.js
function cleanStr (str) {
return str
.replace(/\s/g, '')
.replace(/(\r\n|\n|\r)/gm, '')
}
t.equal(cleanStr(inspect(connection)), cleanStr(`{ url: 'http://localhost:9200/',
id: 'node-id',
headers: { foo: 'bar' },
deadCount: 0,
resurrectTimeout: 0,
_openRequests: 0,
status: 'alive',
roles: { master: true, data: true, ingest: true, ml: false }}`)
)
})
test('connection.toJSON should hide agent, ssl and auth', t => {
t.plan(1)
const connection = new Connection({
url: new URL('http://user:password@localhost:9200'),
id: 'node-id',
headers: { foo: 'bar' }
})
t.same(connection.toJSON(), {
url: 'http://localhost:9200/',
id: 'node-id',
headers: {
foo: 'bar'
},
deadCount: 0,
resurrectTimeout: 0,
_openRequests: 0,
status: 'alive',
roles: {
master: true,
data: true,
ingest: true,
ml: false
}
})
})
// https://github.com/elastic/elasticsearch-js/issues/843
test('Port handling', t => {
t.test('http 80', t => {
const connection = new Connection({
url: new URL('http://localhost:80')
})
t.equal(
connection.buildRequestObject({}).port,
undefined
)
t.end()
})
t.test('https 443', t => {
const connection = new Connection({
url: new URL('https://localhost:443')
})
t.equal(
connection.buildRequestObject({}).port,
undefined
)
t.end()
})
t.end()
})
test('Authorization header', t => {
t.test('None', t => {
const connection = new Connection({
url: new URL('http://localhost:9200')
})
t.same(connection.headers, {})
t.end()
})
t.test('Basic', t => {
const connection = new Connection({
url: new URL('http://localhost:9200'),
auth: { username: 'foo', password: 'bar' }
})
t.same(connection.headers, { authorization: 'Basic Zm9vOmJhcg==' })
t.end()
})
t.test('ApiKey (string)', t => {
const connection = new Connection({
url: new URL('http://localhost:9200'),
auth: { apiKey: 'Zm9vOmJhcg==' }
})
t.same(connection.headers, { authorization: 'ApiKey Zm9vOmJhcg==' })
t.end()
})
t.test('ApiKey (object)', t => {
const connection = new Connection({
url: new URL('http://localhost:9200'),
auth: { apiKey: { id: 'foo', api_key: 'bar' } }
})
t.same(connection.headers, { authorization: 'ApiKey Zm9vOmJhcg==' })
t.end()
})
t.end()
})
test('Should not add agent and ssl to the serialized connection', t => {
const connection = new Connection({
url: new URL('http://localhost:9200')
})
t.equal(
JSON.stringify(connection),
'{"url":"http://localhost:9200/","id":"http://localhost:9200/","headers":{},"deadCount":0,"resurrectTimeout":0,"_openRequests":0,"status":"alive","roles":{"master":true,"data":true,"ingest":true,"ml":false}}'
)
t.end()
})
test('Abort a request syncronously', t => {
t.plan(1)
function handler (req, res) {
t.fail('The server should not be contacted')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
const request = connection.request({
path: '/hello',
method: 'GET'
}, (err, res) => {
t.ok(err instanceof RequestAbortedError)
server.stop()
})
request.abort()
})
})
test('Abort a request asyncronously', t => {
t.plan(1)
function handler (req, res) {
// might be called or not
res.end('ok')
}
buildServer(handler, ({ port }, server) => {
const connection = new Connection({
url: new URL(`http://localhost:${port}`)
})
const request = connection.request({
path: '/hello',
method: 'GET'
}, (err, res) => {
t.ok(err instanceof RequestAbortedError)
server.stop()
})
setImmediate(() => request.abort())
})
})
test('Should correctly resolve request pathname', t => {
t.plan(1)
const connection = new Connection({
url: new URL('http://localhost:80/test')
})
t.equal(
connection.buildRequestObject({
path: 'hello'
}).pathname,
'/test/hello'
)
})
test('Proxy agent (http)', t => {
t.plan(1)
const connection = new Connection({
url: new URL('http://localhost:9200'),
proxy: 'http://localhost:8080'
})
t.ok(connection.agent instanceof hpagent.HttpProxyAgent)
})
test('Proxy agent (https)', t => {
t.plan(1)
const connection = new Connection({
url: new URL('https://localhost:9200'),
proxy: 'http://localhost:8080'
})
t.ok(connection.agent instanceof hpagent.HttpsProxyAgent)
})
test('Abort with a slow body', t => {
t.plan(1)
const connection = new Connection({
url: new URL('https://localhost:9200'),
proxy: 'http://localhost:8080'
})
const slowBody = new Readable({
read (size) {
setTimeout(() => {
this.push('{"size":1, "query":{"match_all":{}}}')
this.push(null) // EOF
}, 1000).unref()
}
})
const request = connection.request({
method: 'GET',
path: '/',
body: slowBody
}, (err, response) => {
t.ok(err instanceof RequestAbortedError)
})
setImmediate(() => request.abort())
})