2015-09-10 19:57:25 +00:00
< ? php
2016-12-03 14:44:34 +00:00
namespace Tests\Wallabag\CoreBundle\Helper ;
2015-09-10 19:57:25 +00:00
2017-07-01 07:52:38 +00:00
use Graby\Graby ;
2017-05-30 15:48:24 +00:00
use Monolog\Handler\TestHandler ;
2017-07-01 07:52:38 +00:00
use Monolog\Logger ;
2017-12-16 21:17:42 +00:00
use PHPUnit\Framework\TestCase ;
2017-07-01 07:52:38 +00:00
use Psr\Log\NullLogger ;
use Symfony\Component\Validator\ConstraintViolation ;
use Symfony\Component\Validator\ConstraintViolationList ;
use Symfony\Component\Validator\Validator\RecursiveValidator ;
2016-02-19 13:22:20 +00:00
use Wallabag\CoreBundle\Entity\Entry ;
2017-07-01 07:52:38 +00:00
use Wallabag\CoreBundle\Helper\ContentProxy ;
2017-05-27 20:08:14 +00:00
use Wallabag\CoreBundle\Helper\RuleBasedTagger ;
2017-07-01 07:52:38 +00:00
use Wallabag\UserBundle\Entity\User ;
2015-09-10 19:57:25 +00:00
2017-12-16 21:17:42 +00:00
class ContentProxyTest extends TestCase
2015-09-10 19:57:25 +00:00
{
2017-05-24 10:57:46 +00:00
private $fetchingErrorMessage = 'wallabag can\'t retrieve contents for this article. Please <a href="http://doc.wallabag.org/en/user/errors_during_fetching.html#how-can-i-help-to-fix-that">troubleshoot this issue</a>.' ;
2016-12-03 14:44:34 +00:00
2016-03-27 18:35:56 +00:00
public function testWithBadUrl ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
2016-04-12 09:36:01 +00:00
-> setMethods ([ 'fetchContent' ])
2016-03-27 18:35:56 +00:00
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
2016-04-12 09:36:01 +00:00
-> willReturn ([
2016-03-27 18:35:56 +00:00
'html' => false ,
'title' => '' ,
'url' => '' ,
'content_type' => '' ,
'language' => '' ,
2016-04-12 09:36:01 +00:00
]);
2016-03-27 18:35:56 +00:00
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://user@:80' );
2016-03-27 18:35:56 +00:00
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://user@:80' , $entry -> getUrl ());
2016-03-27 18:35:56 +00:00
$this -> assertEmpty ( $entry -> getTitle ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( $this -> fetchingErrorMessage , $entry -> getContent ());
2016-03-27 18:35:56 +00:00
$this -> assertEmpty ( $entry -> getPreviewPicture ());
$this -> assertEmpty ( $entry -> getMimetype ());
$this -> assertEmpty ( $entry -> getLanguage ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 0.0 , $entry -> getReadingTime ());
2017-11-21 09:37:36 +00:00
$this -> assertNull ( $entry -> getDomainName ());
2016-03-27 18:35:56 +00:00
}
2015-09-10 19:57:25 +00:00
public function testWithEmptyContent ()
{
2015-10-11 20:27:47 +00:00
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2015-09-10 19:57:25 +00:00
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
2016-04-12 09:36:01 +00:00
-> setMethods ([ 'fetchContent' ])
2015-09-10 19:57:25 +00:00
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
2016-04-12 09:36:01 +00:00
-> willReturn ([
2015-09-20 20:37:27 +00:00
'html' => false ,
'title' => '' ,
'url' => '' ,
'content_type' => '' ,
'language' => '' ,
2016-04-12 09:36:01 +00:00
]);
2015-09-10 19:57:25 +00:00
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
2015-09-10 19:57:25 +00:00
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://0.0.0.0' , $entry -> getUrl ());
2015-09-10 19:57:25 +00:00
$this -> assertEmpty ( $entry -> getTitle ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( $this -> fetchingErrorMessage , $entry -> getContent ());
2015-09-10 19:57:25 +00:00
$this -> assertEmpty ( $entry -> getPreviewPicture ());
$this -> assertEmpty ( $entry -> getMimetype ());
2015-09-20 20:37:27 +00:00
$this -> assertEmpty ( $entry -> getLanguage ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 0.0 , $entry -> getReadingTime ());
$this -> assertSame ( '0.0.0.0' , $entry -> getDomainName ());
2015-09-10 19:57:25 +00:00
}
public function testWithEmptyContentButOG ()
{
2015-10-11 20:27:47 +00:00
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2015-09-10 19:57:25 +00:00
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
2016-04-12 09:36:01 +00:00
-> setMethods ([ 'fetchContent' ])
2015-09-10 19:57:25 +00:00
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
2016-04-12 09:36:01 +00:00
-> willReturn ([
2015-09-20 20:37:27 +00:00
'html' => false ,
'title' => '' ,
'url' => '' ,
'content_type' => '' ,
'language' => '' ,
2016-11-18 14:09:21 +00:00
'status' => '' ,
2016-04-12 09:36:01 +00:00
'open_graph' => [
2015-09-20 20:37:27 +00:00
'og_title' => 'my title' ,
'og_description' => 'desc' ,
2016-04-12 09:36:01 +00:00
],
]);
2015-09-10 19:57:25 +00:00
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://domain.io' );
2015-09-10 19:57:25 +00:00
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://domain.io' , $entry -> getUrl ());
$this -> assertSame ( 'my title' , $entry -> getTitle ());
$this -> assertSame ( $this -> fetchingErrorMessage . '<p><i>But we found a short description: </i></p>desc' , $entry -> getContent ());
2015-09-10 19:57:25 +00:00
$this -> assertEmpty ( $entry -> getPreviewPicture ());
2015-09-20 20:37:27 +00:00
$this -> assertEmpty ( $entry -> getLanguage ());
2016-11-18 14:09:21 +00:00
$this -> assertEmpty ( $entry -> getHttpStatus ());
2015-09-10 19:57:25 +00:00
$this -> assertEmpty ( $entry -> getMimetype ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 0.0 , $entry -> getReadingTime ());
$this -> assertSame ( 'domain.io' , $entry -> getDomainName ());
2015-09-10 19:57:25 +00:00
}
public function testWithContent ()
{
2015-10-11 20:27:47 +00:00
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2015-09-10 19:57:25 +00:00
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
2016-04-12 09:36:01 +00:00
-> setMethods ([ 'fetchContent' ])
2015-09-10 19:57:25 +00:00
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
2016-04-12 09:36:01 +00:00
-> willReturn ([
2015-09-28 17:35:33 +00:00
'html' => str_repeat ( 'this is my content' , 325 ),
2015-09-10 19:57:25 +00:00
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
2015-09-20 20:37:27 +00:00
'language' => 'fr' ,
2016-11-18 14:09:21 +00:00
'status' => '200' ,
2016-04-12 09:36:01 +00:00
'open_graph' => [
2015-09-10 19:57:25 +00:00
'og_title' => 'my OG title' ,
'og_description' => 'OG desc' ,
2015-09-10 20:00:53 +00:00
'og_image' => 'http://3.3.3.3/cover.jpg' ,
2016-04-12 09:36:01 +00:00
],
]);
2015-09-10 19:57:25 +00:00
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
2015-09-10 19:57:25 +00:00
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2015-09-28 17:35:33 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://3.3.3.3/cover.jpg' , $entry -> getPreviewPicture ());
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( '200' , $entry -> getHttpStatus ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2015-09-10 19:57:25 +00:00
}
2015-10-11 20:27:47 +00:00
2017-01-10 16:42:34 +00:00
public function testWithContentAndNoOgImage ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
'status' => '200' ,
'open_graph' => [
'og_title' => 'my OG title' ,
'og_description' => 'OG desc' ,
2017-06-08 19:51:46 +00:00
'og_image' => null ,
2017-01-10 16:42:34 +00:00
],
]);
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2017-05-30 15:48:24 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
2017-01-10 16:42:34 +00:00
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-01-10 16:42:34 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-06-12 14:46:33 +00:00
$this -> assertNull ( $entry -> getPreviewPicture ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( '200' , $entry -> getHttpStatus ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2017-06-08 19:51:46 +00:00
}
public function testWithContentAndBadLanguage ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2017-12-18 09:14:00 +00:00
$validator = $this -> getValidator ( false );
2017-06-30 14:54:26 +00:00
$validator -> expects ( $this -> once ())
2017-06-08 19:51:46 +00:00
-> method ( 'validate' )
2017-06-30 14:54:26 +00:00
-> willReturn ( new ConstraintViolationList ([ new ConstraintViolation ( 'oops' , 'oops' , [], 'oops' , 'language' , 'dontexist' )]));
2017-06-08 19:51:46 +00:00
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'dontexist' ,
'status' => '200' ,
]);
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $validator , $this -> getLogger (), $this -> fetchingErrorMessage );
2017-06-08 19:51:46 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-06-08 19:51:46 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
2017-06-12 14:46:33 +00:00
$this -> assertNull ( $entry -> getLanguage ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( '200' , $entry -> getHttpStatus ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2017-06-08 19:51:46 +00:00
}
public function testWithContentAndBadOgImage ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2017-12-18 09:14:00 +00:00
$validator = $this -> getValidator ( false );
2017-06-08 19:51:46 +00:00
$validator -> expects ( $this -> exactly ( 2 ))
-> method ( 'validate' )
-> will ( $this -> onConsecutiveCalls (
new ConstraintViolationList (),
new ConstraintViolationList ([ new ConstraintViolation ( 'oops' , 'oops' , [], 'oops' , 'url' , 'https://' )])
));
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
'status' => '200' ,
'open_graph' => [
'og_title' => 'my OG title' ,
'og_description' => 'OG desc' ,
'og_image' => 'https://' ,
],
]);
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $validator , $this -> getLogger (), $this -> fetchingErrorMessage );
2017-06-08 19:51:46 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-06-08 19:51:46 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-06-12 14:46:33 +00:00
$this -> assertNull ( $entry -> getPreviewPicture ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( '200' , $entry -> getHttpStatus ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2017-01-10 16:42:34 +00:00
}
2016-03-27 18:35:56 +00:00
public function testWithForcedContent ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy (( new Graby ()), $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage , true );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry (
$entry ,
2017-05-16 21:11:20 +00:00
'http://0.0.0.0' ,
[
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
2017-05-24 14:44:03 +00:00
'date' => '1395635872' ,
'authors' => [ 'Jeremy' , 'Nico' , 'Thomas' ],
'all_headers' => [
'Cache-Control' => 'no-cache' ,
2017-05-29 08:14:01 +00:00
],
2017-05-16 21:11:20 +00:00
]
);
2016-03-27 18:35:56 +00:00
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2016-03-27 18:35:56 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
$this -> assertSame ( '24/03/2014' , $entry -> getPublishedAt () -> format ( 'd/m/Y' ));
2017-05-24 14:44:03 +00:00
$this -> assertContains ( 'Jeremy' , $entry -> getPublishedBy ());
$this -> assertContains ( 'Nico' , $entry -> getPublishedBy ());
$this -> assertContains ( 'Thomas' , $entry -> getPublishedBy ());
2017-11-21 09:37:36 +00:00
$this -> assertNotNull ( $entry -> getHeaders (), 'Headers are stored, so value is not null' );
2017-05-24 14:44:03 +00:00
$this -> assertContains ( 'no-cache' , $entry -> getHeaders ());
}
public function testWithForcedContentAndDatetime ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2017-05-30 15:48:24 +00:00
$logHandler = new TestHandler ();
2017-06-01 09:31:45 +00:00
$logger = new Logger ( 'test' , [ $logHandler ]);
2017-05-30 15:48:24 +00:00
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy (( new Graby ()), $tagger , $this -> getValidator (), $logger , $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
2017-06-01 09:31:45 +00:00
$proxy -> updateEntry (
2016-12-07 03:17:44 +00:00
$entry ,
2017-06-01 09:31:45 +00:00
'http://1.1.1.1' ,
2017-05-24 14:44:03 +00:00
[
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
'date' => '2016-09-08T11:55:58+0200' ,
]
);
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-05-24 14:44:03 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
$this -> assertSame ( '08/09/2016' , $entry -> getPublishedAt () -> format ( 'd/m/Y' ));
2017-05-24 14:44:03 +00:00
}
public function testWithForcedContentAndBadDate ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$logger = new Logger ( 'foo' );
$handler = new TestHandler ();
$logger -> pushHandler ( $handler );
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy (( new Graby ()), $tagger , $this -> getValidator (), $logger , $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry (
$entry ,
2017-06-01 09:31:45 +00:00
'http://1.1.1.1' ,
2017-05-24 14:44:03 +00:00
[
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
'date' => '01 02 2012' ,
]
);
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-05-24 14:44:03 +00:00
$this -> assertContains ( 'this is my content' , $entry -> getContent ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( 4.0 , $entry -> getReadingTime ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2017-05-24 14:44:03 +00:00
$this -> assertNull ( $entry -> getPublishedAt ());
$records = $handler -> getRecords ();
$this -> assertCount ( 1 , $records );
$this -> assertContains ( 'Error while defining date' , $records [ 0 ][ 'message' ]);
2016-03-27 18:35:56 +00:00
}
public function testTaggerThrowException ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' )
-> will ( $this -> throwException ( new \Exception ()));
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy (( new Graby ()), $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2016-12-07 03:17:44 +00:00
$entry = new Entry ( new User ());
2017-06-01 09:31:45 +00:00
$proxy -> updateEntry (
$entry ,
'http://1.1.1.1' ,
[
'html' => str_repeat ( 'this is my content' , 325 ),
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
]
2016-12-07 20:16:49 +00:00
);
2016-03-27 18:35:56 +00:00
$this -> assertCount ( 0 , $entry -> getTags ());
}
2017-05-12 05:53:21 +00:00
public function dataForCrazyHtml ()
{
return [
'script and comment' => [
'<strong>Script inside:</strong> <!--[if gte IE 4]><script>alert(\'lol\');</script><![endif]--><br />' ,
2017-05-29 08:14:01 +00:00
'lol' ,
2017-05-12 05:53:21 +00:00
],
'script' => [
'<strong>Script inside:</strong><script>alert(\'lol\');</script>' ,
2017-05-29 08:14:01 +00:00
'script' ,
2017-05-12 05:53:21 +00:00
],
];
}
/**
* @ dataProvider dataForCrazyHtml
*/
public function testWithCrazyHtmlContent ( $html , $escapedString )
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy (( new Graby ()), $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2017-06-01 09:31:45 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry (
$entry ,
2017-05-12 05:53:21 +00:00
'http://1.1.1.1' ,
[
'html' => $html ,
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1' ,
'content_type' => 'text/html' ,
'language' => 'fr' ,
'status' => '200' ,
'open_graph' => [
'og_title' => 'my OG title' ,
'og_description' => 'OG desc' ,
'og_image' => 'http://3.3.3.3/cover.jpg' ,
],
]
);
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://1.1.1.1' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-05-12 05:53:21 +00:00
$this -> assertNotContains ( $escapedString , $entry -> getContent ());
2017-07-01 07:52:38 +00:00
$this -> assertSame ( 'http://3.3.3.3/cover.jpg' , $entry -> getPreviewPicture ());
$this -> assertSame ( 'text/html' , $entry -> getMimetype ());
$this -> assertSame ( 'fr' , $entry -> getLanguage ());
$this -> assertSame ( '200' , $entry -> getHttpStatus ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2017-05-12 05:53:21 +00:00
}
2017-06-30 15:04:40 +00:00
public function testWithImageAsContent ()
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => '<p><img src="http://1.1.1.1/image.jpg" /></p>' ,
'title' => 'this is my title' ,
'url' => 'http://1.1.1.1/image.jpg' ,
'content_type' => 'image/jpeg' ,
'status' => '200' ,
'open_graph' => [],
]);
2017-11-21 09:37:36 +00:00
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
2017-06-30 15:04:40 +00:00
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
2017-07-03 11:56:39 +00:00
$this -> assertSame ( 'http://1.1.1.1/image.jpg' , $entry -> getUrl ());
$this -> assertSame ( 'this is my title' , $entry -> getTitle ());
2017-06-30 15:04:40 +00:00
$this -> assertContains ( 'http://1.1.1.1/image.jpg' , $entry -> getContent ());
$this -> assertSame ( 'http://1.1.1.1/image.jpg' , $entry -> getPreviewPicture ());
2017-07-03 11:56:39 +00:00
$this -> assertSame ( 'image/jpeg' , $entry -> getMimetype ());
$this -> assertSame ( '200' , $entry -> getHttpStatus ());
$this -> assertSame ( '1.1.1.1' , $entry -> getDomainName ());
2017-06-30 15:04:40 +00:00
}
2018-09-19 11:59:07 +00:00
public function testWebsiteWithValidUTF8Title_doNothing ()
{
// You can use https://www.online-toolz.com/tools/text-hex-convertor.php to convert UTF-8 text <=> hex
// See http://graphemica.com for more info about the characters
// '😻ℤ z' (U+1F63B or F09F98BB; U+2124 or E284A4; U+007A or 7A) in hexadecimal and UTF-8
$actualTitle = $this -> hexToStr ( 'F09F98BB' . 'E284A4' . '7A' );
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => false ,
'title' => $actualTitle ,
'url' => '' ,
'content_type' => 'text/html' ,
'language' => '' ,
]);
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
// '😻ℤ z' (U+1F63B or F09F98BB; U+2124 or E284A4; U+007A or 7A) in hexadecimal and UTF-8
$expectedTitle = 'F09F98BB' . 'E284A4' . '7A' ;
$this -> assertSame ( $expectedTitle , $this -> strToHex ( $entry -> getTitle ()));
}
public function testWebsiteWithInvalidUTF8Title_removeInvalidCharacter ()
{
// See http://graphemica.com for more info about the characters
// 'a€b' (61;80;62) in hexadecimal and WINDOWS-1252 - but 80 is a invalid UTF-8 character.
// The correct UTF-8 € character (U+20AC) is E282AC
$actualTitle = $this -> hexToStr ( '61' . '80' . '62' );
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => false ,
'title' => $actualTitle ,
'url' => '' ,
'content_type' => 'text/html' ,
'language' => '' ,
]);
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
// 'ab' (61;62) because all invalid UTF-8 character (like 80) are removed
$expectedTitle = '61' . '62' ;
$this -> assertSame ( $expectedTitle , $this -> strToHex ( $entry -> getTitle ()));
}
public function testPdfWithUTF16BETitle_convertToUTF8 ()
{
// See http://graphemica.com for more info about the characters
// '😻' (U+1F63B;D83DDE3B) in hexadecimal and as UTF16BE
$actualTitle = $this -> hexToStr ( 'D83DDE3B' );
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => false ,
'title' => $actualTitle ,
'url' => '' ,
'content_type' => 'application/pdf' ,
'language' => '' ,
]);
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
// '😻' (U+1F63B or F09F98BB) in hexadecimal and UTF-8
$expectedTitle = 'F09F98BB' ;
$this -> assertSame ( $expectedTitle , $this -> strToHex ( $entry -> getTitle ()));
}
public function testPdfWithUTF8Title_doNothing ()
{
// See http://graphemica.com for more info about the characters
// '😻' (U+1F63B;D83DDE3B) in hexadecimal and as UTF8
$actualTitle = $this -> hexToStr ( 'F09F98BB' );
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => false ,
'title' => $actualTitle ,
'url' => '' ,
'content_type' => 'application/pdf' ,
'language' => '' ,
]);
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
// '😻' (U+1F63B or F09F98BB) in hexadecimal and UTF-8
$expectedTitle = 'F09F98BB' ;
$this -> assertSame ( $expectedTitle , $this -> strToHex ( $entry -> getTitle ()));
}
public function testPdfWithWINDOWS1252Title_convertToUTF8 ()
{
// See http://graphemica.com for more info about the characters
// '€' (80) in hexadecimal and WINDOWS-1252
$actualTitle = $this -> hexToStr ( '80' );
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => false ,
'title' => $actualTitle ,
'url' => '' ,
'content_type' => 'application/pdf' ,
'language' => '' ,
]);
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
// '€' (U+20AC or E282AC) in hexadecimal and UTF-8
$expectedTitle = 'E282AC' ;
$this -> assertSame ( $expectedTitle , $this -> strToHex ( $entry -> getTitle ()));
}
public function testPdfWithInvalidCharacterInTitle_removeInvalidCharacter ()
{
// See http://graphemica.com for more info about the characters
// '😻ℤ <F09F98BB> z' (U+1F63B or F09F98BB; U+2124 or E284A4; invalid character 81; U+007A or 7A) in hexadecimal and UTF-8
// 0x81 is not a valid character for UTF16, UTF8 and WINDOWS-1252
$actualTitle = $this -> hexToStr ( 'F09F98BB' . 'E284A4' . '81' . '7A' );
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$graby = $this -> getMockBuilder ( 'Graby\Graby' )
-> setMethods ([ 'fetchContent' ])
-> disableOriginalConstructor ()
-> getMock ();
$graby -> expects ( $this -> any ())
-> method ( 'fetchContent' )
-> willReturn ([
'html' => false ,
'title' => $actualTitle ,
'url' => '' ,
'content_type' => 'application/pdf' ,
'language' => '' ,
]);
$proxy = new ContentProxy ( $graby , $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage );
$entry = new Entry ( new User ());
$proxy -> updateEntry ( $entry , 'http://0.0.0.0' );
// '😻ℤ z' (U+1F63B or F09F98BB; U+2124 or E284A4; U+007A or 7A) in hexadecimal and UTF-8
// the 0x81 (represented by <20> ) is invalid for UTF16, UTF8 and WINDOWS-1252 and is removed
$expectedTitle = 'F09F98BB' . 'E284A4' . '7A' ;
$this -> assertSame ( $expectedTitle , $this -> strToHex ( $entry -> getTitle ()));
}
2018-09-06 20:26:20 +00:00
/**
* Data provider for testWithChangedUrl .
*
* Arrays contain the following values :
* $entry_url
* $origin_url
* $content_url
* $expected_entry_url
* $expected_origin_url
* $expected_domain
*/
public function dataForChangedUrl ()
{
return [
'normal' => [
'http://0.0.0.0' ,
null ,
'http://1.1.1.1' ,
'http://1.1.1.1' ,
'http://0.0.0.0' ,
'1.1.1.1' ,
],
'origin already set' => [
'http://0.0.0.0' ,
'http://hello' ,
'http://1.1.1.1' ,
'http://1.1.1.1' ,
'http://hello' ,
'1.1.1.1' ,
],
'trailing slash' => [
'https://example.com/hello-world' ,
null ,
'https://example.com/hello-world/' ,
'https://example.com/hello-world/' ,
null ,
'example.com' ,
],
'query string in fetched content' => [
'https://example.org/hello' ,
null ,
'https://example.org/hello?world=1' ,
2018-10-24 20:27:27 +00:00
'https://example.org/hello?world=1' ,
2018-09-06 20:26:20 +00:00
'https://example.org/hello' ,
'example.org' ,
],
'fragment in fetched content' => [
'https://example.org/hello' ,
null ,
'https://example.org/hello#world' ,
'https://example.org/hello' ,
null ,
'example.org' ,
],
2018-10-22 21:08:58 +00:00
'fragment and query string in fetched content' => [
'https://example.org/hello' ,
null ,
'https://example.org/hello?foo#world' ,
2018-10-24 20:27:27 +00:00
'https://example.org/hello?foo#world' ,
2018-10-22 21:08:58 +00:00
'https://example.org/hello' ,
'example.org' ,
2018-10-22 21:39:31 +00:00
],
'different path and query string in fetch content' => [
'https://example.org/hello' ,
null ,
'https://example.org/world?foo' ,
'https://example.org/world?foo' ,
'https://example.org/hello' ,
'example.org' ,
],
'feedproxy ignore list test' => [
'http://feedproxy.google.com/~r/Wallabag/~3/helloworld' ,
null ,
'https://example.org/hello-wallabag' ,
'https://example.org/hello-wallabag' ,
null ,
'example.org' ,
],
'feedproxy ignore list test with origin url already set' => [
'http://feedproxy.google.com/~r/Wallabag/~3/helloworld' ,
'https://example.org/this-is-source' ,
'https://example.org/hello-wallabag' ,
'https://example.org/hello-wallabag' ,
'https://example.org/this-is-source' ,
'example.org' ,
],
'lemonde ignore pattern test' => [
'http://www.lemonde.fr/tiny/url' ,
null ,
'http://example.com/hello-world' ,
'http://example.com/hello-world' ,
null ,
'example.com' ,
],
2018-09-06 20:26:20 +00:00
];
}
/**
* @ dataProvider dataForChangedUrl
*/
public function testWithChangedUrl ( $entry_url , $origin_url , $content_url , $expected_entry_url , $expected_origin_url , $expected_domain )
{
$tagger = $this -> getTaggerMock ();
$tagger -> expects ( $this -> once ())
-> method ( 'tag' );
$proxy = new ContentProxy (( new Graby ()), $tagger , $this -> getValidator (), $this -> getLogger (), $this -> fetchingErrorMessage , true );
$entry = new Entry ( new User ());
$entry -> setOriginUrl ( $origin_url );
$proxy -> updateEntry (
$entry ,
$entry_url ,
[
'html' => false ,
'title' => '' ,
'url' => $content_url ,
'content_type' => '' ,
'language' => '' ,
],
true
);
$this -> assertSame ( $expected_entry_url , $entry -> getUrl ());
$this -> assertSame ( $expected_domain , $entry -> getDomainName ());
$this -> assertSame ( $expected_origin_url , $entry -> getOriginUrl ());
}
2018-09-19 11:59:07 +00:00
/**
2018-09-23 21:42:05 +00:00
* https :// stackoverflow . com / a / 18506801.
*
2018-09-19 11:59:07 +00:00
* @ param $string
2018-09-23 21:42:05 +00:00
*
2018-09-19 11:59:07 +00:00
* @ return string
*/
2018-09-23 21:42:05 +00:00
private function strToHex ( $string )
{
2018-09-19 11:59:07 +00:00
$hex = '' ;
2018-09-23 21:42:05 +00:00
for ( $i = 0 ; $i < \strlen ( $string ); ++ $i ) {
$ord = \ord ( $string [ $i ]);
2018-09-19 11:59:07 +00:00
$hexCode = dechex ( $ord );
2018-09-23 21:42:05 +00:00
$hex .= substr ( '0' . $hexCode , - 2 );
2018-09-19 11:59:07 +00:00
}
2018-09-23 21:42:05 +00:00
return strtoupper ( $hex );
2018-09-19 11:59:07 +00:00
}
/**
2018-09-23 21:42:05 +00:00
* https :// stackoverflow . com / a / 18506801.
*
2018-09-19 11:59:07 +00:00
* @ param $hex
2018-09-23 21:42:05 +00:00
*
2018-09-19 11:59:07 +00:00
* @ return string
*/
2018-09-23 21:42:05 +00:00
private function hexToStr ( $hex )
{
$string = '' ;
for ( $i = 0 ; $i < \strlen ( $hex ) - 1 ; $i += 2 ) {
$string .= \chr ( hexdec ( $hex [ $i ] . $hex [ $i + 1 ]));
2018-09-19 11:59:07 +00:00
}
2018-09-23 21:42:05 +00:00
2018-09-19 11:59:07 +00:00
return $string ;
}
2015-10-11 20:27:47 +00:00
private function getTaggerMock ()
{
2017-05-27 20:08:14 +00:00
return $this -> getMockBuilder ( RuleBasedTagger :: class )
2016-04-12 09:36:01 +00:00
-> setMethods ([ 'tag' ])
2015-10-11 20:27:47 +00:00
-> disableOriginalConstructor ()
-> getMock ();
}
2015-10-17 15:45:51 +00:00
2015-10-31 15:38:49 +00:00
private function getLogger ()
2015-10-17 15:45:51 +00:00
{
2015-10-31 15:38:49 +00:00
return new NullLogger ();
2015-10-17 15:45:51 +00:00
}
2017-06-08 19:51:46 +00:00
2017-12-18 09:14:00 +00:00
private function getValidator ( $withDefaultMock = true )
2017-06-08 19:51:46 +00:00
{
2017-12-18 09:14:00 +00:00
$mock = $this -> getMockBuilder ( RecursiveValidator :: class )
2017-06-08 19:51:46 +00:00
-> setMethods ([ 'validate' ])
-> disableOriginalConstructor ()
-> getMock ();
2017-12-18 09:14:00 +00:00
if ( $withDefaultMock ) {
$mock -> expects ( $this -> any ())
-> method ( 'validate' )
-> willReturn ( new ConstraintViolationList ());
}
return $mock ;
2017-06-08 19:51:46 +00:00
}
2015-09-10 19:57:25 +00:00
}