—# 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
#
#
# 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.
package
Search::Elasticsearch::TestServer;
$Search::Elasticsearch::TestServer::VERSION
=
'8.12'
;
use
Moo;
use
Search::Elasticsearch();
use
File::Temp();
use
IO::Socket();
use
HTTP::Tiny;
use
namespace::clean;
has
'es_home'
=> (
is
=>
'ro'
,
default
=>
$ENV
{ES_HOME} );
has
'es_version'
=> (
is
=>
'ro'
,
default
=>
$ENV
{ES_VERSION} );
has
'instances'
=> (
is
=>
'ro'
,
default
=> 1 );
has
'http_port'
=> (
is
=>
'ro'
,
default
=> 9600 );
has
'es_port'
=> (
is
=>
'ro'
,
default
=> 9700 );
has
'pids'
=> (
is
=>
'ro'
,
default
=>
sub
{ [] },
clearer
=> 1,
predicate
=> 1
);
has
'dirs'
=> (
is
=>
'ro'
,
default
=>
sub
{ [] } );
has
'conf'
=> (
is
=>
'ro'
,
default
=>
sub
{ [] } );
has
'_starter_pid'
=> (
is
=>
'rw'
,
required
=> 0,
predicate
=> 1 );
#===================================
sub
start {
#===================================
my
$self
=
shift
;
my
$home
=
$self
->es_home
or throw(
'Param'
,
"Missing required param <es_home>"
);
$self
->es_version
or throw(
'Param'
,
"Missing required param <es_version>"
);
my
$instances
=
$self
->instances;
my
$port
=
$self
->http_port;
my
$es_port
=
$self
->es_port;
my
@http
=
map
{
$port
++ } ( 1 ..
$instances
);
my
@transport
=
map
{
$es_port
++ } ( 1 ..
$instances
);
$self
->_check_ports(
@http
,
@transport
);
my
$old_SIGINT
=
$SIG
{INT};
$SIG
{INT} =
sub
{
$self
->
shutdown
;
if
(
ref
$old_SIGINT
eq
'CODE'
) {
return
$old_SIGINT
->();
}
exit
(1);
};
for
( 0 ..
$instances
- 1 ) {
my
$dir
= File::Temp->newdir();
push
@{
$self
->dirs },
$dir
;
$self
->_start_node(
$dir
,
$transport
[
$_
],
$http
[
$_
] );
}
$self
->_check_nodes(
@http
);
}
#===================================
sub
_check_ports {
#===================================
my
$self
=
shift
;
for
my
$port
(
@_
) {
next
unless
IO::Socket::INET->new(
"127.0.0.1:$port"
);
throw(
'Param'
,
"There is already a service running on 127.0.0.1:$port. "
.
"Please shut it down before starting the test server"
);
}
}
#===================================
sub
_check_nodes {
#===================================
my
$self
=
shift
;
my
$http
= HTTP::Tiny->new;
for
my
$node
(
@_
) {
my
$i
= 20;
while
(1) {
last
unless
$i
--;
sleep
1;
}
}
}
#===================================
sub
_start_node {
#===================================
my
(
$self
,
$dir
,
$transport
,
$http
) =
@_
;
my
$pid_file
= File::Temp->new;
my
@config
=
$self
->_command_line(
$pid_file
,
$dir
,
$transport
,
$http
);
my
$int_caught
= 0;
{
local
$SIG
{INT} =
sub
{
$int_caught
++; };
defined
(
my
$pid
=
fork
)
or throw(
'Internal'
,
"Couldn't fork a new process: $!"
);
if
(
$pid
== 0 ) {
throw(
'Internal'
,
"Can't start a new session: $!"
)
if
setsid == -1;
exec
(
@config
) or
die
"Couldn't execute @config: $!"
;
}
else
{
for
( 1 .. 5 ) {
last
if
-s
$pid_file
->filename();
sleep
1;
}
open
my
$pid_fh
,
'<'
,
$pid_file
->filename;
my
$pid
= <
$pid_fh
>;
throw(
'Internal'
,
"No PID file found for Elasticsearch"
)
unless
$pid
;
chomp
$pid
;
push
@{
$self
->{pids} },
$pid
;
$self
->_starter_pid($$);
}
}
$SIG
{INT}->(
'INT'
)
if
$int_caught
;
}
#===================================
sub
guarded_shutdown {
#===================================
my
$self
=
shift
;
if
(
$self
->_has_starter_pid && $$ ==
$self
->_starter_pid ) {
$self
->
shutdown
();
}
}
#===================================
sub
shutdown
{
#===================================
my
$self
=
shift
;
local
$?;
return
unless
$self
->has_pids;
my
$pids
=
$self
->pids;
$self
->clear_pids;
return
unless
@$pids
;
kill
9,
@$pids
;
$self
->clear_dirs;
}
#===================================
sub
_command_line {
#===================================
my
(
$self
,
$pid_file
,
$dir
,
$transport
,
$http
) =
@_
;
my
$version
=
$self
->es_version;
my
$class
=
"Search::Elasticsearch::Client::${version}::TestServer"
;
eval
"require $class"
||
die
$@;
return
$class
->command_line(
@_
);
}
#===================================
sub
clear_dirs {
#===================================
my
$self
=
shift
;
@{
$self
->dirs() } = ();
}
#===================================
sub
DEMOLISH {
shift
->guarded_shutdown }
#===================================
1;
# ABSTRACT: A helper class to launch Elasticsearch nodes
__END__
=pod
=encoding UTF-8
=head1 NAME
Search::Elasticsearch::TestServer - A helper class to launch Elasticsearch nodes
=head1 VERSION
version 8.12
=head1 SYNOPSIS
use Search::Elasticsearch;
use Search::Elasticsearch::TestServer;
my $server = Search::Elasticsearch::TestServer->new(
es_home => '/path/to/elasticsearch', # defaults to $ENV{ES_HOME}
es_version => '6_0' # defaults to $ENV{ES_VERSION}
);
my $nodes = $server->start;
my $es = Search::Elasticsearch->new( nodes => $nodes );
# run tests
$server->shutdown;
=head1 DESCRIPTION
The L<Search::Elasticsearch::TestServer> class can be used to launch one or more
instances of Elasticsearch for testing purposes. The nodes will
be shutdown automatically.
=head1 METHODS
=head2 C<new()>
my $server = Search::Elasticsearch::TestServer->new(
es_home => '/path/to/elasticsearch',
es_version => '6_0',
instances => 1,
http_port => 9600,
es_port => 9700,
conf => ['attr.foo=bar'],
);
Params:
=over
=item * C<es_home>
Required. Must point to the Elasticsearch home directory, which contains
C<./bin/elasticsearch>. Defaults to C<$ENV{ES_HOME}>
=item * C<es_version>
Required. Accepts a version of the client, eg `6_0`, `5_0`, `2_0`, `1_0`, `0_90`.
Defaults to C<$ENV{ES_VERSION}>.
=item * C<instances>
The number of nodes to start. Defaults to 1
=item * C<http_port>
The port to use for HTTP. If multiple instances are started, the C<http_port>
will be incremented for each subsequent instance. Defaults to 9600.
=item * C<es_port>
The port to use for Elasticsearch's internal transport. If multiple instances
are started, the C<es_port> will be incremented for each subsequent instance.
Defaults to 9700
=item * C<conf>
An array containing any extra startup options that should be passed
to Elasticsearch.
=back
=head1 C<start()>
$nodes = $server->start;
Starts the required instances and returns an array ref containing the IP
and port of each node, suitable for passing to L<Search::Elasticsearch/new()>:
$es = Search::Elasticsearch->new( nodes => $nodes );
=head1 C<shutdown()>
$server->shutdown;
Kills the running instances. This will be called automatically when
C<$server> goes out of scope or if the program receives a C<SIGINT>.
=head1 AUTHOR
Enrico Zimuel <enrico.zimuel@elastic.co>
=head1 COPYRIGHT AND LICENSE
This software is Copyright (c) 2024 by Elasticsearch BV.
This is free software, licensed under:
The Apache License, Version 2.0, January 2004
=cut