From 890dbeca3ca507734c184733e128425b0705068a Mon Sep 17 00:00:00 2001
From: Tomas Pecka <tomas.pecka@fit.cvut.cz>
Date: Mon, 6 Apr 2020 14:31:55 +0200
Subject: [PATCH] integrationtest: Report stdout and stderr output from child

---
 .../test-src/testing/TimeoutAqlTest.cpp       | 129 +++++++++++-------
 1 file changed, 83 insertions(+), 46 deletions(-)

diff --git a/alib2integrationtest/test-src/testing/TimeoutAqlTest.cpp b/alib2integrationtest/test-src/testing/TimeoutAqlTest.cpp
index 311b9b1ca6..5637ff9a17 100644
--- a/alib2integrationtest/test-src/testing/TimeoutAqlTest.cpp
+++ b/alib2integrationtest/test-src/testing/TimeoutAqlTest.cpp
@@ -19,6 +19,9 @@
 #define PIPE_RD 0
 #define PIPE_WR 1
 
+#define FD_STDOUT 1
+#define FD_STDERR 2
+
 /* Communication between signal handler and the rest of the program */
 int g_Wakeup [ 2 ];  // pipe for wakeup
 int g_RecvSignal; // signalled flag
@@ -43,16 +46,35 @@ int waitSignalTimeout ( int timeout ) {
 	return g_RecvSignal;
 }
 
+
+std::string readFromFD ( int fd ) {
+	static const size_t BUFSIZE = 64;
+
+	std::string res;
+	int rd;
+	char buf [ BUFSIZE ];
+
+	while ( ( rd = read ( fd, &buf, BUFSIZE - 1 ) ) > 0 ) {
+		res.append ( buf, rd );
+	}
+
+	return res;
+}
+
+void writeToFD ( int fd, const std::string & message, const std::string & errorDesc ) {
+	if ( write ( fd, message.c_str ( ), message.length ( ) ) != ( ssize_t ) message.length ( ) )
+		throw std::runtime_error ( "TimeoutAqlTest: write() to pipe failed (" + errorDesc + ")" );
+}
+
+
 void newSigChild ( int ) {
-	char dummy = 0;
 	g_RecvSignal = 1;
 
 	// write into the pipe so select can read something, this effectively means that SIGCHILD was raised
-	if ( write ( g_Wakeup [ PIPE_WR ], &dummy, 1 ) != 1 )
-		throw std::runtime_error ( "TimeoutAqlTest: write() failure (wakeup signalling)" );
+	writeToFD ( g_Wakeup [ PIPE_WR ], " ", "wakeup signalling" );
 }
 
-int aqlTest ( int fd, const ext::vector < std::string > & queries, unsigned seed ) {
+int aqlTest ( int fd_aql, const ext::vector < std::string > & queries, unsigned seed ) {
 	try {
 		cli::Environment environment;
 
@@ -60,41 +82,27 @@ int aqlTest ( int fd, const ext::vector < std::string > & queries, unsigned seed
 		common::Streams::out = ss;
 		common::Streams::err = ss;
 
+		// seed cli
 		cli::CharSequence sequence { cli::StringLineInterface ( "set seed " + ext::to_string ( seed ) ) };
 		cli::Parser ( cli::Lexer ( std::move ( sequence ) ) ).parse ( ) -> run ( environment );
 
+		// run queries
 		for ( const std::string & q : queries ) {
 			cli::CharSequence querySequence { cli::StringLineInterface ( q ) };
 			cli::Parser ( cli::Lexer ( std::move ( querySequence ) ) ).parse ( ) -> run ( environment );
 
-			if ( write ( fd, ss.str ( ).c_str ( ), ss.str ( ).length ( ) ) != ( ssize_t ) ss.str ( ).length ( ) )
-				throw std::runtime_error ( "TimeoutAqlTest: child output write() failure (child to parent communication)" );
+			writeToFD ( fd_aql, ss.str ( ), "writing aql output" );
 		}
 
 		return cli::ResultInterpret::cli ( environment.getResult ( ) ); /* 0 = OK */
 	} catch ( const std::exception & ) {
 		std::ostringstream oss;
 		alib::ExceptionHandler::handle ( oss );
-		if ( write ( fd, oss.str ( ).c_str ( ), oss.str ( ).length ( ) ) != ( ssize_t ) oss.str ( ).length ( ) )
-			throw std::runtime_error ( "TimeoutAqlTest: write() failure (child to parent communication)" );
+		writeToFD ( fd_aql, oss.str ( ), "writing aql output" );
 		return -1;
 	}
 }
 
-std::string readChildOutput ( int fd ) {
-	static const size_t BUFSIZE = 64;
-
-	std::string res;
-	int rd;
-	char buf [ BUFSIZE ];
-
-	while ( ( rd = read ( fd, &buf, BUFSIZE - 1 ) ) > 0 ) {
-		res.append ( buf, rd );
-	}
-
-	return res;
-}
-
 void _TimeoutAqlTest ( const std::chrono::microseconds & timeout, const ext::vector < std::string > & queries ) {
 	/* Register SIGCHLD handler */
 	struct sigaction act;
@@ -102,75 +110,104 @@ void _TimeoutAqlTest ( const std::chrono::microseconds & timeout, const ext::vec
 	act . sa_handler = newSigChild;
 	sigaction ( SIGCHLD, &act, nullptr );
 
-	/* parent-child communication ( for exceptions ) */
-	int pipefd [ 2 ];
-	if ( pipe ( pipefd ) != 0 )
-		throw std::runtime_error ( "TimeoutAqlTest: Failed to initialize pipe (child to parent communication)" );
+	int pipeAqlOutput [ 2 ]; /* parent-child communication ( aql output ) */
+	int pipeStdout [ 2 ];    /* parent-child communication ( child stdout ) */
+	int pipeStderr [ 2 ];    /* parent-child communication ( child stderr ) */
+
+	if ( pipe ( pipeAqlOutput ) != 0 )
+		throw std::runtime_error ( "TimeoutAqlTest: Failed to initialize pipe (aql output)" );
+
+	if ( pipe ( pipeStdout ) != 0 )
+		throw std::runtime_error ( "TimeoutAqlTest: Failed to initialize pipe (child stdout)" );
+
+	if ( pipe ( pipeStderr ) != 0 )
+		throw std::runtime_error ( "TimeoutAqlTest: Failed to initialize pipe (child stderr)" );
 
 	/* SIGCHLD was not yet raised, initialize communication pipe */
 	g_RecvSignal = 0;
 	if ( pipe ( g_Wakeup ) )
 		throw std::runtime_error ( "TimeoutAqlTest: Failed to initialize pipe (wakeup signalling)" );
 
+	/* random seed for aql */
 	unsigned seed = ext::random_devices::random ( );
 
 	/* do the forking */
-	pid_t x = fork ();
-	if ( x < 0 ) {
-		FAIL ( "Fork error" );
-	} else if ( x == 0 ) {
-		/* child, run the test here */
+	pid_t pid = fork ();
+	REQUIRE ( pid >= 0 );
 
+	if ( pid == 0 ) { /* child, run the test here */
 		act . sa_handler = SIG_DFL;
 		sigaction ( SIGCHLD, &act, nullptr );
 
+		/* close unused ends of pipes in child */
 		close ( g_Wakeup [ PIPE_RD ] );
 		close ( g_Wakeup [ PIPE_WR ] );
-		close ( pipefd [ PIPE_RD ] );
+		close ( pipeAqlOutput [ PIPE_RD ] );
+
+		close ( pipeStdout [ PIPE_RD ] );
+		close ( pipeStderr [ PIPE_RD ] );
 
-		/* just in case ... */
-		close ( 1 );
-		close ( 2 );
+		/* redirect stderr and stdout to pipe */
+		dup2 ( pipeStdout [ PIPE_WR ], FD_STDOUT );
+		dup2 ( pipeStderr [ PIPE_WR ], FD_STDERR );
 
-		exit ( aqlTest ( pipefd [ PIPE_WR ], queries, seed ) );
+		/* run test */
+		exit ( aqlTest ( pipeAqlOutput [ PIPE_WR ], queries, seed ) );
 	}
 
-	close ( pipefd [ PIPE_WR ] );
+	/* close unused ends of pipes in parent */
+	close ( pipeAqlOutput [ PIPE_WR ] );
+	close ( pipeStdout [ PIPE_WR ] );
+	close ( pipeStderr [ PIPE_WR ] );
 
 	/* lets wait the specified time of microseconds, maybe the child will terminate on its own */
 	if ( ! waitSignalTimeout ( timeout.count ( ) ) ) {
 		/* ... and in case it did not ... */
-		kill ( x, SIGTERM );
+		kill ( pid, SIGTERM );
 		while ( ! waitSignalTimeout ( 250000 ) ) /* 1/4 second */
-			kill ( x, SIGKILL );
+			kill ( pid, SIGKILL );
 	}
 
 	/* child termination confirmed */
 	int status;
-	waitpid ( x, &status, 0 );
+	waitpid ( pid, &status, 0 );
 	close ( g_Wakeup [ PIPE_RD ] );
 	close ( g_Wakeup [ PIPE_WR ] );
 
-	std::string childOutput = readChildOutput ( pipefd [ PIPE_RD ] );
+	/* read child outputs */
+	std::string childOutput [ 3 ] = {
+		readFromFD ( pipeAqlOutput [ PIPE_RD ] ),
+		readFromFD ( pipeStdout [ PIPE_RD ] ),
+		readFromFD ( pipeStderr [ PIPE_RD ] )
+	};
 
-	close ( pipefd [ PIPE_RD ] );
-	close ( pipefd [ PIPE_WR ] );
+	/* communication is done */
+	close ( pipeAqlOutput [ PIPE_RD ] );
+	close ( pipeStdout [ PIPE_RD ] );
+	close ( pipeStderr [ PIPE_RD ] );
 
+	/* determine test status */
 	if ( WIFEXITED ( status ) ) {
 		INFO ( "AqlTest failure. Trying to execute: " << "\n" << formatQueries ( queries ) );
 		INFO ( "Seed was: " << seed );
-		INFO ( "Child output was: >" << childOutput << "<" );
+		INFO ( "Child aqlout was: >" << childOutput [ 0 ] << "<" );
+		INFO ( "Child stdout was: >" << childOutput [ 1 ] << "<" );
+		INFO ( "Child stderr was: >" << childOutput [ 2 ] << "<" );
 		REQUIRE ( WEXITSTATUS ( status ) == 0 );
 	} else if ( WIFSIGNALED ( status ) ) {
 		if ( WTERMSIG ( status ) == SIGTERM || WTERMSIG ( status ) == SIGKILL ) { /* killed by timeout control */
 			WARN ( "AqlTest timeout (" << timeout.count ( ) << " us) reached. Trying to execute:\n" << formatQueries ( queries ) <<
 			       "Seed was: " << seed << "\n" <<
-			       "Child output was: >" << childOutput << "<" );
+			       "Child aqlout was: >" << childOutput [ 0 ] << "<\n" <<
+			       "Child stdout was: >" << childOutput [ 1 ] << "<\n" <<
+			       "Child stderr was: >" << childOutput [ 2 ] << "<\n" );
 			CHECK_NOFAIL ( "timeout warning" == nullptr );
 		} else {
 			INFO ( "AqlTest failure. Trying to execute: " << formatQueries ( queries ) );
 			INFO ( "Seed was: " << seed );
-			INFO ( "Child output was: >" << childOutput << "<" );
+			INFO ( "Child aqlout was: >" << childOutput [ 0 ] << "<" );
+			INFO ( "Child stdout was: >" << childOutput [ 1 ] << "<" );
+			INFO ( "Child stderr was: >" << childOutput [ 2 ] << "<" );
 			INFO ( "Child process signaled, signal " << WTERMSIG ( status ) << " (" << strsignal ( WTERMSIG ( status ) ) << ")" );
 			FAIL ( );
 		}
-- 
GitLab