Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[WIP] Higher-Order Marble tests #153

Open
wants to merge 20 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
20 commits
Select commit Hold shift + click to select a range
1a1b2e1
Preliminary marble testing
mbonneau Feb 4, 2017
fbeff95
more marble testing features
martinsik Feb 17, 2017
16e7304
fix correct type for str_repeat()
martinsik Feb 17, 2017
637d3fe
added marker groups support
martinsik Feb 17, 2017
654614a
Added expectObservable and expectSubscriptions to FunctionalTestCase
davidwdan Feb 18, 2017
4b91912
Added exception message support to expectObservable()->toBe()
davidwdan Feb 18, 2017
b2a3ae6
Added dispose support to expectObservable()
davidwdan Feb 18, 2017
3ea6d38
Rename MarbleDiagramError to MarbleDiagramException
mbonneau Feb 20, 2017
6147c65
Grouped marbles now happen at the same time
mbonneau Feb 20, 2017
a619b72
Grouped subscription marbles happen at the same time
mbonneau Feb 20, 2017
6e35541
Added interfaces so we get type hinting for `toBe`
davidwdan Feb 20, 2017
416034c
Fixed continues to actually continue the loop
davidwdan Feb 20, 2017
14b7fd3
Merge branch '2.x' of github.com:ReactiveX/RxPHP into marbleTest
davidwdan Mar 10, 2017
c1d2d4a
Cherry picked higher-order observable test from v1
davidwdan Mar 10, 2017
08b0e0a
use single instance of TestScheduler in all Notification objects
martinsik Mar 11, 2017
eb24667
Materialize the inner observable when converting the OnNextNotificati…
davidwdan Mar 11, 2017
8d219fa
Simplified `materializeObservable`
davidwdan Mar 11, 2017
c153965
Removed unused use statement
davidwdan Mar 11, 2017
ee4ec61
remove redundant test
martinsik Mar 12, 2017
58d185d
`toBe` now does a string comparison
davidwdan Mar 12, 2017
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
52 changes: 52 additions & 0 deletions src/Notification/OnNextObservableNotification.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
<?php

namespace Rx\Notification;

use Rx\Observable;
use Rx\Testing\ColdObservable;
use Rx\Testing\HotObservable;
use Rx\Testing\MockHigherOrderObserver;
use Rx\Testing\TestScheduler;

class OnNextObservableNotification extends OnNextNotification
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: should this live in Rx\Testing instead? It depends on the TestScheduler?

{
/** @var MockHigherOrderObserver */
private $observer;

public function __construct(Observable $value, TestScheduler $scheduler)
{
parent::__construct($value);

$this->observer = new MockHigherOrderObserver($scheduler, $scheduler->getClock());
$value->subscribe($this->observer);
}

public function equals($other): bool
{
$messages1 = $this->getMessages();
/** @var OnNextObservableNotification $other */
$messages2 = $other->getMessages();

if (count($messages1) != count($messages2)) {
return false;
}

for ($i = 0; $i < count($messages1); $i++) {
if (!$messages1[$i]->equals($messages2[$i])) {
return false;
}
}

return true;
}

public function getMessages()
{
return $this->observer->getMessages();
}

public function __toString(): string
{
return '[' . implode(', ', $this->getMessages()) . ']';
}
}
15 changes: 10 additions & 5 deletions src/Testing/ColdObservable.php
Original file line number Diff line number Diff line change
Expand Up @@ -26,13 +26,16 @@ public function __construct(TestScheduler $scheduler, array $messages = [])

protected function _subscribe(ObserverInterface $observer): DisposableInterface
{
$this->subscriptions[] = new Subscription($this->scheduler->getClock());
$index = count($this->subscriptions) - 1;

$currentObservable = $this;
$disposable = new CompositeDisposable();
$scheduler = $this->scheduler;
$isDisposed = false;
$index = null;

if (!($observer instanceof MockHigherOrderObserver)) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do we need all these instanceof checks? Looks wrong.

$this->subscriptions[] = new Subscription($this->scheduler->getClock());
$index = count($this->subscriptions) - 1;
}

foreach ($this->messages as $message) {
$notification = $message->getValue();
Expand All @@ -53,8 +56,10 @@ protected function _subscribe(ObserverInterface $observer): DisposableInterface
$subscriptions = &$this->subscriptions;

return new CallbackDisposable(function () use (&$currentObservable, $index, $observer, $scheduler, &$subscriptions, &$isDisposed) {
$isDisposed = true;
$subscriptions[$index] = new Subscription($subscriptions[$index]->getSubscribed(), $scheduler->getClock());
$isDisposed = true;
if (!($observer instanceof MockHigherOrderObserver)) {
$subscriptions[$index] = new Subscription($subscriptions[$index]->getSubscribed(), $scheduler->getClock());
}
});

}
Expand Down
15 changes: 10 additions & 5 deletions src/Testing/HotObservable.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,17 +48,22 @@ protected function _subscribe(ObserverInterface $observer): DisposableInterface
{
$currentObservable = $this;

$this->observers[] = $observer;
$this->subscriptions[] = new Subscription($this->scheduler->getClock());
$this->observers[] = $observer;
$subscriptions = &$this->subscriptions;
$index = null;

$subscriptions = &$this->subscriptions;
if (!($observer instanceof MockHigherOrderObserver)) {
$this->subscriptions[] = new Subscription($this->scheduler->getClock());
$index = count($this->subscriptions) - 1;
}

$index = count($this->subscriptions) - 1;
$scheduler = $this->scheduler;

return new CallbackDisposable(function () use (&$currentObservable, $index, $observer, $scheduler, &$subscriptions) {
$currentObservable->removeObserver($observer);
$subscriptions[$index] = new Subscription($subscriptions[$index]->getSubscribed(), $scheduler->getClock());
if (!($observer instanceof MockHigherOrderObserver)) {
$subscriptions[$index] = new Subscription($subscriptions[$index]->getSubscribed(), $scheduler->getClock());
}
});
}

Expand Down
7 changes: 7 additions & 0 deletions src/Testing/MockHigherOrderObserver.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
<?php

namespace Rx\Testing;

class MockHigherOrderObserver extends MockObserver
{
}
20 changes: 15 additions & 5 deletions src/Testing/MockObserver.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@
use Rx\Notification\OnCompletedNotification;
use Rx\Notification\OnErrorNotification;
use Rx\Notification\OnNextNotification;
use Rx\Notification\OnNextObservableNotification;
use Rx\Observable;
use Rx\ObserverInterface;

/**
Expand All @@ -16,32 +18,40 @@ class MockObserver implements ObserverInterface
{
private $scheduler;
private $messages = [];
private $startTime = 0;

public function __construct(TestScheduler $scheduler)
public function __construct(TestScheduler $scheduler, int $startTime = 0)
{
$this->scheduler = $scheduler;
$this->startTime = $startTime;
}

public function onNext($value)
{
if ($value instanceof Observable) {
$notification = new OnNextObservableNotification($value, $this->scheduler);
} else {
$notification = new OnNextNotification($value);
}

$this->messages[] = new Recorded(
$this->scheduler->getClock(),
new OnNextNotification($value)
$this->scheduler->getClock() - $this->startTime,
$notification
);
}

public function onError(\Throwable $error)
{
$this->messages[] = new Recorded(
$this->scheduler->getClock(),
$this->scheduler->getClock() - $this->startTime,
new OnErrorNotification($error)
);
}

public function onCompleted()
{
$this->messages[] = new Recorded(
$this->scheduler->getClock(),
$this->scheduler->getClock() - $this->startTime,
new OnCompletedNotification()
);
}
Expand Down
16 changes: 16 additions & 0 deletions test/Rx/Functional/FunctionalTestCase.php
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,19 @@ abstract class FunctionalTestCase extends TestCase
/** @var TestScheduler */
protected $scheduler;

/** @var TestScheduler */
static protected $globalScheduler;

const TIME_FACTOR = 10;

public function setup()
{
$this->scheduler = $this->createTestScheduler();
self::$globalScheduler = $this->scheduler;
}

static function getScheduler() {
return self::$globalScheduler;
}

/**
Expand All @@ -32,6 +40,8 @@ public function setup()
*/
public function assertMessages(array $expected, array $recorded)
{
$this->scheduler->start();

if (count($expected) !== count($recorded)) {
$this->fail(sprintf('Expected message count %d does not match actual count %d.', count($expected), count($recorded)));
}
Expand All @@ -51,6 +61,8 @@ public function assertMessages(array $expected, array $recorded)
*/
public function assertMessagesNotEqual(array $expected, array $recorded)
{
$this->scheduler->start();

if (count($expected) !== count($recorded)) {
$this->assertTrue(true);
return;
Expand All @@ -68,6 +80,8 @@ public function assertMessagesNotEqual(array $expected, array $recorded)

public function assertSubscription(HotObservable $observable, Subscription $expected)
{
$this->scheduler->start();

$subscriptionCount = count($observable->getSubscriptions());

if ($subscriptionCount === 0) {
Expand All @@ -89,6 +103,8 @@ public function assertSubscription(HotObservable $observable, Subscription $expe

public function assertSubscriptions(array $expected, array $recorded)
{
$this->scheduler->start();

if (count($expected) !== count($recorded)) {
$this->fail(sprintf('Expected subscription count %d does not match actual count %d.', count($expected), count($recorded)));
}
Expand Down
Loading