status: Display number of currently connected clients, and whether new clients are allowed
These will be useful for migration status monitoring - replaces "is pass == 7?"
This commit is contained in:
@@ -12,6 +12,9 @@ struct status * status_create( struct server * serve )
|
|||||||
status->size = serve->size;
|
status->size = serve->size;
|
||||||
status->has_control = serve->success;
|
status->has_control = serve->success;
|
||||||
|
|
||||||
|
status->clients_allowed = serve->allow_new_clients;
|
||||||
|
status->num_clients = server_count_clients( serve );
|
||||||
|
|
||||||
server_lock_start_mirror( serve );
|
server_lock_start_mirror( serve );
|
||||||
|
|
||||||
status->is_mirroring = NULL != serve->mirror;
|
status->is_mirroring = NULL != serve->mirror;
|
||||||
@@ -51,6 +54,8 @@ int status_write( struct status * status, int fd )
|
|||||||
PRINT_INT( pid );
|
PRINT_INT( pid );
|
||||||
PRINT_UINT64( size );
|
PRINT_UINT64( size );
|
||||||
PRINT_BOOL( is_mirroring );
|
PRINT_BOOL( is_mirroring );
|
||||||
|
PRINT_BOOL( clients_allowed );
|
||||||
|
PRINT_INT( num_clients );
|
||||||
PRINT_BOOL( has_control );
|
PRINT_BOOL( has_control );
|
||||||
|
|
||||||
if ( status->is_mirroring ) {
|
if ( status->is_mirroring ) {
|
||||||
|
10
src/status.h
10
src/status.h
@@ -30,6 +30,14 @@
|
|||||||
* If the server is started in "serve" mode, this will never be
|
* If the server is started in "serve" mode, this will never be
|
||||||
* false.
|
* false.
|
||||||
*
|
*
|
||||||
|
* clients_allowed:
|
||||||
|
* This will be false if the server is not currently allowing new
|
||||||
|
* connections, for instance, if we're in the migration endgame.
|
||||||
|
*
|
||||||
|
* num_clients:
|
||||||
|
* This tells us how many clients are currently running. If we're in the
|
||||||
|
* migration endgame, it should be 0
|
||||||
|
*
|
||||||
* is_migrating:
|
* is_migrating:
|
||||||
* This will be false when the server is started in either "listen"
|
* This will be false when the server is started in either "listen"
|
||||||
* or "serve" mode. It will become true when a server in "serve"
|
* or "serve" mode. It will become true when a server in "serve"
|
||||||
@@ -73,6 +81,8 @@ struct status {
|
|||||||
pid_t pid;
|
pid_t pid;
|
||||||
uint64_t size;
|
uint64_t size;
|
||||||
int has_control;
|
int has_control;
|
||||||
|
int clients_allowed;
|
||||||
|
int num_clients;
|
||||||
int is_mirroring;
|
int is_mirroring;
|
||||||
int migration_pass;
|
int migration_pass;
|
||||||
uint64_t pass_dirty_bytes;
|
uint64_t pass_dirty_bytes;
|
||||||
|
@@ -9,6 +9,8 @@ struct server* mock_server(void)
|
|||||||
{
|
{
|
||||||
struct server* out = xmalloc( sizeof( struct server ) );
|
struct server* out = xmalloc( sizeof( struct server ) );
|
||||||
out->l_start_mirror = flexthread_mutex_create();
|
out->l_start_mirror = flexthread_mutex_create();
|
||||||
|
out->nbd_client = xmalloc( sizeof( struct client_tbl_entry ) * 4 );
|
||||||
|
out->max_nbd_clients = 4;
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -27,6 +29,7 @@ void destroy_mock_server( struct server* serve )
|
|||||||
}
|
}
|
||||||
|
|
||||||
flexthread_mutex_destroy( serve->l_start_mirror );
|
flexthread_mutex_destroy( serve->l_start_mirror );
|
||||||
|
free( serve->nbd_client );
|
||||||
free( serve );
|
free( serve );
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -73,6 +76,43 @@ START_TEST( test_gets_is_mirroring )
|
|||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
|
START_TEST( test_gets_clients_allowed )
|
||||||
|
{
|
||||||
|
struct server * server = mock_server();
|
||||||
|
struct status * status = status_create( server );
|
||||||
|
|
||||||
|
fail_if( status->clients_allowed, "clients_allowed was set" );
|
||||||
|
status_destroy( status );
|
||||||
|
|
||||||
|
server->allow_new_clients = 1;
|
||||||
|
status = status_create( server );
|
||||||
|
|
||||||
|
fail_unless( status->clients_allowed, "clients_allowed was not set" );
|
||||||
|
status_destroy( status );
|
||||||
|
destroy_mock_server( server );
|
||||||
|
|
||||||
|
}
|
||||||
|
END_TEST
|
||||||
|
|
||||||
|
START_TEST( test_gets_num_clients )
|
||||||
|
{
|
||||||
|
struct server * server = mock_server();
|
||||||
|
struct status * status = status_create( server );
|
||||||
|
|
||||||
|
fail_if( status->num_clients != 0, "num_clients was wrong" );
|
||||||
|
status_destroy( status );
|
||||||
|
|
||||||
|
server->nbd_client[0].thread = 1;
|
||||||
|
server->nbd_client[1].thread = 1;
|
||||||
|
status = status_create( server );
|
||||||
|
|
||||||
|
fail_unless( status->num_clients == 2, "num_clients was wrong" );
|
||||||
|
status_destroy( status );
|
||||||
|
destroy_mock_server( server );
|
||||||
|
|
||||||
|
}
|
||||||
|
END_TEST
|
||||||
|
|
||||||
START_TEST( test_gets_migration_pass )
|
START_TEST( test_gets_migration_pass )
|
||||||
{
|
{
|
||||||
struct server * server = mock_server();
|
struct server * server = mock_server();
|
||||||
@@ -156,57 +196,95 @@ START_TEST( test_gets_migration_statistics )
|
|||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
|
|
||||||
|
#define RENDER_TEST_SETUP \
|
||||||
|
struct status status; \
|
||||||
|
int fds[2]; \
|
||||||
|
pipe( fds );
|
||||||
|
|
||||||
|
void fail_unless_rendered( int fd, char *fragment )
|
||||||
|
{
|
||||||
|
char buf[1024] = {0};
|
||||||
|
char emsg[1024] = {0};
|
||||||
|
char *found = NULL;
|
||||||
|
|
||||||
|
sprintf(emsg, "Fragment: %s not found", fragment );
|
||||||
|
|
||||||
|
fail_unless( read_until_newline( fd, buf, 1024 ) > 0, "Couldn't read" );
|
||||||
|
found = strstr( buf, fragment );
|
||||||
|
fail_if( NULL == found, emsg );
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void fail_if_rendered( int fd, char *fragment )
|
||||||
|
{
|
||||||
|
char buf[1024] = {0};
|
||||||
|
char emsg[1024] = {0};
|
||||||
|
char *found = NULL;
|
||||||
|
|
||||||
|
sprintf(emsg, "Fragment: %s found", fragment );
|
||||||
|
|
||||||
|
fail_unless( read_until_newline( fd, buf, 1024 ) > 0, "Couldn't read" );
|
||||||
|
found = strstr( buf, fragment );
|
||||||
|
fail_unless( NULL == found, emsg );
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
START_TEST( test_renders_has_control )
|
START_TEST( test_renders_has_control )
|
||||||
{
|
{
|
||||||
struct status status;
|
RENDER_TEST_SETUP
|
||||||
int fds[2];
|
|
||||||
pipe(fds);
|
|
||||||
char buf[1024] = {0};
|
|
||||||
|
|
||||||
status.has_control = 1;
|
status.has_control = 1;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "has_control=true" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
|
|
||||||
char *found = strstr( buf, "has_control=true" );
|
|
||||||
fail_if( NULL == found, "has_control=true not found" );
|
|
||||||
|
|
||||||
status.has_control = 0;
|
status.has_control = 0;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "has_control=false" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
found = strstr( buf, "has_control=false" );
|
|
||||||
fail_if( NULL == found, "has_control=false not found" );
|
|
||||||
|
|
||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
|
|
||||||
START_TEST( test_renders_is_mirroring )
|
START_TEST( test_renders_is_mirroring )
|
||||||
{
|
{
|
||||||
struct status status;
|
RENDER_TEST_SETUP
|
||||||
int fds[2];
|
|
||||||
pipe(fds);
|
|
||||||
char buf[1024] = {0};
|
|
||||||
|
|
||||||
status.is_mirroring = 1;
|
status.is_mirroring = 1;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "is_mirroring=true" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
|
|
||||||
char *found = strstr( buf, "is_mirroring=true" );
|
|
||||||
fail_if( NULL == found, "is_mirroring=true not found" );
|
|
||||||
|
|
||||||
status.is_mirroring = 0;
|
status.is_mirroring = 0;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "is_mirroring=false" );
|
||||||
|
}
|
||||||
|
END_TEST
|
||||||
|
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
START_TEST( test_renders_clients_allowed )
|
||||||
"Couldn't read the result" );
|
{
|
||||||
found = strstr( buf, "is_mirroring=false" );
|
RENDER_TEST_SETUP
|
||||||
fail_if( NULL == found, "is_mirroring=false not found" );
|
|
||||||
|
status.clients_allowed = 1;
|
||||||
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "clients_allowed=true" );
|
||||||
|
|
||||||
|
status.clients_allowed = 0;
|
||||||
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "clients_allowed=false" );
|
||||||
|
}
|
||||||
|
END_TEST
|
||||||
|
|
||||||
|
START_TEST( test_renders_num_clients )
|
||||||
|
{
|
||||||
|
RENDER_TEST_SETUP
|
||||||
|
|
||||||
|
status.num_clients = 0;
|
||||||
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "num_clients=0" );
|
||||||
|
|
||||||
|
status.num_clients = 4000;
|
||||||
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "num_clients=4000" );
|
||||||
|
|
||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
@@ -214,74 +292,42 @@ END_TEST
|
|||||||
|
|
||||||
START_TEST( test_renders_pid )
|
START_TEST( test_renders_pid )
|
||||||
{
|
{
|
||||||
struct status status;
|
RENDER_TEST_SETUP
|
||||||
int fds[2];
|
|
||||||
pipe(fds);
|
|
||||||
char buf[1024] = {0};
|
|
||||||
|
|
||||||
status.pid = 42;
|
status.pid = 42;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "pid=42" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
char *found = strstr( buf, "pid=42" );
|
|
||||||
fail_if( NULL == found, "pid=42 not found" );
|
|
||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
START_TEST( test_renders_size )
|
START_TEST( test_renders_size )
|
||||||
{
|
{
|
||||||
struct status status;
|
RENDER_TEST_SETUP
|
||||||
int fds[2];
|
|
||||||
pipe(fds);
|
|
||||||
char buf[1024] = {0};
|
|
||||||
|
|
||||||
status.size = ( (uint64_t)1 << 33 );
|
status.size = ( (uint64_t)1 << 33 );
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "size=8589934592" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
char *found = strstr( buf, "size=8589934592" );
|
|
||||||
fail_if( NULL == found, "size=8589934592 not found" );
|
|
||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
START_TEST( test_renders_migration_pass )
|
START_TEST( test_renders_migration_pass )
|
||||||
{
|
{
|
||||||
struct status status;
|
RENDER_TEST_SETUP
|
||||||
int fds[2];
|
|
||||||
pipe(fds);
|
|
||||||
char buf[1024] = {0};
|
|
||||||
char *found;
|
|
||||||
|
|
||||||
status.is_mirroring = 0;
|
status.is_mirroring = 0;
|
||||||
status.migration_pass = 1;
|
status.migration_pass = 1;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_if_rendered( fds[0], "migration_pass" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
found = strstr( buf, "migration_pass" );
|
|
||||||
fail_if( NULL != found, "migration pass output when not migrating" );
|
|
||||||
|
|
||||||
status.is_mirroring = 1;
|
status.is_mirroring = 1;
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "migration_pass=1" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
found = strstr( buf, "migration_pass=1" );
|
|
||||||
fail_if( NULL == found, "migration pass not output when not migrating" );
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
START_TEST( test_renders_migration_statistics )
|
START_TEST( test_renders_migration_statistics )
|
||||||
{
|
{
|
||||||
struct status status;
|
RENDER_TEST_SETUP
|
||||||
int fds[2];
|
|
||||||
pipe(fds);
|
|
||||||
char buf[1024] = {0};
|
|
||||||
char *found;
|
|
||||||
|
|
||||||
status.is_mirroring = 0;
|
status.is_mirroring = 0;
|
||||||
status.pass_dirty_bytes = 2048;
|
status.pass_dirty_bytes = 2048;
|
||||||
@@ -291,47 +337,41 @@ START_TEST( test_renders_migration_statistics )
|
|||||||
status.migration_speed_limit = 40000001;
|
status.migration_speed_limit = 40000001;
|
||||||
|
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_if_rendered( fds[0], "pass_dirty_bytes" );
|
||||||
|
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
status_write( &status, fds[1] );
|
||||||
"Couldn't read the result" );
|
fail_if_rendered( fds[0], "pass_clean_bytes" );
|
||||||
found = strstr( buf, "pass_dirty_bytes" );
|
|
||||||
fail_if( NULL != found, "pass_dirty_bytes output when not migrating" );
|
|
||||||
|
|
||||||
found = strstr( buf, "pass_clean_bytes" );
|
status_write( &status, fds[1] );
|
||||||
fail_if( NULL != found, "pass_clean_bytes output when not migrating" );
|
fail_if_rendered( fds[0], "migration_duration" );
|
||||||
|
|
||||||
found = strstr( buf, "migration_duration" );
|
status_write( &status, fds[1] );
|
||||||
fail_if( NULL != found, "migration_duration output when not migrating" );
|
fail_if_rendered( fds[0], "migration_speed" );
|
||||||
found = strstr( buf, "migration_speed" );
|
|
||||||
fail_if( NULL != found, "migration_speed output when not migrating" );
|
status_write( &status, fds[1] );
|
||||||
found = strstr( buf, "migration_speed_limit" );
|
fail_if_rendered( fds[0], "migration_speed_limit" );
|
||||||
fail_if( NULL != found, "migration_speed_limit output when not migrating" );
|
|
||||||
|
|
||||||
status.is_mirroring = 1;
|
status.is_mirroring = 1;
|
||||||
status_write( &status, fds[1] );
|
|
||||||
|
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
status_write( &status, fds[1] );
|
||||||
"Couldn't read the result" );
|
fail_unless_rendered( fds[0], "pass_dirty_bytes=2048" );
|
||||||
found = strstr( buf, "pass_dirty_bytes=2048" );
|
|
||||||
fail_if( NULL == found, "pass_dirty_bytes not output when migrating" );
|
status_write( &status, fds[1] );
|
||||||
found = strstr( buf, "pass_clean_bytes=4096" );
|
fail_unless_rendered( fds[0], "pass_clean_bytes=4096" );
|
||||||
fail_if( NULL == found, "pass_clean_bytes not output when migrating" );
|
|
||||||
found = strstr( buf, "migration_duration=8" );
|
status_write( &status, fds[1] );
|
||||||
fail_if( NULL == found, "migration_duration not output when migrating" );
|
fail_unless_rendered( fds[0], "migration_duration=8" );
|
||||||
found = strstr( buf, "migration_speed=40000000" );
|
|
||||||
fail_if( NULL == found, "migration_speed not output when migrating" );
|
status_write( &status, fds[1] );
|
||||||
found = strstr( buf, "migration_speed_limit=40000001" );
|
fail_unless_rendered( fds[0], "migration_speed=40000000" );
|
||||||
fail_if( NULL == found, "migration_speed_limit not output when migrating" );
|
|
||||||
|
status_write( &status, fds[1] );
|
||||||
|
fail_unless_rendered( fds[0], "migration_speed_limit=40000001" );
|
||||||
|
|
||||||
status.migration_speed_limit = UINT64_MAX;
|
status.migration_speed_limit = UINT64_MAX;
|
||||||
|
|
||||||
status_write( &status, fds[1] );
|
status_write( &status, fds[1] );
|
||||||
|
fail_if_rendered( fds[0], "migration_speed_limit" );
|
||||||
fail_unless( read_until_newline( fds[0], buf, 1024 ) > 0,
|
|
||||||
"Couldn't read the result" );
|
|
||||||
|
|
||||||
found = strstr( buf, "migration_speed_limit" );
|
|
||||||
fail_if( NULL != found, "migration_speed_limit output when no migration limit was set" );
|
|
||||||
|
|
||||||
}
|
}
|
||||||
END_TEST
|
END_TEST
|
||||||
|
|
||||||
@@ -345,6 +385,8 @@ Suite *status_suite(void)
|
|||||||
tcase_add_test(tc_create, test_status_create);
|
tcase_add_test(tc_create, test_status_create);
|
||||||
tcase_add_test(tc_create, test_gets_has_control);
|
tcase_add_test(tc_create, test_gets_has_control);
|
||||||
tcase_add_test(tc_create, test_gets_is_mirroring);
|
tcase_add_test(tc_create, test_gets_is_mirroring);
|
||||||
|
tcase_add_test(tc_create, test_gets_clients_allowed);
|
||||||
|
tcase_add_test(tc_create, test_gets_num_clients);
|
||||||
tcase_add_test(tc_create, test_gets_pid);
|
tcase_add_test(tc_create, test_gets_pid);
|
||||||
tcase_add_test(tc_create, test_gets_size);
|
tcase_add_test(tc_create, test_gets_size);
|
||||||
tcase_add_test(tc_create, test_gets_migration_pass);
|
tcase_add_test(tc_create, test_gets_migration_pass);
|
||||||
@@ -353,6 +395,8 @@ Suite *status_suite(void)
|
|||||||
|
|
||||||
tcase_add_test(tc_render, test_renders_has_control);
|
tcase_add_test(tc_render, test_renders_has_control);
|
||||||
tcase_add_test(tc_render, test_renders_is_mirroring);
|
tcase_add_test(tc_render, test_renders_is_mirroring);
|
||||||
|
tcase_add_test(tc_render, test_renders_clients_allowed);
|
||||||
|
tcase_add_test(tc_render, test_renders_num_clients);
|
||||||
tcase_add_test(tc_render, test_renders_pid);
|
tcase_add_test(tc_render, test_renders_pid);
|
||||||
tcase_add_test(tc_render, test_renders_size);
|
tcase_add_test(tc_render, test_renders_size);
|
||||||
tcase_add_test(tc_render, test_renders_migration_pass);
|
tcase_add_test(tc_render, test_renders_migration_pass);
|
||||||
|
Reference in New Issue
Block a user