- { path: ^/api/doc, roles: IS_AUTHENTICATED_ANONYMOUSLY }
- { path: ^/login, roles: IS_AUTHENTICATED_ANONYMOUSLY }
- { path: ^/forgot-password, roles: IS_AUTHENTICATED_ANONYMOUSLY }
+ - { path: /(unread|starred|archive).xml$, roles: IS_AUTHENTICATED_ANONYMOUSLY }
- { path: ^/, roles: ROLE_USER }
$config = new Config($newUser);
$config->setTheme($this->container->getParameter('theme'));
$config->setItemsPerPage($this->container->getParameter('items_on_page'));
- $config->setRssLimit($this->getContainer()->getParameter('rss_limit'));
+ $config->setRssLimit($this->container->getParameter('rss_limit'));
$config->setLanguage($this->container->getParameter('language'));
$em->persist($config);
->findUnreadByUser(
$user->getId(),
0,
- $user->getConfig()->getRssLimit() ?: $this->getContainer()->getParameter('rss_limit')
+ $user->getConfig()->getRssLimit() ?: $this->container->getParameter('rss_limit')
);
return $this->render('WallabagCoreBundle:Entry:entries.xml.twig', array(
->findArchiveByUser(
$user->getId(),
0,
- $user->getConfig()->getRssLimit() ?: $this->getContainer()->getParameter('rss_limit')
+ $user->getConfig()->getRssLimit() ?: $this->container->getParameter('rss_limit')
);
return $this->render('WallabagCoreBundle:Entry:entries.xml.twig', array(
->findStarredByUser(
$user->getId(),
0,
- $user->getConfig()->getRssLimit() ?: $this->getContainer()->getParameter('rss_limit')
+ $user->getConfig()->getRssLimit() ?: $this->container->getParameter('rss_limit')
);
return $this->render('WallabagCoreBundle:Entry:entries.xml.twig', array(
$this->addReference('entry4', $entry4);
+ $entry5 = new Entry($this->getReference('admin-user'));
+ $entry5->setUrl('http://0.0.0.0');
+ $entry5->setTitle('test title entry5');
+ $entry5->setContent('This is my content /o/');
+ $entry5->setStarred(true);
+
+ $manager->persist($entry5);
+
+ $this->addReference('entry5', $entry5);
+
+ $entry6 = new Entry($this->getReference('admin-user'));
+ $entry6->setUrl('http://0.0.0.0');
+ $entry6->setTitle('test title entry6');
+ $entry6->setContent('This is my content /o/');
+ $entry6->setArchived(true);
+
+ $manager->persist($entry6);
+
+ $this->addReference('entry6', $entry6);
+
$manager->flush();
}
* @var integer
*
* @Assert\NotBlank()
+ * @Assert\Range(
+ * min = 1,
+ * max = 100000,
+ * maxMessage = "This will certainly kill the app"
+ * )
* @ORM\Column(name="items_per_page", type="integer", nullable=false)
*/
private $itemsPerPage;
* @var integer
*
* @ORM\Column(name="rss_limit", type="integer", nullable=true)
+ * @Assert\Range(
+ * min = 1,
+ * max = 100000,
+ * maxMessage = "This will certainly kill the app"
+ * )
*/
private $rssLimit;
{
$builder
->add('theme', 'choice', array('choices' => $this->themes))
- ->add('items_per_page', 'text')
+ ->add('items_per_page')
->add('language')
->add('save', 'submit')
;
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
- ->add('rss_limit', 'text')
+ ->add('rss_limit')
->add('save', 'submit')
;
}
<fieldset class="w500p inline">
<div class="row">
<label>Rss links:</label>
- <ul>
- <li><a href="{{ path('unread_rss', {'username': rss.username, 'token': rss.token}) }}">unread</a></li>
- <li><a href="{{ path('starred_rss', {'username': rss.username, 'token': rss.token}) }}">fav</a></li>
- <li><a href="{{ path('archive_rss', {'username': rss.username, 'token': rss.token}) }}">archives</a></li>
- </ul>
+ {% if rss.token %}
+ <ul>
+ <li><a href="{{ path('unread_rss', {'username': rss.username, 'token': rss.token}) }}">unread</a></li>
+ <li><a href="{{ path('starred_rss', {'username': rss.username, 'token': rss.token}) }}">fav</a></li>
+ <li><a href="{{ path('archive_rss', {'username': rss.username, 'token': rss.token}) }}">archives</a></li>
+ </ul>
+ {% else %}
+ <strong>You need to generate a token first.</strong>
+ {% endif %}
</div>
</fieldset>
<pubDate>{{ entry.createdAt|date('D, d M Y H:i:s') }}</pubDate>
<description>
<![CDATA[
- {%- if entry.content| readingTime > 0 %}
+ {%- if entry.content| readingTime > 0 -%}
{% trans %}estimated reading time :{% endtrans %} {{ entry.content| readingTime }} min
- {% else -%}
+ {%- else -%}
{% trans %}estimated reading time :{% endtrans %} < 1 min
- {% endif -%}
+ {%- endif %}
{{ entry.content -}}
]]>
4 = Messages
5 = Article
6 = Media queries
-
+
========================================================================== */
html {
position: absolute;
top: 2em;
left: 50%;
- margin-left: -55px;
+ margin-left: -55px;
}
/* ==========================================================================
1 = Style Guide
========================================================================== */
-::selection {
- color: #FFF;
+::selection {
+ color: #FFF;
background-color: #000;
-}
+}
.desktopHide {
display: none;
position: fixed;
z-index: 20;
top: 0.4em;
- left: 0.6em;
+ left: 0.6em;
}
h2, h3, h4 {
margin: 0;
}
-form input[type="text"], select, form input[type="password"], form input[type="url"], form input[type="email"] {
+form input[type="text"], form input[type="number"], select, form input[type="password"], form input[type="url"], form input[type="email"] {
border: 1px solid #999;
padding: 0.5em 1em;
min-width: 12em;
#bookmarklet {
cursor: move;
-}
+}
h2:after {
content: "";
/* ==========================================================================
2 = Layout
========================================================================== */
-
+
#content {
margin-top: 5em;
min-height: 30em;
/* ==========================================================================
3 = Pictos
========================================================================== */
-
+
@font-face {
font-family: 'icomoon';
src:url('../fonts/icomoon.eot?-s0mcsx');
color: #FFF;
text-decoration: none;
}
-
+
#article_toolbar a:hover, #article_toolbar a:focus {
background-color: #999;
}
#article_toolbar a {
padding: 0.3em 0.4em 0.2em;
}
-
+
#display-mode {
display: none;
}
$this->assertCount(1, $crawler->filter('button[id=config_save]'));
$this->assertCount(1, $crawler->filter('button[id=change_passwd_save]'));
$this->assertCount(1, $crawler->filter('button[id=user_save]'));
+ $this->assertCount(1, $crawler->filter('button[id=new_user_save]'));
+ $this->assertCount(1, $crawler->filter('button[id=rss_config_save]'));
}
public function testUpdate()
$this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(array('_text')));
$this->assertContains('User "wallace" added', $alert[0]);
}
+
+ public function testRssUpdateResetToken()
+ {
+ $this->logInAs('admin');
+ $client = $this->getClient();
+
+ // reset the token
+ $em = $client->getContainer()->get('doctrine.orm.entity_manager');
+ $user = $em
+ ->getRepository('WallabagCoreBundle:User')
+ ->findOneByUsername('admin');
+
+ if (!$user) {
+ $this->markTestSkipped('No user found in db.');
+ }
+
+ $config = $user->getConfig();
+ $config->setRssToken(null);
+ $em->persist($config);
+ $em->flush();
+
+ $crawler = $client->request('GET', '/config');
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+
+ $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(array('_text')));
+ $this->assertContains('You need to generate a token first.', $body[0]);
+
+ $client->request('GET', '/generate-token');
+ $this->assertEquals(302, $client->getResponse()->getStatusCode());
+
+ $crawler = $client->followRedirect();
+
+ $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(array('_text')));
+ $this->assertNotContains('You need to generate a token first.', $body[0]);
+ }
+
+ public function testGenerateTokenAjax()
+ {
+ $this->logInAs('admin');
+ $client = $this->getClient();
+
+ $client->request(
+ 'GET',
+ '/generate-token',
+ array(),
+ array(),
+ array('HTTP_X-Requested-With' => 'XMLHttpRequest')
+ );
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+ $content = json_decode($client->getResponse()->getContent(), true);;
+ $this->assertArrayHasKey('token', $content);
+ }
+
+ public function testRssUpdate()
+ {
+ $this->logInAs('admin');
+ $client = $this->getClient();
+
+ $crawler = $client->request('GET', '/config');
+
+ if (500 == $client->getResponse()->getStatusCode()) {
+ var_export($client->getResponse()->getContent());
+ die();
+ }
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+
+ $form = $crawler->filter('button[id=rss_config_save]')->form();
+
+ $data = array(
+ 'rss_config[rss_limit]' => 12,
+ );
+
+ $client->submit($form, $data);
+
+ $this->assertEquals(302, $client->getResponse()->getStatusCode());
+
+ $crawler = $client->followRedirect();
+
+ $this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(array('_text')));
+ $this->assertContains('RSS information updated', $alert[0]);
+ }
+
+ public function dataForRssFailed()
+ {
+ return array(
+ array(
+ array(
+ 'rss_config[rss_limit]' => 0,
+ ),
+ 'This value should be 1 or more.',
+ ),
+ array(
+ array(
+ 'rss_config[rss_limit]' => 1000000000000,
+ ),
+ 'This will certainly kill the app',
+ ),
+ );
+ }
+
+ /**
+ * @dataProvider dataForRssFailed
+ */
+ public function testRssFailed($data, $expectedMessage)
+ {
+ $this->logInAs('admin');
+ $client = $this->getClient();
+
+ $crawler = $client->request('GET', '/config');
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+
+ $form = $crawler->filter('button[id=rss_config_save]')->form();
+
+ $crawler = $client->submit($form, $data);
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+
+ $this->assertGreaterThan(1, $alert = $crawler->filter('body')->extract(array('_text')));
+ $this->assertContains($expectedMessage, $alert[0]);
+ }
}
--- /dev/null
+<?php
+
+namespace Wallabag\CoreBundle\Tests\Controller;
+
+use Wallabag\CoreBundle\Tests\WallabagTestCase;
+
+class RssControllerTest extends WallabagTestCase
+{
+ public function validateDom($xml, $nb = null)
+ {
+ $doc = new \DOMDocument();
+ $doc->loadXML($xml);
+
+ $xpath = new \DOMXpath($doc);
+
+ if (null === $nb) {
+ $this->assertGreaterThan(0, $xpath->query('//item')->length);
+ } else {
+ $this->assertEquals($nb, $xpath->query('//item')->length);
+ }
+
+ $this->assertEquals(1, $xpath->query('/rss')->length);
+ $this->assertEquals(1, $xpath->query('/rss/channel')->length);
+
+ foreach ($xpath->query('//item') as $item) {
+ $this->assertEquals(1, $xpath->query('title', $item)->length);
+ $this->assertEquals(1, $xpath->query('source', $item)->length);
+ $this->assertEquals(1, $xpath->query('link', $item)->length);
+ $this->assertEquals(1, $xpath->query('guid', $item)->length);
+ $this->assertEquals(1, $xpath->query('pubDate', $item)->length);
+ $this->assertEquals(1, $xpath->query('description', $item)->length);
+ }
+ }
+
+ public function dataForBadUrl()
+ {
+ return array(
+ array(
+ '/admin/YZIOAUZIAO/unread.xml'
+ ),
+ array(
+ '/wallace/YZIOAUZIAO/starred.xml'
+ ),
+ array(
+ '/wallace/YZIOAUZIAO/archives.xml'
+ ),
+ );
+ }
+
+ /**
+ * @dataProvider dataForBadUrl
+ */
+ public function testBadUrl($url)
+ {
+ $client = $this->getClient();
+
+ $client->request('GET', $url);
+
+ $this->assertEquals(404, $client->getResponse()->getStatusCode());
+ }
+
+ public function testUnread()
+ {
+ $client = $this->getClient();
+ $em = $client->getContainer()->get('doctrine.orm.entity_manager');
+ $user = $em
+ ->getRepository('WallabagCoreBundle:User')
+ ->findOneByUsername('admin');
+
+ $config = $user->getConfig();
+ $config->setRssToken('SUPERTOKEN');
+ $config->setRssLimit(2);
+ $em->persist($config);
+ $em->flush();
+
+ $client->request('GET', '/admin/SUPERTOKEN/unread.xml');
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+
+ $this->validateDom($client->getResponse()->getContent(), 2);
+ }
+
+ public function testStarred()
+ {
+ $client = $this->getClient();
+ $em = $client->getContainer()->get('doctrine.orm.entity_manager');
+ $user = $em
+ ->getRepository('WallabagCoreBundle:User')
+ ->findOneByUsername('admin');
+
+ $config = $user->getConfig();
+ $config->setRssToken('SUPERTOKEN');
+ $config->setRssLimit(1);
+ $em->persist($config);
+ $em->flush();
+
+ $client = $this->getClient();
+ $client->request('GET', '/admin/SUPERTOKEN/starred.xml');
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode(), 1);
+
+ $this->validateDom($client->getResponse()->getContent());
+ }
+
+ public function testArchives()
+ {
+ $client = $this->getClient();
+ $em = $client->getContainer()->get('doctrine.orm.entity_manager');
+ $user = $em
+ ->getRepository('WallabagCoreBundle:User')
+ ->findOneByUsername('admin');
+
+ $config = $user->getConfig();
+ $config->setRssToken('SUPERTOKEN');
+ $config->setRssLimit(null);
+ $em->persist($config);
+ $em->flush();
+
+ $client = $this->getClient();
+ $client->request('GET', '/admin/SUPERTOKEN/archive.xml');
+
+ $this->assertEquals(200, $client->getResponse()->getStatusCode());
+
+ $this->validateDom($client->getResponse()->getContent());
+ }
+}
--- /dev/null
+<?php
+
+namespace Wallabag\CoreBundle\Tests\Command;
+
+use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;
+use Wallabag\CoreBundle\ParamConverter\UsernameRssTokenConverter;
+use Sensio\Bundle\FrameworkExtraBundle\Configuration\ParamConverter;
+use Symfony\Component\HttpFoundation\Request;
+use Wallabag\CoreBundle\Entity\User;
+
+class UsernameRssTokenConverterTest extends KernelTestCase
+{
+ public function testSupportsWithNoRegistry()
+ {
+ $params = new ParamConverter(array());
+ $converter = new UsernameRssTokenConverter();
+
+ $this->assertFalse($converter->supports($params));
+ }
+
+ public function testSupportsWithNoRegistryManagers()
+ {
+ $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $registry->expects($this->once())
+ ->method('getManagers')
+ ->will($this->returnValue(array()));
+
+ $params = new ParamConverter(array());
+ $converter = new UsernameRssTokenConverter($registry);
+
+ $this->assertFalse($converter->supports($params));
+ }
+
+ public function testSupportsWithNoConfigurationClass()
+ {
+ $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $registry->expects($this->once())
+ ->method('getManagers')
+ ->will($this->returnValue(array('default' => null)));
+
+ $params = new ParamConverter(array());
+ $converter = new UsernameRssTokenConverter($registry);
+
+ $this->assertFalse($converter->supports($params));
+ }
+
+ public function testSupportsWithNotTheGoodClass()
+ {
+ $meta = $this->getMockBuilder('Doctrine\Common\Persistence\Mapping\ClassMetadata')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $meta->expects($this->once())
+ ->method('getName')
+ ->will($this->returnValue('nothingrelated'));
+
+ $em = $this->getMockBuilder('Doctrine\Common\Persistence\ObjectManager')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $em->expects($this->once())
+ ->method('getClassMetadata')
+ ->with('superclass')
+ ->will($this->returnValue($meta));
+
+ $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $registry->expects($this->once())
+ ->method('getManagers')
+ ->will($this->returnValue(array('default' => null)));
+
+ $registry->expects($this->once())
+ ->method('getManagerForClass')
+ ->with('superclass')
+ ->will($this->returnValue($em));
+
+ $params = new ParamConverter(array('class' => 'superclass'));
+ $converter = new UsernameRssTokenConverter($registry);
+
+ $this->assertFalse($converter->supports($params));
+ }
+
+ public function testSupportsWithGoodClass()
+ {
+ $meta = $this->getMockBuilder('Doctrine\Common\Persistence\Mapping\ClassMetadata')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $meta->expects($this->once())
+ ->method('getName')
+ ->will($this->returnValue('Wallabag\CoreBundle\Entity\User'));
+
+ $em = $this->getMockBuilder('Doctrine\Common\Persistence\ObjectManager')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $em->expects($this->once())
+ ->method('getClassMetadata')
+ ->with('WallabagCoreBundle:User')
+ ->will($this->returnValue($meta));
+
+ $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $registry->expects($this->once())
+ ->method('getManagers')
+ ->will($this->returnValue(array('default' => null)));
+
+ $registry->expects($this->once())
+ ->method('getManagerForClass')
+ ->with('WallabagCoreBundle:User')
+ ->will($this->returnValue($em));
+
+ $params = new ParamConverter(array('class' => 'WallabagCoreBundle:User'));
+ $converter = new UsernameRssTokenConverter($registry);
+
+ $this->assertTrue($converter->supports($params));
+ }
+
+ /**
+ * @expectedException InvalidArgumentException
+ * @expectedExceptionMessage Route attribute is missing
+ */
+ public function testApplyEmptyRequest()
+ {
+ $params = new ParamConverter(array());
+ $converter = new UsernameRssTokenConverter();
+
+ $converter->apply(new Request(), $params);
+ }
+
+ /**
+ * @expectedException Symfony\Component\HttpKernel\Exception\NotFoundHttpException
+ * @expectedExceptionMessage User not found
+ */
+ public function testApplyUserNotFound()
+ {
+ $repo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\UserRepository')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $repo->expects($this->once())
+ ->method('findOneByUsernameAndRsstoken')
+ ->with('test', 'test')
+ ->will($this->returnValue(null));
+
+ $em = $this->getMockBuilder('Doctrine\Common\Persistence\ObjectManager')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $em->expects($this->once())
+ ->method('getRepository')
+ ->with('WallabagCoreBundle:User')
+ ->will($this->returnValue($repo));
+
+ $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $registry->expects($this->once())
+ ->method('getManagerForClass')
+ ->with('WallabagCoreBundle:User')
+ ->will($this->returnValue($em));
+
+ $params = new ParamConverter(array('class' => 'WallabagCoreBundle:User'));
+ $converter = new UsernameRssTokenConverter($registry);
+ $request = new Request(array(), array(), array('username' => 'test', 'token' => 'test'));
+
+ $converter->apply($request, $params);
+ }
+
+ public function testApplyUserFound()
+ {
+ $user = new User();
+
+ $repo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\UserRepository')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $repo->expects($this->once())
+ ->method('findOneByUsernameAndRsstoken')
+ ->with('test', 'test')
+ ->will($this->returnValue($user));
+
+ $em = $this->getMockBuilder('Doctrine\Common\Persistence\ObjectManager')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $em->expects($this->once())
+ ->method('getRepository')
+ ->with('WallabagCoreBundle:User')
+ ->will($this->returnValue($repo));
+
+ $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry')
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $registry->expects($this->once())
+ ->method('getManagerForClass')
+ ->with('WallabagCoreBundle:User')
+ ->will($this->returnValue($em));
+
+ $params = new ParamConverter(array('class' => 'WallabagCoreBundle:User', 'name' => 'user'));
+ $converter = new UsernameRssTokenConverter($registry);
+ $request = new Request(array(), array(), array('username' => 'test', 'token' => 'test'));
+
+ $converter->apply($request, $params);
+
+ $this->assertEquals($user, $request->attributes->get('user'));
+ }
+}