Skip to content
Snippets Groups Projects
Commit 315342b9 authored by Ted Bowman's avatar Ted Bowman
Browse files

Merge branch '3.1.x' into 3441577-core-mr

parents 814c7e78 02406e9b
No related branches found
No related tags found
1 merge request!1057Issue #3441577 add property types
...@@ -263,7 +263,7 @@ class ComposerInspector implements LoggerAwareInterface { ...@@ -263,7 +263,7 @@ class ComposerInspector implements LoggerAwareInterface {
} }
} }
$output = $this->processCallback->getOutput(); $output = $this->processCallback->getOutput();
return isset($output) ? trim($output) : $output; return $output ? trim(implode('', $output)) : NULL;
} }
/** /**
......
...@@ -29,14 +29,14 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar ...@@ -29,14 +29,14 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar
* *
* @var string * @var string
*/ */
private string $outBuffer = ''; private array $outBuffer = [];
/** /**
* The error buffer. * The error buffer.
* *
* @var string * @var string
*/ */
private string $errorBuffer = ''; private array $errorBuffer = [];
/** /**
* Constructs a ProcessOutputCallback object. * Constructs a ProcessOutputCallback object.
...@@ -51,10 +51,10 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar ...@@ -51,10 +51,10 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar
public function __invoke(OutputTypeEnum $type, string $buffer): void { public function __invoke(OutputTypeEnum $type, string $buffer): void {
if ($type === OutputTypeEnum::OUT) { if ($type === OutputTypeEnum::OUT) {
$this->outBuffer .= $buffer; $this->outBuffer[] = $buffer;
} }
elseif ($type === OutputTypeEnum::ERR) { elseif ($type === OutputTypeEnum::ERR) {
$this->errorBuffer .= $buffer; $this->errorBuffer[] = $buffer;
} }
} }
...@@ -66,12 +66,12 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar ...@@ -66,12 +66,12 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar
* @return string|null * @return string|null
* The output or NULL if there is none. * The output or NULL if there is none.
*/ */
public function getOutput(): ?string { public function getOutput(): array {
$error_output = $this->getErrorOutput(); $error_output = $this->getErrorOutput();
if ($error_output) { if ($error_output) {
$this->logger->warning($error_output); $this->logger->warning(implode('', $error_output));
} }
return trim($this->outBuffer) !== '' ? $this->outBuffer : NULL; return $this->outBuffer;
} }
/** /**
...@@ -82,8 +82,8 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar ...@@ -82,8 +82,8 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar
*/ */
public function parseJsonOutput(): mixed { public function parseJsonOutput(): mixed {
$output = $this->getOutput(); $output = $this->getOutput();
if ($output !== NULL) { if ($output) {
return json_decode($output, TRUE, flags: JSON_THROW_ON_ERROR); return json_decode(trim(implode('', $output)), TRUE, flags: JSON_THROW_ON_ERROR);
} }
return NULL; return NULL;
} }
...@@ -94,8 +94,22 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar ...@@ -94,8 +94,22 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar
* @return string|null * @return string|null
* The error output or NULL if there isn't any. * The error output or NULL if there isn't any.
*/ */
public function getErrorOutput(): ?string { public function getErrorOutput(): array {
return trim($this->errorBuffer) !== '' ? $this->errorBuffer : NULL; return $this->errorBuffer;
}
/**
* {@inheritdoc}
*/
public function clearErrorOutput(): void {
$this->errorBuffer = [];
}
/**
* {@inheritdoc}
*/
public function clearOutput(): void {
$this->outBuffer = [];
} }
/** /**
...@@ -104,8 +118,8 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar ...@@ -104,8 +118,8 @@ final class ProcessOutputCallback implements OutputCallbackInterface, LoggerAwar
* @return self * @return self
*/ */
public function reset(): self { public function reset(): self {
$this->errorBuffer = ''; $this->clearErrorOutput();
$this->outBuffer = ''; $this->clearOutput();
return $this; return $this;
} }
......
...@@ -445,6 +445,34 @@ class FixtureManipulator { ...@@ -445,6 +445,34 @@ class FixtureManipulator {
} }
} }
/**
* {@inheritdoc}
*/
public function clearErrorOutput(): void {
throw new \LogicException("Unexpected call to clearErrorOutput().");
}
/**
* {@inheritdoc}
*/
public function clearOutput(): void {
throw new \LogicException("Unexpected call to clearOutput().");
}
/**
* {@inheritdoc}
*/
public function getErrorOutput(): array {
throw new \LogicException("Unexpected call to getErrorOutput().");
}
/**
* {@inheritdoc}
*/
public function getOutput(): array {
throw new \LogicException("Unexpected call to getOutput().");
}
}; };
/** @var \PhpTuf\ComposerStager\API\Process\Service\ComposerProcessRunnerInterface $runner */ /** @var \PhpTuf\ComposerStager\API\Process\Service\ComposerProcessRunnerInterface $runner */
$runner = \Drupal::service(ComposerProcessRunnerInterface::class); $runner = \Drupal::service(ComposerProcessRunnerInterface::class);
......
...@@ -63,6 +63,18 @@ abstract class TemplateProjectTestBase extends QuickStartTestBase { ...@@ -63,6 +63,18 @@ abstract class TemplateProjectTestBase extends QuickStartTestBase {
*/ */
protected const MAX_EXECUTION_TIME = 20; protected const MAX_EXECUTION_TIME = 20;
/**
* {@inheritdoc}
*/
protected function setUp(): void {
// Build tests cannot be run if Sqlite minimum version is not met.
$sqlite = (new \PDO('sqlite::memory:'))->query('select sqlite_version()')->fetch()[0];
if (version_compare($sqlite, Tasks::SQLITE_MINIMUM_VERSION) < 0) {
$this->markTestSkipped();
}
parent::setUp();
}
/** /**
* {@inheritdoc} * {@inheritdoc}
*/ */
......
...@@ -10,7 +10,6 @@ use Drupal\fixture_manipulator\ActiveFixtureManipulator; ...@@ -10,7 +10,6 @@ use Drupal\fixture_manipulator\ActiveFixtureManipulator;
use Drupal\package_manager\ComposerInspector; use Drupal\package_manager\ComposerInspector;
use Drupal\package_manager\Exception\ComposerNotReadyException; use Drupal\package_manager\Exception\ComposerNotReadyException;
use Drupal\package_manager\InstalledPackage; use Drupal\package_manager\InstalledPackage;
use Drupal\package_manager\ProcessOutputCallback;
use Drupal\package_manager\InstalledPackagesList; use Drupal\package_manager\InstalledPackagesList;
use Drupal\Tests\package_manager\Traits\InstalledPackagesListTrait; use Drupal\Tests\package_manager\Traits\InstalledPackagesListTrait;
use Drupal\package_manager\PathLocator; use Drupal\package_manager\PathLocator;
...@@ -19,6 +18,7 @@ use PhpTuf\ComposerStager\API\Exception\RuntimeException; ...@@ -19,6 +18,7 @@ use PhpTuf\ComposerStager\API\Exception\RuntimeException;
use PhpTuf\ComposerStager\API\Path\Factory\PathFactoryInterface; use PhpTuf\ComposerStager\API\Path\Factory\PathFactoryInterface;
use PhpTuf\ComposerStager\API\Precondition\Service\ComposerIsAvailableInterface; use PhpTuf\ComposerStager\API\Precondition\Service\ComposerIsAvailableInterface;
use PhpTuf\ComposerStager\API\Process\Service\ComposerProcessRunnerInterface; use PhpTuf\ComposerStager\API\Process\Service\ComposerProcessRunnerInterface;
use PhpTuf\ComposerStager\API\Process\Service\OutputCallbackInterface;
use PhpTuf\ComposerStager\API\Process\Value\OutputTypeEnum; use PhpTuf\ComposerStager\API\Process\Value\OutputTypeEnum;
use Prophecy\Argument; use Prophecy\Argument;
use Prophecy\Prophecy\ObjectProphecy; use Prophecy\Prophecy\ObjectProphecy;
...@@ -250,10 +250,18 @@ class ComposerInspectorTest extends PackageManagerKernelTestBase { ...@@ -250,10 +250,18 @@ class ComposerInspectorTest extends PackageManagerKernelTestBase {
public function testVersionCheck(?string $reported_version, ?string $expected_message): void { public function testVersionCheck(?string $reported_version, ?string $expected_message): void {
$runner = $this->mockComposerRunner($reported_version); $runner = $this->mockComposerRunner($reported_version);
// Mock the ComposerIsAvailableInterface so that if it uses the Composer
// runner it will not affect the test expectations.
$composerPrecondition = $this->prophesize(ComposerIsAvailableInterface::class);
$composerPrecondition
->assertIsFulfilled(Argument::cetera())
->shouldBeCalledOnce();
$this->container->set(ComposerIsAvailableInterface::class, $composerPrecondition->reveal());
// The result of the version check is statically cached, so the runner // The result of the version check is statically cached, so the runner
// should only be called once, even though we call validate() twice in this // should only be called once, even though we call validate() twice in this
// test. // test.
$runner->getMethodProphecies('run')[0]->shouldBeCalledOnce(); $runner->getMethodProphecies('run')[0]->withArguments([['--format=json'], NULL, [], Argument::any()])->shouldBeCalledOnce();
// The runner should be called with `validate` as the first argument, but // The runner should be called with `validate` as the first argument, but
// it won't affect the outcome of this test. // it won't affect the outcome of this test.
$runner->run(Argument::withEntry(0, 'validate')); $runner->run(Argument::withEntry(0, 'validate'));
...@@ -535,7 +543,7 @@ class ComposerInspectorTest extends PackageManagerKernelTestBase { ...@@ -535,7 +543,7 @@ class ComposerInspectorTest extends PackageManagerKernelTestBase {
]); ]);
$callback = end($arguments_passed_to_runner); $callback = end($arguments_passed_to_runner);
assert($callback instanceof ProcessOutputCallback); assert($callback instanceof OutputCallbackInterface);
$callback(OutputTypeEnum::OUT, $command_output); $callback(OutputTypeEnum::OUT, $command_output);
}; };
......
...@@ -38,13 +38,13 @@ class ProcessOutputCallbackTest extends UnitTestCase { ...@@ -38,13 +38,13 @@ class ProcessOutputCallbackTest extends UnitTestCase {
$error_text = 'What happened?'; $error_text = 'What happened?';
$callback(OutputTypeEnum::ERR, $error_text); $callback(OutputTypeEnum::ERR, $error_text);
$this->assertSame($error_text, $callback->getErrorOutput()); $this->assertSame([$error_text], $callback->getErrorOutput());
// The error should not yet be logged. // The error should not yet be logged.
$this->assertEmpty($logger->records); $this->assertEmpty($logger->records);
// There should be no output data, but calling getOutput() should log the // There should be no output data, but calling getOutput() should log the
// error. // error.
$this->assertNull($callback->getOutput()); $this->assertSame([], $callback->getOutput());
$this->assertNull($callback->parseJsonOutput()); $this->assertNull($callback->parseJsonOutput());
$this->assertTrue($logger->hasWarning($error_text)); $this->assertTrue($logger->hasWarning($error_text));
...@@ -63,8 +63,8 @@ class ProcessOutputCallbackTest extends UnitTestCase { ...@@ -63,8 +63,8 @@ class ProcessOutputCallbackTest extends UnitTestCase {
$callback->setLogger($logger); $callback->setLogger($logger);
// The buffers should initially be empty, and nothing should be logged. // The buffers should initially be empty, and nothing should be logged.
$this->assertNull($callback->getOutput()); $this->assertSame([], $callback->getOutput());
$this->assertNull($callback->getErrorOutput()); $this->assertSame([], $callback->getErrorOutput());
$this->assertNull($callback->parseJsonOutput()); $this->assertNull($callback->parseJsonOutput());
$this->assertEmpty($logger->records); $this->assertEmpty($logger->records);
...@@ -77,13 +77,15 @@ class ProcessOutputCallbackTest extends UnitTestCase { ...@@ -77,13 +77,15 @@ class ProcessOutputCallbackTest extends UnitTestCase {
$json = json_encode($data, JSON_PRETTY_PRINT); $json = json_encode($data, JSON_PRETTY_PRINT);
// Ensure the JSON is a multi-line string. // Ensure the JSON is a multi-line string.
$this->assertGreaterThan(1, substr_count($json, "\n")); $this->assertGreaterThan(1, substr_count($json, "\n"));
$expected_output = [];
foreach (explode("\n", $json) as $line) { foreach (explode("\n", $json) as $line) {
$callback(OutputTypeEnum::OUT, "$line\n"); $callback(OutputTypeEnum::OUT, "$line\n");
$expected_output[] = "$line\n";
} }
$this->assertSame("$json\n", $callback->getOutput()); $this->assertSame($expected_output, $callback->getOutput());
// Ensure that parseJsonOutput() can parse the data without errors. // Ensure that parseJsonOutput() can parse the data without errors.
$this->assertSame($data, $callback->parseJsonOutput()); $this->assertSame($data, $callback->parseJsonOutput());
$this->assertNull($callback->getErrorOutput()); $this->assertSame([], $callback->getErrorOutput());
$this->assertEmpty($logger->records); $this->assertEmpty($logger->records);
// If we send error output, it should be logged, but we should still be able // If we send error output, it should be logged, but we should still be able
...@@ -91,34 +93,37 @@ class ProcessOutputCallbackTest extends UnitTestCase { ...@@ -91,34 +93,37 @@ class ProcessOutputCallbackTest extends UnitTestCase {
$callback(OutputTypeEnum::ERR, 'Oh no, what happened?'); $callback(OutputTypeEnum::ERR, 'Oh no, what happened?');
$callback(OutputTypeEnum::ERR, 'Really what happened?!'); $callback(OutputTypeEnum::ERR, 'Really what happened?!');
$this->assertSame($data, $callback->parseJsonOutput()); $this->assertSame($data, $callback->parseJsonOutput());
$this->assertSame('Oh no, what happened?Really what happened?!', $callback->getErrorOutput()); $expected_error = ['Oh no, what happened?', 'Really what happened?!'];
$this->assertSame($expected_error, $callback->getErrorOutput());
$this->assertTrue($logger->hasWarning('Oh no, what happened?Really what happened?!')); $this->assertTrue($logger->hasWarning('Oh no, what happened?Really what happened?!'));
// Send more output and error data to the callback; they should be appended // Send more output and error data to the callback; they should be appended
// to the data we previously sent. // to the data we previously sent.
$callback(OutputTypeEnum::OUT, '{}'); $callback(OutputTypeEnum::OUT, '{}');
$expected_output[] = '{}';
$callback(OutputTypeEnum::ERR, 'new Error 1!'); $callback(OutputTypeEnum::ERR, 'new Error 1!');
$callback(OutputTypeEnum::ERR, 'new Error 2!'); $callback(OutputTypeEnum::ERR, 'new Error 2!');
$expected_error[] = 'new Error 1!';
$expected_error[] = 'new Error 2!';
// The output buffer will no longer be valid JSON, so don't try to parse it. // The output buffer will no longer be valid JSON, so don't try to parse it.
$this->assertSame("$json\n{}", $callback->getOutput()); $this->assertSame($expected_output, $callback->getOutput());
$expected_error = 'Oh no, what happened?Really what happened?!new Error 1!new Error 2!';
$this->assertSame($expected_error, $callback->getErrorOutput()); $this->assertSame($expected_error, $callback->getErrorOutput());
$this->assertTrue($logger->hasWarning($expected_error)); $this->assertTrue($logger->hasWarning(implode('', $expected_error)));
// The previously logged error output should still be there. // The previously logged error output should still be there.
$this->assertTrue($logger->hasWarning('Oh no, what happened?Really what happened?!')); $this->assertTrue($logger->hasWarning('Oh no, what happened?Really what happened?!'));
// Clear all stored output and errors. // Clear all stored output and errors.
$callback->reset(); $callback->reset();
$this->assertNull($callback->getOutput()); $this->assertSame([], $callback->getOutput());
$this->assertNull($callback->getErrorOutput()); $this->assertSame([], $callback->getErrorOutput());
$this->assertNull($callback->parseJsonOutput()); $this->assertNull($callback->parseJsonOutput());
// Send more output and error data. // Send more output and error data.
$callback(OutputTypeEnum::OUT, 'Bonjour!'); $callback(OutputTypeEnum::OUT, 'Bonjour!');
$callback(OutputTypeEnum::ERR, 'You continue to annoy me.'); $callback(OutputTypeEnum::ERR, 'You continue to annoy me.');
// We should now only see the stuff we just sent... // We should now only see the stuff we just sent...
$this->assertSame('Bonjour!', $callback->getOutput()); $this->assertSame(['Bonjour!'], $callback->getOutput());
$this->assertSame('You continue to annoy me.', $callback->getErrorOutput()); $this->assertSame(['You continue to annoy me.'], $callback->getErrorOutput());
$this->assertTrue($logger->hasWarning('You continue to annoy me.')); $this->assertTrue($logger->hasWarning('You continue to annoy me.'));
// ...but the previously logged errors should still be there. // ...but the previously logged errors should still be there.
$this->assertTrue($logger->hasWarning('Oh no, what happened?Really what happened?!new Error 1!new Error 2!')); $this->assertTrue($logger->hasWarning('Oh no, what happened?Really what happened?!new Error 1!new Error 2!'));
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment