2017-01-05 15:58:24 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Shaarli\Api\Controllers;
|
|
|
|
|
2017-10-22 13:43:19 +02:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2017-01-05 15:58:24 +01:00
|
|
|
use Shaarli\Config\ConfigManager;
|
2019-01-12 23:55:38 +01:00
|
|
|
use Shaarli\History;
|
2017-01-05 15:58:24 +01:00
|
|
|
use Slim\Container;
|
|
|
|
use Slim\Http\Environment;
|
|
|
|
use Slim\Http\Request;
|
|
|
|
use Slim\Http\Response;
|
2017-10-22 13:43:19 +02:00
|
|
|
use Slim\Router;
|
2017-01-05 15:58:24 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Class PostLinkTest
|
|
|
|
*
|
|
|
|
* Test POST Link REST API service.
|
|
|
|
*
|
|
|
|
* @package Shaarli\Api\Controllers
|
|
|
|
*/
|
2017-10-22 13:43:19 +02:00
|
|
|
class PostLinkTest extends TestCase
|
2017-01-05 15:58:24 +01:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var string datastore to test write operations
|
|
|
|
*/
|
|
|
|
protected static $testDatastore = 'sandbox/datastore.php';
|
|
|
|
|
2017-05-07 16:50:20 +02:00
|
|
|
/**
|
|
|
|
* @var string datastore to test write operations
|
|
|
|
*/
|
|
|
|
protected static $testHistory = 'sandbox/history.php';
|
|
|
|
|
2017-01-05 15:58:24 +01:00
|
|
|
/**
|
|
|
|
* @var ConfigManager instance
|
|
|
|
*/
|
|
|
|
protected $conf;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \ReferenceLinkDB instance.
|
|
|
|
*/
|
|
|
|
protected $refDB = null;
|
|
|
|
|
2017-05-07 16:50:20 +02:00
|
|
|
/**
|
2019-01-12 23:55:38 +01:00
|
|
|
* @var HistoryController instance.
|
2017-05-07 16:50:20 +02:00
|
|
|
*/
|
|
|
|
protected $history;
|
|
|
|
|
2017-01-05 15:58:24 +01:00
|
|
|
/**
|
|
|
|
* @var Container instance.
|
|
|
|
*/
|
|
|
|
protected $container;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var Links controller instance.
|
|
|
|
*/
|
|
|
|
protected $controller;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of JSON field per link.
|
|
|
|
*/
|
|
|
|
const NB_FIELDS_LINK = 9;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Before every test, instantiate a new Api with its config, plugins and links.
|
|
|
|
*/
|
|
|
|
public function setUp()
|
|
|
|
{
|
|
|
|
$this->conf = new ConfigManager('tests/utils/config/configJson.json.php');
|
|
|
|
$this->refDB = new \ReferenceLinkDB();
|
|
|
|
$this->refDB->write(self::$testDatastore);
|
|
|
|
|
2017-05-07 16:50:20 +02:00
|
|
|
$refHistory = new \ReferenceHistory();
|
|
|
|
$refHistory->write(self::$testHistory);
|
2019-01-12 23:55:38 +01:00
|
|
|
$this->history = new History(self::$testHistory);
|
2017-05-07 16:50:20 +02:00
|
|
|
|
2017-01-05 15:58:24 +01:00
|
|
|
$this->container = new Container();
|
|
|
|
$this->container['conf'] = $this->conf;
|
2018-12-03 01:10:39 +01:00
|
|
|
$this->container['db'] = new \Shaarli\Bookmark\LinkDB(self::$testDatastore, true, false);
|
2019-01-12 23:55:38 +01:00
|
|
|
$this->container['history'] = new History(self::$testHistory);
|
2017-01-05 15:58:24 +01:00
|
|
|
|
|
|
|
$this->controller = new Links($this->container);
|
|
|
|
|
2017-10-22 13:43:19 +02:00
|
|
|
$mock = $this->createMock(Router::class);
|
2017-01-05 15:58:24 +01:00
|
|
|
$mock->expects($this->any())
|
|
|
|
->method('relativePathFor')
|
|
|
|
->willReturn('api/v1/links/1');
|
|
|
|
|
|
|
|
// affect @property-read... seems to work
|
|
|
|
$this->controller->getCi()->router = $mock;
|
|
|
|
|
|
|
|
// Used by index_url().
|
|
|
|
$this->controller->getCi()['environment'] = [
|
|
|
|
'SERVER_NAME' => 'domain.tld',
|
|
|
|
'SERVER_PORT' => 80,
|
|
|
|
'SCRIPT_NAME' => '/',
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* After every test, remove the test datastore.
|
|
|
|
*/
|
|
|
|
public function tearDown()
|
|
|
|
{
|
|
|
|
@unlink(self::$testDatastore);
|
2017-05-07 16:50:20 +02:00
|
|
|
@unlink(self::$testHistory);
|
2017-01-05 15:58:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test link creation without any field: creates a blank note.
|
|
|
|
*/
|
|
|
|
public function testPostLinkMinimal()
|
|
|
|
{
|
|
|
|
$env = Environment::mock([
|
|
|
|
'REQUEST_METHOD' => 'POST',
|
|
|
|
]);
|
|
|
|
|
|
|
|
$request = Request::createFromEnvironment($env);
|
|
|
|
|
|
|
|
$response = $this->controller->postLink($request, new Response());
|
|
|
|
$this->assertEquals(201, $response->getStatusCode());
|
|
|
|
$this->assertEquals('api/v1/links/1', $response->getHeader('Location')[0]);
|
|
|
|
$data = json_decode((string) $response->getBody(), true);
|
|
|
|
$this->assertEquals(self::NB_FIELDS_LINK, count($data));
|
|
|
|
$this->assertEquals(43, $data['id']);
|
2019-01-12 23:19:15 +01:00
|
|
|
$this->assertRegExp('/[\w_-]{6}/', $data['shorturl']);
|
2017-01-05 15:58:24 +01:00
|
|
|
$this->assertEquals('http://domain.tld/?' . $data['shorturl'], $data['url']);
|
|
|
|
$this->assertEquals('?' . $data['shorturl'], $data['title']);
|
|
|
|
$this->assertEquals('', $data['description']);
|
|
|
|
$this->assertEquals([], $data['tags']);
|
|
|
|
$this->assertEquals(false, $data['private']);
|
2018-10-13 01:40:04 +02:00
|
|
|
$this->assertTrue(
|
|
|
|
new \DateTime('5 seconds ago') < \DateTime::createFromFormat(\DateTime::ATOM, $data['created'])
|
|
|
|
);
|
2017-01-05 15:58:24 +01:00
|
|
|
$this->assertEquals('', $data['updated']);
|
2017-05-07 16:50:20 +02:00
|
|
|
|
|
|
|
$historyEntry = $this->history->getHistory()[0];
|
2019-01-12 23:55:38 +01:00
|
|
|
$this->assertEquals(History::CREATED, $historyEntry['event']);
|
2017-05-07 16:50:20 +02:00
|
|
|
$this->assertTrue(
|
|
|
|
(new \DateTime())->add(\DateInterval::createFromDateString('-5 seconds')) < $historyEntry['datetime']
|
|
|
|
);
|
|
|
|
$this->assertEquals(43, $historyEntry['id']);
|
2017-01-05 15:58:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test link creation with all available fields.
|
|
|
|
*/
|
|
|
|
public function testPostLinkFull()
|
|
|
|
{
|
|
|
|
$link = [
|
|
|
|
'url' => 'website.tld/test?foo=bar',
|
|
|
|
'title' => 'new entry',
|
|
|
|
'description' => 'shaare description',
|
|
|
|
'tags' => ['one', 'two'],
|
|
|
|
'private' => true,
|
|
|
|
];
|
|
|
|
$env = Environment::mock([
|
|
|
|
'REQUEST_METHOD' => 'POST',
|
|
|
|
'CONTENT_TYPE' => 'application/json'
|
|
|
|
]);
|
|
|
|
|
|
|
|
$request = Request::createFromEnvironment($env);
|
|
|
|
$request = $request->withParsedBody($link);
|
|
|
|
$response = $this->controller->postLink($request, new Response());
|
|
|
|
|
|
|
|
$this->assertEquals(201, $response->getStatusCode());
|
|
|
|
$this->assertEquals('api/v1/links/1', $response->getHeader('Location')[0]);
|
|
|
|
$data = json_decode((string) $response->getBody(), true);
|
|
|
|
$this->assertEquals(self::NB_FIELDS_LINK, count($data));
|
|
|
|
$this->assertEquals(43, $data['id']);
|
2019-01-12 23:19:15 +01:00
|
|
|
$this->assertRegExp('/[\w_-]{6}/', $data['shorturl']);
|
2017-01-05 15:58:24 +01:00
|
|
|
$this->assertEquals('http://' . $link['url'], $data['url']);
|
|
|
|
$this->assertEquals($link['title'], $data['title']);
|
|
|
|
$this->assertEquals($link['description'], $data['description']);
|
|
|
|
$this->assertEquals($link['tags'], $data['tags']);
|
|
|
|
$this->assertEquals(true, $data['private']);
|
2018-10-13 01:40:04 +02:00
|
|
|
$this->assertTrue(
|
|
|
|
new \DateTime('2 seconds ago') < \DateTime::createFromFormat(\DateTime::ATOM, $data['created'])
|
|
|
|
);
|
2017-01-05 15:58:24 +01:00
|
|
|
$this->assertEquals('', $data['updated']);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test link creation with an existing link (duplicate URL). Should return a 409 HTTP error and the existing link.
|
|
|
|
*/
|
|
|
|
public function testPostLinkDuplicate()
|
|
|
|
{
|
|
|
|
$link = [
|
|
|
|
'url' => 'mediagoblin.org/',
|
|
|
|
'title' => 'new entry',
|
|
|
|
'description' => 'shaare description',
|
|
|
|
'tags' => ['one', 'two'],
|
|
|
|
'private' => true,
|
|
|
|
];
|
|
|
|
$env = Environment::mock([
|
|
|
|
'REQUEST_METHOD' => 'POST',
|
|
|
|
'CONTENT_TYPE' => 'application/json'
|
|
|
|
]);
|
|
|
|
|
|
|
|
$request = Request::createFromEnvironment($env);
|
|
|
|
$request = $request->withParsedBody($link);
|
|
|
|
$response = $this->controller->postLink($request, new Response());
|
|
|
|
|
|
|
|
$this->assertEquals(409, $response->getStatusCode());
|
|
|
|
$data = json_decode((string) $response->getBody(), true);
|
|
|
|
$this->assertEquals(self::NB_FIELDS_LINK, count($data));
|
|
|
|
$this->assertEquals(7, $data['id']);
|
|
|
|
$this->assertEquals('IuWvgA', $data['shorturl']);
|
|
|
|
$this->assertEquals('http://mediagoblin.org/', $data['url']);
|
|
|
|
$this->assertEquals('MediaGoblin', $data['title']);
|
|
|
|
$this->assertEquals('A free software media publishing platform #hashtagOther', $data['description']);
|
|
|
|
$this->assertEquals(['gnu', 'media', 'web', '.hidden', 'hashtag'], $data['tags']);
|
|
|
|
$this->assertEquals(false, $data['private']);
|
|
|
|
$this->assertEquals(
|
2018-12-03 01:10:39 +01:00
|
|
|
\DateTime::createFromFormat(\Shaarli\Bookmark\LinkDB::LINK_DATE_FORMAT, '20130614_184135'),
|
2017-01-05 15:58:24 +01:00
|
|
|
\DateTime::createFromFormat(\DateTime::ATOM, $data['created'])
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
2018-12-03 01:10:39 +01:00
|
|
|
\DateTime::createFromFormat(\Shaarli\Bookmark\LinkDB::LINK_DATE_FORMAT, '20130615_184230'),
|
2017-01-05 15:58:24 +01:00
|
|
|
\DateTime::createFromFormat(\DateTime::ATOM, $data['updated'])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|