2019-01-10 15:20:05 +00:00
|
|
|
<?php
|
2019-02-16 17:51:24 +00:00
|
|
|
|
2019-01-10 15:20:05 +00:00
|
|
|
namespace ActivityPub\Test\Controllers;
|
|
|
|
|
2019-01-20 20:02:37 +00:00
|
|
|
use ActivityPub\Auth\AuthService;
|
2019-01-23 14:28:15 +00:00
|
|
|
use ActivityPub\Controllers\GetController;
|
2019-03-23 16:57:51 +00:00
|
|
|
use ActivityPub\Objects\BlockService;
|
2019-01-20 20:02:37 +00:00
|
|
|
use ActivityPub\Objects\CollectionsService;
|
2019-02-16 17:51:24 +00:00
|
|
|
use ActivityPub\Objects\ContextProvider;
|
2019-01-10 15:20:05 +00:00
|
|
|
use ActivityPub\Objects\ObjectsService;
|
2019-02-16 17:51:24 +00:00
|
|
|
use ActivityPub\Test\TestConfig\APTestCase;
|
2019-01-23 16:37:05 +00:00
|
|
|
use ActivityPub\Test\TestUtils\TestActivityPubObject;
|
2019-02-07 03:48:00 +00:00
|
|
|
use ActivityPub\Utils\SimpleDateTimeProvider;
|
2019-02-16 17:57:26 +00:00
|
|
|
use Doctrine\ORM\EntityManager;
|
2019-03-23 17:35:30 +00:00
|
|
|
use Exception;
|
2019-01-29 02:24:51 +00:00
|
|
|
use GuzzleHttp\Client;
|
2019-01-10 15:20:05 +00:00
|
|
|
use Symfony\Component\HttpFoundation\Request;
|
|
|
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
2019-02-16 17:51:24 +00:00
|
|
|
use Symfony\Component\HttpKernel\Exception\UnauthorizedHttpException;
|
2019-01-10 15:20:05 +00:00
|
|
|
|
2019-02-07 03:48:00 +00:00
|
|
|
class GetControllerTest extends APTestCase
|
2019-01-10 15:20:05 +00:00
|
|
|
{
|
2019-02-16 17:51:24 +00:00
|
|
|
/**
|
|
|
|
* @var GetController
|
|
|
|
*/
|
|
|
|
private $getController;
|
2019-03-23 17:35:30 +00:00
|
|
|
|
2019-02-16 17:51:24 +00:00
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $objects;
|
|
|
|
|
|
|
|
public function setUp()
|
|
|
|
{
|
|
|
|
$this->objects = self::getObjects();
|
|
|
|
$objectsService = $this->getMock( ObjectsService::class );
|
|
|
|
$objectsService->method( 'dereference' )->will(
|
|
|
|
$this->returnCallback( function ( $uri ) {
|
|
|
|
if ( array_key_exists( $uri, $this->objects ) ) {
|
2019-03-23 17:35:30 +00:00
|
|
|
return $this->objects[$uri];
|
2019-02-16 17:51:24 +00:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
} )
|
|
|
|
);
|
2019-03-30 15:27:03 +00:00
|
|
|
$objectsService->method( 'update' )->will(
|
|
|
|
$this->returnCallback( function ( $uri ) {
|
|
|
|
if ( array_key_exists( $uri, $this->objects ) ) {
|
|
|
|
return $this->objects[$uri];
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
} )
|
|
|
|
);
|
2019-02-16 17:51:24 +00:00
|
|
|
$authService = new AuthService();
|
|
|
|
$contextProvider = new ContextProvider();
|
|
|
|
$httpClient = $this->getMock( Client::class );
|
|
|
|
$collectionsService = new CollectionsService(
|
2019-02-16 17:57:26 +00:00
|
|
|
4,
|
|
|
|
$authService,
|
|
|
|
$contextProvider,
|
|
|
|
$httpClient,
|
|
|
|
new SimpleDateTimeProvider(),
|
|
|
|
$this->getMock( EntityManager::class ),
|
|
|
|
$objectsService
|
2019-02-16 17:51:24 +00:00
|
|
|
);
|
2019-03-23 16:57:51 +00:00
|
|
|
$blockService = $this->getMock( BlockService::class );
|
2019-03-23 17:35:30 +00:00
|
|
|
$blockService->method( 'getBlockedActorIds' )->will(
|
|
|
|
$this->returnValue( array( 'https://elsewhere.com/actors/blocked' ) )
|
|
|
|
);
|
2019-02-16 17:51:24 +00:00
|
|
|
$this->getController = new GetController(
|
2019-03-23 16:57:51 +00:00
|
|
|
$objectsService, $collectionsService, $authService, $blockService
|
2019-02-16 17:51:24 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-02-07 03:48:00 +00:00
|
|
|
private static function getObjects()
|
|
|
|
{
|
2019-03-23 17:35:30 +00:00
|
|
|
$actorObject = TestActivityPubObject::fromArray( array(
|
|
|
|
'id' => 'https://example.com/actors/1',
|
|
|
|
'inbox' => array(
|
|
|
|
'id' => 'https://example.com/actors/1/inbox',
|
|
|
|
'type' => 'OrderedCollection',
|
|
|
|
'orderedItems' => array(
|
|
|
|
array(
|
|
|
|
'id' => 'https://elsewhere.com/objects/1',
|
|
|
|
'actor' => 'https://elsewhere.com/actors/blocked',
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'id' => 'https://elsewhere.com/objects/2',
|
|
|
|
'actor' => 'https://elsewhere.com/actors/notblocked',
|
|
|
|
)
|
|
|
|
),
|
|
|
|
),
|
|
|
|
) );
|
|
|
|
$inboxObject = $actorObject['inbox'];
|
2019-02-07 03:48:00 +00:00
|
|
|
return array(
|
2019-03-23 17:35:30 +00:00
|
|
|
'https://example.com/objects/1' => TestActivityPubObject::fromArray( array(
|
2019-02-07 03:48:00 +00:00
|
|
|
'id' => 'https://example.com/objects/1',
|
|
|
|
'object' => array(
|
|
|
|
'id' => 'https://example.com/objects/2',
|
|
|
|
'type' => 'Note',
|
|
|
|
),
|
|
|
|
'audience' => array( 'https://www.w3.org/ns/activitystreams#Public' ),
|
|
|
|
'type' => 'Create',
|
2019-03-23 17:35:30 +00:00
|
|
|
) ),
|
|
|
|
'https://example.com/objects/2' => TestActivityPubObject::fromArray( array(
|
2019-01-10 15:20:05 +00:00
|
|
|
'id' => 'https://example.com/objects/2',
|
2019-02-07 03:48:00 +00:00
|
|
|
'object' => array(
|
|
|
|
'id' => 'https://example.com/objects/3',
|
|
|
|
'type' => 'Note',
|
|
|
|
),
|
|
|
|
'to' => array( 'https://example.com/actor/1' ),
|
|
|
|
'type' => 'Create',
|
|
|
|
'actor' => array(
|
|
|
|
'id' => 'https://example.com/actor/2',
|
|
|
|
),
|
2019-03-23 17:35:30 +00:00
|
|
|
) ),
|
|
|
|
'https://example.com/objects/3' => TestActivityPubObject::fromArray( array(
|
2019-01-19 22:22:58 +00:00
|
|
|
'id' => 'https://example.com/objects/3',
|
2019-02-07 03:48:00 +00:00
|
|
|
'object' => array(
|
|
|
|
'id' => 'https://example.com/objects/2',
|
|
|
|
'type' => 'Note',
|
|
|
|
),
|
|
|
|
'type' => 'Like',
|
|
|
|
'actor' => array(
|
|
|
|
'id' => 'https://example.com/actor/2',
|
|
|
|
),
|
2019-03-23 17:35:30 +00:00
|
|
|
) ),
|
|
|
|
'https://example.com/objects/4' => TestActivityPubObject::fromArray( array(
|
2019-02-07 03:48:00 +00:00
|
|
|
'id' => 'https://example.com/objects/4',
|
|
|
|
'type' => 'Tombstone',
|
2019-03-23 17:35:30 +00:00
|
|
|
) ),
|
|
|
|
'https://example.com/actors/1' => $actorObject,
|
|
|
|
'https://example.com/actors/1/inbox' => $inboxObject,
|
2019-02-07 03:48:00 +00:00
|
|
|
);
|
|
|
|
}
|
2019-01-10 15:20:05 +00:00
|
|
|
|
2019-03-23 17:35:30 +00:00
|
|
|
private function getObjectArray( $id )
|
|
|
|
{
|
|
|
|
$obj = $this->objects[$id];
|
|
|
|
return $obj->asArray();
|
|
|
|
}
|
|
|
|
|
2019-01-10 15:20:05 +00:00
|
|
|
public function testItRendersPersistedObject()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/1' );
|
2019-01-23 14:28:15 +00:00
|
|
|
$response = $this->getController->handle( $request );
|
2019-01-10 15:20:05 +00:00
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
2019-03-23 17:35:30 +00:00
|
|
|
json_encode( $this->getObjectArray('https://example.com/objects/1' ) ),
|
2019-01-10 15:20:05 +00:00
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testItThrowsNotFound()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/notreal' );
|
2019-02-07 03:48:00 +00:00
|
|
|
$this->setExpectedException( NotFoundHttpException::class );
|
2019-01-23 14:28:15 +00:00
|
|
|
$this->getController->handle( $request );
|
2019-01-10 15:20:05 +00:00
|
|
|
}
|
2019-01-18 22:13:46 +00:00
|
|
|
|
|
|
|
public function testItDeniesAccess()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/2' );
|
2019-02-07 03:48:00 +00:00
|
|
|
$this->setExpectedException( UnauthorizedHttpException::class );
|
2019-01-23 14:28:15 +00:00
|
|
|
$this->getController->handle( $request );
|
2019-01-18 22:13:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testItAllowsAccessToAuthedActor()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/2' );
|
|
|
|
$request->attributes->set( 'actor', 'https://example.com/actor/1' );
|
2019-01-23 14:28:15 +00:00
|
|
|
$response = $this->getController->handle( $request );
|
2019-01-18 22:13:46 +00:00
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
2019-03-23 17:35:30 +00:00
|
|
|
json_encode( $this->getObjectArray( 'https://example.com/objects/2' ) ),
|
2019-01-18 22:13:46 +00:00
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testItAllowsAccessToAttributedActor()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/2' );
|
|
|
|
$request->attributes->set( 'actor', 'https://example.com/actor/2' );
|
2019-01-23 14:28:15 +00:00
|
|
|
$response = $this->getController->handle( $request );
|
2019-01-18 22:13:46 +00:00
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
2019-03-23 17:35:30 +00:00
|
|
|
json_encode( $this->getObjectArray( 'https://example.com/objects/2' ) ),
|
2019-01-18 22:13:46 +00:00
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
|
|
|
}
|
2019-01-19 22:22:58 +00:00
|
|
|
|
|
|
|
public function testItAllowsAccessToNoAudienceObject()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/3' );
|
2019-01-23 14:28:15 +00:00
|
|
|
$response = $this->getController->handle( $request );
|
2019-01-19 22:22:58 +00:00
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
2019-03-23 17:35:30 +00:00
|
|
|
json_encode( $this->getObjectArray( 'https://example.com/objects/3' ) ),
|
2019-01-19 22:22:58 +00:00
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
|
|
|
}
|
2019-01-20 20:02:37 +00:00
|
|
|
|
|
|
|
public function testItDisregardsQueryParams()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/1?foo=bar&baz=qux' );
|
2019-01-23 14:28:15 +00:00
|
|
|
$response = $this->getController->handle( $request );
|
2019-01-20 20:02:37 +00:00
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
2019-03-23 17:35:30 +00:00
|
|
|
json_encode( $this->getObjectArray( 'https://example.com/objects/1' ) ),
|
2019-01-20 20:02:37 +00:00
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
|
|
|
}
|
2019-01-31 21:16:38 +00:00
|
|
|
|
|
|
|
public function testItReturns410ForTombstones()
|
|
|
|
{
|
|
|
|
$request = Request::create( 'https://example.com/objects/4' );
|
|
|
|
$response = $this->getController->handle( $request );
|
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
2019-03-23 17:35:30 +00:00
|
|
|
json_encode( $this->getObjectArray( 'https://example.com/objects/4' ) ),
|
2019-01-31 21:16:38 +00:00
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
|
|
|
$this->assertEquals( 410, $response->getStatusCode() );
|
|
|
|
}
|
2019-03-23 16:57:51 +00:00
|
|
|
|
|
|
|
public function testItFiltersInboxForBlockedActors()
|
|
|
|
{
|
2019-03-23 17:35:30 +00:00
|
|
|
$request = Request::create( 'https://example.com/actors/1/inbox' );
|
|
|
|
$response = $this->getController->handle( $request );
|
|
|
|
$this->assertNotNull( $response );
|
|
|
|
$this->assertEquals(
|
|
|
|
json_encode(
|
|
|
|
array(
|
|
|
|
'id' => 'https://example.com/actors/1/inbox',
|
|
|
|
'type' => 'OrderedCollection',
|
|
|
|
'first' => array(
|
|
|
|
'@context' => array(
|
|
|
|
'https://www.w3.org/ns/activitystreams',
|
|
|
|
'https://w3id.org/security/v1',
|
|
|
|
),
|
2019-03-30 15:27:03 +00:00
|
|
|
'id' => 'https://example.com/actors/1/inbox?offset=0&sort=desc',
|
2019-03-23 17:35:30 +00:00
|
|
|
'type' => 'OrderedCollectionPage',
|
|
|
|
'orderedItems' => array(
|
|
|
|
array(
|
|
|
|
'id' => 'https://elsewhere.com/objects/2',
|
|
|
|
'actor' => 'https://elsewhere.com/actors/notblocked',
|
|
|
|
),
|
|
|
|
),
|
|
|
|
'partOf' => 'https://example.com/actors/1/inbox',
|
|
|
|
'startIndex' => 0,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
),
|
|
|
|
$response->getContent()
|
|
|
|
);
|
|
|
|
$this->assertEquals( 'application/json', $response->headers->get( 'Content-Type' ) );
|
2019-03-23 16:57:51 +00:00
|
|
|
}
|
2019-01-10 15:20:05 +00:00
|
|
|
}
|
2019-02-07 03:48:00 +00:00
|
|
|
|