mirror of
https://we.phorge.it/source/arcanist.git
synced 2025-01-08 22:01:02 +01:00
Provide "MethodCallFuture" to fix exception semantics in mixed-future contexts
Summary: Ref T13666. Currently, "PhabricatorRepository->newConduitFuture()" sometimes returns a real "ConduitFuture" (when repository clustering is configured) and sometimes returns a degenerate "ImmediateFuture" (when repository clustering is not configured). To populate the "ImmediateFuture", the Conduit method is called inline without any special exception handling. This means that the two pathways have significantly different exception behavior: - On the "ImmediateFuture" pathway, the "newConduitFuture()" method itself may raise an exception related to resolving the call (e.g., invalid parameters). - On the "ConduitFuture" pathway, exceptions are raised only once the future is "resolve()"'d. The second behavior is the correct behavior (and the primary behavior of upstream test environments, since all my stuff and the Phacility stuff is clustered). Prepare to put both pathways on the second behavior by introducing a "MethodCallFuture", which can move the `$conduit_call->execute()` call (and thus its exception handling) to future resolution time. See also the followup diff in Phabricator to actually enact this change. Test Plan: Added unit tests and made them pass, see also next change. Reviewers: cspeckmim Reviewed By: cspeckmim Maniphest Tasks: T13666 Differential Revision: https://secure.phabricator.com/D21720
This commit is contained in:
parent
82016c00e1
commit
f993b1fbda
3 changed files with 97 additions and 0 deletions
|
@ -632,6 +632,8 @@ phutil_register_library_map(array(
|
|||
'LinesOfALargeFile' => 'filesystem/linesofalarge/LinesOfALargeFile.php',
|
||||
'LinesOfALargeFileTestCase' => 'filesystem/linesofalarge/__tests__/LinesOfALargeFileTestCase.php',
|
||||
'MFilterTestHelper' => 'utils/__tests__/MFilterTestHelper.php',
|
||||
'MethodCallFuture' => 'future/MethodCallFuture.php',
|
||||
'MethodCallFutureTestCase' => 'future/__tests__/MethodCallFutureTestCase.php',
|
||||
'NoseTestEngine' => 'unit/engine/NoseTestEngine.php',
|
||||
'PHPASTParserTestCase' => 'parser/xhpast/__tests__/PHPASTParserTestCase.php',
|
||||
'PhageAction' => 'phage/action/PhageAction.php',
|
||||
|
@ -1695,6 +1697,8 @@ phutil_register_library_map(array(
|
|||
'LinesOfALargeFile' => 'LinesOfALarge',
|
||||
'LinesOfALargeFileTestCase' => 'PhutilTestCase',
|
||||
'MFilterTestHelper' => 'Phobject',
|
||||
'MethodCallFuture' => 'Future',
|
||||
'MethodCallFutureTestCase' => 'PhutilTestCase',
|
||||
'NoseTestEngine' => 'ArcanistUnitTestEngine',
|
||||
'PHPASTParserTestCase' => 'PhutilTestCase',
|
||||
'PhageAction' => 'Phobject',
|
||||
|
|
36
src/future/MethodCallFuture.php
Normal file
36
src/future/MethodCallFuture.php
Normal file
|
@ -0,0 +1,36 @@
|
|||
<?php
|
||||
|
||||
/**
|
||||
* Degenerate future which resolves by calling a method.
|
||||
*
|
||||
* $future = new MethodCallFuture($calculator, 'add', 1, 2);
|
||||
*
|
||||
* This future is similar to @{class:ImmediateFuture}, but may make it easier
|
||||
* to implement exception behavior correctly. See T13666.
|
||||
*/
|
||||
final class MethodCallFuture extends Future {
|
||||
|
||||
private $callObject;
|
||||
private $callMethod;
|
||||
private $callArgv;
|
||||
|
||||
public function __construct($object, $method /* , ...*/ ) {
|
||||
$argv = func_get_args();
|
||||
|
||||
$this->callObject = $object;
|
||||
$this->callMethod = $method;
|
||||
$this->callArgv = array_slice($argv, 2);
|
||||
}
|
||||
|
||||
public function isReady() {
|
||||
|
||||
$call = array($this->callObject, $this->callMethod);
|
||||
$argv = $this->callArgv;
|
||||
|
||||
$result = call_user_func_array($call, $argv);
|
||||
$this->setResult($result);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
}
|
57
src/future/__tests__/MethodCallFutureTestCase.php
Normal file
57
src/future/__tests__/MethodCallFutureTestCase.php
Normal file
|
@ -0,0 +1,57 @@
|
|||
<?php
|
||||
|
||||
final class MethodCallFutureTestCase extends PhutilTestCase {
|
||||
|
||||
public function testMethodCallFutureSums() {
|
||||
$future = new MethodCallFuture($this, 'getSum', 1, 2, 3);
|
||||
$result = $future->resolve();
|
||||
|
||||
$this->assertEqual(6, $result, pht('MethodCallFuture: getSum(1, 2, 3)'));
|
||||
|
||||
$future = new MethodCallFuture($this, 'getSum');
|
||||
$result = $future->resolve();
|
||||
|
||||
$this->assertEqual(0, $result, pht('MethodCallFuture: getSum()'));
|
||||
}
|
||||
|
||||
public function testMethodCallFutureExceptions() {
|
||||
$future = new MethodCallFuture($this, 'raiseException');
|
||||
|
||||
// See T13666. Using "FutureIterator" to advance the future until it is
|
||||
// ready to resolve should NOT throw an exception.
|
||||
|
||||
foreach (new FutureIterator(array($future)) as $resolvable) {
|
||||
// Continue below...
|
||||
}
|
||||
|
||||
$caught = null;
|
||||
try {
|
||||
$future->resolve();
|
||||
} catch (PhutilMethodNotImplementedException $ex) {
|
||||
$caught = $ex;
|
||||
}
|
||||
|
||||
$this->assertTrue(
|
||||
($caught instanceof PhutilMethodNotImplementedException),
|
||||
pht('MethodCallFuture: exceptions raise at resolution.'));
|
||||
}
|
||||
|
||||
public function getSum(/* ... */) {
|
||||
$args = func_get_args();
|
||||
|
||||
$sum = 0;
|
||||
foreach ($args as $arg) {
|
||||
$sum += $arg;
|
||||
}
|
||||
|
||||
return $sum;
|
||||
}
|
||||
|
||||
public function raiseException() {
|
||||
// We just want to throw any narrow exception so the test isn't catching
|
||||
// too broad an exception type. This is simulating some exception during
|
||||
// future resolution.
|
||||
throw new PhutilMethodNotImplementedException();
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in a new issue