This is quite an advanced question, perhaps knowledge of Symfony and Behat may not be necessary to understand the problem.
So in order to test the input and output of an interactive CLI app bin/albumgrab
I've written in PHP using the Symfony Console component, I've set up my Behat feature context to build an expect
script, run via exec
.
This exec
command is run in PHP via Symfony Process
.
$expect = <<<BASH
exec expect -c '
set timeout 180
spawn bin/albumgrab
expect "Please enter the name of the directory"
send "/tmp/php-london\n"
expect "Please enter the URL to the first image"
send "https://www.facebook.com/PeeHPLondon/photos/pb.7119218495.-2207520000.1430669248./10153559172718496/?type=3&src=https%3A%2F%2Ffbcdn-sphotos-g-a.akamaihd.net%2Fhphotos-ak-xfp1%2Fv%2Ft1.0-9%2F10986697_10153559172718496_5727444485530442900_n.jpg%3Foh%3Dc47770f4cd15fecc6888bcd504899087%26oe%3D55DA9CB0%26__gda__%3D1439174101_7c78a93bf247dbad6c56681b6db5309c&size=960%2C959&fbid=10153559172718496\\n"
interact
'
BASH;
$process = new Symfony\Component\Process\Process($expect);
$process->mustRun();
However, when it gets past the second input, it seems to exit but successfully.
Calling:
$process->setTty(true);
Makes it run all the way through, but will print straight to stdout and I can no longer capture the output to make an assertion, not even with PHP's output buffering.
I figured PTY would be more suitable anyway:
$process->setPty(true);
As it was the solution to this StackOverflow question. However, this is not supported everywhere, at least not on Mac OS X.
You can see what I've attempted so far in Github: https://github.com/adamelso/albumgrab/pull/13/files and the Travis output for the last attempt https://travis-ci.org/adamelso/albumgrab/jobs/61137499
So my main question is why it keeps exiting with 0 early and how to prevent it?
As per the answer to Expect Script wait Command Hangs you need to wait for EOF instead of using the interact
command:
set timeout 180
spawn ./bin/albumgrab
expect "Please enter the name of the directory"
send "/tmp/php-london\n"
expect "Please enter the URL to the first image"
send "https://www.facebook.com/PeeHPLondon/photos/pb.7119218495.-2207520000.1430669248./10153559172718496/\n"
expect EOF
Here's a full script I've used to test it on an OS X:
<?php
require_once __DIR__.'/vendor/autoload.php';
use Symfony\Component\Process\Process;
$expect = <<<BASH
exec expect -c '
set timeout 180
spawn ./bin/albumgrab
expect "Please enter the name of the directory"
send "/tmp/php-london\n"
expect "Please enter the URL to the first image"
send "https://www.facebook.com/PeeHPLondon/photos/pb.7119218495.-2207520000.1430669248./10153559172718496/?type=3&src=https%3A%2F%2Ffbcdn-sphotos-g-a.akamaihd.net%2Fhphotos-ak-xfp1%2Fv%2Ft1.0-9%2F10986697_10153559172718496_5727444485530442900_n.jpg%3Foh%3Dc47770f4cd15fecc6888bcd504899087%26oe%3D55DA9CB0%26__gda__%3D1439174101_7c78a93bf247dbad6c56681b6db5309c&size=960%2C959&fbid=10153559172718496\\n"
expect EOF
'
BASH;
$process = new Process($expect);
$process->setPty(true);
$process->start();
$process->wait(function ($type, $buffer) {
if (Process::ERR === $type) {
echo 'ERR > '.$buffer;
} else {
echo 'OUT > '.$buffer;
}
});
if (!$process->isSuccessful()) {
throw new \RuntimeException($process->getErrorOutput());
}