From 23634d5d842dabcf5d7475e2becb7e127824239e Mon Sep 17 00:00:00 2001 From: Jeremy Benoist Date: Wed, 1 Jun 2016 21:27:35 +0200 Subject: Jump to Symfony 3.1 --- .../Controller/AnnotationControllerTest.php | 120 ++++ .../WallabagAnnotationTestCase.php | 63 ++ .../Controller/WallabagRestControllerTest.php | 513 ++++++++++++++++ tests/Wallabag/ApiBundle/WallabagApiTestCase.php | 51 ++ .../CoreBundle/Command/InstallCommandTest.php | 279 +++++++++ .../CoreBundle/Command/TagAllCommandTest.php | 60 ++ .../CoreBundle/Controller/ConfigControllerTest.php | 652 ++++++++++++++++++++ .../Controller/DeveloperControllerTest.php | 71 +++ .../CoreBundle/Controller/EntryControllerTest.php | 665 +++++++++++++++++++++ .../CoreBundle/Controller/ExportControllerTest.php | 251 ++++++++ .../CoreBundle/Controller/RssControllerTest.php | 126 ++++ .../Controller/SecurityControllerTest.php | 72 +++ .../Controller/SettingsControllerTest.php | 32 + .../CoreBundle/Controller/StaticControllerTest.php | 28 + .../CoreBundle/Controller/TagControllerTest.php | 128 ++++ .../EventListener/LocaleListenerTest.php | 82 +++ .../RegistrationConfirmedListenerTest.php | 91 +++ .../EventListener/UserLocaleListenerTest.php | 58 ++ .../StringToListTransformerTest.php | 50 ++ .../CoreBundle/Helper/ContentProxyTest.php | 318 ++++++++++ tests/Wallabag/CoreBundle/Helper/RedirectTest.php | 55 ++ .../CoreBundle/Helper/RuleBasedTaggerTest.php | 212 +++++++ .../CoreBundle/Mock/InstallCommandMock.php | 22 + .../UsernameRssTokenConverterTest.php | 219 +++++++ .../Subscriber/TablePrefixSubscriberTest.php | 114 ++++ .../CoreBundle/Twig/WallabagExtensionTest.php | 17 + tests/Wallabag/CoreBundle/WallabagCoreTestCase.php | 51 ++ .../Controller/ImportControllerTest.php | 29 + .../Controller/PocketControllerTest.php | 65 ++ .../Controller/WallabagV1ControllerTest.php | 129 ++++ .../Controller/WallabagV2ControllerTest.php | 95 +++ .../ImportBundle/Import/ImportChainTest.php | 21 + .../ImportBundle/Import/ImportCompilerPassTest.php | 47 ++ .../ImportBundle/Import/PocketImportTest.php | 393 ++++++++++++ .../ImportBundle/Import/WallabagV1ImportTest.php | 150 +++++ .../ImportBundle/Import/WallabagV2ImportTest.php | 146 +++++ tests/Wallabag/ImportBundle/fixtures/test.html | 0 tests/Wallabag/ImportBundle/fixtures/test.txt | 0 .../ImportBundle/fixtures/wallabag-v1-read.json | 53 ++ .../ImportBundle/fixtures/wallabag-v1.json | 69 +++ .../ImportBundle/fixtures/wallabag-v2-read.json | 28 + .../ImportBundle/fixtures/wallabag-v2.json | 346 +++++++++++ .../UserBundle/Mailer/AuthCodeMailerTest.php | 84 +++ 43 files changed, 6055 insertions(+) create mode 100644 tests/Wallabag/AnnotationBundle/Controller/AnnotationControllerTest.php create mode 100644 tests/Wallabag/AnnotationBundle/WallabagAnnotationTestCase.php create mode 100644 tests/Wallabag/ApiBundle/Controller/WallabagRestControllerTest.php create mode 100644 tests/Wallabag/ApiBundle/WallabagApiTestCase.php create mode 100644 tests/Wallabag/CoreBundle/Command/InstallCommandTest.php create mode 100644 tests/Wallabag/CoreBundle/Command/TagAllCommandTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/ConfigControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/DeveloperControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/EntryControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/ExportControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/RssControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/SecurityControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/SettingsControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/StaticControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/Controller/TagControllerTest.php create mode 100644 tests/Wallabag/CoreBundle/EventListener/LocaleListenerTest.php create mode 100644 tests/Wallabag/CoreBundle/EventListener/RegistrationConfirmedListenerTest.php create mode 100644 tests/Wallabag/CoreBundle/EventListener/UserLocaleListenerTest.php create mode 100644 tests/Wallabag/CoreBundle/Form/DataTransformer/StringToListTransformerTest.php create mode 100644 tests/Wallabag/CoreBundle/Helper/ContentProxyTest.php create mode 100644 tests/Wallabag/CoreBundle/Helper/RedirectTest.php create mode 100644 tests/Wallabag/CoreBundle/Helper/RuleBasedTaggerTest.php create mode 100644 tests/Wallabag/CoreBundle/Mock/InstallCommandMock.php create mode 100644 tests/Wallabag/CoreBundle/ParamConverter/UsernameRssTokenConverterTest.php create mode 100644 tests/Wallabag/CoreBundle/Subscriber/TablePrefixSubscriberTest.php create mode 100644 tests/Wallabag/CoreBundle/Twig/WallabagExtensionTest.php create mode 100644 tests/Wallabag/CoreBundle/WallabagCoreTestCase.php create mode 100644 tests/Wallabag/ImportBundle/Controller/ImportControllerTest.php create mode 100644 tests/Wallabag/ImportBundle/Controller/PocketControllerTest.php create mode 100644 tests/Wallabag/ImportBundle/Controller/WallabagV1ControllerTest.php create mode 100644 tests/Wallabag/ImportBundle/Controller/WallabagV2ControllerTest.php create mode 100644 tests/Wallabag/ImportBundle/Import/ImportChainTest.php create mode 100644 tests/Wallabag/ImportBundle/Import/ImportCompilerPassTest.php create mode 100644 tests/Wallabag/ImportBundle/Import/PocketImportTest.php create mode 100644 tests/Wallabag/ImportBundle/Import/WallabagV1ImportTest.php create mode 100644 tests/Wallabag/ImportBundle/Import/WallabagV2ImportTest.php create mode 100644 tests/Wallabag/ImportBundle/fixtures/test.html create mode 100644 tests/Wallabag/ImportBundle/fixtures/test.txt create mode 100644 tests/Wallabag/ImportBundle/fixtures/wallabag-v1-read.json create mode 100644 tests/Wallabag/ImportBundle/fixtures/wallabag-v1.json create mode 100644 tests/Wallabag/ImportBundle/fixtures/wallabag-v2-read.json create mode 100644 tests/Wallabag/ImportBundle/fixtures/wallabag-v2.json create mode 100644 tests/Wallabag/UserBundle/Mailer/AuthCodeMailerTest.php (limited to 'tests') diff --git a/tests/Wallabag/AnnotationBundle/Controller/AnnotationControllerTest.php b/tests/Wallabag/AnnotationBundle/Controller/AnnotationControllerTest.php new file mode 100644 index 00000000..70849f74 --- /dev/null +++ b/tests/Wallabag/AnnotationBundle/Controller/AnnotationControllerTest.php @@ -0,0 +1,120 @@ +client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagAnnotationBundle:Annotation') + ->findOneByUsername('admin'); + + if (!$annotation) { + $this->markTestSkipped('No content found in db.'); + } + + $this->logInAs('admin'); + $crawler = $this->client->request('GET', 'annotations/'.$annotation->getEntry()->getId().'.json'); + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + $this->assertEquals(1, $content['total']); + $this->assertEquals($annotation->getText(), $content['rows'][0]['text']); + } + + public function testSetAnnotation() + { + $this->logInAs('admin'); + + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('admin'); + + $headers = ['CONTENT_TYPE' => 'application/json']; + $content = json_encode([ + 'text' => 'my annotation', + 'quote' => 'my quote', + 'ranges' => ['start' => '', 'startOffset' => 24, 'end' => '', 'endOffset' => 31], + ]); + $crawler = $this->client->request('POST', 'annotations/'.$entry->getId().'.json', [], [], $headers, $content); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals('Big boss', $content['user']); + $this->assertEquals('v1.0', $content['annotator_schema_version']); + $this->assertEquals('my annotation', $content['text']); + $this->assertEquals('my quote', $content['quote']); + + $annotation = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagAnnotationBundle:Annotation') + ->findLastAnnotationByPageId($entry->getId(), 1); + + $this->assertEquals('my annotation', $annotation->getText()); + } + + public function testEditAnnotation() + { + $annotation = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagAnnotationBundle:Annotation') + ->findOneByUsername('admin'); + + $this->logInAs('admin'); + + $headers = ['CONTENT_TYPE' => 'application/json']; + $content = json_encode([ + 'text' => 'a modified annotation', + ]); + $crawler = $this->client->request('PUT', 'annotations/'.$annotation->getId().'.json', [], [], $headers, $content); + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals('Big boss', $content['user']); + $this->assertEquals('v1.0', $content['annotator_schema_version']); + $this->assertEquals('a modified annotation', $content['text']); + $this->assertEquals('my quote', $content['quote']); + + $annotationUpdated = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagAnnotationBundle:Annotation') + ->findOneById($annotation->getId()); + $this->assertEquals('a modified annotation', $annotationUpdated->getText()); + } + + public function testDeleteAnnotation() + { + $annotation = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagAnnotationBundle:Annotation') + ->findOneByUsername('admin'); + + $this->logInAs('admin'); + + $headers = ['CONTENT_TYPE' => 'application/json']; + $content = json_encode([ + 'text' => 'a modified annotation', + ]); + $crawler = $this->client->request('DELETE', 'annotations/'.$annotation->getId().'.json', [], [], $headers, $content); + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals('a modified annotation', $content['text']); + + $annotationDeleted = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagAnnotationBundle:Annotation') + ->findOneById($annotation->getId()); + + $this->assertNull($annotationDeleted); + } +} diff --git a/tests/Wallabag/AnnotationBundle/WallabagAnnotationTestCase.php b/tests/Wallabag/AnnotationBundle/WallabagAnnotationTestCase.php new file mode 100644 index 00000000..82790a5c --- /dev/null +++ b/tests/Wallabag/AnnotationBundle/WallabagAnnotationTestCase.php @@ -0,0 +1,63 @@ +client = $this->createAuthorizedClient(); + } + + public function logInAs($username) + { + $crawler = $this->client->request('GET', '/login'); + $form = $crawler->filter('button[type=submit]')->form(); + $data = [ + '_username' => $username, + '_password' => 'mypassword', + ]; + + $this->client->submit($form, $data); + } + + /** + * @return Client + */ + protected function createAuthorizedClient() + { + $client = static::createClient(); + $container = $client->getContainer(); + + /** @var $userManager \FOS\UserBundle\Doctrine\UserManager */ + $userManager = $container->get('fos_user.user_manager'); + /** @var $loginManager \FOS\UserBundle\Security\LoginManager */ + $loginManager = $container->get('fos_user.security.login_manager'); + $firewallName = $container->getParameter('fos_user.firewall_name'); + + $this->user = $userManager->findUserBy(['username' => 'admin']); + $loginManager->loginUser($firewallName, $this->user); + + // save the login token into the session and put it in a cookie + $container->get('session')->set('_security_'.$firewallName, serialize($container->get('security.token_storage')->getToken())); + $container->get('session')->save(); + + $session = $container->get('session'); + $client->getCookieJar()->set(new Cookie($session->getName(), $session->getId())); + + return $client; + } +} diff --git a/tests/Wallabag/ApiBundle/Controller/WallabagRestControllerTest.php b/tests/Wallabag/ApiBundle/Controller/WallabagRestControllerTest.php new file mode 100644 index 00000000..c39cc357 --- /dev/null +++ b/tests/Wallabag/ApiBundle/Controller/WallabagRestControllerTest.php @@ -0,0 +1,513 @@ +client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneBy(['user' => 1, 'isArchived' => false]); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $this->client->request('GET', '/api/entries/'.$entry->getId().'.json'); + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals($entry->getTitle(), $content['title']); + $this->assertEquals($entry->getUrl(), $content['url']); + $this->assertCount(count($entry->getTags()), $content['tags']); + $this->assertEquals($entry->getUserName(), $content['user_name']); + $this->assertEquals($entry->getUserEmail(), $content['user_email']); + $this->assertEquals($entry->getUserId(), $content['user_id']); + + $this->assertTrue( + $this->client->getResponse()->headers->contains( + 'Content-Type', + 'application/json' + ) + ); + } + + public function testGetOneEntryWrongUser() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneBy(['user' => 2, 'isArchived' => false]); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $this->client->request('GET', '/api/entries/'.$entry->getId().'.json'); + + $this->assertEquals(403, $this->client->getResponse()->getStatusCode()); + } + + public function testGetEntries() + { + $this->client->request('GET', '/api/entries'); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThanOrEqual(1, count($content)); + $this->assertNotEmpty($content['_embedded']['items']); + $this->assertGreaterThanOrEqual(1, $content['total']); + $this->assertEquals(1, $content['page']); + $this->assertGreaterThanOrEqual(1, $content['pages']); + + $this->assertTrue( + $this->client->getResponse()->headers->contains( + 'Content-Type', + 'application/json' + ) + ); + } + + public function testGetStarredEntries() + { + $this->client->request('GET', '/api/entries', ['star' => 1, 'sort' => 'updated']); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThanOrEqual(1, count($content)); + $this->assertNotEmpty($content['_embedded']['items']); + $this->assertGreaterThanOrEqual(1, $content['total']); + $this->assertEquals(1, $content['page']); + $this->assertGreaterThanOrEqual(1, $content['pages']); + + $this->assertTrue( + $this->client->getResponse()->headers->contains( + 'Content-Type', + 'application/json' + ) + ); + } + + public function testGetArchiveEntries() + { + $this->client->request('GET', '/api/entries', ['archive' => 1]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThanOrEqual(1, count($content)); + $this->assertNotEmpty($content['_embedded']['items']); + $this->assertGreaterThanOrEqual(1, $content['total']); + $this->assertEquals(1, $content['page']); + $this->assertGreaterThanOrEqual(1, $content['pages']); + + $this->assertTrue( + $this->client->getResponse()->headers->contains( + 'Content-Type', + 'application/json' + ) + ); + } + + public function testDeleteEntry() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUser(1); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $this->client->request('DELETE', '/api/entries/'.$entry->getId().'.json'); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals($entry->getTitle(), $content['title']); + $this->assertEquals($entry->getUrl(), $content['url']); + + // We'll try to delete this entry again + $this->client->request('DELETE', '/api/entries/'.$entry->getId().'.json'); + + $this->assertEquals(404, $this->client->getResponse()->getStatusCode()); + } + + public function testPostEntry() + { + $this->client->request('POST', '/api/entries.json', [ + 'url' => 'http://www.lemonde.fr/pixels/article/2015/03/28/plongee-dans-l-univers-d-ingress-le-jeu-de-google-aux-frontieres-du-reel_4601155_4408996.html', + 'tags' => 'google', + 'title' => 'New title for my article', + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThan(0, $content['id']); + $this->assertEquals('http://www.lemonde.fr/pixels/article/2015/03/28/plongee-dans-l-univers-d-ingress-le-jeu-de-google-aux-frontieres-du-reel_4601155_4408996.html', $content['url']); + $this->assertEquals(false, $content['is_archived']); + $this->assertEquals(false, $content['is_starred']); + $this->assertEquals('New title for my article', $content['title']); + $this->assertEquals(1, $content['user_id']); + $this->assertCount(1, $content['tags']); + } + + public function testPostSameEntry() + { + $this->client->request('POST', '/api/entries.json', [ + 'url' => 'http://www.lemonde.fr/pixels/article/2015/03/28/plongee-dans-l-univers-d-ingress-le-jeu-de-google-aux-frontieres-du-reel_4601155_4408996.html', + 'archive' => '1', + 'tags' => 'google, apple', + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThan(0, $content['id']); + $this->assertEquals('http://www.lemonde.fr/pixels/article/2015/03/28/plongee-dans-l-univers-d-ingress-le-jeu-de-google-aux-frontieres-du-reel_4601155_4408996.html', $content['url']); + $this->assertEquals(true, $content['is_archived']); + $this->assertEquals(false, $content['is_starred']); + $this->assertCount(2, $content['tags']); + } + + public function testPostArchivedAndStarredEntry() + { + $this->client->request('POST', '/api/entries.json', [ + 'url' => 'http://www.lemonde.fr/idees/article/2016/02/08/preserver-la-liberte-d-expression-sur-les-reseaux-sociaux_4861503_3232.html', + 'archive' => '1', + 'starred' => '1', + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThan(0, $content['id']); + $this->assertEquals('http://www.lemonde.fr/idees/article/2016/02/08/preserver-la-liberte-d-expression-sur-les-reseaux-sociaux_4861503_3232.html', $content['url']); + $this->assertEquals(true, $content['is_archived']); + $this->assertEquals(true, $content['is_starred']); + $this->assertEquals(1, $content['user_id']); + } + + public function testPostArchivedAndStarredEntryWithoutQuotes() + { + $this->client->request('POST', '/api/entries.json', [ + 'url' => 'http://www.lemonde.fr/idees/article/2016/02/08/preserver-la-liberte-d-expression-sur-les-reseaux-sociaux_4861503_3232.html', + 'archive' => 0, + 'starred' => 1, + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThan(0, $content['id']); + $this->assertEquals('http://www.lemonde.fr/idees/article/2016/02/08/preserver-la-liberte-d-expression-sur-les-reseaux-sociaux_4861503_3232.html', $content['url']); + $this->assertEquals(false, $content['is_archived']); + $this->assertEquals(true, $content['is_starred']); + } + + public function testPatchEntry() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUser(1); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + // hydrate the tags relations + $nbTags = count($entry->getTags()); + + $this->client->request('PATCH', '/api/entries/'.$entry->getId().'.json', [ + 'title' => 'New awesome title', + 'tags' => 'new tag '.uniqid(), + 'starred' => '1', + 'archive' => '0', + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals($entry->getId(), $content['id']); + $this->assertEquals($entry->getUrl(), $content['url']); + $this->assertEquals('New awesome title', $content['title']); + $this->assertGreaterThan($nbTags, count($content['tags'])); + $this->assertEquals(1, $content['user_id']); + } + + public function testPatchEntryWithoutQuotes() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUser(1); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + // hydrate the tags relations + $nbTags = count($entry->getTags()); + + $this->client->request('PATCH', '/api/entries/'.$entry->getId().'.json', [ + 'title' => 'New awesome title', + 'tags' => 'new tag '.uniqid(), + 'starred' => 1, + 'archive' => 0, + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals($entry->getId(), $content['id']); + $this->assertEquals($entry->getUrl(), $content['url']); + $this->assertEquals('New awesome title', $content['title']); + $this->assertGreaterThan($nbTags, count($content['tags'])); + } + + public function testGetTagsEntry() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneWithTags(1); + + $entry = $entry[0]; + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $tags = []; + foreach ($entry->getTags() as $tag) { + $tags[] = ['id' => $tag->getId(), 'label' => $tag->getLabel(), 'slug' => $tag->getSlug()]; + } + + $this->client->request('GET', '/api/entries/'.$entry->getId().'/tags'); + + $this->assertEquals(json_encode($tags, JSON_HEX_QUOT), $this->client->getResponse()->getContent()); + } + + public function testPostTagsOnEntry() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUser(1); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $nbTags = count($entry->getTags()); + + $newTags = 'tag1,tag2,tag3'; + + $this->client->request('POST', '/api/entries/'.$entry->getId().'/tags', ['tags' => $newTags]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertArrayHasKey('tags', $content); + $this->assertEquals($nbTags + 3, count($content['tags'])); + + $entryDB = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->find($entry->getId()); + + $tagsInDB = []; + foreach ($entryDB->getTags()->toArray() as $tag) { + $tagsInDB[$tag->getId()] = $tag->getLabel(); + } + + foreach (explode(',', $newTags) as $tag) { + $this->assertContains($tag, $tagsInDB); + } + } + + public function testDeleteOneTagEntry() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneWithTags(1); + $entry = $entry[0]; + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + // hydrate the tags relations + $nbTags = count($entry->getTags()); + $tag = $entry->getTags()[0]; + + $this->client->request('DELETE', '/api/entries/'.$entry->getId().'/tags/'.$tag->getId().'.json'); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertArrayHasKey('tags', $content); + $this->assertEquals($nbTags - 1, count($content['tags'])); + } + + public function testGetUserTags() + { + $this->client->request('GET', '/api/tags.json'); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertGreaterThan(0, $content); + $this->assertArrayHasKey('id', $content[0]); + $this->assertArrayHasKey('label', $content[0]); + + return end($content); + } + + /** + * @depends testGetUserTags + */ + public function testDeleteUserTag($tag) + { + $this->client->request('DELETE', '/api/tags/'.$tag['id'].'.json'); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertArrayHasKey('label', $content); + $this->assertEquals($tag['label'], $content['label']); + $this->assertEquals($tag['slug'], $content['slug']); + + $entries = $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findAllByTagId($this->user->getId(), $tag['id']); + + $this->assertCount(0, $entries); + } + + public function testGetVersion() + { + $this->client->request('GET', '/api/version'); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals($this->client->getContainer()->getParameter('wallabag_core.version'), $content); + } + + public function testSaveIsArchivedAfterPost() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneBy(['user' => 1, 'isArchived' => true]); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $this->client->request('POST', '/api/entries.json', [ + 'url' => $entry->getUrl(), + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals(true, $content['is_archived']); + } + + public function testSaveIsStarredAfterPost() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneBy(['user' => 1, 'isStarred' => true]); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $this->client->request('POST', '/api/entries.json', [ + 'url' => $entry->getUrl(), + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals(true, $content['is_starred']); + } + + public function testSaveIsArchivedAfterPatch() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneBy(['user' => 1, 'isArchived' => true]); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + + $this->client->request('PATCH', '/api/entries/'.$entry->getId().'.json', [ + 'title' => $entry->getTitle().'++', + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals(true, $content['is_archived']); + } + + public function testSaveIsStarredAfterPatch() + { + $entry = $this->client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneBy(['user' => 1, 'isStarred' => true]); + + if (!$entry) { + $this->markTestSkipped('No content found in db.'); + } + $this->client->request('PATCH', '/api/entries/'.$entry->getId().'.json', [ + 'title' => $entry->getTitle().'++', + ]); + + $this->assertEquals(200, $this->client->getResponse()->getStatusCode()); + + $content = json_decode($this->client->getResponse()->getContent(), true); + + $this->assertEquals(true, $content['is_starred']); + } +} diff --git a/tests/Wallabag/ApiBundle/WallabagApiTestCase.php b/tests/Wallabag/ApiBundle/WallabagApiTestCase.php new file mode 100644 index 00000000..cf9b3347 --- /dev/null +++ b/tests/Wallabag/ApiBundle/WallabagApiTestCase.php @@ -0,0 +1,51 @@ +client = $this->createAuthorizedClient(); + } + + /** + * @return Client + */ + protected function createAuthorizedClient() + { + $client = static::createClient(); + $container = $client->getContainer(); + + /** @var $userManager \FOS\UserBundle\Doctrine\UserManager */ + $userManager = $container->get('fos_user.user_manager'); + /** @var $loginManager \FOS\UserBundle\Security\LoginManager */ + $loginManager = $container->get('fos_user.security.login_manager'); + $firewallName = $container->getParameter('fos_user.firewall_name'); + + $this->user = $userManager->findUserBy(['username' => 'admin']); + $loginManager->loginUser($firewallName, $this->user); + + // save the login token into the session and put it in a cookie + $container->get('session')->set('_security_'.$firewallName, serialize($container->get('security.token_storage')->getToken())); + $container->get('session')->save(); + + $session = $container->get('session'); + $client->getCookieJar()->set(new Cookie($session->getName(), $session->getId())); + + return $client; + } +} diff --git a/tests/Wallabag/CoreBundle/Command/InstallCommandTest.php b/tests/Wallabag/CoreBundle/Command/InstallCommandTest.php new file mode 100644 index 00000000..6c6ce087 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Command/InstallCommandTest.php @@ -0,0 +1,279 @@ +getClient()->getContainer()->get('doctrine')->getConnection()->getDriver() instanceof \Doctrine\DBAL\Driver\PDOPgSql\Driver) { + /* + * LOG: statement: CREATE DATABASE "wallabag" + * ERROR: source database "template1" is being accessed by other users + * DETAIL: There is 1 other session using the database. + * STATEMENT: CREATE DATABASE "wallabag" + * FATAL: database "wallabag" does not exist + * + * http://stackoverflow.com/a/14374832/569101 + */ + $this->markTestSkipped('PostgreSQL spotted: can find a good way to drop current database, skipping.'); + } + } + + public static function tearDownAfterClass() + { + $application = new Application(static::$kernel); + $application->setAutoExit(false); + + $code = $application->run(new ArrayInput([ + 'command' => 'doctrine:fixtures:load', + '--no-interaction' => true, + '--env' => 'test', + ]), new NullOutput()); + } + + public function testRunInstallCommand() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new InstallCommandMock()); + + $command = $application->find('wallabag:install'); + + // We mock the QuestionHelper + $question = $this->getMockBuilder('Symfony\Component\Console\Helper\QuestionHelper') + ->disableOriginalConstructor() + ->getMock(); + $question->expects($this->any()) + ->method('ask') + ->will($this->returnValue('yes_'.uniqid('', true))); + + // We override the standard helper with our mock + $command->getHelperSet()->set($question, 'question'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + ]); + + $this->assertContains('Checking system requirements.', $tester->getDisplay()); + $this->assertContains('Setting up database.', $tester->getDisplay()); + $this->assertContains('Administration setup.', $tester->getDisplay()); + $this->assertContains('Config setup.', $tester->getDisplay()); + $this->assertContains('Installing assets.', $tester->getDisplay()); + } + + public function testRunInstallCommandWithReset() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new InstallCommandMock()); + + $command = $application->find('wallabag:install'); + + // We mock the QuestionHelper + $question = $this->getMockBuilder('Symfony\Component\Console\Helper\QuestionHelper') + ->disableOriginalConstructor() + ->getMock(); + $question->expects($this->any()) + ->method('ask') + ->will($this->returnValue('yes_'.uniqid('', true))); + + // We override the standard helper with our mock + $command->getHelperSet()->set($question, 'question'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + '--reset' => true, + ]); + + $this->assertContains('Checking system requirements.', $tester->getDisplay()); + $this->assertContains('Setting up database.', $tester->getDisplay()); + $this->assertContains('Droping database, creating database and schema, clearing the cache', $tester->getDisplay()); + $this->assertContains('Administration setup.', $tester->getDisplay()); + $this->assertContains('Config setup.', $tester->getDisplay()); + $this->assertContains('Installing assets.', $tester->getDisplay()); + + // we force to reset everything + $this->assertContains('Droping database, creating database and schema, clearing the cache', $tester->getDisplay()); + } + + public function testRunInstallCommandWithDatabaseRemoved() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new DropDatabaseDoctrineCommand()); + + // drop database first, so the install command won't ask to reset things + $command = $application->find('doctrine:database:drop'); + $command->run(new ArrayInput([ + 'command' => 'doctrine:database:drop', + '--force' => true, + ]), new NullOutput()); + + // start a new application to avoid lagging connexion to pgsql + $client = static::createClient(); + $application = new Application($client->getKernel()); + $application->add(new InstallCommand()); + + $command = $application->find('wallabag:install'); + + // We mock the QuestionHelper + $question = $this->getMockBuilder('Symfony\Component\Console\Helper\QuestionHelper') + ->disableOriginalConstructor() + ->getMock(); + $question->expects($this->any()) + ->method('ask') + ->will($this->returnValue('yes_'.uniqid('', true))); + + // We override the standard helper with our mock + $command->getHelperSet()->set($question, 'question'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + ]); + + $this->assertContains('Checking system requirements.', $tester->getDisplay()); + $this->assertContains('Setting up database.', $tester->getDisplay()); + $this->assertContains('Administration setup.', $tester->getDisplay()); + $this->assertContains('Config setup.', $tester->getDisplay()); + $this->assertContains('Installing assets.', $tester->getDisplay()); + + // the current database doesn't already exist + $this->assertContains('Creating database and schema, clearing the cache', $tester->getDisplay()); + } + + public function testRunInstallCommandChooseResetSchema() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new InstallCommandMock()); + + $command = $application->find('wallabag:install'); + + // We mock the QuestionHelper + $question = $this->getMockBuilder('Symfony\Component\Console\Helper\QuestionHelper') + ->disableOriginalConstructor() + ->getMock(); + + $question->expects($this->exactly(3)) + ->method('ask') + ->will($this->onConsecutiveCalls( + false, // don't want to reset the entire database + true, // do want to reset the schema + false // don't want to create a new user + )); + + // We override the standard helper with our mock + $command->getHelperSet()->set($question, 'question'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + ]); + + $this->assertContains('Checking system requirements.', $tester->getDisplay()); + $this->assertContains('Setting up database.', $tester->getDisplay()); + $this->assertContains('Administration setup.', $tester->getDisplay()); + $this->assertContains('Config setup.', $tester->getDisplay()); + $this->assertContains('Installing assets.', $tester->getDisplay()); + + $this->assertContains('Droping schema and creating schema', $tester->getDisplay()); + } + + public function testRunInstallCommandChooseNothing() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new InstallCommand()); + $application->add(new DropDatabaseDoctrineCommand()); + $application->add(new CreateDatabaseDoctrineCommand()); + + // drop database first, so the install command won't ask to reset things + $command = new DropDatabaseDoctrineCommand(); + $command->setApplication($application); + $command->run(new ArrayInput([ + 'command' => 'doctrine:database:drop', + '--force' => true, + ]), new NullOutput()); + + $this->getClient()->getContainer()->get('doctrine')->getConnection()->close(); + + $command = new CreateDatabaseDoctrineCommand(); + $command->setApplication($application); + $command->run(new ArrayInput([ + 'command' => 'doctrine:database:create', + '--env' => 'test', + ]), new NullOutput()); + + $command = $application->find('wallabag:install'); + + // We mock the QuestionHelper + $question = $this->getMockBuilder('Symfony\Component\Console\Helper\QuestionHelper') + ->disableOriginalConstructor() + ->getMock(); + + $question->expects($this->exactly(2)) + ->method('ask') + ->will($this->onConsecutiveCalls( + false, // don't want to reset the entire database + false // don't want to create a new user + )); + + // We override the standard helper with our mock + $command->getHelperSet()->set($question, 'question'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + ]); + + $this->assertContains('Checking system requirements.', $tester->getDisplay()); + $this->assertContains('Setting up database.', $tester->getDisplay()); + $this->assertContains('Administration setup.', $tester->getDisplay()); + $this->assertContains('Config setup.', $tester->getDisplay()); + $this->assertContains('Installing assets.', $tester->getDisplay()); + + $this->assertContains('Creating schema', $tester->getDisplay()); + } + + public function testRunInstallCommandNoInteraction() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new InstallCommandMock()); + + $command = $application->find('wallabag:install'); + + // We mock the QuestionHelper + $question = $this->getMockBuilder('Symfony\Component\Console\Helper\QuestionHelper') + ->disableOriginalConstructor() + ->getMock(); + $question->expects($this->any()) + ->method('ask') + ->will($this->returnValue('yes_'.uniqid('', true))); + + // We override the standard helper with our mock + $command->getHelperSet()->set($question, 'question'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + '--no-interaction' => true, + ]); + + $this->assertContains('Checking system requirements.', $tester->getDisplay()); + $this->assertContains('Setting up database.', $tester->getDisplay()); + $this->assertContains('Administration setup.', $tester->getDisplay()); + $this->assertContains('Config setup.', $tester->getDisplay()); + $this->assertContains('Installing assets.', $tester->getDisplay()); + } +} diff --git a/tests/Wallabag/CoreBundle/Command/TagAllCommandTest.php b/tests/Wallabag/CoreBundle/Command/TagAllCommandTest.php new file mode 100644 index 00000000..ec31708f --- /dev/null +++ b/tests/Wallabag/CoreBundle/Command/TagAllCommandTest.php @@ -0,0 +1,60 @@ +getClient()->getKernel()); + $application->add(new TagAllCommand()); + + $command = $application->find('wallabag:tag:all'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + ]); + } + + public function testRunTagAllCommandWithBadUsername() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new TagAllCommand()); + + $command = $application->find('wallabag:tag:all'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + 'username' => 'unknown', + ]); + + $this->assertContains('User "unknown" not found', $tester->getDisplay()); + } + + public function testRunTagAllCommand() + { + $application = new Application($this->getClient()->getKernel()); + $application->add(new TagAllCommand()); + + $command = $application->find('wallabag:tag:all'); + + $tester = new CommandTester($command); + $tester->execute([ + 'command' => $command->getName(), + 'username' => 'admin', + ]); + + $this->assertContains('Tagging entries for user « admin »... Done', $tester->getDisplay()); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/ConfigControllerTest.php b/tests/Wallabag/CoreBundle/Controller/ConfigControllerTest.php new file mode 100644 index 00000000..7193f9b0 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/ConfigControllerTest.php @@ -0,0 +1,652 @@ +getClient(); + + $client->request('GET', '/new'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('login', $client->getResponse()->headers->get('location')); + } + + public function testIndex() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $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=update_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->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=config_save]')->form(); + + $data = [ + 'config[theme]' => 'baggy', + 'config[items_per_page]' => '30', + 'config[reading_speed]' => '0.5', + 'config[language]' => 'en', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(['_text'])); + $this->assertContains('flashes.config.notice.config_saved', $alert[0]); + } + + public function testChangeReadingSpeed() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + $form = $crawler->filter('button[id=submit-filter]')->form(); + $dataFilters = [ + 'entry_filter[readingTime][right_number]' => 22, + 'entry_filter[readingTime][left_number]' => 22, + ]; + $crawler = $client->submit($form, $dataFilters); + $this->assertCount(1, $crawler->filter('div[class=entry]')); + + // Change reading speed + $crawler = $client->request('GET', '/config'); + $form = $crawler->filter('button[id=config_save]')->form(); + $data = [ + 'config[reading_speed]' => '2', + ]; + $client->submit($form, $data); + + // Is the entry still available via filters? + $crawler = $client->request('GET', '/unread/list'); + $form = $crawler->filter('button[id=submit-filter]')->form(); + $crawler = $client->submit($form, $dataFilters); + $this->assertCount(0, $crawler->filter('div[class=entry]')); + + // Restore old configuration + $crawler = $client->request('GET', '/config'); + $form = $crawler->filter('button[id=config_save]')->form(); + $data = [ + 'config[reading_speed]' => '0.5', + ]; + $client->submit($form, $data); + } + + public function dataForUpdateFailed() + { + return [ + [[ + 'config[theme]' => 'baggy', + 'config[items_per_page]' => '', + 'config[language]' => 'en', + ]], + ]; + } + + /** + * @dataProvider dataForUpdateFailed + */ + public function testUpdateFailed($data) + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=config_save]')->form(); + + $crawler = $client->submit($form, $data); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertGreaterThan(1, $alert = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('This value should not be blank', $alert[0]); + } + + public function dataForChangePasswordFailed() + { + return [ + [ + [ + 'change_passwd[old_password]' => 'material', + 'change_passwd[new_password][first]' => '', + 'change_passwd[new_password][second]' => '', + ], + 'validator.password_wrong_value', + ], + [ + [ + 'change_passwd[old_password]' => 'mypassword', + 'change_passwd[new_password][first]' => '', + 'change_passwd[new_password][second]' => '', + ], + 'This value should not be blank', + ], + [ + [ + 'change_passwd[old_password]' => 'mypassword', + 'change_passwd[new_password][first]' => 'hop', + 'change_passwd[new_password][second]' => '', + ], + 'validator.password_must_match', + ], + [ + [ + 'change_passwd[old_password]' => 'mypassword', + 'change_passwd[new_password][first]' => 'hop', + 'change_passwd[new_password][second]' => 'hop', + ], + 'validator.password_too_short', + ], + ]; + } + + /** + * @dataProvider dataForChangePasswordFailed + */ + public function testChangePasswordFailed($data, $expectedMessage) + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=change_passwd_save]')->form(); + + $crawler = $client->submit($form, $data); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertGreaterThan(1, $alert = $crawler->filter('body')->extract(['_text'])); + $this->assertContains($expectedMessage, $alert[0]); + } + + public function testChangePassword() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=change_passwd_save]')->form(); + + $data = [ + 'change_passwd[old_password]' => 'mypassword', + 'change_passwd[new_password][first]' => 'mypassword', + 'change_passwd[new_password][second]' => 'mypassword', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(['_text'])); + $this->assertContains('flashes.config.notice.password_updated', $alert[0]); + } + + public function dataForUserFailed() + { + return [ + [ + [ + 'update_user[name]' => '', + 'update_user[email]' => '', + ], + 'fos_user.email.blank', + ], + [ + [ + 'update_user[name]' => '', + 'update_user[email]' => 'test', + ], + 'fos_user.email.invalid', + ], + ]; + } + + /** + * @dataProvider dataForUserFailed + */ + public function testUserFailed($data, $expectedMessage) + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=update_user_save]')->form(); + + $crawler = $client->submit($form, $data); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertGreaterThan(1, $alert = $crawler->filter('body')->extract(['_text'])); + $this->assertContains($expectedMessage, $alert[0]); + } + + public function testUserUpdate() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=update_user_save]')->form(); + + $data = [ + 'update_user[name]' => 'new name', + 'update_user[email]' => 'admin@wallabag.io', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $alert = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('flashes.config.notice.user_updated', $alert[0]); + } + + public function dataForNewUserFailed() + { + return [ + [ + [ + 'new_user[username]' => '', + 'new_user[plainPassword][first]' => '', + 'new_user[plainPassword][second]' => '', + 'new_user[email]' => '', + ], + 'fos_user.username.blank', + ], + [ + [ + 'new_user[username]' => 'a', + 'new_user[plainPassword][first]' => 'mypassword', + 'new_user[plainPassword][second]' => 'mypassword', + 'new_user[email]' => '', + ], + 'fos_user.username.short', + ], + [ + [ + 'new_user[username]' => 'wallace', + 'new_user[plainPassword][first]' => 'mypassword', + 'new_user[plainPassword][second]' => 'mypassword', + 'new_user[email]' => 'test', + ], + 'fos_user.email.invalid', + ], + [ + [ + 'new_user[username]' => 'admin', + 'new_user[plainPassword][first]' => 'wallacewallace', + 'new_user[plainPassword][second]' => 'wallacewallace', + 'new_user[email]' => 'wallace@wallace.me', + ], + 'fos_user.username.already_used', + ], + [ + [ + 'new_user[username]' => 'wallace', + 'new_user[plainPassword][first]' => 'mypassword1', + 'new_user[plainPassword][second]' => 'mypassword2', + 'new_user[email]' => 'wallace@wallace.me', + ], + 'validator.password_must_match', + ], + ]; + } + + /** + * @dataProvider dataForNewUserFailed + */ + public function testNewUserFailed($data, $expectedMessage) + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=new_user_save]')->form(); + + $crawler = $client->submit($form, $data); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertGreaterThan(1, $alert = $crawler->filter('body')->extract(['_text'])); + $this->assertContains($expectedMessage, $alert[0]); + } + + public function testNewUserCreated() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=new_user_save]')->form(); + + $data = [ + 'new_user[username]' => 'wallace', + 'new_user[plainPassword][first]' => 'wallace1', + 'new_user[plainPassword][second]' => 'wallace1', + 'new_user[email]' => 'wallace@wallace.me', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(['_text'])); + $this->assertContains('flashes.config.notice.user_added', $alert[0]); + + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $user = $em + ->getRepository('WallabagUserBundle:User') + ->findOneByUsername('wallace'); + + $this->assertTrue(false !== $user); + $this->assertTrue($user->isEnabled()); + $this->assertEquals('material', $user->getConfig()->getTheme()); + $this->assertEquals(12, $user->getConfig()->getItemsPerPage()); + $this->assertEquals(50, $user->getConfig()->getRssLimit()); + $this->assertEquals('en', $user->getConfig()->getLanguage()); + $this->assertEquals(1, $user->getConfig()->getReadingSpeed()); + } + + public function testRssUpdateResetToken() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + // reset the token + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $user = $em + ->getRepository('WallabagUserBundle: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(['_text'])); + $this->assertContains('config.form_rss.no_token', $body[0]); + + $client->request('GET', '/generate-token'); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertNotContains('config.form_rss.no_token', $body[0]); + } + + public function testGenerateTokenAjax() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request( + 'GET', + '/generate-token', + [], + [], + ['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'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=rss_config_save]')->form(); + + $data = [ + '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(['_text'])); + $this->assertContains('flashes.config.notice.rss_updated', $alert[0]); + } + + public function dataForRssFailed() + { + return [ + [ + [ + 'rss_config[rss_limit]' => 0, + ], + 'This value should be 1 or more.', + ], + [ + [ + 'rss_config[rss_limit]' => 1000000000000, + ], + 'validator.rss_limit_too_hight', + ], + ]; + } + + /** + * @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(['_text'])); + $this->assertContains($expectedMessage, $alert[0]); + } + + public function testTaggingRuleCreation() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertTrue($client->getResponse()->isSuccessful()); + + $form = $crawler->filter('button[id=tagging_rule_save]')->form(); + + $data = [ + 'tagging_rule[rule]' => 'readingTime <= 3', + 'tagging_rule[tags]' => 'short reading', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(['_text'])); + $this->assertContains('flashes.config.notice.tagging_rules_updated', $alert[0]); + + $deleteLink = $crawler->filter('.delete')->last()->link(); + + $crawler = $client->click($deleteLink); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + $this->assertGreaterThan(1, $alert = $crawler->filter('div.messages.success')->extract(['_text'])); + $this->assertContains('flashes.config.notice.tagging_rules_deleted', $alert[0]); + } + + public function dataForTaggingRuleFailed() + { + return [ + [ + [ + 'tagging_rule[rule]' => 'unknownVar <= 3', + 'tagging_rule[tags]' => 'cool tag', + ], + [ + 'The variable', + 'does not exist.', + ], + ], + [ + [ + 'tagging_rule[rule]' => 'length(domainName) <= 42', + 'tagging_rule[tags]' => 'cool tag', + ], + [ + 'The operator', + 'does not exist.', + ], + ], + ]; + } + + /** + * @dataProvider dataForTaggingRuleFailed + */ + public function testTaggingRuleCreationFail($data, $messages) + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/config'); + + $this->assertTrue($client->getResponse()->isSuccessful()); + + $form = $crawler->filter('button[id=tagging_rule_save]')->form(); + + $crawler = $client->submit($form, $data); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + + foreach ($messages as $message) { + $this->assertContains($message, $body[0]); + } + } + + public function testDeletingTaggingRuleFromAnOtherUser() + { + $this->logInAs('bob'); + $client = $this->getClient(); + + $rule = $client->getContainer()->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:TaggingRule') + ->findAll()[0]; + + $crawler = $client->request('GET', '/tagging-rule/delete/'.$rule->getId()); + + $this->assertEquals(403, $client->getResponse()->getStatusCode()); + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('You can not access this tagging rule', $body[0]); + } + + public function testDemoMode() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $config = $client->getContainer()->get('craue_config'); + $config->set('demo_mode_enabled', 1); + $config->set('demo_mode_username', 'admin'); + + $crawler = $client->request('GET', '/config'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[id=change_passwd_save]')->form(); + + $data = [ + 'change_passwd[old_password]' => 'mypassword', + 'change_passwd[new_password][first]' => 'mypassword', + 'change_passwd[new_password][second]' => 'mypassword', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('flashes.config.notice.password_not_updated_demo', $client->getContainer()->get('session')->getFlashBag()->get('notice')[0]); + + $config->set('demo_mode_enabled', 0); + $config->set('demo_mode_username', 'wallabag'); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/DeveloperControllerTest.php b/tests/Wallabag/CoreBundle/Controller/DeveloperControllerTest.php new file mode 100644 index 00000000..79452ace --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/DeveloperControllerTest.php @@ -0,0 +1,71 @@ +logInAs('admin'); + $client = $this->getClient(); + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $nbClients = $em->getRepository('WallabagApiBundle:Client')->findAll(); + + $crawler = $client->request('GET', '/developer/client/create'); + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[type=submit]')->form(); + + $client->submit($form); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $newNbClients = $em->getRepository('WallabagApiBundle:Client')->findAll(); + $this->assertGreaterThan(count($nbClients), count($newNbClients)); + } + + public function testListingClient() + { + $this->logInAs('admin'); + $client = $this->getClient(); + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $nbClients = $em->getRepository('WallabagApiBundle:Client')->findAll(); + + $crawler = $client->request('GET', '/developer'); + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertEquals(count($nbClients), $crawler->filter('ul[class=collapsible] li')->count()); + } + + public function testDeveloperHowto() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/developer/howto/first-app'); + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testRemoveClient() + { + $this->logInAs('admin'); + $client = $this->getClient(); + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $nbClients = $em->getRepository('WallabagApiBundle:Client')->findAll(); + + $crawler = $client->request('GET', '/developer'); + + $link = $crawler + ->filter('div[class=collapsible-body] p a') + ->eq(0) + ->link() + ; + + $client->click($link); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $newNbClients = $em->getRepository('WallabagApiBundle:Client')->findAll(); + $this->assertGreaterThan(count($newNbClients), count($nbClients)); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/EntryControllerTest.php b/tests/Wallabag/CoreBundle/Controller/EntryControllerTest.php new file mode 100644 index 00000000..bea771bc --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/EntryControllerTest.php @@ -0,0 +1,665 @@ +getClient(); + + $client->request('GET', '/new'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('login', $client->getResponse()->headers->get('location')); + } + + public function testQuickstart() + { + $this->logInAs('empty'); + $client = $this->getClient(); + + $client->request('GET', '/unread/list'); + $crawler = $client->followRedirect(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('quickstart.intro.paragraph_1', $body[0]); + + // Test if quickstart is disabled when user has 1 entry + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('form[name=entry]')->form(); + + $data = [ + 'entry[url]' => $this->url, + ]; + + $client->submit($form, $data); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $client->followRedirect(); + + $crawler = $client->request('GET', '/unread/list'); + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('entry.list.number_on_the_page', $body[0]); + } + + public function testGetNew() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertCount(1, $crawler->filter('input[type=url]')); + $this->assertCount(1, $crawler->filter('form[name=entry]')); + } + + public function testPostNewViaBookmarklet() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/'); + + $this->assertCount(4, $crawler->filter('div[class=entry]')); + + // Good URL + $client->request('GET', '/bookmarklet', ['url' => $this->url]); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $client->followRedirect(); + $crawler = $client->request('GET', '/'); + $this->assertCount(5, $crawler->filter('div[class=entry]')); + + $em = $client->getContainer() + ->get('doctrine.orm.entity_manager'); + $entry = $em + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + $em->remove($entry); + $em->flush(); + } + + public function testPostNewEmpty() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('form[name=entry]')->form(); + + $crawler = $client->submit($form); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertCount(1, $alert = $crawler->filter('form ul li')->extract(['_text'])); + $this->assertEquals('This value should not be blank.', $alert[0]); + } + + /** + * This test will require an internet connection. + */ + public function testPostNewOk() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('form[name=entry]')->form(); + + $data = [ + 'entry[url]' => $this->url, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $this->assertInstanceOf('Wallabag\CoreBundle\Entity\Entry', $content); + $this->assertEquals($this->url, $content->getUrl()); + $this->assertContains('Google', $content->getTitle()); + } + + public function testPostNewOkUrlExist() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('form[name=entry]')->form(); + + $data = [ + 'entry[url]' => $this->url, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('/view/', $client->getResponse()->getTargetUrl()); + } + + /** + * This test will require an internet connection. + */ + public function testPostNewThatWillBeTagged() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('form[name=entry]')->form(); + + $data = [ + 'entry[url]' => $url = 'https://github.com/wallabag/wallabag', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('/', $client->getResponse()->getTargetUrl()); + + $em = $client->getContainer() + ->get('doctrine.orm.entity_manager'); + $entry = $em + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUrl($url); + $tags = $entry->getTags(); + + $this->assertCount(1, $tags); + $this->assertEquals('wallabag', $tags[0]->getLabel()); + + $em->remove($entry); + $em->flush(); + + // and now re-submit it to test the cascade persistence for tags after entry removal + // related https://github.com/wallabag/wallabag/issues/2121 + $crawler = $client->request('GET', '/new'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('form[name=entry]')->form(); + + $data = [ + 'entry[url]' => $url = 'https://github.com/wallabag/wallabag/tree/master', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('/', $client->getResponse()->getTargetUrl()); + + $entry = $em + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUrl($url); + + $tags = $entry->getTags(); + + $this->assertCount(1, $tags); + $this->assertEquals('wallabag', $tags[0]->getLabel()); + + $em->remove($entry); + $em->flush(); + } + + public function testArchive() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/archive/list'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testStarred() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/starred/list'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testRangeException() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/all/list/900'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertEquals('/all/list', $client->getResponse()->getTargetUrl()); + } + + /** + * @depends testPostNewOk + */ + public function testView() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $crawler = $client->request('GET', '/view/'.$content->getId()); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains($content->getTitle(), $body[0]); + } + + /** + * @depends testPostNewOk + * + * This test will require an internet connection. + */ + public function testReload() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + // empty content + $content->setContent(''); + $client->getContainer()->get('doctrine.orm.entity_manager')->persist($content); + $client->getContainer()->get('doctrine.orm.entity_manager')->flush(); + + $client->request('GET', '/reload/'.$content->getId()); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $this->assertNotEmpty($content->getContent()); + } + + public function testEdit() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $crawler = $client->request('GET', '/edit/'.$content->getId()); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $this->assertCount(1, $crawler->filter('input[id=entry_title]')); + $this->assertCount(1, $crawler->filter('button[id=entry_save]')); + } + + public function testEditUpdate() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $crawler = $client->request('GET', '/edit/'.$content->getId()); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $form = $crawler->filter('button[type=submit]')->form(); + + $data = [ + 'entry[title]' => 'My updated title hehe :)', + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $alert = $crawler->filter('div[id=article] h1')->extract(['_text'])); + $this->assertContains('My updated title hehe :)', $alert[0]); + } + + public function testToggleArchive() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $client->request('GET', '/archive/'.$content->getId()); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $res = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->find($content->getId()); + + $this->assertEquals($res->isArchived(), true); + } + + public function testToggleStar() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $client->request('GET', '/star/'.$content->getId()); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $res = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneById($content->getId()); + + $this->assertEquals($res->isStarred(), true); + } + + public function testDelete() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId($this->url, $this->getLoggedInUserId()); + + $client->request('GET', '/delete/'.$content->getId()); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $client->request('GET', '/delete/'.$content->getId()); + + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + } + + /** + * It will create a new entry. + * Browse to it. + * Then remove it. + * + * And it'll check that user won't be redirected to the view page of the content when it had been removed + */ + public function testViewAndDelete() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + // add a new content to be removed later + $user = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagUserBundle:User') + ->findOneByUserName('admin'); + + $content = new Entry($user); + $content->setUrl('http://1.1.1.1/entry'); + $content->setReadingTime(12); + $content->setDomainName('domain.io'); + $content->setMimetype('text/html'); + $content->setTitle('test title entry'); + $content->setContent('This is my content /o/'); + $content->setArchived(true); + $content->setLanguage('fr'); + + $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->persist($content); + $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->flush(); + + $client->request('GET', '/view/'.$content->getId()); + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $client->request('GET', '/delete/'.$content->getId()); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $client->followRedirect(); + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testViewOtherUserEntry() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('bob'); + + $client->request('GET', '/view/'.$content->getId()); + + $this->assertEquals(403, $client->getResponse()->getStatusCode()); + } + + public function testFilterOnReadingTime() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + + $form = $crawler->filter('button[id=submit-filter]')->form(); + + $data = [ + 'entry_filter[readingTime][right_number]' => 22, + 'entry_filter[readingTime][left_number]' => 22, + ]; + + $crawler = $client->submit($form, $data); + + $this->assertCount(1, $crawler->filter('div[class=entry]')); + } + + public function testFilterOnUnreadStatus() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/all/list'); + + $form = $crawler->filter('button[id=submit-filter]')->form(); + + $data = [ + 'entry_filter[isUnread]' => true, + ]; + + $crawler = $client->submit($form, $data); + + $this->assertCount(4, $crawler->filter('div[class=entry]')); + } + + public function testFilterOnCreationDate() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + + $form = $crawler->filter('button[id=submit-filter]')->form(); + + $data = [ + 'entry_filter[createdAt][left_date]' => date('d/m/Y'), + 'entry_filter[createdAt][right_date]' => date('d/m/Y', strtotime('+1 day')), + ]; + + $crawler = $client->submit($form, $data); + + $this->assertCount(5, $crawler->filter('div[class=entry]')); + + $data = [ + 'entry_filter[createdAt][left_date]' => date('d/m/Y'), + 'entry_filter[createdAt][right_date]' => date('d/m/Y'), + ]; + + $crawler = $client->submit($form, $data); + + $this->assertCount(5, $crawler->filter('div[class=entry]')); + + $data = [ + 'entry_filter[createdAt][left_date]' => '01/01/1970', + 'entry_filter[createdAt][right_date]' => '01/01/1970', + ]; + + $crawler = $client->submit($form, $data); + + $this->assertCount(0, $crawler->filter('div[class=entry]')); + } + + public function testPaginationWithFilter() + { + $this->logInAs('admin'); + $client = $this->getClient(); + $crawler = $client->request('GET', '/config'); + + $form = $crawler->filter('button[id=config_save]')->form(); + + $data = [ + 'config[items_per_page]' => '1', + ]; + + $client->submit($form, $data); + + $parameters = '?entry_filter%5BreadingTime%5D%5Bleft_number%5D=&entry_filter%5BreadingTime%5D%5Bright_number%5D='; + + $client->request('GET', 'unread/list'.$parameters); + + $this->assertContains($parameters, $client->getResponse()->getContent()); + + // reset pagination + $crawler = $client->request('GET', '/config'); + $form = $crawler->filter('button[id=config_save]')->form(); + $data = [ + 'config[items_per_page]' => '12', + ]; + $client->submit($form, $data); + } + + public function testFilterOnDomainName() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + $form = $crawler->filter('button[id=submit-filter]')->form(); + $data = [ + 'entry_filter[domainName]' => 'domain', + ]; + + $crawler = $client->submit($form, $data); + $this->assertCount(5, $crawler->filter('div[class=entry]')); + + $form = $crawler->filter('button[id=submit-filter]')->form(); + $data = [ + 'entry_filter[domainName]' => 'wallabag', + ]; + + $crawler = $client->submit($form, $data); + $this->assertCount(0, $crawler->filter('div[class=entry]')); + } + + public function testFilterOnStatus() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + $form = $crawler->filter('button[id=submit-filter]')->form(); + $form['entry_filter[isArchived]']->tick(); + $form['entry_filter[isStarred]']->untick(); + + $crawler = $client->submit($form); + $this->assertCount(1, $crawler->filter('div[class=entry]')); + + $form = $crawler->filter('button[id=submit-filter]')->form(); + $form['entry_filter[isArchived]']->untick(); + $form['entry_filter[isStarred]']->tick(); + + $crawler = $client->submit($form); + $this->assertCount(1, $crawler->filter('div[class=entry]')); + } + + public function testPreviewPictureFilter() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + $form = $crawler->filter('button[id=submit-filter]')->form(); + $form['entry_filter[previewPicture]']->tick(); + + $crawler = $client->submit($form); + $this->assertCount(1, $crawler->filter('div[class=entry]')); + } + + public function testFilterOnLanguage() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/unread/list'); + $form = $crawler->filter('button[id=submit-filter]')->form(); + $data = [ + 'entry_filter[language]' => 'fr', + ]; + + $crawler = $client->submit($form, $data); + $this->assertCount(2, $crawler->filter('div[class=entry]')); + + $form = $crawler->filter('button[id=submit-filter]')->form(); + $data = [ + 'entry_filter[language]' => 'en', + ]; + + $crawler = $client->submit($form, $data); + $this->assertCount(2, $crawler->filter('div[class=entry]')); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/ExportControllerTest.php b/tests/Wallabag/CoreBundle/Controller/ExportControllerTest.php new file mode 100644 index 00000000..b22156c3 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/ExportControllerTest.php @@ -0,0 +1,251 @@ +getClient(); + + $client->request('GET', '/export/unread.csv'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('login', $client->getResponse()->headers->get('location')); + } + + public function testUnknownCategoryExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/export/awesomeness.epub'); + + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + } + + public function testUnknownFormatExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/export/unread.xslx'); + + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + } + + public function testUnsupportedFormatExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/export/unread.doc'); + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('admin'); + + $client->request('GET', '/export/'.$content->getId().'.doc'); + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + } + + public function testBadEntryId() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/export/0.mobi'); + + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + } + + public function testEpubExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + ob_start(); + $crawler = $client->request('GET', '/export/archive.epub'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('application/epub+zip', $headers->get('content-type')); + $this->assertEquals('attachment; filename="Archive articles.epub"', $headers->get('content-disposition')); + $this->assertEquals('binary', $headers->get('content-transfer-encoding')); + } + + public function testMobiExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('admin'); + + ob_start(); + $crawler = $client->request('GET', '/export/'.$content->getId().'.mobi'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('application/x-mobipocket-ebook', $headers->get('content-type')); + $this->assertEquals('attachment; filename="'.preg_replace('/[^A-Za-z0-9\-]/', '', $content->getTitle()).'.mobi"', $headers->get('content-disposition')); + $this->assertEquals('binary', $headers->get('content-transfer-encoding')); + } + + public function testPdfExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + ob_start(); + $crawler = $client->request('GET', '/export/all.pdf'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('application/pdf', $headers->get('content-type')); + $this->assertEquals('attachment; filename="All articles.pdf"', $headers->get('content-disposition')); + $this->assertEquals('binary', $headers->get('content-transfer-encoding')); + } + + public function testTxtExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + ob_start(); + $crawler = $client->request('GET', '/export/all.txt'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('text/plain; charset=UTF-8', $headers->get('content-type')); + $this->assertEquals('attachment; filename="All articles.txt"', $headers->get('content-disposition')); + $this->assertEquals('UTF-8', $headers->get('content-transfer-encoding')); + } + + public function testCsvExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + // to be sure results are the same + $contentInDB = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->createQueryBuilder('e') + ->leftJoin('e.user', 'u') + ->where('u.username = :username')->setParameter('username', 'admin') + ->andWhere('e.isArchived = true') + ->getQuery() + ->getArrayResult(); + + ob_start(); + $crawler = $client->request('GET', '/export/archive.csv'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('application/csv', $headers->get('content-type')); + $this->assertEquals('attachment; filename="Archive articles.csv"', $headers->get('content-disposition')); + $this->assertEquals('UTF-8', $headers->get('content-transfer-encoding')); + + $csv = str_getcsv($client->getResponse()->getContent(), "\n"); + + $this->assertGreaterThan(1, $csv); + // +1 for title line + $this->assertEquals(count($contentInDB) + 1, count($csv)); + $this->assertEquals('Title;URL;Content;Tags;"MIME Type";Language', $csv[0]); + } + + public function testJsonExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + // to be sure results are the same + $contentInDB = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->createQueryBuilder('e') + ->leftJoin('e.user', 'u') + ->where('u.username = :username')->setParameter('username', 'admin') + ->getQuery() + ->getArrayResult(); + + ob_start(); + $crawler = $client->request('GET', '/export/all.json'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('application/json', $headers->get('content-type')); + $this->assertEquals('attachment; filename="All articles.json"', $headers->get('content-disposition')); + $this->assertEquals('UTF-8', $headers->get('content-transfer-encoding')); + + $content = json_decode($client->getResponse()->getContent(), true); + $this->assertEquals(count($contentInDB), count($content)); + $this->assertArrayHasKey('id', $content[0]); + $this->assertArrayHasKey('title', $content[0]); + $this->assertArrayHasKey('url', $content[0]); + $this->assertArrayHasKey('is_archived', $content[0]); + $this->assertArrayHasKey('is_starred', $content[0]); + $this->assertArrayHasKey('content', $content[0]); + $this->assertArrayHasKey('mimetype', $content[0]); + $this->assertArrayHasKey('language', $content[0]); + $this->assertArrayHasKey('reading_time', $content[0]); + $this->assertArrayHasKey('domain_name', $content[0]); + $this->assertArrayHasKey('tags', $content[0]); + } + + public function testXmlExport() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + // to be sure results are the same + $contentInDB = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->createQueryBuilder('e') + ->leftJoin('e.user', 'u') + ->where('u.username = :username')->setParameter('username', 'admin') + ->andWhere('e.isArchived = false') + ->getQuery() + ->getArrayResult(); + + ob_start(); + $crawler = $client->request('GET', '/export/unread.xml'); + ob_end_clean(); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + + $headers = $client->getResponse()->headers; + $this->assertEquals('application/xml', $headers->get('content-type')); + $this->assertEquals('attachment; filename="Unread articles.xml"', $headers->get('content-disposition')); + $this->assertEquals('UTF-8', $headers->get('content-transfer-encoding')); + + $content = new \SimpleXMLElement($client->getResponse()->getContent()); + $this->assertGreaterThan(0, $content->count()); + $this->assertEquals(count($contentInDB), $content->count()); + $this->assertNotEmpty('id', (string) $content->entry[0]->id); + $this->assertNotEmpty('title', (string) $content->entry[0]->title); + $this->assertNotEmpty('url', (string) $content->entry[0]->url); + $this->assertNotEmpty('content', (string) $content->entry[0]->content); + $this->assertNotEmpty('domain_name', (string) $content->entry[0]->domain_name); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/RssControllerTest.php b/tests/Wallabag/CoreBundle/Controller/RssControllerTest.php new file mode 100644 index 00000000..fb6fe06a --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/RssControllerTest.php @@ -0,0 +1,126 @@ +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 [ + [ + '/admin/YZIOAUZIAO/unread.xml', + ], + [ + '/wallace/YZIOAUZIAO/starred.xml', + ], + [ + '/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('WallabagUserBundle: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('WallabagUserBundle: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('WallabagUserBundle: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()); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/SecurityControllerTest.php b/tests/Wallabag/CoreBundle/Controller/SecurityControllerTest.php new file mode 100644 index 00000000..f503ff4b --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/SecurityControllerTest.php @@ -0,0 +1,72 @@ +logInAs('admin'); + $client = $this->getClient(); + $client->followRedirects(); + + $crawler = $client->request('GET', '/config'); + $this->assertContains('config.form_rss.description', $crawler->filter('body')->extract(['_text'])[0]); + } + + public function testLoginWith2Factor() + { + $client = $this->getClient(); + + if (!$client->getContainer()->getParameter('twofactor_auth')) { + $this->markTestSkipped('twofactor_auth is not enabled.'); + + return; + } + + $client->followRedirects(); + + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $user = $em + ->getRepository('WallabagUserBundle:User') + ->findOneByUsername('admin'); + $user->setTwoFactorAuthentication(true); + $em->persist($user); + $em->flush(); + + $this->logInAs('admin'); + $crawler = $client->request('GET', '/config'); + $this->assertContains('scheb_two_factor.trusted', $crawler->filter('body')->extract(['_text'])[0]); + + // restore user + $user = $em + ->getRepository('WallabagUserBundle:User') + ->findOneByUsername('admin'); + $user->setTwoFactorAuthentication(false); + $em->persist($user); + $em->flush(); + } + + public function testTrustedComputer() + { + $client = $this->getClient(); + + if (!$client->getContainer()->getParameter('twofactor_auth')) { + $this->markTestSkipped('twofactor_auth is not enabled.'); + + return; + } + + $em = $client->getContainer()->get('doctrine.orm.entity_manager'); + $user = $em + ->getRepository('WallabagUserBundle:User') + ->findOneByUsername('admin'); + + $date = new \DateTime(); + $user->addTrustedComputer('ABCDEF', $date->add(new \DateInterval('P1M'))); + $this->assertTrue($user->isTrustedComputer('ABCDEF')); + $this->assertFalse($user->isTrustedComputer('FEDCBA')); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/SettingsControllerTest.php b/tests/Wallabag/CoreBundle/Controller/SettingsControllerTest.php new file mode 100644 index 00000000..9b8b5702 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/SettingsControllerTest.php @@ -0,0 +1,32 @@ +logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/settings'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testSettingsWithNormalUser() + { + $this->logInAs('bob'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/settings'); + + $this->assertEquals(403, $client->getResponse()->getStatusCode()); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/StaticControllerTest.php b/tests/Wallabag/CoreBundle/Controller/StaticControllerTest.php new file mode 100644 index 00000000..98a37b50 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/StaticControllerTest.php @@ -0,0 +1,28 @@ +logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/about'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testHowto() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/howto'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } +} diff --git a/tests/Wallabag/CoreBundle/Controller/TagControllerTest.php b/tests/Wallabag/CoreBundle/Controller/TagControllerTest.php new file mode 100644 index 00000000..a019d36c --- /dev/null +++ b/tests/Wallabag/CoreBundle/Controller/TagControllerTest.php @@ -0,0 +1,128 @@ +logInAs('admin'); + $client = $this->getClient(); + + $client->request('GET', '/tag/list'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + } + + public function testAddTagToEntry() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $entry = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('admin'); + + $crawler = $client->request('GET', '/view/'.$entry->getId()); + + $form = $crawler->filter('form[name=tag]')->form(); + + $data = [ + 'tag[label]' => $this->tagName, + ]; + + $client->submit($form, $data); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $this->assertEquals(1, count($entry->getTags())); + + # tag already exists and already assigned + $client->submit($form, $data); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $newEntry = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->find($entry->getId()); + + $this->assertEquals(1, count($newEntry->getTags())); + + # tag already exists but still not assigned to this entry + $data = [ + 'tag[label]' => 'foo', + ]; + + $client->submit($form, $data); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $newEntry = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->find($entry->getId()); + + $this->assertEquals(2, count($newEntry->getTags())); + } + + public function testAddMultipleTagToEntry() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $entry = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('admin'); + + $crawler = $client->request('GET', '/view/'.$entry->getId()); + + $form = $crawler->filter('form[name=tag]')->form(); + + $data = [ + 'tag[label]' => 'foo2, bar2', + ]; + + $client->submit($form, $data); + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $newEntry = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->find($entry->getId()); + + $tags = $newEntry->getTags()->toArray(); + $this->assertGreaterThanOrEqual(2, count($tags)); + $this->assertNotEquals(false, array_search('foo2', $tags), 'Tag foo2 is assigned to the entry'); + $this->assertNotEquals(false, array_search('bar2', $tags), 'Tag bar2 is assigned to the entry'); + } + + public function testRemoveTagFromEntry() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $entry = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findOneByUsernameAndNotArchived('admin'); + + $tag = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Tag') + ->findOneByEntryAndTagLabel($entry, $this->tagName); + + $client->request('GET', '/remove-tag/'.$entry->getId().'/'.$tag->getId()); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $this->assertNotContains($this->tagName, $entry->getTags()); + + $client->request('GET', '/remove-tag/'.$entry->getId().'/'.$tag->getId()); + + $this->assertEquals(404, $client->getResponse()->getStatusCode()); + } +} diff --git a/tests/Wallabag/CoreBundle/EventListener/LocaleListenerTest.php b/tests/Wallabag/CoreBundle/EventListener/LocaleListenerTest.php new file mode 100644 index 00000000..2a7f9390 --- /dev/null +++ b/tests/Wallabag/CoreBundle/EventListener/LocaleListenerTest.php @@ -0,0 +1,82 @@ +getMock('Symfony\Component\HttpKernel\HttpKernelInterface'), $request, HttpKernelInterface::MASTER_REQUEST); + } + + public function testWithoutSession() + { + $request = Request::create('/'); + + $listener = new LocaleListener('fr'); + $event = $this->getEvent($request); + + $listener->onKernelRequest($event); + $this->assertEquals('en', $request->getLocale()); + } + + public function testWithPreviousSession() + { + $request = Request::create('/'); + // generate a previous session + $request->cookies->set('MOCKSESSID', 'foo'); + $request->setSession(new Session(new MockArraySessionStorage())); + + $listener = new LocaleListener('fr'); + $event = $this->getEvent($request); + + $listener->onKernelRequest($event); + $this->assertEquals('fr', $request->getLocale()); + } + + public function testLocaleFromRequestAttribute() + { + $request = Request::create('/'); + // generate a previous session + $request->cookies->set('MOCKSESSID', 'foo'); + $request->setSession(new Session(new MockArraySessionStorage())); + $request->attributes->set('_locale', 'es'); + + $listener = new LocaleListener('fr'); + $event = $this->getEvent($request); + + $listener->onKernelRequest($event); + $this->assertEquals('en', $request->getLocale()); + $this->assertEquals('es', $request->getSession()->get('_locale')); + } + + public function testSubscribedEvents() + { + $request = Request::create('/'); + // generate a previous session + $request->cookies->set('MOCKSESSID', 'foo'); + $request->setSession(new Session(new MockArraySessionStorage())); + + $listener = new LocaleListener('fr'); + $event = $this->getEvent($request); + + $dispatcher = new EventDispatcher(); + $dispatcher->addSubscriber($listener); + + $dispatcher->dispatch( + KernelEvents::REQUEST, + $event + ); + + $this->assertEquals('fr', $request->getLocale()); + } +} diff --git a/tests/Wallabag/CoreBundle/EventListener/RegistrationConfirmedListenerTest.php b/tests/Wallabag/CoreBundle/EventListener/RegistrationConfirmedListenerTest.php new file mode 100644 index 00000000..e45722fa --- /dev/null +++ b/tests/Wallabag/CoreBundle/EventListener/RegistrationConfirmedListenerTest.php @@ -0,0 +1,91 @@ +em = $this->getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $this->listener = new RegistrationConfirmedListener( + $this->em, + 'baggy', + 20, + 50, + 'fr' + ); + + $this->dispatcher = new EventDispatcher(); + $this->dispatcher->addSubscriber($this->listener); + + $this->request = Request::create('/'); + $this->response = Response::create(); + } + + public function testWithInvalidUser() + { + $user = new User(); + $user->setEnabled(false); + + $event = new FilterUserResponseEvent( + $user, + $this->request, + $this->response + ); + + $this->em->expects($this->never())->method('persist'); + $this->em->expects($this->never())->method('flush'); + + $this->dispatcher->dispatch( + FOSUserEvents::REGISTRATION_CONFIRMED, + $event + ); + } + + public function testWithValidUser() + { + $user = new User(); + $user->setEnabled(true); + + $event = new FilterUserResponseEvent( + $user, + $this->request, + $this->response + ); + + $config = new Config($user); + $config->setTheme('baggy'); + $config->setItemsPerPage(20); + $config->setRssLimit(50); + $config->setLanguage('fr'); + + $this->em->expects($this->once()) + ->method('persist') + ->will($this->returnValue($config)); + $this->em->expects($this->once()) + ->method('flush'); + + $this->dispatcher->dispatch( + FOSUserEvents::REGISTRATION_CONFIRMED, + $event + ); + } +} diff --git a/tests/Wallabag/CoreBundle/EventListener/UserLocaleListenerTest.php b/tests/Wallabag/CoreBundle/EventListener/UserLocaleListenerTest.php new file mode 100644 index 00000000..e9ac7c1d --- /dev/null +++ b/tests/Wallabag/CoreBundle/EventListener/UserLocaleListenerTest.php @@ -0,0 +1,58 @@ +setEnabled(true); + + $config = new Config($user); + $config->setLanguage('fr'); + + $user->setConfig($config); + + $userToken = new UsernamePasswordToken($user, '', 'test'); + $request = Request::create('/'); + $event = new InteractiveLoginEvent($request, $userToken); + + $listener->onInteractiveLogin($event); + + $this->assertEquals('fr', $session->get('_locale')); + } + + public function testWithoutLanguage() + { + $session = new Session(new MockArraySessionStorage()); + $listener = new UserLocaleListener($session); + + $user = new User(); + $user->setEnabled(true); + + $config = new Config($user); + + $user->setConfig($config); + + $userToken = new UsernamePasswordToken($user, '', 'test'); + $request = Request::create('/'); + $event = new InteractiveLoginEvent($request, $userToken); + + $listener->onInteractiveLogin($event); + + $this->assertEquals('', $session->get('_locale')); + } +} diff --git a/tests/Wallabag/CoreBundle/Form/DataTransformer/StringToListTransformerTest.php b/tests/Wallabag/CoreBundle/Form/DataTransformer/StringToListTransformerTest.php new file mode 100644 index 00000000..0ec98c1f --- /dev/null +++ b/tests/Wallabag/CoreBundle/Form/DataTransformer/StringToListTransformerTest.php @@ -0,0 +1,50 @@ +assertSame($expectedResult, $transformer->transform($inputData)); + } + + public function transformProvider() + { + return [ + [null, ''], + [[], ''], + [['single value'], 'single value'], + [['first value', 'second value'], 'first value,second value'], + ]; + } + + /** + * @dataProvider reverseTransformProvider + */ + public function testReverseTransformWithValidData($inputData, $expectedResult) + { + $transformer = new StringToListTransformer(); + + $this->assertSame($expectedResult, $transformer->reverseTransform($inputData)); + } + + public function reverseTransformProvider() + { + return [ + [null, null], + ['', []], + ['single value', ['single value']], + ['first value,second value', ['first value', 'second value']], + ['first value, second value', ['first value', 'second value']], + ['first value, , second value', ['first value', 'second value']], + ]; + } +} diff --git a/tests/Wallabag/CoreBundle/Helper/ContentProxyTest.php b/tests/Wallabag/CoreBundle/Helper/ContentProxyTest.php new file mode 100644 index 00000000..7abb0737 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Helper/ContentProxyTest.php @@ -0,0 +1,318 @@ +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' => '', + 'url' => '', + 'content_type' => '', + 'language' => '', + ]); + + $proxy = new ContentProxy($graby, $tagger, $this->getTagRepositoryMock(), $this->getLogger()); + $entry = $proxy->updateEntry(new Entry(new User()), 'http://user@:80'); + + $this->assertEquals('http://user@:80', $entry->getUrl()); + $this->assertEmpty($entry->getTitle()); + $this->assertEquals('

Unable to retrieve readable content.

', $entry->getContent()); + $this->assertEmpty($entry->getPreviewPicture()); + $this->assertEmpty($entry->getMimetype()); + $this->assertEmpty($entry->getLanguage()); + $this->assertEquals(0.0, $entry->getReadingTime()); + $this->assertEquals(false, $entry->getDomainName()); + } + + public function testWithEmptyContent() + { + $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' => '', + 'url' => '', + 'content_type' => '', + 'language' => '', + ]); + + $proxy = new ContentProxy($graby, $tagger, $this->getTagRepositoryMock(), $this->getLogger()); + $entry = $proxy->updateEntry(new Entry(new User()), 'http://0.0.0.0'); + + $this->assertEquals('http://0.0.0.0', $entry->getUrl()); + $this->assertEmpty($entry->getTitle()); + $this->assertEquals('

Unable to retrieve readable content.

', $entry->getContent()); + $this->assertEmpty($entry->getPreviewPicture()); + $this->assertEmpty($entry->getMimetype()); + $this->assertEmpty($entry->getLanguage()); + $this->assertEquals(0.0, $entry->getReadingTime()); + $this->assertEquals('0.0.0.0', $entry->getDomainName()); + } + + public function testWithEmptyContentButOG() + { + $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' => '', + 'url' => '', + 'content_type' => '', + 'language' => '', + 'open_graph' => [ + 'og_title' => 'my title', + 'og_description' => 'desc', + ], + ]); + + $proxy = new ContentProxy($graby, $tagger, $this->getTagRepositoryMock(), $this->getLogger()); + $entry = $proxy->updateEntry(new Entry(new User()), 'http://domain.io'); + + $this->assertEquals('http://domain.io', $entry->getUrl()); + $this->assertEquals('my title', $entry->getTitle()); + $this->assertEquals('

Unable to retrieve readable content.

But we found a short description:

desc', $entry->getContent()); + $this->assertEmpty($entry->getPreviewPicture()); + $this->assertEmpty($entry->getLanguage()); + $this->assertEmpty($entry->getMimetype()); + $this->assertEquals(0.0, $entry->getReadingTime()); + $this->assertEquals('domain.io', $entry->getDomainName()); + } + + public function testWithContent() + { + $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', + 'open_graph' => [ + 'og_title' => 'my OG title', + 'og_description' => 'OG desc', + 'og_image' => 'http://3.3.3.3/cover.jpg', + ], + ]); + + $proxy = new ContentProxy($graby, $tagger, $this->getTagRepositoryMock(), $this->getLogger()); + $entry = $proxy->updateEntry(new Entry(new User()), 'http://0.0.0.0'); + + $this->assertEquals('http://1.1.1.1', $entry->getUrl()); + $this->assertEquals('this is my title', $entry->getTitle()); + $this->assertContains('this is my content', $entry->getContent()); + $this->assertEquals('http://3.3.3.3/cover.jpg', $entry->getPreviewPicture()); + $this->assertEquals('text/html', $entry->getMimetype()); + $this->assertEquals('fr', $entry->getLanguage()); + $this->assertEquals(4.0, $entry->getReadingTime()); + $this->assertEquals('1.1.1.1', $entry->getDomainName()); + } + + public function testWithForcedContent() + { + $tagger = $this->getTaggerMock(); + $tagger->expects($this->once()) + ->method('tag'); + + $graby = $this->getMockBuilder('Graby\Graby')->getMock(); + + $proxy = new ContentProxy($graby, $tagger, $this->getTagRepositoryMock(), $this->getLogger()); + $entry = $proxy->updateEntry(new Entry(new User()), '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', + ]); + + $this->assertEquals('http://1.1.1.1', $entry->getUrl()); + $this->assertEquals('this is my title', $entry->getTitle()); + $this->assertContains('this is my content', $entry->getContent()); + $this->assertEquals('text/html', $entry->getMimetype()); + $this->assertEquals('fr', $entry->getLanguage()); + $this->assertEquals(4.0, $entry->getReadingTime()); + $this->assertEquals('1.1.1.1', $entry->getDomainName()); + } + + public function testTaggerThrowException() + { + $graby = $this->getMockBuilder('Graby\Graby') + ->disableOriginalConstructor() + ->getMock(); + + $tagger = $this->getTaggerMock(); + $tagger->expects($this->once()) + ->method('tag') + ->will($this->throwException(new \Exception())); + + $tagRepo = $this->getTagRepositoryMock(); + $proxy = new ContentProxy($graby, $tagger, $tagRepo, $this->getLogger()); + + $entry = $proxy->updateEntry(new Entry(new User()), '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', + ]); + + $this->assertCount(0, $entry->getTags()); + } + + public function testAssignTagsWithArrayAndExtraSpaces() + { + $graby = $this->getMockBuilder('Graby\Graby') + ->disableOriginalConstructor() + ->getMock(); + + $tagRepo = $this->getTagRepositoryMock(); + $proxy = new ContentProxy($graby, $this->getTaggerMock(), $tagRepo, $this->getLogger()); + + $entry = new Entry(new User()); + + $proxy->assignTagsToEntry($entry, [' tag1', 'tag2 ']); + + $this->assertCount(2, $entry->getTags()); + $this->assertEquals('tag1', $entry->getTags()[0]->getLabel()); + $this->assertEquals('tag2', $entry->getTags()[1]->getLabel()); + } + + public function testAssignTagsWithString() + { + $graby = $this->getMockBuilder('Graby\Graby') + ->disableOriginalConstructor() + ->getMock(); + + $tagRepo = $this->getTagRepositoryMock(); + $proxy = new ContentProxy($graby, $this->getTaggerMock(), $tagRepo, $this->getLogger()); + + $entry = new Entry(new User()); + + $proxy->assignTagsToEntry($entry, 'tag1, tag2'); + + $this->assertCount(2, $entry->getTags()); + $this->assertEquals('tag1', $entry->getTags()[0]->getLabel()); + $this->assertEquals('tag2', $entry->getTags()[1]->getLabel()); + } + + public function testAssignTagsWithEmptyArray() + { + $graby = $this->getMockBuilder('Graby\Graby') + ->disableOriginalConstructor() + ->getMock(); + + $tagRepo = $this->getTagRepositoryMock(); + $proxy = new ContentProxy($graby, $this->getTaggerMock(), $tagRepo, $this->getLogger()); + + $entry = new Entry(new User()); + + $proxy->assignTagsToEntry($entry, []); + + $this->assertCount(0, $entry->getTags()); + } + + public function testAssignTagsWithEmptyString() + { + $graby = $this->getMockBuilder('Graby\Graby') + ->disableOriginalConstructor() + ->getMock(); + + $tagRepo = $this->getTagRepositoryMock(); + $proxy = new ContentProxy($graby, $this->getTaggerMock(), $tagRepo, $this->getLogger()); + + $entry = new Entry(new User()); + + $proxy->assignTagsToEntry($entry, ''); + + $this->assertCount(0, $entry->getTags()); + } + + public function testAssignTagsAlreadyAssigned() + { + $graby = $this->getMockBuilder('Graby\Graby') + ->disableOriginalConstructor() + ->getMock(); + + $tagRepo = $this->getTagRepositoryMock(); + $proxy = new ContentProxy($graby, $this->getTaggerMock(), $tagRepo, $this->getLogger()); + + $tagEntity = new Tag(); + $tagEntity->setLabel('tag1'); + + $entry = new Entry(new User()); + $entry->addTag($tagEntity); + + $proxy->assignTagsToEntry($entry, 'tag1, tag2'); + + $this->assertCount(2, $entry->getTags()); + $this->assertEquals('tag1', $entry->getTags()[0]->getLabel()); + $this->assertEquals('tag2', $entry->getTags()[1]->getLabel()); + } + + private function getTaggerMock() + { + return $this->getMockBuilder('Wallabag\CoreBundle\Helper\RuleBasedTagger') + ->setMethods(['tag']) + ->disableOriginalConstructor() + ->getMock(); + } + + private function getTagRepositoryMock() + { + return $this->getMockBuilder('Wallabag\CoreBundle\Repository\TagRepository') + ->disableOriginalConstructor() + ->getMock(); + } + + private function getLogger() + { + return new NullLogger(); + } +} diff --git a/tests/Wallabag/CoreBundle/Helper/RedirectTest.php b/tests/Wallabag/CoreBundle/Helper/RedirectTest.php new file mode 100644 index 00000000..f339f75e --- /dev/null +++ b/tests/Wallabag/CoreBundle/Helper/RedirectTest.php @@ -0,0 +1,55 @@ +routerMock = $this->getRouterMock(); + $this->redirect = new Redirect($this->routerMock); + } + + public function testRedirectToNullWithFallback() + { + $redirectUrl = $this->redirect->to(null, 'fallback'); + + $this->assertEquals('fallback', $redirectUrl); + } + + public function testRedirectToNullWithoutFallback() + { + $redirectUrl = $this->redirect->to(null); + + $this->assertEquals($this->routerMock->generate('homepage'), $redirectUrl); + } + + public function testRedirectToValidUrl() + { + $redirectUrl = $this->redirect->to('/unread/list'); + + $this->assertEquals('/unread/list', $redirectUrl); + } + + private function getRouterMock() + { + $mock = $this->getMockBuilder('Symfony\Component\Routing\Router') + ->disableOriginalConstructor() + ->getMock(); + + $mock->expects($this->any()) + ->method('generate') + ->with('homepage') + ->willReturn('homepage'); + + return $mock; + } +} diff --git a/tests/Wallabag/CoreBundle/Helper/RuleBasedTaggerTest.php b/tests/Wallabag/CoreBundle/Helper/RuleBasedTaggerTest.php new file mode 100644 index 00000000..17b08c2a --- /dev/null +++ b/tests/Wallabag/CoreBundle/Helper/RuleBasedTaggerTest.php @@ -0,0 +1,212 @@ +rulerz = $this->getRulerZMock(); + $this->tagRepository = $this->getTagRepositoryMock(); + $this->entryRepository = $this->getEntryRepositoryMock(); + + $this->tagger = new RuleBasedTagger($this->rulerz, $this->tagRepository, $this->entryRepository); + } + + public function testTagWithNoRule() + { + $entry = new Entry($this->getUser()); + + $this->tagger->tag($entry); + + $this->assertTrue($entry->getTags()->isEmpty()); + } + + public function testTagWithNoMatchingRule() + { + $taggingRule = $this->getTaggingRule('rule as string', ['foo', 'bar']); + $user = $this->getUser([$taggingRule]); + $entry = new Entry($user); + + $this->rulerz + ->expects($this->once()) + ->method('satisfies') + ->with($entry, 'rule as string') + ->willReturn(false); + + $this->tagger->tag($entry); + + $this->assertTrue($entry->getTags()->isEmpty()); + } + + public function testTagWithAMatchingRule() + { + $taggingRule = $this->getTaggingRule('rule as string', ['foo', 'bar']); + $user = $this->getUser([$taggingRule]); + $entry = new Entry($user); + + $this->rulerz + ->expects($this->once()) + ->method('satisfies') + ->with($entry, 'rule as string') + ->willReturn(true); + + $this->tagger->tag($entry); + + $this->assertFalse($entry->getTags()->isEmpty()); + + $tags = $entry->getTags(); + $this->assertSame('foo', $tags[0]->getLabel()); + $this->assertSame('bar', $tags[1]->getLabel()); + } + + public function testTagWithAMixOfMatchingRules() + { + $taggingRule = $this->getTaggingRule('bla bla', ['hey']); + $otherTaggingRule = $this->getTaggingRule('rule as string', ['foo']); + + $user = $this->getUser([$taggingRule, $otherTaggingRule]); + $entry = new Entry($user); + + $this->rulerz + ->method('satisfies') + ->will($this->onConsecutiveCalls(false, true)); + + $this->tagger->tag($entry); + + $this->assertFalse($entry->getTags()->isEmpty()); + + $tags = $entry->getTags(); + $this->assertSame('foo', $tags[0]->getLabel()); + } + + public function testWhenTheTagExists() + { + $taggingRule = $this->getTaggingRule('rule as string', ['foo']); + $user = $this->getUser([$taggingRule]); + $entry = new Entry($user); + $tag = new Tag(); + + $this->rulerz + ->expects($this->once()) + ->method('satisfies') + ->with($entry, 'rule as string') + ->willReturn(true); + + $this->tagRepository + ->expects($this->once()) + // the method `findOneByLabel` doesn't exist, EntityRepository will then call `_call` method + // to magically call the `findOneBy` with ['label' => 'foo'] + ->method('__call') + ->willReturn($tag); + + $this->tagger->tag($entry); + + $this->assertFalse($entry->getTags()->isEmpty()); + + $tags = $entry->getTags(); + $this->assertSame($tag, $tags[0]); + } + + public function testSameTagWithDifferentfMatchingRules() + { + $taggingRule = $this->getTaggingRule('bla bla', ['hey']); + $otherTaggingRule = $this->getTaggingRule('rule as string', ['hey']); + + $user = $this->getUser([$taggingRule, $otherTaggingRule]); + $entry = new Entry($user); + + $this->rulerz + ->method('satisfies') + ->willReturn(true); + + $this->tagger->tag($entry); + + $this->assertFalse($entry->getTags()->isEmpty()); + + $tags = $entry->getTags(); + $this->assertCount(1, $tags); + } + + public function testTagAllEntriesForAUser() + { + $taggingRule = $this->getTaggingRule('bla bla', ['hey']); + + $user = $this->getUser([$taggingRule]); + + $this->rulerz + ->method('satisfies') + ->willReturn(true); + + $this->rulerz + ->method('filter') + ->willReturn([new Entry($user), new Entry($user)]); + + $entries = $this->tagger->tagAllForUser($user); + + $this->assertCount(2, $entries); + + foreach ($entries as $entry) { + $tags = $entry->getTags(); + + $this->assertCount(1, $tags); + $this->assertEquals('hey', $tags[0]->getLabel()); + } + } + + private function getUser(array $taggingRules = []) + { + $user = new User(); + $config = new Config($user); + + $user->setConfig($config); + + foreach ($taggingRules as $rule) { + $config->addTaggingRule($rule); + } + + return $user; + } + + private function getTaggingRule($rule, array $tags) + { + $taggingRule = new TaggingRule(); + $taggingRule->setRule($rule); + $taggingRule->setTags($tags); + + return $taggingRule; + } + + private function getRulerZMock() + { + return $this->getMockBuilder('RulerZ\RulerZ') + ->disableOriginalConstructor() + ->getMock(); + } + + private function getTagRepositoryMock() + { + return $this->getMockBuilder('Wallabag\CoreBundle\Repository\TagRepository') + ->disableOriginalConstructor() + ->getMock(); + } + + private function getEntryRepositoryMock() + { + return $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + } +} diff --git a/tests/Wallabag/CoreBundle/Mock/InstallCommandMock.php b/tests/Wallabag/CoreBundle/Mock/InstallCommandMock.php new file mode 100644 index 00000000..5806bd4d --- /dev/null +++ b/tests/Wallabag/CoreBundle/Mock/InstallCommandMock.php @@ -0,0 +1,22 @@ +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([])); + + $params = new ParamConverter([]); + $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(['default' => null])); + + $params = new ParamConverter([]); + $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(['default' => null])); + + $registry->expects($this->once()) + ->method('getManagerForClass') + ->with('superclass') + ->will($this->returnValue($em)); + + $params = new ParamConverter(['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\UserBundle\Entity\User')); + + $em = $this->getMockBuilder('Doctrine\Common\Persistence\ObjectManager') + ->disableOriginalConstructor() + ->getMock(); + + $em->expects($this->once()) + ->method('getClassMetadata') + ->with('WallabagUserBundle:User') + ->will($this->returnValue($meta)); + + $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry') + ->disableOriginalConstructor() + ->getMock(); + + $registry->expects($this->once()) + ->method('getManagers') + ->will($this->returnValue(['default' => null])); + + $registry->expects($this->once()) + ->method('getManagerForClass') + ->with('WallabagUserBundle:User') + ->will($this->returnValue($em)); + + $params = new ParamConverter(['class' => 'WallabagUserBundle:User']); + $converter = new UsernameRssTokenConverter($registry); + + $this->assertTrue($converter->supports($params)); + } + + /** + * @expectedException InvalidArgumentException + * @expectedExceptionMessage Route attribute is missing + */ + public function testApplyEmptyRequest() + { + $params = new ParamConverter([]); + $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\UserBundle\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('WallabagUserBundle:User') + ->will($this->returnValue($repo)); + + $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry') + ->disableOriginalConstructor() + ->getMock(); + + $registry->expects($this->once()) + ->method('getManagerForClass') + ->with('WallabagUserBundle:User') + ->will($this->returnValue($em)); + + $params = new ParamConverter(['class' => 'WallabagUserBundle:User']); + $converter = new UsernameRssTokenConverter($registry); + $request = new Request([], [], ['username' => 'test', 'token' => 'test']); + + $converter->apply($request, $params); + } + + public function testApplyUserFound() + { + $user = new User(); + + $repo = $this->getMockBuilder('Wallabag\UserBundle\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('WallabagUserBundle:User') + ->will($this->returnValue($repo)); + + $registry = $this->getMockBuilder('Doctrine\Common\Persistence\ManagerRegistry') + ->disableOriginalConstructor() + ->getMock(); + + $registry->expects($this->once()) + ->method('getManagerForClass') + ->with('WallabagUserBundle:User') + ->will($this->returnValue($em)); + + $params = new ParamConverter(['class' => 'WallabagUserBundle:User', 'name' => 'user']); + $converter = new UsernameRssTokenConverter($registry); + $request = new Request([], [], ['username' => 'test', 'token' => 'test']); + + $converter->apply($request, $params); + + $this->assertEquals($user, $request->attributes->get('user')); + } +} diff --git a/tests/Wallabag/CoreBundle/Subscriber/TablePrefixSubscriberTest.php b/tests/Wallabag/CoreBundle/Subscriber/TablePrefixSubscriberTest.php new file mode 100644 index 00000000..4ae76703 --- /dev/null +++ b/tests/Wallabag/CoreBundle/Subscriber/TablePrefixSubscriberTest.php @@ -0,0 +1,114 @@ +getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $subscriber = new TablePrefixSubscriber($prefix); + + $metaClass = new ClassMetadata($entityName); + $metaClass->setPrimaryTable(['name' => $tableName]); + + $metaDataEvent = new LoadClassMetadataEventArgs($metaClass, $em); + + $this->assertEquals($tableNameExpected, $metaDataEvent->getClassMetadata()->getTableName()); + + $subscriber->loadClassMetadata($metaDataEvent); + + $this->assertEquals($finalTableName, $metaDataEvent->getClassMetadata()->getTableName()); + $this->assertEquals($finalTableNameQuoted, $metaDataEvent->getClassMetadata()->getQuotedTableName($platform)); + } + + /** + * @dataProvider dataForPrefix + */ + public function testSubscribedEvents($prefix, $entityName, $tableName, $tableNameExpected, $finalTableName, $finalTableNameQuoted, $platform) + { + $em = $this->getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $metaClass = new ClassMetadata($entityName); + $metaClass->setPrimaryTable(['name' => $tableName]); + + $metaDataEvent = new LoadClassMetadataEventArgs($metaClass, $em); + + $subscriber = new TablePrefixSubscriber($prefix); + + $evm = new EventManager(); + $evm->addEventSubscriber($subscriber); + + $evm->dispatchEvent('loadClassMetadata', $metaDataEvent); + + $this->assertEquals($finalTableName, $metaDataEvent->getClassMetadata()->getTableName()); + $this->assertEquals($finalTableNameQuoted, $metaDataEvent->getClassMetadata()->getQuotedTableName($platform)); + } + + public function testPrefixManyToMany() + { + $em = $this->getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $subscriber = new TablePrefixSubscriber('yo_'); + + $metaClass = new ClassMetadata('Wallabag\UserBundle\Entity\Entry'); + $metaClass->setPrimaryTable(['name' => 'entry']); + $metaClass->mapManyToMany([ + 'fieldName' => 'tags', + 'joinTable' => ['name' => null, 'schema' => null], + 'targetEntity' => 'Tag', + 'mappedBy' => null, + 'inversedBy' => 'entries', + 'cascade' => ['persist'], + 'indexBy' => null, + 'orphanRemoval' => false, + 'fetch' => 2, + ]); + + $metaDataEvent = new LoadClassMetadataEventArgs($metaClass, $em); + + $this->assertEquals('entry', $metaDataEvent->getClassMetadata()->getTableName()); + + $subscriber->loadClassMetadata($metaDataEvent); + + $this->assertEquals('yo_entry', $metaDataEvent->getClassMetadata()->getTableName()); + $this->assertEquals('yo_entry_tag', $metaDataEvent->getClassMetadata()->associationMappings['tags']['joinTable']['name']); + $this->assertEquals('yo_entry', $metaDataEvent->getClassMetadata()->getQuotedTableName(new \Doctrine\DBAL\Platforms\MySqlPlatform())); + } +} diff --git a/tests/Wallabag/CoreBundle/Twig/WallabagExtensionTest.php b/tests/Wallabag/CoreBundle/Twig/WallabagExtensionTest.php new file mode 100644 index 00000000..8ec2a75a --- /dev/null +++ b/tests/Wallabag/CoreBundle/Twig/WallabagExtensionTest.php @@ -0,0 +1,17 @@ +assertEquals('lemonde.fr', $extension->removeWww('www.lemonde.fr')); + $this->assertEquals('lemonde.fr', $extension->removeWww('lemonde.fr')); + $this->assertEquals('gist.github.com', $extension->removeWww('gist.github.com')); + } +} diff --git a/tests/Wallabag/CoreBundle/WallabagCoreTestCase.php b/tests/Wallabag/CoreBundle/WallabagCoreTestCase.php new file mode 100644 index 00000000..c69e8330 --- /dev/null +++ b/tests/Wallabag/CoreBundle/WallabagCoreTestCase.php @@ -0,0 +1,51 @@ +client; + } + + public function setUp() + { + parent::setUp(); + + $this->client = static::createClient(); + } + + public function logInAs($username) + { + $crawler = $this->client->request('GET', '/login'); + $form = $crawler->filter('button[type=submit]')->form(); + $data = [ + '_username' => $username, + '_password' => 'mypassword', + ]; + + $this->client->submit($form, $data); + } + + /** + * Return the user id of the logged in user. + * You should be sure that you called `logInAs` before. + * + * @return int + */ + public function getLoggedInUserId() + { + $token = static::$kernel->getContainer()->get('security.token_storage')->getToken(); + + if (null !== $token) { + return $token->getUser()->getId(); + } + + throw new \RuntimeException('No logged in User.'); + } +} diff --git a/tests/Wallabag/ImportBundle/Controller/ImportControllerTest.php b/tests/Wallabag/ImportBundle/Controller/ImportControllerTest.php new file mode 100644 index 00000000..96b5300b --- /dev/null +++ b/tests/Wallabag/ImportBundle/Controller/ImportControllerTest.php @@ -0,0 +1,29 @@ +getClient(); + + $client->request('GET', '/import/'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('login', $client->getResponse()->headers->get('location')); + } + + public function testImportList() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertEquals(3, $crawler->filter('blockquote')->count()); + } +} diff --git a/tests/Wallabag/ImportBundle/Controller/PocketControllerTest.php b/tests/Wallabag/ImportBundle/Controller/PocketControllerTest.php new file mode 100644 index 00000000..6aaf1b57 --- /dev/null +++ b/tests/Wallabag/ImportBundle/Controller/PocketControllerTest.php @@ -0,0 +1,65 @@ +logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/pocket'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertEquals(1, $crawler->filter('button[type=submit]')->count()); + } + + public function testImportPocketAuthBadToken() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/pocket/auth'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + } + + public function testImportPocketAuth() + { + $this->markTestSkipped('PocketImport: Find a way to properly mock a service.'); + + $this->logInAs('admin'); + $client = $this->getClient(); + + $pocketImport = $this->getMockBuilder('Wallabag\ImportBundle\Import\PocketImport') + ->disableOriginalConstructor() + ->getMock(); + + $pocketImport + ->expects($this->once()) + ->method('getRequestToken') + ->willReturn('token'); + + $client->getContainer()->set('wallabag_import.pocket.import', $pocketImport); + + $crawler = $client->request('GET', '/import/pocket/auth'); + + $this->assertEquals(301, $client->getResponse()->getStatusCode()); + $this->assertContains('getpocket.com/auth/authorize', $client->getResponse()->headers->get('location')); + } + + public function testImportPocketCallbackWithBadToken() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/pocket/callback'); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + $this->assertContains('import/pocket', $client->getResponse()->headers->get('location')); + $this->assertEquals('flashes.import.notice.failed', $client->getContainer()->get('session')->getFlashBag()->peek('notice')[0]); + } +} diff --git a/tests/Wallabag/ImportBundle/Controller/WallabagV1ControllerTest.php b/tests/Wallabag/ImportBundle/Controller/WallabagV1ControllerTest.php new file mode 100644 index 00000000..c1025b41 --- /dev/null +++ b/tests/Wallabag/ImportBundle/Controller/WallabagV1ControllerTest.php @@ -0,0 +1,129 @@ +logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v1'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertEquals(1, $crawler->filter('form[name=upload_import_file] > button[type=submit]')->count()); + $this->assertEquals(1, $crawler->filter('input[type=file]')->count()); + } + + public function testImportWallabagWithFile() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v1'); + $form = $crawler->filter('form[name=upload_import_file] > button[type=submit]')->form(); + + $file = new UploadedFile(__DIR__.'/../fixtures/wallabag-v1.json', 'wallabag-v1.json'); + + $data = [ + 'upload_import_file[file]' => $file, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId( + 'http://www.framablog.org/index.php/post/2014/02/05/Framabag-service-libre-gratuit-interview-developpeur', + $this->getLoggedInUserId() + ); + + $tag = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Tag') + ->findOneByLabel('Framabag'); + + $this->assertTrue($content->getTags()->contains($tag)); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('flashes.import.notice.summary', $body[0]); + } + + public function testImportWallabagWithFileAndMarkAllAsRead() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v1'); + $form = $crawler->filter('form[name=upload_import_file] > button[type=submit]')->form(); + + $file = new UploadedFile(__DIR__.'/../fixtures/wallabag-v1-read.json', 'wallabag-v1-read.json'); + + $data = [ + 'upload_import_file[file]' => $file, + 'upload_import_file[mark_as_read]' => 1, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $content1 = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId( + 'http://gilbert.pellegrom.me/recreating-the-square-slider', + $this->getLoggedInUserId() + ); + + $this->assertTrue($content1->isArchived()); + + $content2 = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId( + 'https://www.wallabag.org/features/', + $this->getLoggedInUserId() + ); + + $this->assertTrue($content2->isArchived()); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('flashes.import.notice.summary', $body[0]); + } + + public function testImportWallabagWithEmptyFile() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v1'); + $form = $crawler->filter('form[name=upload_import_file] > button[type=submit]')->form(); + + $file = new UploadedFile(__DIR__.'/../fixtures/test.txt', 'test.txt'); + + $data = [ + 'upload_import_file[file]' => $file, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('flashes.import.notice.failed', $body[0]); + } +} diff --git a/tests/Wallabag/ImportBundle/Controller/WallabagV2ControllerTest.php b/tests/Wallabag/ImportBundle/Controller/WallabagV2ControllerTest.php new file mode 100644 index 00000000..d8d2c8bf --- /dev/null +++ b/tests/Wallabag/ImportBundle/Controller/WallabagV2ControllerTest.php @@ -0,0 +1,95 @@ +logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v2'); + + $this->assertEquals(200, $client->getResponse()->getStatusCode()); + $this->assertEquals(1, $crawler->filter('form[name=upload_import_file] > button[type=submit]')->count()); + $this->assertEquals(1, $crawler->filter('input[type=file]')->count()); + } + + public function testImportWallabagWithFile() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v2'); + $form = $crawler->filter('form[name=upload_import_file] > button[type=submit]')->form(); + + $file = new UploadedFile(__DIR__.'/../fixtures/wallabag-v2.json', 'wallabag-v2.json'); + + $data = [ + 'upload_import_file[file]' => $file, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('flashes.import.notice.summary', $body[0]); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId( + 'http://www.liberation.fr/planete/2015/10/26/refugies-l-ue-va-creer-100-000-places-d-accueil-dans-les-balkans_1408867', + $this->getLoggedInUserId() + ); + + $this->assertEmpty($content->getMimetype()); + $this->assertEmpty($content->getPreviewPicture()); + $this->assertEmpty($content->getLanguage()); + $this->assertEquals(0, count($content->getTags())); + + $content = $client->getContainer() + ->get('doctrine.orm.entity_manager') + ->getRepository('WallabagCoreBundle:Entry') + ->findByUrlAndUserId( + 'https://www.mediapart.fr/', + $this->getLoggedInUserId() + ); + + $this->assertNotEmpty($content->getMimetype()); + $this->assertNotEmpty($content->getPreviewPicture()); + $this->assertNotEmpty($content->getLanguage()); + $this->assertEquals(2, count($content->getTags())); + } + + public function testImportWallabagWithEmptyFile() + { + $this->logInAs('admin'); + $client = $this->getClient(); + + $crawler = $client->request('GET', '/import/wallabag-v2'); + $form = $crawler->filter('form[name=upload_import_file] > button[type=submit]')->form(); + + $file = new UploadedFile(__DIR__.'/../fixtures/test.txt', 'test.txt'); + + $data = [ + 'upload_import_file[file]' => $file, + ]; + + $client->submit($form, $data); + + $this->assertEquals(302, $client->getResponse()->getStatusCode()); + + $crawler = $client->followRedirect(); + + $this->assertGreaterThan(1, $body = $crawler->filter('body')->extract(['_text'])); + $this->assertContains('flashes.import.notice.failed', $body[0]); + } +} diff --git a/tests/Wallabag/ImportBundle/Import/ImportChainTest.php b/tests/Wallabag/ImportBundle/Import/ImportChainTest.php new file mode 100644 index 00000000..32568ce5 --- /dev/null +++ b/tests/Wallabag/ImportBundle/Import/ImportChainTest.php @@ -0,0 +1,21 @@ +getMockBuilder('Wallabag\ImportBundle\Import\ImportInterface') + ->disableOriginalConstructor() + ->getMock(); + + $importChain = new ImportChain(); + $importChain->addImport($import, 'alias'); + + $this->assertCount(1, $importChain->getAll()); + $this->assertEquals($import, $importChain->getAll()['alias']); + } +} diff --git a/tests/Wallabag/ImportBundle/Import/ImportCompilerPassTest.php b/tests/Wallabag/ImportBundle/Import/ImportCompilerPassTest.php new file mode 100644 index 00000000..71a007a9 --- /dev/null +++ b/tests/Wallabag/ImportBundle/Import/ImportCompilerPassTest.php @@ -0,0 +1,47 @@ +process($container); + + $this->assertNull($res); + } + + public function testProcess() + { + $container = new ContainerBuilder(); + $container + ->register('wallabag_import.chain') + ->setPublic(false) + ; + + $container + ->register('foo') + ->addTag('wallabag_import.import', ['alias' => 'pocket']) + ; + + $this->process($container); + + $this->assertTrue($container->hasDefinition('wallabag_import.chain')); + + $definition = $container->getDefinition('wallabag_import.chain'); + $this->assertTrue($definition->hasMethodCall('addImport')); + + $calls = $definition->getMethodCalls(); + $this->assertEquals('pocket', $calls[0][1][1]); + } + + protected function process(ContainerBuilder $container) + { + $repeatedPass = new ImportCompilerPass(); + $repeatedPass->process($container); + } +} diff --git a/tests/Wallabag/ImportBundle/Import/PocketImportTest.php b/tests/Wallabag/ImportBundle/Import/PocketImportTest.php new file mode 100644 index 00000000..41f9b51f --- /dev/null +++ b/tests/Wallabag/ImportBundle/Import/PocketImportTest.php @@ -0,0 +1,393 @@ +accessToken; + } +} + +class PocketImportTest extends \PHPUnit_Framework_TestCase +{ + protected $token; + protected $user; + protected $em; + protected $contentProxy; + protected $logHandler; + + private function getPocketImport($consumerKey = 'ConsumerKey') + { + $this->user = new User(); + + $this->tokenStorage = $this->getMockBuilder('Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface') + ->disableOriginalConstructor() + ->getMock(); + + $token = $this->getMockBuilder('Symfony\Component\Security\Core\Authentication\Token\TokenInterface') + ->disableOriginalConstructor() + ->getMock(); + + $this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy') + ->disableOriginalConstructor() + ->getMock(); + + $token->expects($this->once()) + ->method('getUser') + ->willReturn($this->user); + + $this->tokenStorage->expects($this->once()) + ->method('getToken') + ->willReturn($token); + + $this->em = $this->getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $config = $this->getMockBuilder('Craue\ConfigBundle\Util\Config') + ->disableOriginalConstructor() + ->getMock(); + + $config->expects($this->any()) + ->method('get') + ->with('pocket_consumer_key') + ->willReturn($consumerKey); + + $pocket = new PocketImportMock( + $this->tokenStorage, + $this->em, + $this->contentProxy, + $config + ); + + $this->logHandler = new TestHandler(); + $logger = new Logger('test', [$this->logHandler]); + $pocket->setLogger($logger); + + return $pocket; + } + + public function testInit() + { + $pocketImport = $this->getPocketImport(); + + $this->assertEquals('Pocket', $pocketImport->getName()); + $this->assertNotEmpty($pocketImport->getUrl()); + $this->assertEquals('import.pocket.description', $pocketImport->getDescription()); + } + + public function testOAuthRequest() + { + $client = new Client(); + + $mock = new Mock([ + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(json_encode(['code' => 'wunderbar_code']))), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + $pocketImport->setClient($client); + + $code = $pocketImport->getRequestToken('http://0.0.0.0/redirect'); + + $this->assertEquals('wunderbar_code', $code); + } + + public function testOAuthRequestBadResponse() + { + $client = new Client(); + + $mock = new Mock([ + new Response(403), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + $pocketImport->setClient($client); + + $code = $pocketImport->getRequestToken('http://0.0.0.0/redirect'); + + $this->assertFalse($code); + + $records = $this->logHandler->getRecords(); + $this->assertContains('PocketImport: Failed to request token', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } + + public function testOAuthAuthorize() + { + $client = new Client(); + + $mock = new Mock([ + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(json_encode(['access_token' => 'wunderbar_token']))), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + $pocketImport->setClient($client); + + $res = $pocketImport->authorize('wunderbar_code'); + + $this->assertTrue($res); + $this->assertEquals('wunderbar_token', $pocketImport->getAccessToken()); + } + + public function testOAuthAuthorizeBadResponse() + { + $client = new Client(); + + $mock = new Mock([ + new Response(403), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + $pocketImport->setClient($client); + + $res = $pocketImport->authorize('wunderbar_code'); + + $this->assertFalse($res); + + $records = $this->logHandler->getRecords(); + $this->assertContains('PocketImport: Failed to authorize client', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } + + /** + * Will sample results from https://getpocket.com/developer/docs/v3/retrieve. + */ + public function testImport() + { + $client = new Client(); + + $mock = new Mock([ + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(json_encode(['access_token' => 'wunderbar_token']))), + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(' + { + "status": 1, + "list": { + "229279689": { + "item_id": "229279689", + "resolved_id": "229279689", + "given_url": "http://www.grantland.com/blog/the-triangle/post/_/id/38347/ryder-cup-preview", + "given_title": "The Massive Ryder Cup Preview - The Triangle Blog - Grantland", + "favorite": "1", + "status": "1", + "resolved_title": "The Massive Ryder Cup Preview", + "resolved_url": "http://www.grantland.com/blog/the-triangle/post/_/id/38347/ryder-cup-preview", + "excerpt": "The list of things I love about the Ryder Cup is so long that it could fill a (tedious) novel, and golf fans can probably guess most of them.", + "is_article": "1", + "has_video": "1", + "has_image": "1", + "word_count": "3197", + "images": { + "1": { + "item_id": "229279689", + "image_id": "1", + "src": "http://a.espncdn.com/combiner/i?img=/photo/2012/0927/grant_g_ryder_cr_640.jpg&w=640&h=360", + "width": "0", + "height": "0", + "credit": "Jamie Squire/Getty Images", + "caption": "" + } + }, + "videos": { + "1": { + "item_id": "229279689", + "video_id": "1", + "src": "http://www.youtube.com/v/Er34PbFkVGk?version=3&hl=en_US&rel=0", + "width": "420", + "height": "315", + "type": "1", + "vid": "Er34PbFkVGk" + } + }, + "tags": { + "grantland": { + "item_id": "1147652870", + "tag": "grantland" + }, + "Ryder Cup": { + "item_id": "1147652870", + "tag": "Ryder Cup" + } + } + }, + "229279690": { + "item_id": "229279689", + "resolved_id": "229279689", + "given_url": "http://www.grantland.com/blog/the-triangle/post/_/id/38347/ryder-cup-preview", + "given_title": "The Massive Ryder Cup Preview - The Triangle Blog - Grantland", + "favorite": "1", + "status": "1", + "resolved_title": "The Massive Ryder Cup Preview", + "resolved_url": "http://www.grantland.com/blog/the-triangle/post/_/id/38347/ryder-cup-preview", + "excerpt": "The list of things I love about the Ryder Cup is so long that it could fill a (tedious) novel, and golf fans can probably guess most of them.", + "is_article": "1", + "has_video": "0", + "has_image": "0", + "word_count": "3197" + } + } + } + ')), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + + $entryRepo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + + $entryRepo->expects($this->exactly(2)) + ->method('findByUrlAndUserId') + ->will($this->onConsecutiveCalls(false, true)); + + $this->em + ->expects($this->exactly(2)) + ->method('getRepository') + ->willReturn($entryRepo); + + $entry = new Entry($this->user); + + $this->contentProxy + ->expects($this->once()) + ->method('updateEntry') + ->willReturn($entry); + + $pocketImport->setClient($client); + $pocketImport->authorize('wunderbar_code'); + + $res = $pocketImport->import(); + + $this->assertTrue($res); + $this->assertEquals(['skipped' => 1, 'imported' => 1], $pocketImport->getSummary()); + } + + /** + * Will sample results from https://getpocket.com/developer/docs/v3/retrieve. + */ + public function testImportAndMarkAllAsRead() + { + $client = new Client(); + + $mock = new Mock([ + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(json_encode(['access_token' => 'wunderbar_token']))), + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(' + { + "status": 1, + "list": { + "229279689": { + "item_id": "229279689", + "resolved_id": "229279689", + "given_url": "http://www.grantland.com/blog/the-triangle/post/_/id/38347/ryder-cup-preview", + "given_title": "The Massive Ryder Cup Preview - The Triangle Blog - Grantland", + "favorite": "1", + "status": "1", + "excerpt": "The list of things I love about the Ryder Cup is so long that it could fill a (tedious) novel, and golf fans can probably guess most of them.", + "is_article": "1", + "has_video": "1", + "has_image": "1", + "word_count": "3197" + }, + "229279690": { + "item_id": "229279689", + "resolved_id": "229279689", + "given_url": "http://www.grantland.com/blog/the-triangle/post/_/id/38347/ryder-cup-preview/2", + "given_title": "The Massive Ryder Cup Preview - The Triangle Blog - Grantland", + "favorite": "1", + "status": "0", + "excerpt": "The list of things I love about the Ryder Cup is so long that it could fill a (tedious) novel, and golf fans can probably guess most of them.", + "is_article": "1", + "has_video": "0", + "has_image": "0", + "word_count": "3197" + } + } + } + ')), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + + $entryRepo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + + $entryRepo->expects($this->exactly(2)) + ->method('findByUrlAndUserId') + ->will($this->onConsecutiveCalls(false, false)); + + $this->em + ->expects($this->exactly(2)) + ->method('getRepository') + ->willReturn($entryRepo); + + // check that every entry persisted are archived + $this->em + ->expects($this->any()) + ->method('persist') + ->with($this->callback(function ($persistedEntry) { + return $persistedEntry->isArchived(); + })); + + $entry = new Entry($this->user); + + $this->contentProxy + ->expects($this->exactly(2)) + ->method('updateEntry') + ->willReturn($entry); + + $pocketImport->setClient($client); + $pocketImport->authorize('wunderbar_code'); + + $res = $pocketImport->setMarkAsRead(true)->import(); + + $this->assertTrue($res); + $this->assertEquals(['skipped' => 0, 'imported' => 2], $pocketImport->getSummary()); + } + + public function testImportBadResponse() + { + $client = new Client(); + + $mock = new Mock([ + new Response(200, ['Content-Type' => 'application/json'], Stream::factory(json_encode(['access_token' => 'wunderbar_token']))), + new Response(403), + ]); + + $client->getEmitter()->attach($mock); + + $pocketImport = $this->getPocketImport(); + $pocketImport->setClient($client); + $pocketImport->authorize('wunderbar_code'); + + $res = $pocketImport->import(); + + $this->assertFalse($res); + + $records = $this->logHandler->getRecords(); + $this->assertContains('PocketImport: Failed to import', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } +} diff --git a/tests/Wallabag/ImportBundle/Import/WallabagV1ImportTest.php b/tests/Wallabag/ImportBundle/Import/WallabagV1ImportTest.php new file mode 100644 index 00000000..bdc47dac --- /dev/null +++ b/tests/Wallabag/ImportBundle/Import/WallabagV1ImportTest.php @@ -0,0 +1,150 @@ +user = new User(); + + $this->em = $this->getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy') + ->disableOriginalConstructor() + ->getMock(); + + $wallabag = new WallabagV1Import($this->em, $this->contentProxy); + + $this->logHandler = new TestHandler(); + $logger = new Logger('test', [$this->logHandler]); + $wallabag->setLogger($logger); + + if (false === $unsetUser) { + $wallabag->setUser($this->user); + } + + return $wallabag; + } + + public function testInit() + { + $wallabagV1Import = $this->getWallabagV1Import(); + + $this->assertEquals('wallabag v1', $wallabagV1Import->getName()); + $this->assertNotEmpty($wallabagV1Import->getUrl()); + $this->assertEquals('import.wallabag_v1.description', $wallabagV1Import->getDescription()); + } + + public function testImport() + { + $wallabagV1Import = $this->getWallabagV1Import(); + $wallabagV1Import->setFilepath(__DIR__.'/../fixtures/wallabag-v1.json'); + + $entryRepo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + + $entryRepo->expects($this->exactly(4)) + ->method('findByUrlAndUserId') + ->will($this->onConsecutiveCalls(false, true, false, false)); + + $this->em + ->expects($this->any()) + ->method('getRepository') + ->willReturn($entryRepo); + + $entry = $this->getMockBuilder('Wallabag\CoreBundle\Entity\Entry') + ->disableOriginalConstructor() + ->getMock(); + + $this->contentProxy + ->expects($this->exactly(3)) + ->method('updateEntry') + ->willReturn($entry); + + $res = $wallabagV1Import->import(); + + $this->assertTrue($res); + $this->assertEquals(['skipped' => 1, 'imported' => 3], $wallabagV1Import->getSummary()); + } + + public function testImportAndMarkAllAsRead() + { + $wallabagV1Import = $this->getWallabagV1Import(); + $wallabagV1Import->setFilepath(__DIR__.'/../fixtures/wallabag-v1-read.json'); + + $entryRepo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + + $entryRepo->expects($this->exactly(3)) + ->method('findByUrlAndUserId') + ->will($this->onConsecutiveCalls(false, false, false)); + + $this->em + ->expects($this->any()) + ->method('getRepository') + ->willReturn($entryRepo); + + $this->contentProxy + ->expects($this->exactly(3)) + ->method('updateEntry') + ->willReturn(new Entry($this->user)); + + // check that every entry persisted are archived + $this->em + ->expects($this->any()) + ->method('persist') + ->with($this->callback(function ($persistedEntry) { + return $persistedEntry->isArchived(); + })); + + $res = $wallabagV1Import->setMarkAsRead(true)->import(); + + $this->assertTrue($res); + + $this->assertEquals(['skipped' => 0, 'imported' => 3], $wallabagV1Import->getSummary()); + } + + public function testImportBadFile() + { + $wallabagV1Import = $this->getWallabagV1Import(); + $wallabagV1Import->setFilepath(__DIR__.'/../fixtures/wallabag-v1.jsonx'); + + $res = $wallabagV1Import->import(); + + $this->assertFalse($res); + + $records = $this->logHandler->getRecords(); + $this->assertContains('WallabagImport: unable to read file', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } + + public function testImportUserNotDefined() + { + $wallabagV1Import = $this->getWallabagV1Import(true); + $wallabagV1Import->setFilepath(__DIR__.'/../fixtures/wallabag-v1.json'); + + $res = $wallabagV1Import->import(); + + $this->assertFalse($res); + + $records = $this->logHandler->getRecords(); + $this->assertContains('WallabagImport: user is not defined', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } +} diff --git a/tests/Wallabag/ImportBundle/Import/WallabagV2ImportTest.php b/tests/Wallabag/ImportBundle/Import/WallabagV2ImportTest.php new file mode 100644 index 00000000..8ec66b12 --- /dev/null +++ b/tests/Wallabag/ImportBundle/Import/WallabagV2ImportTest.php @@ -0,0 +1,146 @@ +user = new User(); + + $this->em = $this->getMockBuilder('Doctrine\ORM\EntityManager') + ->disableOriginalConstructor() + ->getMock(); + + $this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy') + ->disableOriginalConstructor() + ->getMock(); + + $wallabag = new WallabagV2Import($this->em, $this->contentProxy); + + $this->logHandler = new TestHandler(); + $logger = new Logger('test', [$this->logHandler]); + $wallabag->setLogger($logger); + + if (false === $unsetUser) { + $wallabag->setUser($this->user); + } + + return $wallabag; + } + + public function testInit() + { + $wallabagV2Import = $this->getWallabagV2Import(); + + $this->assertEquals('wallabag v2', $wallabagV2Import->getName()); + $this->assertNotEmpty($wallabagV2Import->getUrl()); + $this->assertEquals('import.wallabag_v2.description', $wallabagV2Import->getDescription()); + } + + public function testImport() + { + $wallabagV2Import = $this->getWallabagV2Import(); + $wallabagV2Import->setFilepath(__DIR__.'/../fixtures/wallabag-v2.json'); + + $entryRepo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + + $entryRepo->expects($this->exactly(24)) + ->method('findByUrlAndUserId') + ->will($this->onConsecutiveCalls(false, true, false)); + + $this->em + ->expects($this->any()) + ->method('getRepository') + ->willReturn($entryRepo); + + $this->contentProxy + ->expects($this->exactly(2)) + ->method('updateEntry') + ->willReturn(new Entry($this->user)); + + $res = $wallabagV2Import->import(); + + $this->assertTrue($res); + $this->assertEquals(['skipped' => 22, 'imported' => 2], $wallabagV2Import->getSummary()); + } + + public function testImportAndMarkAllAsRead() + { + $wallabagV2Import = $this->getWallabagV2Import(); + $wallabagV2Import->setFilepath(__DIR__.'/../fixtures/wallabag-v2-read.json'); + + $entryRepo = $this->getMockBuilder('Wallabag\CoreBundle\Repository\EntryRepository') + ->disableOriginalConstructor() + ->getMock(); + + $entryRepo->expects($this->exactly(2)) + ->method('findByUrlAndUserId') + ->will($this->onConsecutiveCalls(false, false)); + + $this->em + ->expects($this->any()) + ->method('getRepository') + ->willReturn($entryRepo); + + $this->contentProxy + ->expects($this->exactly(2)) + ->method('updateEntry') + ->willReturn(new Entry($this->user)); + + // check that every entry persisted are archived + $this->em + ->expects($this->any()) + ->method('persist') + ->with($this->callback(function ($persistedEntry) { + return $persistedEntry->isArchived(); + })); + + $res = $wallabagV2Import->setMarkAsRead(true)->import(); + + $this->assertTrue($res); + + $this->assertEquals(['skipped' => 0, 'imported' => 2], $wallabagV2Import->getSummary()); + } + + public function testImportBadFile() + { + $wallabagV1Import = $this->getWallabagV2Import(); + $wallabagV1Import->setFilepath(__DIR__.'/../fixtures/wallabag-v2.jsonx'); + + $res = $wallabagV1Import->import(); + + $this->assertFalse($res); + + $records = $this->logHandler->getRecords(); + $this->assertContains('WallabagImport: unable to read file', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } + + public function testImportUserNotDefined() + { + $wallabagV1Import = $this->getWallabagV2Import(true); + $wallabagV1Import->setFilepath(__DIR__.'/../fixtures/wallabag-v2.json'); + + $res = $wallabagV1Import->import(); + + $this->assertFalse($res); + + $records = $this->logHandler->getRecords(); + $this->assertContains('WallabagImport: user is not defined', $records[0]['message']); + $this->assertEquals('ERROR', $records[0]['level_name']); + } +} diff --git a/tests/Wallabag/ImportBundle/fixtures/test.html b/tests/Wallabag/ImportBundle/fixtures/test.html new file mode 100644 index 00000000..e69de29b diff --git a/tests/Wallabag/ImportBundle/fixtures/test.txt b/tests/Wallabag/ImportBundle/fixtures/test.txt new file mode 100644 index 00000000..e69de29b diff --git a/tests/Wallabag/ImportBundle/fixtures/wallabag-v1-read.json b/tests/Wallabag/ImportBundle/fixtures/wallabag-v1-read.json new file mode 100644 index 00000000..c4d1cf58 --- /dev/null +++ b/tests/Wallabag/ImportBundle/fixtures/wallabag-v1-read.json @@ -0,0 +1,53 @@ +[ + { + "0": "3", + "1": "Features - wallabag", + "2": "https://www.wallabag.org/features/", + "3": "0", + "4": "1", + "5": "\n\t\t

Here are some features. If one is missing, you can open a new issue.

\n", + "6": "1", + "id": "3", + "title": "Features - wallabag", + "url": "https://www.wallabag.org/features/", + "is_read": "0", + "is_fav": "1", + "content": "\n\t\t

Here are some features. If one is missing, you can open a new issue.

\n", + "user_id": "1", + "tags": "" + }, + { + "0": "10", + "1": "Recreating The Square Slider", + "2": "http://gilbert.pellegrom.me/recreating-the-square-slider", + "3": "0", + "4": "0", + "5": "

The new Square site is lovely and I really like the slider they have on the homepage. So I decided to try and recreate it in a simple and reusable way.

\n\n

\n Demo | Download\n

\n\n\n\n

The HTML

\n\n
<div class=\"square-slider\">  \n    <div class=\"slide slide1\">\n        <div class=\"content light\">\n            <h3>Recreating The Square Slider</h3>\n            <p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec ac eros et augue vulputate \n            aliquet pellentesque vitae tortor. Pellentesque mi velit, euismod nec semper</p>\n        </div>\n        <img src=\"images/asset1.png\" alt=\"\" class=\"asset\" />\n    </div>\n    <div class=\"slide slide2\">\n        <div class=\"content dark\">\n            <h3>Looks Amazing Right?</h3>\n            <p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec ac eros et augue vulputate \n            aliquet pellentesque vitae tortor. Pellentesque mi velit, euismod nec semper</p>\n        </div>\n        <img src=\"images/asset2.png\" alt=\"\" class=\"asset\" />\n    </div>\n    <div class=\"slide slide3 inverted\">\n        <div class=\"content light\">\n            <h3>And Simple To Use</h3>\n            <p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec ac eros et augue vulputate \n            aliquet pellentesque vitae tortor. Pellentesque mi velit, euismod nec semper</p>\n        </div>\n        <img src=\"images/asset3.png\" alt=\"\" class=\"asset\" />\n    </div>\n    <a href=\"#\" class=\"prev\">Prev</a>\n    <a href=\"#\" class=\"next\">Next</a>\n    <div class=\"overlay\"></div>\n</div>
\n\n

The CSS

\n\n
.square-slider {  \n    overflow: hidden;\n    position: relative;\n    background: #fff;\n}\n.square-slider .slide {\n    position: absolute;\n    top: 0;\n    left: 0;\n    width: 100%;\n    height: 100%;\n    display: none;\n    opacity: 0;\n    -moz-opacity: 0;\n    -moz-transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    -webkit-transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    -o-transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    -moz-transform: translate3d(0, 0, 0);\n    -webkit-transform: translate3d(0, 0, 0);\n    -o-transform: translate3d(0, 0, 0);\n    -ms-transform: translate3d(0, 0, 0);\n    transform: translate3d(0, 0, 0);\n}\n.square-slider .slide:first-child { display: block; }\n.square-slider .slide:first-child,\n.square-slider .slide.active {\n    opacity: 1;\n    -moz-opacity: 1;\n}\n.square-slider .slide .content {\n    position: absolute;\n    top: 40%;\n    left: 50%;\n    margin-left: -450px;\n    width: 360px;\n    text-shadow: 0 1px 1px rgba(0,0,0,0.3);\n    z-index: 7;\n    -webkit-transition-property: -webkit-transform,opacity;\n    -moz-transition-property: -moz-transform,opacity;\n    -webkit-transition-duration: 800ms,700ms;\n    -moz-transition-duration: 800ms,700ms;\n    -webkit-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -moz-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -webkit-transform: translate3d(-30px, 0, 0);\n    -moz-transform: translate(-30px, 0);\n}\n.square-slider .slide.inverted .content {\n    left: auto;\n    right: 50%;\n    margin-left: 0;\n    margin-right: -450px;\n    -webkit-transform: translate3d(30px, 0, 0);\n    -moz-transform: translate(30px, 0);\n}\n.square-slider .slide.active .content {\n    -webkit-transform: translate3d(0, 0, 0);\n    -moz-transform: translate(0, 0);\n}\n.square-slider .slide .asset {\n    position: absolute;\n    bottom: 0;\n    left: 50%;\n    -webkit-transition-property: -webkit-transform,opacity;\n    -moz-transition-property: -moz-transform,opacity;\n    -webkit-transition-duration: 800ms,700ms;\n    -moz-transition-duration: 800ms,700ms;\n    -webkit-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -moz-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -webkit-transform: translate3d(0, 0, 0);\n    -moz-transform: translate(0, 0);\n}\n.square-slider .slide.inverted .asset {\n    left: auto;\n    right: 50%;\n}\n.square-slider .slide.active .asset {\n    -webkit-transform: translate3d(-7px, 3px, 0);\n    -moz-transform: translate(-7px, 3px);\n}\n.square-slider .slide.inverted.active .asset {\n    -webkit-transform: translate3d(7px, 3px, 0);\n    -moz-transform: translate(7px, 3px);\n}\n.square-slider .prev,\n.square-slider .next {\n    background: url(images/nav.png) no-repeat;\n    display: block;\n    width: 67px;\n    height: 67px;\n    position: absolute;\n    top: 50%;\n    margin-top: -30px;\n    z-index: 10;\n    border: 0;\n    text-indent: -9999px;\n    display: none;\n    opacity: 0.6;\n    -moz-opacity: 0.6;\n    -webkit-transition: opacity 0.5s ease-in;\n    -moz-transition: opacity 0.5s ease-in;\n    -ms-transition: opacity 0.5s ease-in;\n    -o-transition: opacity 0.5s ease-in;\n    transition: opacity 0.5s ease-in;\n}\n.square-slider .prev { \n    left: 40px; \n    background-position: 0 100%;\n}\n.square-slider .next { right: 40px; }\n.square-slider .prev:hover,\n.square-slider .next:hover {\n    opacity: 1;\n    -moz-opacity: 1;\n}\n.square-slider .overlay {\n    position: absolute;\n    top: 0;\n    left: -100%;\n    width: 300%;\n    height: 100%;\n    z-index: 5;\n    -moz-box-shadow: inset 0px 0px 10px rgba(0,0,0,0.3);\n    -webkit-box-shadow: inset 0px 0px 10px rgba(0,0,0,0.3);\n    box-shadow: inset 0px 0px 10px rgba(0,0,0,0.3);\n}\n\n\n.square-slider {\n    width: 100%;\n    height: 550px;\n    margin: 40px auto;\n}\n.square-slider .slide .content.light { color: #fff; }\n.square-slider .slide .content.dark { \n    color: #333; \n    text-shadow: 0 1px 1px rgba(255,255,255,0.3);\n}\n.square-slider .slide1 { background: url(images/bg1.jpg) no-repeat 50% 50%; }\n.square-slider .slide2 { background: url(images/bg2.jpg) no-repeat 50% 50%; }\n.square-slider .slide3 { background: url(images/bg3.jpg) no-repeat 50% 50%; }
\n\n

The Javascript (jQuery)

\n\n
(function($){\n\n    $('.square-slider').each(function(){\n        var slider = $(this),\n            slides = slider.find('.slide'),\n            currentSlide = 0;\n\n        slides.show();\n        $(slides[currentSlide]).addClass('active');\n        $('.next,.prev', slider).show();\n\n        $('.prev', slider).on('click', function(){\n            slides.removeClass('active');\n            currentSlide--;\n            if(currentSlide < 0) currentSlide = slides.length - 1;\n            $(slides[currentSlide]).addClass('active');\n            return false;\n        });\n\n        $('.next', slider).on('click', function(){\n            slides.removeClass('active');\n            currentSlide++;\n            if(currentSlide > slides.length - 1) currentSlide = 0;\n            $(slides[currentSlide]).addClass('active');\n            return false;\n        });\n    });\n\n})(window.jQuery);
\n\n

A Few Notes

\n\n

Enjoy.

\n ", + "6": "1", + "id": "10", + "title": "Recreating The Square Slider", + "url": "http://gilbert.pellegrom.me/recreating-the-square-slider", + "is_read": "0", + "is_fav": "0", + "content": "

The new Square site is lovely and I really like the slider they have on the homepage. So I decided to try and recreate it in a simple and reusable way.

\n\n

\n Demo | Download\n

\n\n\n\n

The HTML

\n\n
<div class=\"square-slider\">  \n    <div class=\"slide slide1\">\n        <div class=\"content light\">\n            <h3>Recreating The Square Slider</h3>\n            <p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec ac eros et augue vulputate \n            aliquet pellentesque vitae tortor. Pellentesque mi velit, euismod nec semper</p>\n        </div>\n        <img src=\"images/asset1.png\" alt=\"\" class=\"asset\" />\n    </div>\n    <div class=\"slide slide2\">\n        <div class=\"content dark\">\n            <h3>Looks Amazing Right?</h3>\n            <p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec ac eros et augue vulputate \n            aliquet pellentesque vitae tortor. Pellentesque mi velit, euismod nec semper</p>\n        </div>\n        <img src=\"images/asset2.png\" alt=\"\" class=\"asset\" />\n    </div>\n    <div class=\"slide slide3 inverted\">\n        <div class=\"content light\">\n            <h3>And Simple To Use</h3>\n            <p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec ac eros et augue vulputate \n            aliquet pellentesque vitae tortor. Pellentesque mi velit, euismod nec semper</p>\n        </div>\n        <img src=\"images/asset3.png\" alt=\"\" class=\"asset\" />\n    </div>\n    <a href=\"#\" class=\"prev\">Prev</a>\n    <a href=\"#\" class=\"next\">Next</a>\n    <div class=\"overlay\"></div>\n</div>
\n\n

The CSS

\n\n
.square-slider {  \n    overflow: hidden;\n    position: relative;\n    background: #fff;\n}\n.square-slider .slide {\n    position: absolute;\n    top: 0;\n    left: 0;\n    width: 100%;\n    height: 100%;\n    display: none;\n    opacity: 0;\n    -moz-opacity: 0;\n    -moz-transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    -webkit-transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    -o-transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    transition: opacity 800ms cubic-bezier(0.51, 0.01, 0.37, 0.98) 100ms;\n    -moz-transform: translate3d(0, 0, 0);\n    -webkit-transform: translate3d(0, 0, 0);\n    -o-transform: translate3d(0, 0, 0);\n    -ms-transform: translate3d(0, 0, 0);\n    transform: translate3d(0, 0, 0);\n}\n.square-slider .slide:first-child { display: block; }\n.square-slider .slide:first-child,\n.square-slider .slide.active {\n    opacity: 1;\n    -moz-opacity: 1;\n}\n.square-slider .slide .content {\n    position: absolute;\n    top: 40%;\n    left: 50%;\n    margin-left: -450px;\n    width: 360px;\n    text-shadow: 0 1px 1px rgba(0,0,0,0.3);\n    z-index: 7;\n    -webkit-transition-property: -webkit-transform,opacity;\n    -moz-transition-property: -moz-transform,opacity;\n    -webkit-transition-duration: 800ms,700ms;\n    -moz-transition-duration: 800ms,700ms;\n    -webkit-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -moz-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -webkit-transform: translate3d(-30px, 0, 0);\n    -moz-transform: translate(-30px, 0);\n}\n.square-slider .slide.inverted .content {\n    left: auto;\n    right: 50%;\n    margin-left: 0;\n    margin-right: -450px;\n    -webkit-transform: translate3d(30px, 0, 0);\n    -moz-transform: translate(30px, 0);\n}\n.square-slider .slide.active .content {\n    -webkit-transform: translate3d(0, 0, 0);\n    -moz-transform: translate(0, 0);\n}\n.square-slider .slide .asset {\n    position: absolute;\n    bottom: 0;\n    left: 50%;\n    -webkit-transition-property: -webkit-transform,opacity;\n    -moz-transition-property: -moz-transform,opacity;\n    -webkit-transition-duration: 800ms,700ms;\n    -moz-transition-duration: 800ms,700ms;\n    -webkit-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -moz-transition-timing-function: cubic-bezier(0.51, 0.01, 0.37, 0.98);\n    -webkit-transform: translate3d(0, 0, 0);\n    -moz-transform: translate(0, 0);\n}\n.square-slider .slide.inverted .asset {\n    left: auto;\n    right: 50%;\n}\n.square-slider .slide.active .asset {\n    -webkit-transform: translate3d(-7px, 3px, 0);\n    -moz-transform: translate(-7px, 3px);\n}\n.square-slider .slide.inverted.active .asset {\n    -webkit-transform: translate3d(7px, 3px, 0);\n    -moz-transform: translate(7px, 3px);\n}\n.square-slider .prev,\n.square-slider .next {\n    background: url(images/nav.png) no-repeat;\n    display: block;\n    width: 67px;\n    height: 67px;\n    position: absolute;\n    top: 50%;\n    margin-top: -30px;\n    z-index: 10;\n    border: 0;\n    text-indent: -9999px;\n    display: none;\n    opacity: 0.6;\n    -moz-opacity: 0.6;\n    -webkit-transition: opacity 0.5s ease-in;\n    -moz-transition: opacity 0.5s ease-in;\n    -ms-transition: opacity 0.5s ease-in;\n    -o-transition: opacity 0.5s ease-in;\n    transition: opacity 0.5s ease-in;\n}\n.square-slider .prev { \n    left: 40px; \n    background-position: 0 100%;\n}\n.square-slider .next { right: 40px; }\n.square-slider .prev:hover,\n.square-slider .next:hover {\n    opacity: 1;\n    -moz-opacity: 1;\n}\n.square-slider .overlay {\n    position: absolute;\n    top: 0;\n    left: -100%;\n    width: 300%;\n    height: 100%;\n    z-index: 5;\n    -moz-box-shadow: inset 0px 0px 10px rgba(0,0,0,0.3);\n    -webkit-box-shadow: inset 0px 0px 10px rgba(0,0,0,0.3);\n    box-shadow: inset 0px 0px 10px rgba(0,0,0,0.3);\n}\n\n\n.square-slider {\n    width: 100%;\n    height: 550px;\n    margin: 40px auto;\n}\n.square-slider .slide .content.light { color: #fff; }\n.square-slider .slide .content.dark { \n    color: #333; \n    text-shadow: 0 1px 1px rgba(255,255,255,0.3);\n}\n.square-slider .slide1 { background: url(images/bg1.jpg) no-repeat 50% 50%; }\n.square-slider .slide2 { background: url(images/bg2.jpg) no-repeat 50% 50%; }\n.square-slider .slide3 { background: url(images/bg3.jpg) no-repeat 50% 50%; }
\n\n

The Javascript (jQuery)

\n\n
(function($){\n\n    $('.square-slider').each(function(){\n        var slider = $(this),\n            slides = slider.find('.slide'),\n            currentSlide = 0;\n\n        slides.show();\n        $(slides[currentSlide]).addClass('active');\n        $('.next,.prev', slider).show();\n\n        $('.prev', slider).on('click', function(){\n            slides.removeClass('active');\n            currentSlide--;\n            if(currentSlide < 0) currentSlide = slides.length - 1;\n            $(slides[currentSlide]).addClass('active');\n            return false;\n        });\n\n        $('.next', slider).on('click', function(){\n            slides.removeClass('active');\n            currentSlide++;\n            if(currentSlide > slides.length - 1) currentSlide = 0;\n            $(slides[currentSlide]).addClass('active');\n            return false;\n        });\n    });\n\n})(window.jQuery);
\n\n

A Few Notes

\n\n

Enjoy.

\n ", + "user_id": "1", + "tags": "" + }, + { + "0": "11", + "1": "J’aime le logiciel libre", + "2": "http://framablog.org/2015/02/14/jaime-le-logiciel-libre/", + "3": "0", + "4": "0", + "5": "\n

Aujourd’hui, c’est la Saint Valentin, et l’occasion de déclarer son amour des logiciels libres !

\n

\"ilovefs-banner-extralarge\"

\n

Framasoft vous a déjà proposé son adaptation délirante de poèmes pour l’occasion, et voici une petite bande-dessinée qui synthétise l’événement :

\n

\"dm_001_jaime_le_logiciel_libre\"

\n

Cette bande-dessinée est extraite du nouveau blog Grise Bouille hébergé par Framasoft.

\n

Crédit : Simon Gee Giraudot (Creative Commons By-Sa)

\n
\"\"

Gee

\"\"

Auteur/dessinateur de bandes dessinées (Le Geektionnerd, Superflu, Bastards Inc, etc.) et doctorant en informatique sur son temps salarié.

", + "6": "1", + "id": "11", + "title": "J’aime le logiciel libre", + "url": "http://framablog.org/2015/02/14/jaime-le-logiciel-libre/", + "is_read": "0", + "is_fav": "0", + "content": "\n

Aujourd’hui, c’est la Saint Valentin, et l’occasion de déclarer son amour des logiciels libres !

\n

\"ilovefs-banner-extralarge\"

\n

Framasoft vous a déjà proposé son adaptation délirante de poèmes pour l’occasion, et voici une petite bande-dessinée qui synthétise l’événement :

\n

\"dm_001_jaime_le_logiciel_libre\"

\n

Cette bande-dessinée est extraite du nouveau blog Grise Bouille hébergé par Framasoft.

\n

Crédit : Simon Gee Giraudot (Creative Commons By-Sa)

\n
\"\"

Gee

\"\"

Auteur/dessinateur de bandes dessinées (Le Geektionnerd, Superflu, Bastards Inc, etc.) et doctorant en informatique sur son temps salarié.

", + "user_id": "1", + "tags": "framasoft,tag" + } +] diff --git a/tests/Wallabag/ImportBundle/fixtures/wallabag-v1.json b/tests/Wallabag/ImportBundle/fixtures/wallabag-v1.json new file mode 100644 index 00000000..f298469f --- /dev/null +++ b/tests/Wallabag/ImportBundle/fixtures/wallabag-v1.json @@ -0,0 +1,69 @@ +[ + { + "0": "1", + "1": "Framabag, un nouveau service libre et gratuit", + "2": "http://www.framablog.org/index.php/post/2014/02/05/Framabag-service-libre-gratuit-interview-developpeur", + "3": "0", + "4": "0", + "5": "\n

Une interview de Nicolas, son développeur.

\n

Il ne vous a sûrement pas échappé que notre consommation de contenus du Web est terriblement chronophage et particulièrement frustrante tout à la fois : non seulement nous passons beaucoup (trop ?) de temps en ligne à explorer les mines aurifères de la toile, y détectant pépites et filons, mais nous sommes surtout constamment en manque. Même si nous ne sommes pas dans le zapping frénétique si facilement dénoncé par les doctes psychologues qui pontifient sur les dangers du numérique pour les jeunes cervelles, il nous vient souvent le goût amer de l’inachevé : pas le temps de tout lire (TL;DR est devenu le clin d’œil mi-figue mi-raisin d’une génération de lecteurs pressés), pas trop le temps de réfléchir non plus hélas, pas le temps de suivre la ribambelle de liens associés à un article…

\n

Pour nous donner bonne conscience, nous rangeons scrupuleusement un marque-page de plus dans un sous-dossier qui en comporte déjà 256, nous notons un élément de plus dans la toujours ridiculement longue toudouliste, bref nous remettons à plus tard, c’est-à-dire le plus souvent aux introuvables calendes grecques, le soin de lire vraiment un article jugé intéressant, de regarder une vidéo signalée par les rézossocios, de lire un chapitre entier d’un ouvrage disponible en ligne…

\n

Alors bien sûr, à défaut de nous donner tout le temps qui serait nécessaire, des solutions existent pour nous permettre de « lire plus tard » en sauvegardant le précieux pollen de nos butinages de site en site, et d’en faire ultérieurement votre miel ; c’est bel et bon mais les ruches sont un peu distantes, ça s’appelle le cloud (nos amis techies m’ont bien expliqué mais j’ai seulement compris que des trucs à moi sont sur des machines lointaines, ça ne me rassure pas trop) et elles sont souvent propriétaires, ne laissant entrer que les utilisateurs consommateurs payants et qui consentent à leurs conditions. Sans compter que de gros bourdons viennent profiter plus ou moins discrètement de toutes ces traces de nous-mêmes qui permettent de monétiser notre profil : si je collecte sur ces services (ne les nommons pas, justement) une série d’articles sur l’idée de Nature chez Diderot, je recevrai diverses sollicitations pour devenir client de la boutique Nature & Découverte du boulevard Diderot. Et si d’aventure les programmes de la NSA moulinent sur le service, je serai peut-être un jour dans une liste des militants naturistes indésirables sur les vols de la PanAm (je ne sais plus trop si je plaisante là, finalement…)

\n

La bonne idée : « se constituer un réservoir de documents sélectionnés à parcourir plus tard » appelait donc une autre bonne idée, celle d’avoir le contrôle de ce réservoir, de notre collection personnelle. C’est Nicolas Lœuillet, ci-dessous interviewé, qui s’y est collé avec une belle application appelée euh… oui, appelée Wallabag.

\n

Framasoft soutient d’autant plus son initiative qu’on lui a cherché des misères pour une histoire de nom et qu’il est possible d’installer soi-même une copie de Wallabag sur son propre site.

\n

Le petit plus de Framasoft, réseau toujours désireux de vous proposer des alternatives libératrices, c’est de vous proposer (sur inscription préalable) un accès au Framabag, autrement dit votre Wallabag sur un serveur Frama* avec notre garantie de confidentialité. Comme pour le Framanews, nous vous accueillons volontiers dans la limite de nos capacités, en vous invitant à vous lancer dans votre auto-hébergement de Wallabag.
Cet article est trop long ? Mettez-le dans votre Framabag et hop.

\n

Framablog : Salut Nicolas… Tu peux te présenter brièvement ?

\n

Salut ! Développeur PHP depuis quelques années maintenant (10 ans), j’ai voulu me remettre à niveau techniquement parlant (depuis 3 ans, j’ai pas mal lâché le clavier). Pour mes besoins persos, j’ai donc créé un petit projet pour remplacer une solution propriétaire existante. Sans aucune prétention, j’ai hébergé ce projet sur Github et comme c’est la seule solution open source de ce type, le nombre de personnes intéressées a augmenté …

\n

Les utilisateurs de services Framasoft ne le savent pas forcément, mais tu as déjà pas mal participé à la FramaGalaxie, non ?

\n

En effet. J’ai commencé un plugin pour Framanews, ttrss-purge-accounts, qui permet de nettoyer la base de données de comptes plus utilisés. Mais ce plugin a besoin d’être terminé à 100% pour être intégré au sein de Framanews (et donc de Tiny Tiny RSS), si quelqu’un souhaite m’aider, il n’y a aucun souci.
J’ai aussi fait 1 ou 2 apparitions dans des traductions pour Framablog. Rien d’extraordinaire, je ne suis pas bilingue, ça me permet de m’entraîner.

\n

Parlons de suite de ce qui fâche : ton application Wallabag, elle s’appellait pas “Poche”, avant ? Tu nous racontes l’histoire ?

\n

Euh en effet … Déjà, pourquoi poche ? Parce que l’un des trois « ténors » sur le marché s’appelle Pocket. Comme mon appli n’était destinée qu’à mon usage personnel au départ, je ne me suis pas torturé bien longtemps.

\n

Cet été, on a failli changer de nom, quand il y a eu de plus en plus d’utilisateurs. Et puis on s’est dit que poche, c’était pas mal, ça sonnait bien français et puis avec les quelques dizaines d’utilisateurs, on ne gênerait personne.

\n

C’est sans compter avec les sociétés américaines et leur fâcheuse manie de vouloir envoyer leurs avocats à tout bout de champ. Le 23 janvier, j’ai reçu un email de la part du cabinet d’avocats de Pocket me demandant de changer le nom, le logo, de ne plus utiliser le terme “read-it-later” (« lisez le plus tard ») et de ne plus dire que Pocket n’est pas gratuit (tout est parti d’un tweet où je qualifie Pocket de « non free » à savoir non libre). Bref, même si je semblais dans mon droit, j’ai quand même pris la décision de changer de nom et Wallabag est né, suite aux dizaines de propositions de nom reçues. C’est un mélange entre le wallaby (de la famille des kangourous, qui stockent dans leur poche ce qui leur est cher) et bag (les termes sac / sacoche / besace sont énormément revenus). Mais maintenant, on va de l’avant, plus de temps à perdre avec ça, on a du pain sur la planche.
\"wallaby crédit photo William Warby qui autorise explicitement toute réutilisation.

\n

Bon, alors explique-moi ce que je vais pouvoir faire avec Framabag…

\n

Alors Framabag, ça te permet de te créer un compte gratuitement et librement pour pouvoir utiliser Wallabag. Seule ton adresse email est nécessaire, on se charge d’installer et de mettre à jour Wallabag pour toi. Tu peux d’ailleurs profiter d’autres services proposés par Framasoft ici.

\n

À ce jour, il y a 834 comptes créés sur Framabag.

\n

Vous avez vraiment conçu ce service afin qu’on puisse l’utiliser avec un maximum d’outils, non ?

\n

Autour de l’application web, il existe déjà des applications pour smartphones (Android et Windows Phone), des extensions Firefox et Google Chrome.

\n

Comme Wallabag possède des flux RSS, c’est facile de lire les articles sauvegardés sur sa liseuse (si celle-ci permet de lire des flux RSS). Calibre (« logiciel de lecture, de gestion de bibliothèques et de conversion de fichiers numériques de type ebook ou livre électronique »,nous dit ubuntu-fr.org) intègre depuis quelques semaines maintenant la possibilité de récupérer les articles non lus, pratique pour faire un fichier ePub !

\n

D’autres applications web permettent l’intégration avec Wallabag (FreshRSS, Leed et Tiny Tiny RSS pour les agrégateurs de flux). L’API qui sera disponible dans la prochaine version de Wallabag permettra encore plus d’interactivité.

\n

Y a-t-il un mode de lecture hors ligne ou est-ce que c’est prévu pour les prochaines versions ?

\n

Il y a un pseudo mode hors ligne, disponible avec l’application Android. On peut récupérer (via un flux RSS) les articles non lus que l’on a sauvegardés. Une fois déconnecté, on peut continuer à lire sur son smartphone ou sa tablette les articles. Par contre, il manque des fonctionnalités : quand tu marques un article comme lu, ce n’est pas synchronisé avec la version web de Wallabag. J’espère que je suis presque clair dans mes explications.

\n

Pour la v2, qui est déjà en cours de développement, où je suis bien aidé par Vincent Jousse, on aura la possibilité d’avoir un vrai mode hors ligne.

\n

Alors si on veut aider / participer / trifouiller le code / vous envoyer des retours, on fait comment ?

\n

On peut aider de plusieurs façons :

\n

Le mot de la fin…?

\n

Merci à Framasoft d’accueillir et de soutenir Wallabag !

\n

La route est encore bien longue pour ne plus utiliser de solutions propriétaires, mais on devrait y arriver, non ?

\n

\"framasoft
hackez Gégé !

\n", + "6": "1", + "id": "1", + "title": "Framabag, un nouveau service libre et gratuit", + "url": "http://www.framablog.org/index.php/post/2014/02/05/Framabag-service-libre-gratuit-interview-developpeur", + "is_read": "0", + "is_fav": "0", + "content": "\n

Une interview de Nicolas, son développeur.

\n

Il ne vous a sûrement pas échappé que notre consommation de contenus du Web est terriblement chronophage et particulièrement frustrante tout à la fois : non seulement nous passons beaucoup (trop ?) de temps en ligne à explorer les mines aurifères de la toile, y détectant pépites et filons, mais nous sommes surtout constamment en manque. Même si nous ne sommes pas dans le zapping frénétique si facilement dénoncé par les doctes psychologues qui pontifient sur les dangers du numérique pour les jeunes cervelles, il nous vient souvent le goût amer de l’inachevé : pas le temps de tout lire (TL;DR est devenu le clin d’œil mi-figue mi-raisin d’une génération de lecteurs pressés), pas trop le temps de réfléchir non plus hélas, pas le temps de suivre la ribambelle de liens associés à un article…

\n

Pour nous donner bonne conscience, nous rangeons scrupuleusement un marque-page de plus dans un sous-dossier qui en comporte déjà 256, nous notons un élément de plus dans la toujours ridiculement longue toudouliste, bref nous remettons à plus tard, c’est-à-dire le plus souvent aux introuvables calendes grecques, le soin de lire vraiment un article jugé intéressant, de regarder une vidéo signalée par les rézossocios, de lire un chapitre entier d’un ouvrage disponible en ligne…

\n

Alors bien sûr, à défaut de nous donner tout le temps qui serait nécessaire, des solutions existent pour nous permettre de « lire plus tard » en sauvegardant le précieux pollen de nos butinages de site en site, et d’en faire ultérieurement votre miel ; c’est bel et bon mais les ruches sont un peu distantes, ça s’appelle le cloud (nos amis techies m’ont bien expliqué mais j’ai seulement compris que des trucs à moi sont sur des machines lointaines, ça ne me rassure pas trop) et elles sont souvent propriétaires, ne laissant entrer que les utilisateurs consommateurs payants et qui consentent à leurs conditions. Sans compter que de gros bourdons viennent profiter plus ou moins discrètement de toutes ces traces de nous-mêmes qui permettent de monétiser notre profil : si je collecte sur ces services (ne les nommons pas, justement) une série d’articles sur l’idée de Nature chez Diderot, je recevrai diverses sollicitations pour devenir client de la boutique Nature & Découverte du boulevard Diderot. Et si d’aventure les programmes de la NSA moulinent sur le service, je serai peut-être un jour dans une liste des militants naturistes indésirables sur les vols de la PanAm (je ne sais plus trop si je plaisante là, finalement…)

\n

La bonne idée : « se constituer un réservoir de documents sélectionnés à parcourir plus tard » appelait donc une autre bonne idée, celle d’avoir le contrôle de ce réservoir, de notre collection personnelle. C’est Nicolas Lœuillet, ci-dessous interviewé, qui s’y est collé avec une belle application appelée euh… oui, appelée Wallabag.

\n

Framasoft soutient d’autant plus son initiative qu’on lui a cherché des misères pour une histoire de nom et qu’il est possible d’installer soi-même une copie de Wallabag sur son propre site.

\n

Le petit plus de Framasoft, réseau toujours désireux de vous proposer des alternatives libératrices, c’est de vous proposer (sur inscription préalable) un accès au Framabag, autrement dit votre Wallabag sur un serveur Frama* avec notre garantie de confidentialité. Comme pour le Framanews, nous vous accueillons volontiers dans la limite de nos capacités, en vous invitant à vous lancer dans votre auto-hébergement de Wallabag.
Cet article est trop long ? Mettez-le dans votre Framabag et hop.

\n

Framablog : Salut Nicolas… Tu peux te présenter brièvement ?

\n

Salut ! Développeur PHP depuis quelques années maintenant (10 ans), j’ai voulu me remettre à niveau techniquement parlant (depuis 3 ans, j’ai pas mal lâché le clavier). Pour mes besoins persos, j’ai donc créé un petit projet pour remplacer une solution propriétaire existante. Sans aucune prétention, j’ai hébergé ce projet sur Github et comme c’est la seule solution open source de ce type, le nombre de personnes intéressées a augmenté …

\n

Les utilisateurs de services Framasoft ne le savent pas forcément, mais tu as déjà pas mal participé à la FramaGalaxie, non ?

\n

En effet. J’ai commencé un plugin pour Framanews, ttrss-purge-accounts, qui permet de nettoyer la base de données de comptes plus utilisés. Mais ce plugin a besoin d’être terminé à 100% pour être intégré au sein de Framanews (et donc de Tiny Tiny RSS), si quelqu’un souhaite m’aider, il n’y a aucun souci.
J’ai aussi fait 1 ou 2 apparitions dans des traductions pour Framablog. Rien d’extraordinaire, je ne suis pas bilingue, ça me permet de m’entraîner.

\n

Parlons de suite de ce qui fâche : ton application Wallabag, elle s’appellait pas “Poche”, avant ? Tu nous racontes l’histoire ?

\n

Euh en effet … Déjà, pourquoi poche ? Parce que l’un des trois « ténors » sur le marché s’appelle Pocket. Comme mon appli n’était destinée qu’à mon usage personnel au départ, je ne me suis pas torturé bien longtemps.

\n

Cet été, on a failli changer de nom, quand il y a eu de plus en plus d’utilisateurs. Et puis on s’est dit que poche, c’était pas mal, ça sonnait bien français et puis avec les quelques dizaines d’utilisateurs, on ne gênerait personne.

\n

C’est sans compter avec les sociétés américaines et leur fâcheuse manie de vouloir envoyer leurs avocats à tout bout de champ. Le 23 janvier, j’ai reçu un email de la part du cabinet d’avocats de Pocket me demandant de changer le nom, le logo, de ne plus utiliser le terme “read-it-later” (« lisez le plus tard ») et de ne plus dire que Pocket n’est pas gratuit (tout est parti d’un tweet où je qualifie Pocket de « non free » à savoir non libre). Bref, même si je semblais dans mon droit, j’ai quand même pris la décision de changer de nom et Wallabag est né, suite aux dizaines de propositions de nom reçues. C’est un mélange entre le wallaby (de la famille des kangourous, qui stockent dans leur poche ce qui leur est cher) et bag (les termes sac / sacoche / besace sont énormément revenus). Mais maintenant, on va de l’avant, plus de temps à perdre avec ça, on a du pain sur la planche.
\"wallaby crédit photo William Warby qui autorise explicitement toute réutilisation.

\n

Bon, alors explique-moi ce que je vais pouvoir faire avec Framabag…

\n

Alors Framabag, ça te permet de te créer un compte gratuitement et librement pour pouvoir utiliser Wallabag. Seule ton adresse email est nécessaire, on se charge d’installer et de mettre à jour Wallabag pour toi. Tu peux d’ailleurs profiter d’autres services proposés par Framasoft ici.

\n

À ce jour, il y a 834 comptes créés sur Framabag.

\n

Vous avez vraiment conçu ce service afin qu’on puisse l’utiliser avec un maximum d’outils, non ?

\n

Autour de l’application web, il existe déjà des applications pour smartphones (Android et Windows Phone), des extensions Firefox et Google Chrome.

\n

Comme Wallabag possède des flux RSS, c’est facile de lire les articles sauvegardés sur sa liseuse (si celle-ci permet de lire des flux RSS). Calibre (« logiciel de lecture, de gestion de bibliothèques et de conversion de fichiers numériques de type ebook ou livre électronique »,nous dit ubuntu-fr.org) intègre depuis quelques semaines maintenant la possibilité de récupérer les articles non lus, pratique pour faire un fichier ePub !

\n

D’autres applications web permettent l’intégration avec Wallabag (FreshRSS, Leed et Tiny Tiny RSS pour les agrégateurs de flux). L’API qui sera disponible dans la prochaine version de Wallabag permettra encore plus d’interactivité.

\n

Y a-t-il un mode de lecture hors ligne ou est-ce que c’est prévu pour les prochaines versions ?

\n

Il y a un pseudo mode hors ligne, disponible avec l’application Android. On peut récupérer (via un flux RSS) les articles non lus que l’on a sauvegardés. Une fois déconnecté, on peut continuer à lire sur son smartphone ou sa tablette les articles. Par contre, il manque des fonctionnalités : quand tu marques un article comme lu, ce n’est pas synchronisé avec la version web de Wallabag. J’espère que je suis presque clair dans mes explications.

\n

Pour la v2, qui est déjà en cours de développement, où je suis bien aidé par Vincent Jousse, on aura la possibilité d’avoir un vrai mode hors ligne.

\n

Alors si on veut aider / participer / trifouiller le code / vous envoyer des retours, on fait comment ?

\n

On peut aider de plusieurs façons :

\n

Le mot de la fin…?

\n

Merci à Framasoft d’accueillir et de soutenir Wallabag !

\n

La route est encore bien longue pour ne plus utiliser de solutions propriétaires, mais on devrait y arriver, non ?

\n

\"framasoft
hackez Gégé !

\n", + "user_id": "1", + "tags":"Framabag" + }, + { + "0": "2", + "1": "wallabag/wallabag", + "2": "https://github.com/wallabag/wallabag", + "3": "1", + "4": "0", + "5": "README.md

wallabag is a self hostable application allowing you to not miss any content anymore. Click, save, read it when you can. It extracts content so that you can read it when you have time.

\n

More informations on our website: wallabag.org

\n

License

\n

Copyright © 2010-2014 Nicolas Lœuillet nicolas@loeuillet.org This work is free. You can redistribute it and/or modify it under the terms of the Do What The Fuck You Want To Public License, Version 2, as published by Sam Hocevar. See the COPYING file for more details.

\n", + "6": "1", + "id": "2", + "title": "wallabag/wallabag", + "url": "https://github.com/wallabag/wallabag", + "is_read": "1", + "is_fav": "0", + "content": "README.md

wallabag is a self hostable application allowing you to not miss any content anymore. Click, save, read it when you can. It extracts content so that you can read it when you have time.

\n

More informations on our website: wallabag.org

\n

License

\n

Copyright © 2010-2014 Nicolas Lœuillet nicolas@loeuillet.org This work is free. You can redistribute it and/or modify it under the terms of the Do What The Fuck You Want To Public License, Version 2, as published by Sam Hocevar. See the COPYING file for more details.

\n", + "user_id": "1", + "tags":"" + }, + { + "0": "3", + "1": "a self hostable application for saving web pages | wallabag", + "2": "https://www.wallabag.org/", + "3": "1", + "4": "0", + "5": "\n
\n
\n

wallabag (formerly poche) is a self hostable application for saving web pages. Unlike other services, wallabag is free (as in freedom) and open source.

\n
\n\n
\n
\n
\n

With this application you will not miss content anymore. Click, save, read it when you want. It saves the content you select so that you can read it when you have time.

\n
\n\n
\n
\n
\n

How it works

\n

Thanks to the bookmarklet or third-party applications, you save an article in your wallabag to read it later. Then, when you open your wallabag, you can comfortably read your articles.

\n

How to use wallabag

\n

There are two ways to use wallabag: you can install it on your web server or you can create an account at Framabag (we install and upgrade wallabag for you).

\n
\n\n
\n", + "6": "1", + "id": "3", + "title": "a self hostable application for saving web pages | wallabag", + "url": "https://www.wallabag.org/", + "is_read": "1", + "is_fav": "0", + "content": "\n
\n
\n

wallabag (formerly poche) is a self hostable application for saving web pages. Unlike other services, wallabag is free (as in freedom) and open source.

\n
\n\n
\n
\n
\n

With this application you will not miss content anymore. Click, save, read it when you want. It saves the content you select so that you can read it when you have time.

\n
\n\n
\n
\n
\n

How it works

\n

Thanks to the bookmarklet or third-party applications, you save an article in your wallabag to read it later. Then, when you open your wallabag, you can comfortably read your articles.

\n

How to use wallabag

\n

There are two ways to use wallabag: you can install it on your web server or you can create an account at Framabag (we install and upgrade wallabag for you).

\n
\n\n
\n", + "user_id": "1" + }, + { + "0": "4", + "1": "Sans titre", + "2": "http:\/\/www.konradlischka.info\/2016\/01\/blog\/wie-ein-deutsches-start-up-mit-wagniskapital-die-marktluecke-fuer-lokalen-digitaljournalismus-schliessen-will\/", + "3": "0", + "4": "0", + "5": "[unable to retrieve full-text content]", + "6": "1", + "id": "4", + "title": "Sans titre", + "url": "http:\/\/www.konradlischka.info\/2016\/01\/blog\/wie-ein-deutsches-start-up-mit-wagniskapital-die-marktluecke-fuer-lokalen-digitaljournalismus-schliessen-will\/", + "is_read": "0", + "is_fav": "0", + "content": "[unable to retrieve full-text content]", + "user_id": "1", + "tags": "" + } +] diff --git a/tests/Wallabag/ImportBundle/fixtures/wallabag-v2-read.json b/tests/Wallabag/ImportBundle/fixtures/wallabag-v2-read.json new file mode 100644 index 00000000..3fa0bddf --- /dev/null +++ b/tests/Wallabag/ImportBundle/fixtures/wallabag-v2-read.json @@ -0,0 +1,28 @@ +[ + { + "id": 4668, + "title": "Wikimedia Foundation removes The Diary of Anne Frank due to copyright law requirements « Wikimedia blog", + "url": "https://blog.wikimedia.org/2016/02/10/anne-frank-diary-removal/", + "is_archived": true, + "is_starred": false, + "content": "

\"AnneFrankSchoolPhoto\"
Anne Frank in 1940. Photo by Collectie Anne Frank Stichting Amsterdam, public domain.

\n

Today, in an unfortunate example of the overreach of the United States’ current copyright law, the Wikimedia Foundation removed the Dutch-language text of The Diary of a Young Girl—more commonly known in English as the Diary of Anne Frank—from Wikisource.[1]

\n

We took this action to comply with the United States’ Digital Millennium Copyright Act (DMCA), as we believe the diary is still under US copyright protection under the law as it is currently written. Nevertheless, our removal serves as an excellent example of why the law should be changed to prevent repeated extensions of copyright terms, an issue that has plagued our communities for years.

\n

What prompted us to remove the diary?

\n

The deletion was required because the Foundation is under the jurisdiction of US law and is therefore subject to the DMCA, specifically title 17, chapter 5, section 512 of the United States Code. As we noted in 2013, “The location of the servers, incorporation, and headquarters are just three of many factors that establish US jurisdiction … if infringing content is linked to or embedded in Wikimedia projects, then  the Foundation may still be subject to liability for such use—either as a direct or contributory infringer.

\n

Based on email discussions sent to the Wikimedia Foundation at legal[at]wikimedia.org, we determined that the Wikimedia Foundation had either “actual knowledge” (i in the statute quoted below) or what is commonly called “red flag knowledge” (ii in the statute quoted below) that the Anne Frank text was hosted on Wikisource and was under copyright. The statute section states that a service provider is only protected by the DMCA when it:

\n

(i) does not have actual knowledge that the material or an activity using the material on the system or network is infringing;

\n

(ii) in the absence of such actual knowledge, is not aware of facts or circumstances from which infringing activity is apparent; or

\n

(The rest applies when we get a proper DMCA takedown notice.)

\n

Of particular concern, the US’ 9th Circuit Court of Appeals stated in their ruling for UMG Recordings, Inc. v. Shelter Capital Partners LLC that in circumstances where a hosting provider (like the Wikimedia Foundation) is informed by a third party (like an unrelated user) about infringing copyrighted content, that would likely constitute either actual or red flag knowledge under the DMCA.

\n

We believe, based on the detail and specificity contained in the emails, that we received that we had actual knowledge sufficient for the DMCA to require us to perform a takedown even in the absence of a demand letter.

\n

How is the diary still copyrighted?

\n

You may wonder why or how the Anne Frank text is copyrighted at all, as Anne Frank died in February 1945. With 70 years having passed since her death, the text may have passed into public domain in the Netherlands on January 1, 2016, where it was first published, although there is still some dispute about this.

\n

However, in the United States, the Anne Frank original text will be under copyright until 2042. This is the result of several factors coming together, and the English-language Wikipedia has actually covered this issue with a multi-part test on its non-US copyrights content guideline.

\n

In short, there are three major laws that together make the diary still copyrighted:

\n
  1. In general, the U.S. copyright for works published before 1978 is 95 years from date of publication. This came about because copyrights in the U.S. were originally for 28 years, with the ability to then extend that for a second 28 years (making a total of 56). Starting with the 1976 Copyright Act and extending to several more acts, the renewal became automatic and was extended. Today, the total term of works published before 1978 is 95 years from date of publication.
  2. \n
  3. Foreign works of countries that are treaty partners to the United States are covered as if they were US works.
  4. \n
  5. Even if a country was not a treaty partner under copyright law at the time of a publication, the 1994 Uruguay Round Agreements Act (URAA) restored copyright to works that:\n
    • had been published in a foreign country
    • \n
    • were still under copyright in that country in 1996
    • \n
    • and would have had U.S. copyright but for the fact they were published abroad.
    • \n
  6. \n
\n

Court challenges to the URAA have all failed, with the most notable (Golan v. Holder) resulting in a Supreme Court ruling that upheld the URAA.

\n

What that means for Anne Frank’s diary is unfortunately simple: no matter how it wound up in the United States and regardless of what formal copyright notices they used, the US grants it copyright until the year 2042, or 95 years after its original publication in 1947.

\n

Under current copyright law, this remains true regardless of its copyright status anywhere else in the world and regardless of whether it may have been in the public domain in the United States in the past.

\n

Jacob Rogers, Legal Counsel*
Wikimedia Foundation

\n

*Special thanks to Anisha Mangalick, Legal Fellow, for her assistance in this matter.

\n

[1] The diary text was originally located at https://nl.wikisource.org/wiki/Het_Achterhuis_(Anne_Frank).

\n

This article was edited to clarify that it is not just the location of the Wikimedia Foundation’s servers that determine whether we fall in US jurisdiction.

\n\t\t\t\t\t\t\t\t\t\t\t", + "mimetype": "text/html", + "language": "en", + "reading_time": 4, + "domain_name": "blog.wikimedia.org", + "tags": [] + }, + { + "id": 4667, + "title": "Tails - Tails 2.0.1 is out", + "url": "https://tails.boum.org/news/version_2.0.1/index.en.html", + "is_archived": false, + "is_starred": false, + "content": "
\n

This release fixes numerous security issues. All users must upgrade as soon as possible.

\n\n

New features

\n

Upgrades and changes

\n

Fixed problems

\n\n

See the current list of known issues.

\n

Go to the download or upgrade page.

\n

If your Tails does not boot after an automatic upgrade, please upgrade your Tails manually.

\n

The next Tails release is scheduled for March 08.

\n

Have a look at our roadmap to see where we are heading to.

\n

We need your help and there are many ways to contribute to Tails (donating is only one of them). Come talk to us!

\n
\n

Tags: announce

\n

Pages linking to this one: inc/stable i386 release notes security/Numerous security holes in 2.0

\n

Last edited Sat 13 Feb 2016 02:23:58 PM CET

\n
", + "mimetype": "text/html", + "language": "en", + "reading_time": 1, + "domain_name": "tails.boum.org", + "tags": [] + } +] diff --git a/tests/Wallabag/ImportBundle/fixtures/wallabag-v2.json b/tests/Wallabag/ImportBundle/fixtures/wallabag-v2.json new file mode 100644 index 00000000..37c59668 --- /dev/null +++ b/tests/Wallabag/ImportBundle/fixtures/wallabag-v2.json @@ -0,0 +1,346 @@ +[ + { + "id": "23", + "title": "Site d'information français d'actualités indépendant et participatif en ligne | Mediapart", + "url": "https://www.mediapart.fr/", + "is_archived": false, + "is_starred": false, + "content": "
Édition CAMédia\n

Deux nouvelles éditions pour débattre dans le club sur la laïcité et sur la démocratie

\n

18 janv. 2016 | Par

\n

CAMédia après un échange sur « l'éthique du débat » a lancé deux discussions , l'une sur le thème de la laïcité, l'autre ( encore en cours) sur celui de la démocratie. Nous sommes heureux de pouvoir signaler la création de deux nouvelles éditions participatives sur ces thèmes. Nous vous invitons à les lire et à participer à leurs débats.

\n
\n

De l'importance de rêver, éloge du merveilleux

\n

17 janv. 2016 | Par

\n

Je parlerai ici des rêves comme moteur de vie, de ces rêves qui vous rattachent et vous font espérer à ce qu’il y a de plus humain dans l’homme, même au milieu de la plus noire des détresses.

\n
\n

Fin(s) d'une toute-puissance

\n

18 janv. 2016 | Par

\n

En ce début d’année, je recommande la lecture du dernier ouvrage de Guillaume Duval, La France ne sera jamais plus une grande puissance ? Tant mieux !

\n
\n

L’Allier, département de destruction massive du tissu culturel

\n

18 janv. 2016 | Par

\n

Les temps sont durs pour les petites structures, les associations culturelles qui, de bourgades en villages, travaillent au cœur des régions. Leurs subventions sont souvent revues à la baisse. Le département de l’Allier les a carrément supprimées. Pour favoriser « l’événementiel ».

\n
Édition Les invités de Mediapart\n

La démocratie déjà attaquée par la coopération réglementaire transatlantique

\n

18 janv. 2016 | Par

\n

Lora Verheecke et David Lundy travaillent pour Corporate Europe Observatory, une ONG basée à Bruxelles qui enquête sur le pouvoir des lobbies des grandes entreprises sur la politique de l’Union européenne. Ils révèlent que depuis 25 ans le projet de « coopération réglementaire » mené par l’Union européenne et les États-Unis a été dominé par les grandes entreprises. ET que le TTIP cherche à entériner ce projet.

\n
\n

2016, une année test pour Jacob Zuma et son gouvernement

\n

18 janv. 2016 | Par

\n

Les turbulences de l’an passé ont toutes les chances de continuer à troubler le climat politique et social de l’Afrique du Sud en 2016. La situation exige des changements profonds dans la conduite des affaires du pays. Jacob Zuma tout en admettant la nécessité de ces changements, est-il l’homme de la situation ? Son gouvernement répondra-t-il aux attentes des citoyens sud-africains ?

\n
\n

Un mal fou (janvier 2016)

\n

14 janv. 2016 | Par

\n

J’ai une fringale d’aventure, d’aventures à venir. J’ai la fringale de la fringale des aventures et soudain, rupture. Je n’y arrive plus, tout est bloqué, tout empêché. Faut dire que depuis un an environ, tout est devenu plus compliqué. Ecrire va de moins en moins de soi.

\n
\n

Redoublement : le changement à bas bruit ?

\n

17 janv. 2016 | Par

\n

S’il est une caractéristique de la forme scolaire française bien établie dans la culture des personnels, des élèves et des parents, c’est bien le redoublement, censé sanctionner des résultats insuffisants pour envisager le passage dans la classe supérieure. Or, en ce domaine, l’évolution est nette.

\n
\n

Samedi-sciences (196): des chasseurs de mammouths en Arctique il y a 45 000 ans

\n

16 janv. 2016 | Par Michel de Pracontal

\n

Les restes d’un mammouth retrouvés en Arctique sibérien, datés de 45 000 ans, portent les traces de blessures infligées par des chasseurs humains. Les scientifiques pensaient jusqu’ici que notre espèce ne s’était pas aventurée dans cette région glaciale il y a plus de 30 000 ou 35 0000 ans. En réalité, des hommes ont réussi à survivre en Arctique au moins 10 000 ans plus tôt que l’on croyait.

\n
\n

De la démocratie, du citoyen et de l'éthique

\n

14 janv. 2016 | Par

\n

Trois ouvrages sont parus au Seuil, qui font état de la nécessité d’intégrer le citoyen dans la gouvernance de la nation. Non pas à titre consultatif mais doté d’un pouvoir délibératif pour constituer une contre-force face aux clans politico-financiers qui dominent la vie publique.

\n
", + "mimetype": "text/html", + "language": "fr", + "reading_time": 3, + "domain_name": "www.mediapart.fr", + "preview_picture": "https://www.mediapart.fr/images/social/800/mediapart.png", + "tags": [ + "mediapart", + "blog" + ] + }, + { + "id": 22, + "title": "Réfugiés: l'UE va créer 100 000 places d'accueil dans les Balkans", + "url": "http://www.liberation.fr/planete/2015/10/26/refugies-l-ue-va-creer-100-000-places-d-accueil-dans-les-balkans_1408867", + "is_archived": false, + "is_starred": false, + "content": "

Pour un sommet sur les réfugiés qui devait se concentrer sur des «mesures opérationnelles immédiates» dans les Balkans, la réunion, dimanche à Bruxelles, de 11 chefs d’Etat et de gouvernement, dont 8 Européens, a été agitée. Dès leur arrivée, Viktor Orbán (Hongrie) et Aléxis Tsípras (Grèce) se sont jeté des anathèmes. Le Premier ministre grec a dénoncé l’attitude «not in my backyard» (pas de ça chez moi) de certains Etats européens, alors que son pays est montré du doigt par d’autres dirigeants, dont Orbán : ils reprochent à la Grèce de ne pas suffisamment contrôler ses frontières avec la Turquie et ne pas montrer assez de zèle dans l’enregistrement des demandeurs d’asile.

\n

Le sommet, convoqué par la Commission européenne, sur suggestion de l’Allemagne, aura au moins permis à ces 11 Etats – Autriche, Bulgarie, Croatie, Allemagne, Grèce, Hongrie, Roumanie, Slovénie côté européen, et 3 pays «non UE», Albanie, Macédoine et Serbie – de discuter ensemble.

\n

400 policiers européens en Slovénie

\n

L’objectif, rappelé par Angela Merkel, était de trouver une «réponse coordonnée» à la crise des réfugiés. Quelques mesures ont été annoncées : 100 000 places d’accueil seront créées, dont 50 000 en Grèce, et le reste le long de la route des Balkans. 400 officiers de police de pays européens partiront en Slovénie, actuellement submergée, pour aider au contrôle des frontières. Frontex, l’agence européenne de surveillance des frontières, s’impliquera aux frontières gréco-macédonienne et gréco-albanaise pour des contrôles et identifications.

\n

Ce sommet est intervenu dans un contexte de fortes tensions, marqué par des fermetures de frontières bloquant les réfugiés dans des zones tampon. Ces obstacles ont été partiellement levés ces derniers jours, les autorités tentant d’organiser un «corridor» informel vers l’Allemagne, qui pourtant durcit sa politique d’accueil et souhaite désormais ralentir le flux. Mais la situation des réfugiés est catastrophique. L’ONG Human Rights Watch craint que des réfugiés ne meurent dans les Balkans. Des groupes de centaines, voire de milliers de personnes, bloqués près des postes-frontières, se retrouvent dans des conditions humanitaires intenables.

\n

Depuis mi-septembre, 250 000 personnes ont traversé les Balkans. En une semaine, la Slovénie a vu 60 000 réfugiés fouler le sol de son territoire. Dimanche, 15 000 personnes ont transité en Slovénie.

\n

Des zones tampon

\n

L’enjeu principal du sommet, aux yeux de nombreux Etats de l’Union européenne, était aussi que les pays des Balkans «prennent leur part» face à la crise : qu’ils accueillent et enregistrent davantage de réfugiés. Ces Etats craignent que l’Autriche ou l’Allemagne ne ferment leurs frontières et fassent de leurs pays des «zones tampon», comme s’en inquiétait Boyko Borissov, Premier ministre bulgare.

\n

« Aujourd’hui, plusieurs Etats du nord de l’Europe veulent que l’on enregistre les migrants puis que l’on détermine leur éligibilité au statut de réfugié, explique Marc Pierini, du think tank Carnegie Europe. La difficulté, c’est que les gens sont en mouvement. Pour le faire, il faut se poser quelque part. La crainte des pays intermédiaires, donc ceux des Balkans, est qu’on enregistre ces personnes sur leur territoire et qu’ils soient contraints de rester sur leur sol. Donc les pays des Balkans ne sont pas désireux d’accueillir ces réfugiés et ces derniers veulent avancer.»

\n

Le sommet a élaboré quelques principes. L’idée générale est de rendre effective la «logique de hotspot» : un enregistrement des demandeurs d’asile à leur point d’entrée dans l’Union européenne, suivi de l’expulsion de ceux qui ne correspondraient pas aux critères de la Convention de Genève, et la répartition des autres, via le mécanisme de relocalisation.

\n

Dans ce cadre, l’enregistrement des demandeurs d’asile est un élément clé. «Pas d’enregistrement, pas de droit», a prévenu le président de la Commission européenne, Jean-Claude Juncker, dimanche soir. Les Etats ont tenu à rappeler que les migrants qui refusent de demander l’asile à la frontière peuvent se voir refuser l’entrée dans un pays.

\n

Et les Etats «décourageront les mouvements de réfugiés» de frontière en frontière. La politique consistant à laisser passer les migrants vers un autre pays est officiellement jugée «inacceptable».

\n

Se jeter dans la gueule du loup

\n

Voilà pour la théorie. En pratique, la relocalisation ne devrait concerner que 160 000 réfugiés en deux ans, alors que près de 700 000 personnes sont arrivées en Europe depuis le début de l’année. De plus, les Etats ne jouent pas le jeu. La semaine passée, seules 854 places de relocalisation avaient été proposées.

\n

Dans ce contexte, il est probable que les Etats des Balkans ne s’impliqueront pas outre mesure dans les solutions proposées, craignant de devoir «garder» les réfugiés alors que l’Union européenne tarde à mettre en œuvre leur répartition.

\n

Quant aux réfugiés, ils préfèrent traverser les frontières par eux-mêmes, plutôt que de se jeter dans ces «hotspots», considérés comme la gueule du loup.

\nCédric Vallet", + "mimetype": "", + "language": "", + "reading_time": 4, + "domain_name": "www.liberation.fr", + "tags": [] + }, + { + "id": "21", + "title": "No title found", + "url": "http://news.nationalgeographic.com/2016/02/160211-albatrosses-mothers-babies-animals-science/&sf20739758=1", + "is_archived": false, + "is_starred": true, + "content": "Oh, what a shame, no content", + "mimetype": "", + "language": "", + "reading_time": 4, + "domain_name": "news.nationalgeographic.com", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 0, + "id": 612, + "title": "Échecs", + "url": "https://fr.wikipedia.org/wiki/Échecs", + "content": "
Un article de Wikipédia, l'encyclopédie libre.
\n\t\t\t\t\t\t\t\t\n\t\t\t\t\t\t\t\t\t\t\t\t\n\t\t\t\t
\"Fairytale Vous lisez un « article de qualité ».
\n
\"Page Cet article concerne le jeu appelé « les échecs ». Pour d'autres emplois du mot, voir échec.
\n
\n
\n
Jeu d’échecs
jeu de société
\n
\n
\"Description
\n\n\n\n\n\n\n
Formatplateau
Mécanismestratégie combinatoire abstrait
Joueur(s)2
\n\n\n\n\n
Données clés
habileté
\nphysique

\"\" Non
réflexion
\ndécision

\"\" Oui
générateur
\nde hasard

\"\" Non
info. compl.
\net parfaite

\"\" Oui
\n
\n\n
\n
\"\"\n\n
\n
\n
\n
\"\"\n
\n\nEnluminure, Liber de Moribus, vers 1300.
\n
\n
\n
\n
\"\"\n
\n\nJoueurs sur un échiquier géant
\n
\n
\n
\n
\"\"\n
\n\nTable échiquier au parc de la Tête d'Or à Lyon, France.
\n
\n
\n

Le jeu d’échecs (prononcer [eʃɛk]) oppose deux joueurs de part et d’autre d’un plateau ou tablier appelé échiquier composé de soixante-quatre cases claires et sombres nommées les cases blanches et les cases noires. Les joueurs jouent à tour de rôle en déplaçant l'une de leurs seize pièces (ou deux pièces en cas de roque), claires pour le camp des blancs, sombres pour le camp des noirs. Chaque joueur possède au départ un roi, une dame, deux tours, deux fous, deux cavaliers et huit pions. Le but du jeu est d'infliger à son adversaire un échec et mat, une situation dans laquelle le roi d'un joueur est en prise sans qu'il soit possible d'y remédier.

\n

Le jeu a été introduit dans le Sud de l'Europe à partir du Xe siècle par les Arabes, mais on ignore où il fut inventé exactement. Il dérive du shatranj ou chatrang qui lui-même est la version perse du chaturanga de l'Inde classique. Les règles actuelles se fixent à partir de la fin du XVe siècle. Le jeu d’échecs est l'un des jeux de réflexion les plus populaires au monde. Il est pratiqué par des millions de gens sous de multiples formes : en famille, entre amis, dans des lieux publics, en club, en tournoi, par correspondance, contre des machines spécialisées, entre ordinateurs, entre programmes, sur Internet, aux niveaux amateur et professionnel. Depuis son introduction en Europe, le jeu d'échecs jouit d'un prestige et d'une aura particulière qui du « jeu des rois » l’a fait devenir peu à peu « le roi des jeux » ou encore « le noble jeu », en référence à sa dimension tactique et à sa notoriété mondiale. Il a très largement inspiré la culture, en particulier la peinture, la littérature et le cinéma.

\n

La compétition aux échecs existe depuis les origines. On en trouve trace à la cour d'Haroun ar-Rachid au VIIIe siècle. Le premier tournoi de l'ère moderne a lieu à Londres en marge de l'Exposition universelle de 1851. La compétition est régie par la Fédération internationale des échecs (FIDE). Parallèlement, l'Association of Chess Professionals défend les intérêts des joueurs professionnels. Le premier champion du monde d'échecs est Wilhelm Steinitz en 1886 ; le champion en titre est le Norvégien Magnus Carlsen depuis 2013.

\n

Une théorie du jeu, développée depuis son invention et de façon intensive par les joueurs de premier plan de l'époque moderne, est transmise au travers d'une littérature échiquéenne abondante. La théorie des jeux (mathématique) décrit quant à elle les échecs comme un jeu de stratégie combinatoire abstrait de réflexion pure, fini, sans cycle et à information complète et parfaite. L'absence de cycle est garantie par les règles de nulle : répétition de position, règle des cinquante coups et impossibilité de mater.

\n

Un des objectifs des premiers informaticiens a été de mettre au point des machines capables de jouer aux échecs. De nos jours, le jeu est profondément influencé par les capacités des programmes joueurs d'échecs, ainsi que par la possibilité de jouer sur Internet. En 1997, Deep Blue devient le premier ordinateur à battre un champion du monde en titre dans un match qui l'oppose à Garry Kasparov.

\n

La composition échiquéenne, la forme artistique du jeu, a produit des centaines de milliers de problèmes dans de multiples genres. Cette discipline est également sous l'égide de la FIDE, qui organise des concours spécifiques pour les compositeurs de problème et les solutionnistes. Elle édite l'Album FIDE, un recueil trisannuel des meilleures compositions.

\n

\n\n

\n

Règles du jeu[ | modifier le code]

\n
Article détaillé : Règles du jeu d'échecs.
\n

Présentation[ | modifier le code]

\n
\n
\"ChessStartingPosition.jpg\"\n
\n\n
\n
\n
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Tour
\n
\"Cavalier
\n
\"Fou
\n
\"Reine
\n
\"Roi
\n
\"Fou
\n
\"Cavalier
\n
\"Tour
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Tour
\n
\"Cavalier
\n
\"Fou
\n
\"Reine
\n
\"Roi
\n
\"Fou
\n
\"Cavalier
\n
\"Tour
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Position initiale d'une partie d'échecs.
\n
\n
\n

Une partie d'échecs commence dans la position initiale ci-contre, les blancs jouent le premier coup puis les joueurs jouent à tour de rôle en déplaçant à chaque fois une de leurs pièces (deux dans le cas d'un roque)[G 1]. Chaque pièce se déplace de façon spécifique, il n'est pas possible de jouer sur une case occupée par une pièce de son propre camp. Lorsqu'une pièce adverse se trouve sur la case d'arrivée de la pièce jouée, elle est capturée et retirée de l'échiquier. Gagner du matériel (des pièces) est un moyen pour gagner la partie, mais ne suffit pas toujours pour y parvenir.

\n

Il existe des règles spéciales lors du déplacement de certaines pièces : le roque, qui permet le déplacement simultané du roi et de l'une des tours ; la prise en passant, qui permet une capture particulière des pions ; et la promotion des pions, qui permet de les transformer en une pièce maîtresse de son choix (sauf le roi) lorsqu'ils atteignent la dernière rangée de l'échiquier[G 2].

\n

Lorsqu'un roi est menacé de capture, on dit qu'il est en échec. Si cette menace est imparable (on peut tenter de parer la menace en déplaçant le roi, en interposant une pièce ou en capturant la pièce attaquante) on dit qu'il y a échec et mat et la partie se termine sur la victoire du joueur qui mate. Il est interdit de mettre son propre roi en échec ou de le faire passer sur une ligne d'échec pendant le roque. Il est également interdit de roquer quand le roi est en échec sur sa case de départ. Si cela arrive (par inadvertance entre débutants) on doit reprendre le coup[G 3].

\n

Si un camp ne peut plus jouer aucun coup légal (cela arrive par exemple avec un roi seul et l'ensemble de ses pions bloqués) et si son roi n'est pas en échec, on dit alors qu'il s'agit d'une position de pat. Quel que soit le matériel dont le camp adverse dispose, la partie est déclarée nulle, c'est-à-dire sans vainqueur[G 4].

\n

Le but du jeu est donc d'infliger un échec et mat à son adversaire. Le terme échec et mat vient de sāh māta (en persan, soit as-sāh māt(a) الشّاهُ ماتَ en arabe), « le roi est mort », pour indiquer la défaite du roi. Le mot sāh (« roi » en persan) est à l'origine du mot échec et du nom des échecs dans un grand nombre de langues[1].

\n
\n

Déplacements des pièces[ | modifier le code]

\n

Chaque pièce peut se déplacer au choix du joueur sur l'une des cases marquées d'une croix. Hormis le pion, elles capturent une pièce adverse qui se trouve sur leur trajectoire, sans pouvoir aller au-delà. À l'exception de la prise en passant, la pièce qui capture prend la place de la pièce capturée, cette dernière étant définitivement retirée de l'échiquier.

\n
  • Le fou, la tour et la dame sont des pièces à longue portée (ou pièces de lignes) : elles peuvent se déplacer le long de lignes. Chaque camp possède deux fous : ils se déplacent toujours sur les cases d'une même couleur, en diagonales; chaque camp possède donc un fou de cases blanches, et un fou de cases noires.
  • \n
Déplacements du fou de cases blanches\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Case
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\"Fou
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
Déplacements de la tour\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Tour
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
Déplacements de la dame\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Reine
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
\n
  • Le roi se déplace d'une seule case à la fois, il dispose d'une règle de déplacement spéciale : le roque.
  • \n
  • Le cavalier ne peut être intercepté par aucune des pièces autour de lui, il saute jusqu'à sa case d'arrivée.
  • \n
Déplacements du roi\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\"Case
\n
\"Roi
\n
\"Case
\n
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\n
\"Case
\n
\"Roi
\n
\"Case
\n
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
Déplacements du cavalier\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\"Cavalier
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\"Cavalier
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
La tour peut capturer la dame\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Reine
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Tour
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
\n
  • Le pion peut se déplacer sur les cases marquées d'une croix (sans pouvoir y capturer une pièce adverse), et peut capturer sur les cases marquées d'un rond (sans pouvoir s'y déplacer si elles sont vides).
  • \n
Chacun des pions peut se déplacer de deux cases à la fois lors de son tout premier déplacement (ex. les pions f2 et g7 dans les diagrammes ci-dessous). Par contre, déplacer deux pions d'une case en un seul coup (une légende due à une mauvaise traduction d'un livre allemand[2]) est interdit dans la règle officielle du jeu d'échecs. Les pions disposent d'une règle de capture spéciale : la prise en passant.
\n
Les pions ne peuvent jamais reculer, les pions blancs se dirigent vers la huitième rangée, les pions noirs se dirigent vers la première rangée, et sont obligatoirement promus dès qu'ils l'atteignent.
\n
Déplacements d'un pion blanc\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
Déplacements d'un pion noir\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\"Pion
\n
\n
\n
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
Le pion peut capturer le cavalier\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\"Cavalier
\n
\"Case
\n
\"Case
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
\n
\n
\n

Règles spéciales[ | modifier le code]

\n

Roque[ | modifier le code]

\n
Article détaillé : Roque (échecs).
\n
\n
\"\"\n
\n\n
Le petit roque, noté 0-0
\n
\n
\n
\n
\n
\"\"\n
\n\n
Le grand roque, noté 0-0-0
\n
\n
\n
\n

Le roque consiste à déplacer en un seul coup le roi et l'une des tours. Il y a deux façons de roquer :

\n
  • avec le roi et la tour de la colonne h, ce déplacement s'appelle le « petit roque » ;
  • \n
  • avec le roi et la tour de la colonne a, ce déplacement s'appelle le « grand roque » car la tour effectue un déplacement plus grand (une case de plus).
  • \n

Dans les deux cas, on procède ainsi : on déplace d'abord le roi de deux cases vers la tour puis, avec la même main, on fait passer la tour de l'autre côté, juste à côté du roi (voir le diagramme ci-contre).

\n

Les conditions suivantes sont nécessaires pour pouvoir roquer :

\n
  1. aucune pièce ne se trouve entre le roi et la tour concernée ;
  2. \n
  3. le roi et la tour concernée n'ont encore jamais joué ;
  4. \n
  5. le roi n'est pas en échec ;
  6. \n
  7. la case traversée par le roi n'est contrôlée par aucune pièce adverse.
  8. \n

Remarques :

\n
  • La dernière règle s'explique ainsi : le roi joue deux coups en un ; par exemple, dans le petit roque, le Roi va en f1, puis en g1. Donc, conformément aux règles, il ne peut se mettre en échec sur la case intermédiaire f1, ni sur la case g1 (le roi n'a pas le droit de se mettre en échec de lui-même).
  • \n
  • Le roi et la tour ne devant jamais avoir joué, chaque camp ne peut faire qu'un seul roque dans une partie, que ce soit un petit ou un grand roque.
  • \n
  • La tour, par contre, peut être attaquée par une pièce adverse : la case a1 (a8 pour les noirs) lors du grand roque, et h1 (h8 pour les noirs) lors du petit roque peut être contrôlée par une pièce adverse. Lors du grand roque, la case b1 (b8 pour les noirs) peut, elle aussi, être contrôlée par une pièce adverse, puisque le roi n'y va pas.
  • \n
\n

Prise en passant[ | modifier le code]

\n
Article détaillé : Prise en passant.
\n
\n
\"\"\n
\n\n
La prise en passant, notée e.p.
\n
\n
\n
\n

La prise en passant peut intervenir lorsqu'un camp vient de jouer un pion de deux cases (c'est possible lors d'un tout premier déplacement du pion) et, ce faisant, évite la confrontation avec un pion adverse. Dans l'exemple ci-contre, les blancs jouant a2-a4 évitent la rencontre entre le pion blanc a2 et le pion noir b4.

\n

Toutefois, la règle du déplacement d'un pion de deux cases s'interprète ainsi : le pion joue deux coups en un, tout d'abord un coup d'une case (a2-a3 dans notre exemple), puis un second coup du même pion d'une case (a3-a4). Dans ces conditions le camp adverse peut considérer qu'après le premier coup il est en droit lui-même de capturer le pion déplacé : c'est ce qu'il fait effectivement grâce à la prise en passant, bxa3 dans l'exemple.

\n

De façon cohérente, le pion capture sur la première case, c'est-à-dire a3 dans notre exemple, et le pion capturé est bien retiré de l'échiquier.

\n

Remarques :

\n
  • La prise en passant n'est pas obligatoire.
  • \n
  • Le camp qui prend en passant doit le faire immédiatement, au coup suivant cette possibilité disparait.
  • \n
  • La prise en passant est notée (de façon optionnelle) en ajoutant e.p. après le coup, exemple : bxa3 e.p.
  • \n
\n

Pat[ | modifier le code]

\n
Article détaillé : Pat (échecs).
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Roi
\n
\"Tour
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Roi
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
Les blancs au trait sont pat
\n
\n
\n
\n

Le pat est une situation particulière dans laquelle un camp au trait ne peut jouer aucun coup légal, sans pour autant que son roi soit en échec. La partie se termine immédiatement et elle est déclarée nulle, c'est-à-dire sans vainqueur.

\n

Dans le diagramme ci-contre, les blancs au trait n'ont aucun coup légal car on n'a pas le droit de se mettre en échec volontairement, et le pion blanc est bloqué. Puisqu'il n'y a pas échec, c'est un pat et la partie est déclarée nulle.

\n

Promotion[ | modifier le code]

\n
Article détaillé : Promotion (échecs).
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\"Roi
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Roi
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
\n
Promotion et sous-promotion
\n
\n
\n
\n

La promotion du pion[G 5] consiste à le transformer, au choix du joueur et indépendamment des pièces antérieurement perdues, en dame, en tour, en fou ou en cavalier de même couleur lorsqu'il atteint la dernière rangée de l'échiquier (la huitième pour les blancs et la première pour les noirs). Dans le diagramme ci-contre, les blancs peuvent jouer leur pion en f8 et le transformer en dame, en tour, en fou ou en cavalier. Lorsqu'un pion atteint la dernière rangée, il est obligatoire de le promouvoir immédiatement, on ne peut ni le laisser inchangé ni reporter la promotion à plus tard.

\n

La sous-promotion consiste, lors de la promotion, à choisir une pièce autre que la dame, qui est normalement privilégiée car c'est la pièce la plus puissante du jeu. C'est parfois utile comme dans le diagramme ci-contre, en effet on se rend compte qu'après avoir joué f8=D, le roi noir ne dispose d'aucun coup légal. Les noirs, n'ayant pas d'autre pièce à jouer, sont pat et la partie est nulle, bien que les blancs aient une dame en plus.

\n

En conséquence, les blancs choisissent de faire une sous-promotion en tour : f8=T, les noirs ne sont pas pat car ils peuvent jouer Rg7 et les blancs gagnent cette finale théorique facile. Si les blancs choisissaient de sous-promouvoir leur pion en fou ou en cavalier la partie serait nulle car il n'est pas possible de mater avec R + F contre R seul, ou R + C contre R seul.

\n
\n

Fin de la partie[ | modifier le code]

\n
\n
\"\"\n
\n\nFin de partie dans un style théâtral.
\n
\n
\n

Toutes les parties ne se terminent pas nécessairement par un échec et mat.

\n

Les parties peuvent se terminer par une victoire pour un camp associé à une défaite pour l'autre camp. Plusieurs cas de figure peuvent se présenter :

\n
  • échec et mat,
  • \n
  • abandon d'un joueur,
  • \n
  • perte au temps : dans une partie à la pendule, un des deux joueurs peut être à court de temps de réflexion et finir par perdre pour dépassement de son quota, cela même si sa position est nettement supérieure,
  • \n
  • par décision de l'arbitre, pour non-respect du règlement (retard ou absence à une partie, tricherie, sonnerie de téléphone pendant la partie).
  • \n

Les parties peuvent se terminer par une partie nulle, c'est-à-dire sans vainqueur. Plusieurs cas de figure peuvent se présenter :

\n
  • Par accord mutuel entre les deux joueurs pendant la partie.
  • \n
  • À partir de la troisième répétition d'une même position avec le même joueur ayant le trait. Cette nulle est obtenue uniquement si un joueur l'exige, l'autre joueur ne peut pas s'y opposer.
  • \n
  • En vertu de la règle des 50 coups.
  • \n
  • Par l'impossibilité de mater : s'il n'existe aucune suite de coups (légaux) qui peut mener au mat de l'un ou de l'autre joueur. Cette impossibilité de mater met fin à la partie immédiatement, aucun joueur ne peut s'y opposer.
  • \n
  • Lorsque survient un pat.
  • \n
  • Lorsqu'un joueur perd au temps et que l'autre joueur n'a pas suffisamment de matériel pour gagner. Cette nulle est obtenue automatiquement : aucun joueur ne peut s'y opposer.
  • \n

Notation des parties[ | modifier le code]

\n
Article détaillé : Notation algébrique.
\n
\n
\"\"\n
Notation algébrique des coups. À l'intersection de la colonne g et de la rangée numéro 5 se trouve la case g5.
\n
\n
\n
\n
\"\"\n
\n\nUne feuille de partie Réti contre Capablanca, en 1924.
\n
\n
\n

En compétition, il est obligatoire de noter les coups joués[3], afin de permettre le contrôle de la partie par l'arbitre, son archivage par l'organisateur et sa publication dans des livres, revues, sites web ou bases de données. À cette fin, divers systèmes de notation ont été proposés et utilisés, dont la notation descriptive, très populaire dans les pays anglo-saxons et hispaniques. De nos jours, on utilise mondialement la notation algébrique abrégée, qui est le système officiel de la FIDE[4].

\n

Dans la notation algébrique, chaque colonne de l'échiquier est désignée par une lettre de a à h et chaque rangée est désignée par un chiffre de 1 à 8, la case a1 étant placée à la gauche des blancs. Les cases de l’échiquier peuvent donc être désignées par la combinaison d'une lettre et d'un chiffre (voir la case g5 sur le diagramme ci-contre).

\n

Pour l'enregistrement de la partie, on utilise habituellement un formulaire ad hoc, appelé feuille de partie, dont le format peut varier (voir un exemple ci-contre).

\n

Pour chaque coup on note :

\n
  • le numéro du coup, suivi d'un point (puis, de façon optionnelle une espace)
  • \n
  • l'initiale de la pièce jouée (R pour Roi, D pour Dame, T pour Tour, F pour Fou et C pour Cavalier, l'initiale du pion étant omise, bien qu’anciennement utilisée)
  • \n
  • la case d'arrivée de la pièce jouée (une lettre + un chiffre)
  • \n

Exemples :

\n
  • 1. Cf3 indique qu'au premier coup des blancs, ceux-ci ont joué leur cavalier de la case g1 à la case f3 (g1 étant la case initiale du cavalier au début du jeu).
  • \n
  • 1. e4 indique qu'au premier coup des blancs, ceux-ci ont joué leur pion de la case e2 à la case e4 (la lettre identifiant le pion n'est pas indiquée).
  • \n

On fait suivre le coup noir sans répéter le numéro du coup.

\n

Exemples :

\n
  • 1. Cf3 Cf6 indique qu'au premier coup des noirs, ceux-ci ont joué leur cavalier de la case g8 à la case f6
  • \n
  • 1. e4 e5 indique qu'au premier coup des noirs, ceux-ci ont joué leur pion de la case e7 à la case e5
  • \n

On n'indique pas la case de départ de la pièce, en général ce n'est pas nécessaire car une seule pièce du type mentionné peut atteindre la case d'arrivée. En cas d'ambiguïté, on ajoute devant la case d'arrivée une lettre ou un chiffre permettant d'identifier la colonne ou la rangée de départ de la pièce concernée.

\n

Exemple :

\n
  • 1. e4 e5 2. Cc3 Cc6 3. Cge2 indique que c'est le cavalier venant de la case g1 qui se déplace en e2 (et non celui étant en c3 dans la position initiale).
  • \n

Lorsque la pièce jouée capture une pièce adverse, on le mentionne en ajoutant une croix entre l'initiale de la pièce et la case d'arrivée.

\n

Exemple :

\n
  • 1. Cf3 e5 2. Cxe5 indique que le cavalier en f3 capture le pion noir en e5.
  • \n

Lorsqu'on indique un coup noir après un commentaire écrit, on le fait précéder d'un point de suspension.

\n

Exemple :

\n
  • 1. e4 ouverture du pion roi, 1…e5 (les noirs viennent de jouer leur pion en e5).
  • \n

Notation des coups spéciaux[ | modifier le code]

\n

Le roque est noté 0-0 pour le petit roque, et 0-0-0 pour le grand roque.

\n

La prise en passant se note comme une prise normale, on mentionne la case d'arrivée du pion. On peut ajouter la mention e.p. après le coup, de façon optionnelle pour faciliter la lecture.

\n

La promotion d'un pion en pièce se note en indiquant le type de pièce en laquelle le pion est promu soit à la fin du coup (exemple : e8D, noté aussi e8=D).

\n

Lorsque le roi adverse se trouve en échec, on ajoute communément un « + » à la suite du coup, exemple : Dh4+.

\n

Si le roi est échec et mat, on utilise traditionnellement le symbole « ≠ » (éventuellement précédé d'une espace), ou plus récemment le symbole « # », ou bien on écrit mat. Exemple : Dxf7≠, Dxf7 # ou Dxf7 mat.

\n

Le signe « ++ » est également utilisé pour indiquer un échec et mat selon le règlement de la FIDE. Certains auteurs l'utilisent cependant pour marquer un échec double.

\n

Notation avec figurines[ | modifier le code]

\n

Dans de nombreuses revues internationales, les initiales des pièces sont remplacées par des figurines schématisant chaque pièce, contournant ainsi le barrage de la langue. D'autre part, la notation est parfois encore abrégée en omettant le signe de la prise (x) et le numéro de rangée pour les prises de pion (ainsi, exd4 devient exd, ou ed, pour autant qu'il n'y ait pas d'ambiguïté possible).

\n

Les figurines ressemblent à ceci : \"Chess \"Chess \"Chess \"Chess \"Chess \"Chess.

\n

Annotation des parties[ | modifier le code]

\n
Article détaillé : Annotation (échecs).
\n

Lors d'une analyse de partie, le commentateur a souvent besoin de donner son avis sur un coup joué. On a donc intégré au système de notation des symboles, insérés juste après le coup, permettant de donner de manière simple un avis sur le coup.

\n

Les plus fréquemment utilisés par les joueurs sont :

\n
  • ! : bon coup. C'est souvent un petit avantage (voir plus bas).
  • \n
  • !! : très bon coup. C'est souvent un avantage décisif (voir plus bas).
  • \n
  • ? : mauvais coup.
  • \n
  • ?? : très mauvais coup. Conduit généralement à la perte de la partie.
  • \n

D'autres symboles sont possibles :

\n
  • !? : coup intéressant
  • \n
  • ?! : coup douteux
  • \n
  • N : Nouveauté théorique : un coup inédit dans la « théorie des ouvertures », à un haut niveau de compétition (généralement entre grands maîtres).
  • \n

De même, il est souvent utile, à la fin de l'analyse d'une variante, de donner un avis sur la position résultant de cette suite de coups. Là aussi, des symboles ont été intégrés à la notation pour faciliter cette tâche :

\n
  • +- : avantage décisif aux blancs
  • \n
  • += : léger avantage aux blancs
  • \n
  • = : position équilibrée
  • \n
  • =+ : léger avantage aux noirs
  • \n
  • -+ : avantage décisif aux noirs
  • \n
  • : position incertaine
  • \n
  • =/∞ : avec compensation pour un désavantage matériel
  • \n

Notation informatique (PGN et FEN)[ | modifier le code]

\n
Article détaillé : Portable Game Notation.
\n

Le format PGN vise à standardiser le format utilisé pour décrire une partie d'échecs à destination des programmes informatiques. Il se compose d'une partie d'en-têtes qui donnent des informations au sujet des joueurs, de la date et du lieu de la partie, de la cadence, etc.

\n

Ces en-têtes sont suivis par les coups joués, décrits en format SAN (Standard Algebraic Notation). Le format SAN, qui fait partie de la spécification PGN, est très similaire à la notation algébrique abrégée en langue anglaise (K=Roi, Q=Dame, B=Fou, N=Cavalier, R=Tour) mais en diffère cependant quelque peu (par exemple, en cas de promotion, le signe = est obligatoire: e8=Q tandis qu'en notation algébrique abrégée, ce signe est omis : e8Q).

\n

Le standard FEN (Forsyth-Edwards Notation) est utilisé pour décrire une position.

\n

Chess Query Language (CQL) est un langage de requête qui permet d'extraire des parties ou des positions d'une base de données de parties d'échecs.

\n

Principes de jeu[ | modifier le code]

\n

La stratégie concerne l'évaluation globale de la position et l'établissement de plans à long terme, par exemple le positionnement des pièces et leur coordination, ou l'attaque dans un secteur donné de l’échiquier, alors que la tactique concerne la réalisation de manœuvres immédiates qui découlent des éléments stratégiques mis en place. Le grand maitre Xavier Tartacover, a dit un jour à ce sujet, que : « La Tactique consiste à savoir ce qu'il faut faire quand il y a quelque chose à faire. La Stratégie consiste à savoir ce qu'il faut faire quand il n'y a rien à faire ! »

\n

On distingue généralement trois phases dans le déroulement d'une partie d'échecs : l'ouverture qui dure de 10 à 25 coups et pendant laquelle les joueurs développent leurs pièces en prévision de la bataille à venir ; le milieu de partie qui est en général la période la plus combative avec éventuellement des attaques directes sur les rois ; et enfin la finale, lorsque le matériel est réduit, les rois y prennent une part plus active et la promotion des pions est souvent un objectif décisif. Chacune de ces phases fait intervenir à des degrés divers des éléments tactiques, stratégiques et psychologiques.

\n

Stratégie[ | modifier le code]

\n
Article détaillé : Stratégie échiquéenne.
\n\n\n\n
Visualisation de la structure de pions
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Tour
\n
\n
\"Fou
\n
\n
\"Tour
\n
\n
\"Roi
\n
\n
\"Pion
\n
\"Pion
\n
\n
\"Cavalier
\n
\n
\"Pion
\n
\"Fou
\n
\"Pion
\n
\n
\n
\"Pion
\n
\"Tour
\n
\n
\"Cavalier
\n
\"Pion
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\"Cavalier
\n
\n
\"Fou
\n
\"Cavalier
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\"Roi
\n
\n
\n
\"Fou
\n
\n
\"Tour
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Une partie Tarrasch – Euwe de 1922
\nAprès 12…Te8
\n
\n
\n
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\"Pion
\n
\n
\"Pion
\n
\n
\n
\"Pion
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
et son squelette de pion :
\nLa formation Rauzer
\n
\n
\n

L'étape la plus élémentaire dans l'évaluation de la position consiste à compter le matériel de chaque camp. L'expérience permet d'attribuer à chaque type de pièce un certain nombre de points, 1 point pour chaque pion, 3 points pour un cavalier ou un fou, 5 points pour une tour et 9 points pour la dame. Les cavaliers valent un peu plus que les fous dans les positions fermées (encombrées) typiquement en début de partie et à l'inverse les fous valent davantage que les cavaliers dans les positions ouvertes ou en fin de partie. Par ailleurs, deux tours (10 points) valent généralement plus qu'une dame (9 points). Ce décompte est une bonne illustration de la valeur relative des pièces mais les joueurs expérimentés n'ont pas besoin de s'y livrer, ils savent à tout moment où ils en sont. Pour une évaluation précise on prend en compte des considérations positionnelles, par exemple des pions avancés sont un atout ou inversement une faiblesse s'ils sont difficiles à soutenir, une paire de fous (contre fou + cavalier) est appréciée pour sa facilité à contrôler à la fois les cases blanches et les cases noires de l'échiquier.

\n

Un autre facteur important dans l'évaluation de la position est la prise en compte de la structure de pions, également appelée squelette de pions, ou la répartition dissymétrique des pions sur chaque aile de l'échiquier. Les pions sont peu mobiles et leur configuration détermine largement la stratégie de la partie. Les faiblesses créées dans leur structure (pions isolés, doublés, arriérés, trous dans la chaîne de pions) sont souvent permanentes, aussi doivent-elles être soigneusement évitées ou bien compensées, par exemple par des possibilités d'attaque.

\n

Le diagramme ci-contre, tiré d'une partie Siegbert Tarrasch - Max Euwe de 1922, montre la difficulté qu'il peut y avoir à évaluer certaines positions. En effet l'intuition de nombreux joueurs est ici prise en défaut : Le fou noir est bloqué par son propre pion en e5 et les blancs peuvent exploiter le trou en d6, cependant l'expérience montre que la faiblesse blanche en d4 est plus grave encore : la théorie considère que les noirs ont de meilleures perspectives[5].

\n

Tactique[ | modifier le code]

\n
Article détaillé : Tactique échiquéenne.
\n
Botvinnik - Ioudovitch,
championnat de l'URSS 1933[6]
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Tour
\n
\n
\"Fou
\n
\n
\n
\"Tour
\n
\n
\n
\n
\"Pion
\n
\n
\"Cavalier
\n
\"Reine
\n
\n
\"Fou
\n
\n
\n
\"Cavalier
\n
\"Pion
\n
\n
\"Pion
\n
\n
\"Roi
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\"Cavalier
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\"Fou
\n
\n
\n
\n
\n
\n
\"Reine
\n
\n
\"Fou
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\"Tour
\n
\"Tour
\n
\n
\n
\"Roi
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Exemple d'un sacrifice de pièce qui expose le roi noir. Après 1. Fh5+ les noirs abandonnent car le mat est inévitable, par exemple 1…Rxh5 2. Cg3+ Rh4 3. De4+ Tf4 4. Dxf4≠, ou 1…Rh7 2. Cf6+ Rh8 3. Dh7≠.
\n
\n
\n

La tactique concerne habituellement des actions à très court terme, au point qu'elles peuvent être complètement calculées par le joueur[G 6]. La profondeur du calcul, c'est-à-dire le nombre de coups de la variante la plus longue, dépend des capacités du joueur, ou de la puissance de l'ordinateur le cas échéant. Dans les positions tranquilles, avec de nombreuses alternatives de part et d'autre, il y a peu de chances qu'un calcul profond soit possible, alors que dans les positions comportant un nombre limité de coups forcés, les joueurs les plus forts sont à même de calculer de très longues séquences de coups.

\n

Des suites forcées d'un ou deux coups, les menaces, échanges de pièces, attaques doubles, etc. peuvent être enchaînés dans des combinaisons : des séquences de manœuvres souvent forcées pour l'un ou l'autre des deux camps. Les théoriciens ont décrit un grand nombre de méthodes élémentaires et de manœuvres caractéristiques comme le clouage, la fourchette, l'enfilade, la batterie, l'attaque à la découverte et en particulier l'échec à la découverte, le coup intermédiaire (ou zwischenzug), la déviation, le leurre, le sacrifice, le minage, la surcharge, l'interception[G 7].

\n

Ouverture[ | modifier le code]

\n
Article détaillé : Ouverture (échecs).
\n

L'ouverture est le nom donné aux premiers coups d'une partie[G 8]. On donne aux ouvertures reconnues des noms comme la partie espagnole ou la défense sicilienne mais également la partie des quatre cavaliers. Un grand nombre d'ouvrages spécialisés les répertorient, comme, par exemple, l'Encyclopédie des ouvertures d'échecs.

\n

Il existe des dizaines d'ouvertures aux styles très variés, certaines sont tranquilles comme le début Réti alors que d'autres, comme le gambit letton, sont très agressives. Les variantes comportent en général de 10 à 15 coups, mais certaines variantes, dans lesquelles on estime que ne sont joués que les meilleurs coups de part et d'autre, peuvent comporter jusqu'à 30 ou 35 coups. Les joueurs professionnels passent des années à étudier les ouvertures et continuent à les approfondir leur carrière durant, participant eux-mêmes à leur étude systématique. En effet, au plus haut niveau de jeu le début de partie se présente comme un duel de connaissances entre deux compétiteurs ainsi qu'un laboratoire permanent permettant de tester les idées nouvelles.

\n

Les ouvertures poursuivent toutes des buts stratégiques similaires :

\n
  • le développement des pièces (leur mise en jeu),
  • \n
  • l'occupation ou le contrôle du centre,
  • \n
  • la mise en sécurité du roi,
  • \n
  • l'établissement d'une bonne structure de pions.
  • \n

La plupart des joueurs et des théoriciens considèrent que le fait de jouer en premier donne aux blancs un petit avantage. Dans l'ouverture l'objectif des noirs est de neutraliser cet avantage ou alors de trouver des compensations dans une position déséquilibrée.

\n

Milieu de partie[ | modifier le code]

\n
Article détaillé : Milieu de partie.
\n

Le milieu de partie ou milieu de jeu débute lorsque la plupart des pièces ont été développées. Le recours à la théorie des ouvertures n'étant plus de mise, les joueurs doivent évaluer leur position, concevoir des plans basés sur ses caractéristiques, et dans le même temps tenir compte des possibilités tactiques[G 9].

\n

Certains plans ou thèmes stratégiques liés aux structures de pions découlent directement de l'ouverture, par exemple l'attaque de minorité, qui consiste à avancer des pions de l'aile dame alors que l'adversaire possède plus de pions sur cette aile. L'étude des ouvertures doit donc être menée en parallèle de la préparation des plans possibles dans le milieu de partie.

\n

Le milieu de partie est la phase de la partie dans laquelle l'attaque sur le roi prend le plus d'importance, bien que ce thème ne soit pas à négliger dans les autres phases du jeu. Un exemple classique est le sacrifice double de la partie Lasker - Bauer 1889.

\n

Une autre question stratégique importante dans le milieu de partie est de savoir quand il est opportun d'entrer en finale, c'est-à-dire simplifier la position en échangeant du matériel. Par exemple, un avantage matériel même minime permet souvent le gain, mais seulement en finale. Le camp le plus fort doit donc trouver un moyen de forcer son adversaire à jouer une finale favorable. Il doit pour cela éviter les cas connus comme donnant la nulle malgré la différence de matériel, par exemple la plupart des positions avec roi, fou et pion contre roi et fou avec des fous de couleurs opposées (l'un sur cases blanches et l'autre sur cases noires) ou roi, tour et cavalier contre roi et tour.

\n

Finale[ | modifier le code]

\n
Article détaillé : Finale (échecs).
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\"Roi
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Roi
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Un exemple de zugzwang réciproque : avoir le trait dans cette position est désavantageux pour les blancs comme pour les noirs.
\n
\n
\n

La fin de partie, ou finale, est la phase de la partie qui se déroule lorsqu'il ne reste que quelques pièces sur l'échiquier[G 10]. Il y a trois différences stratégiques avec les étapes précédentes :

\n
  • Lors de la finale les pions prennent une importance particulière, les finales se résument souvent à tenter de promouvoir les pions en les amenant sur la dernière rangée de l'échiquier.
  • \n
  • Le roi, qui doit être protégé pendant le milieu de partie à cause de la menace de se faire mater, devient une pièce puissante en finale. Il est souvent amené au centre de l'échiquier où il peut protéger ses pions, attaquer les pions adverses et gêner les mouvements du roi adverse.
  • \n
  • Le zugzwang, situation où tous les coups légaux sont défavorables alors que passer son tour n'est pas possible aux échecs, est souvent un facteur de première importance dans les finales. C'est rarement le cas en milieu ou en début de partie, car un zugzwang ne se produit généralement que lorsqu'il reste peu de matériel. Par exemple, le diagramme ci-contre est un zugzwang réciproque (un zugzwang pour les deux camps) : si les noirs ont le trait ils sont obligés de jouer 1…Rb7 et ils laissent ainsi les blancs promouvoir leur pion en dame après 2.Rd7 ; si les blancs ont le trait ils doivent soit jouer 1.Rc6 qui pate le roi noir, soit perdre leur pion en jouant tout autre coup, dans les deux cas ils concèdent la partie nulle.
  • \n

Les finales sont classées en fonction du type de pièces qui restent sur l'échiquier. Les mats de base sont les positions dans lesquelles un camp possède un roi seul et l'autre camp une ou deux pièces en mesure de mater, en combinant les efforts de ces pièces et du roi. Par exemple, les finales de pions ne comportent que des rois et des pions dans les deux camps et la tâche du camp le plus fort consiste à promouvoir un pion. Les finales plus complexes sont classées en fonction des pièces sur l'échiquier en-dehors des rois, par exemple tour et pion contre tour. Toutes les finales de six pièces ou moins au total, rois inclus, ont été entièrement analysées par ordinateur. Le résultat de ces analyses forme les tables de finales.

\n

Parties[ | modifier le code]

\n

Miniatures[ | modifier le code]

\n
Mat du berger\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Tour
\n
\n
\"Fou
\n
\"Reine
\n
\"Roi
\n
\"Fou
\n
\n
\"Tour
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\"Reine
\n
\"Pion
\n
\"Pion
\n
\n
\n
\"Cavalier
\n
\n
\n
\"Cavalier
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\"Fou
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Tour
\n
\"Cavalier
\n
\"Fou
\n
\n
\"Roi
\n
\n
\"Cavalier
\n
\"Tour
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Après 3. Dxf7 mat
\n
\n
\n

Une miniature est généralement définie comme une partie d'échecs qui se termine en moins de 20 coups[G 11]. Cela peut être une partie entre débutants, comme le coup du berger reproduit ci-dessous, ou bien une partie terminée rapidement entre forts joueurs.

\n
  • Le coup du berger est une partie de débutants, elle exploite la faiblesse du pion noir f7, qui n'est défendu que par le roi. La légende dit qu'il aurait été inventé par un berger ayant été défié par un roi. Le coup du berger permet de battre très rapidement les joueurs débutants. Voir le diagramme ci-contre, la partie se déroule généralement ainsi : 1. e4 e5 2. Fc4 Cc6 3. Dh5 Cf6?? 4. Dxf7 mat
  • \n
  • Le mat du lion, appelé également mat du sot ou mat de l'écolier, est la partie la plus courte qu'il est possible de jouer, elle est gagnée par les noirs en seulement deux coups : 1.g4 e5 2.f3 Dh4 mat.
  • \n
  • Le mat de Legal est quant à lui déjà plus sophistiqué.
  • \n

Partie commentée[ | modifier le code]

\n
Article détaillé : Partie immortelle.
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Tour
\n
\"Cavalier
\n
\"Fou
\n
\n
\"Roi
\n
\"Fou
\n
\n
\"Tour
\n
\"Pion
\n
\n
\n
\"Pion
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\"Pion
\n
\n
\n
\"Cavalier
\n
\n
\n
\n
\"Fou
\n
\n
\n
\n
\"Cavalier
\n
\"Reine
\n
\n
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\n
\"Pion
\n
\"Tour
\n
\"Cavalier
\n
\"Fou
\n
\"Reine
\n
\n
\"Roi
\n
\"Tour
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Après 11.Tg1!
\n
\n
\n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\"Tour
\n
\n
\"Fou
\n
\n
\"Roi
\n
\n
\"Cavalier
\n
\"Tour
\n
\"Pion
\n
\n
\n
\"Pion
\n
\n
\"Pion
\n
\"Pion
\n
\"Pion
\n
\"Cavalier
\n
\n
\n
\"Fou
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\"Cavalier
\n
\"Pion
\n
\"Cavalier
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\"Pion
\n
\n
\"Reine
\n
\n
\n
\"Pion
\n
\n
\"Pion
\n
\n
\"Roi
\n
\n
\n
\n
\"Reine
\n
\n
\n
\n
\n
\n
\"Fou
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Après 20…Ca6
\nLes blancs matent en 3 coups.
\n
\n
\n

Cette partie a opposé Adolf Anderssen à Lionel Kieseritzky à Londres, en 1851.

\n

1. e4 e5 2. f4

\n

Le principe de cette ouverture, le gambit du roi, est de sacrifier un pion dès le deuxième coup pour attaquer.

\n

2… exf4 3. Fc4 Dh4+ 4. Rf1 b5

\n

C'est Kieseritzky qui a découvert ce coup. Le but est d'écarter le fou du roi de la diagonale a2-g8, tout en préparant une attaque ultérieure de pions.

\n

5. Fxb5 Cf6 6. Cf3 Dh6

\n

Ici, les noirs se trompent. La place de la dame est en h5. Ce coup vient à l'encontre de la suite logique du coup en 5.

\n

7. d3 Ch5 8. Ch4! Dg5 9. Cf5! c6 10. g4 Cf6

\n

Les noirs sont maintenant acculés à la défensive.

\n

11. Tg1!

\n

Ce sacrifice du fou ôte tout espoir de contre-attaque aux noirs. Les pièces noires développées doivent retourner à leur base.

\n

11…cxb5 12. h4 Dg6 13. h5 Dg5 14. Df3 Cg8

\n

À cause de 15. Fxf4, les noirs sont contraints d'assurer une case de retraite pour leur dame.

\n

15. Fxf4 Df6 16. Cc3

\n

Toutes les pièces noires sont revenues à leur base, ou presque.

\n

16…Fc5 17. Cd5! Dxb2 18. Fd6! Fxg1

\n

Les noirs ne peuvent prendre le Fd6, car la suite est forcée : 18…Fxd6 19. Cxd6+ Rd8 20. Cxf7+ Re8 21. Cd6+ Rd8 22. Df8 mat. Les blancs ont une telle avance de développement que la décision ne saurait tarder.

\n

19. e5!

\n

La dame noire est privée de la grande diagonale. Une menace de mat, commençant par 20. Cxg7+, est aussi dans l'air.

\n

19…Dxa1+ 20. Re2 Ca6

\n

Kieseritzky s'imagine que la menace de mat est écartée, car la case c7 est protégée. C'est maintenant qu'Anderssen le surprend.

\n

21. Cxg7+ Rd8 22. Df6+!! Cxf6 23. Fe7 mat

\n

La coordination des trois pièces blanches tout comme la position des pièces noires, toutes présentes sur l'échiquier mais mal coordonnées, ont valu à cette partie le qualificatif « d'Immortelle » par le commentateur Falkbeer, qui publia une analyse détaillée de cette partie en 1855 dans la magazine Wiener Schachzeitung[7]. Il fit remarquer que la position finale est un mat modèle, ce à quoi fut certainement sensible Anderssen qui était également un compositeur de problèmes d'échecs.

\n

Parties célèbres[ | modifier le code]

\n\n

Compétition[ | modifier le code]

\n

Jeu à la pendule[ | modifier le code]

\n
Article détaillé : Pendule d'échecs.
\n

Une partie d'échecs pouvant durer plusieurs heures, il est nécessaire de limiter et de décompter le temps de réflexion de chacun des joueurs.

\n

Au début, chaque coup devait être joué dans un temps imparti (5 minutes par coup, par exemple). Ensuite, l'utilisation d'une pendule ad hoc a permis d'attribuer un temps de réflexion global pour la durée de la partie, ou bien pour un nombre déterminé de coups, par exemple 40 coups en deux heures.

\n

Pendule[ | modifier le code]

\n
\n
\"\"\n
\n\nPendule classique
\n
\n
\n
\n
\"\"\n
\n\nPendule électronique
\n
\n
\n

La pendule d'échecs est un boîtier juxtaposant deux horloges identiques, mécaniques ou électroniques, commandées par deux boutons reliés par une bascule. Elle est toujours utilisée dans les compétitions homologuées par la FIDE[8]. Après avoir joué son coup, le joueur au trait appuie (avec la main qui a déplacé la pièce) sur le bouton de l'horloge situé de son côté. Cela stoppe son horloge, relève le bouton de son adversaire et remet en marche l'horloge de celui-ci.

\n

Dans le cas d'une pendule mécanique, le cadran de chaque horloge est équipé d'un drapeau, petite pièce de plastique ou de métal libre mobile autour d'un axe placé à la gauche du nombre 12. Ce drapeau est progressivement soulevé lorsque l'aiguille des minutes approche du 12 de l'horloge, puis retombe brusquement lorsqu'elle l'atteint précisément. Si la chute du drapeau se produit avant que le joueur ait effectué le nombre de coups exigé par la cadence en vigueur, celui-ci perd immédiatement la partie, sauf si l'adversaire dispose d'un matériel insuffisant pour mater, auquel cas la partie se conclut par une nulle.

\n

Les pendules électroniques permettent une plus grande précision lors des phases de Zeitnot et autorisent d'autres cadences de jeu, notamment celles avec incrément (cadences « Fischer » ou « Bronstein »). La polyvalence des pendules électroniques leur permet aussi d'être utilisées dans d'autres jeux, comme le shōgi, le jeu de go ou le Scrabble.

\n

L'arbitre choisit de placer la pendule du côté de l'échiquier qui lui convient. Souvent, le joueur qui a les noirs peut choisir le côté de la table où il s'installe. Néanmoins, la décision finale revient à l'arbitre.

\n

Cadences de jeu[ | modifier le code]

\n

Une cadence est composée d'une ou plusieurs périodes. Une période est définie par un nombre minimal de coups à jouer en un certain temps. La fin d'une période est appelé contrôle de temps.

\n

La cadence habituelle des parties en compétition est d'une heure et trente minutes pour quarante coups, puis trente minutes pour la fin de la partie, avec un incrément de trente secondes dès le premier coup. Avant la généralisation des pendules électroniques, la cadence usuelle était de deux heures pour quarante coups, puis une heure KO.

\n

Le temps imparti à chacun des joueurs permet de répartir les parties en plusieurs classes. Chacune d'elle a ses règles spécifiques :

\n
  • blitz (de l'allemand « éclair ») : partie de moins de quinze minutes par joueur, comptées pour soixante coups si la cadence prévoit un incrément[9] ;
  • \n
  • partie rapide : partie de quinze à soixante minutes par joueur, comptées pour soixante coups si la cadence prévoit un incrément[9] ;
  • \n
  • la cadence de tournoi ou longue : pour la FIDE, c'est une partie de deux heures KO au minimum, ou deux heures pour soixante coups si la cadence prévoit un incrément. Cependant, une cadence inférieure est acceptable dans les compétitions ouvertes seulement aux joueurs dont le classement Elo est limité : 1 h 30 au minimum si tous les joueurs ont moins de 2200, 1 h au minimum si tous les joueurs ont moins de 1600[10] ;
  • \n
  • les parties par correspondance durent plusieurs semaines, la cadence généralement adoptée par l'ICCF est de cinquante jours pour dix coups ;
  • \n
  • les parties amicales sont souvent jouées sans décompte du temps.
  • \n

Fédération internationale des échecs (FIDE)[ | modifier le code]

\n\n

La FIDE[11] fixe les règles du jeu[12], publie le classement Elo international[13], octroie les titres de grand maître international, maître international, maître FIDE et leurs pendants féminins[14], ainsi que les titres d'arbitre FIDE et d'arbitre international[15]. Elle organise également les Olympiades d'échecs et le championnat du monde d'échecs. Les membres de la FIDE sont les fédérations nationales, telle la Fédération française des échecs.

\n

La FIDE a une commission permanente pour la composition échiquéenne qui gère le domaine des problèmes d'échecs et en particulier les compétitions liées aux problèmes d'échecs.

\n

Les joueurs par correspondance dépendent de la Fédération internationale du jeu d’échecs par correspondance (ICCF), qui reprend les règles de la FIDE mais dont le classement Elo est indépendant.

\n

Arbitrage[ | modifier le code]

\n
Article détaillé : Arbitre d'échecs.
\n

Les parties de compétition sont supervisées par des arbitres qui garantissent le respect des règles du jeu.

\n

On peut classer les arbitres en deux grandes catégories :

\n
  • les arbitres de niveau national avec plusieurs gradations selon leur avancement ;
  • \n
  • les arbitres reconnus par la FIDE : les arbitres FIDE et les arbitres internationaux[15].
  • \n

En France, il existe quatre niveaux d’arbitres, de AF4 à AF1, ce dernier étant le niveau plus élevé. Il existe également un titre d'Arbitre Fédéral Jeune pour les 12-16 ans. Le site de la Fédération française des échecs propose une rubrique sur l'arbitrage[16].

\n

Tricherie[ | modifier le code]

\n
Article détaillé : Tricherie aux échecs.
\n

Plusieurs moyens permettent de tricher aux échecs. Les plus fréquents sont le non-respect d'une règle du jeu en espérant qu'il ne sera pas sanctionné par l'arbitre, l'utilisation discrète d'un programme d'échecs, la communication avec un complice. Il existe aussi des cas d'abus du système de classement Elo et d'obtention de titres de grand maître international ou d'autres titres. Un tricheur est normalement exclu de la compétition dans laquelle il a triché ; il peut aussi être interdit de toute compétition pour une durée déterminée.

\n

Systèmes d'appariement[ | modifier le code]

\n
Articles détaillés : Système suisse et Table de Berger.
\n

La plupart des tournois d'échecs au niveau amateur se jouent au système suisse. Ce système permet à tous les joueurs de jouer toutes les rondes, et donne un classement général en fin de tournoi qui désigne clairement le vainqueur. Les compétitions de haut niveau sont généralement jouées avec un petit nombre de joueurs au format toutes rondes (chaque participant rencontre tous les autres) en utilisant la table de Berger. Les coupes par élimination directe sont rares ; cette formule se rencontre essentiellement dans le cadre de la coupe du monde d'échecs.

\n

Champions du monde[ | modifier le code]

\n\n
Article détaillé : Championnat du monde d'échecs.
\n

Après sa victoire sur Johannes Zukertort en 1886, Wilhelm Steinitz fut le premier champion du monde officiel. Ensuite, le titre fut décerné à qui battait, en match, le champion du monde[G 12]. Le tenant du titre choisissait le prétendant parmi les meilleurs joueurs ou parmi ceux qui viendraient avec le meilleur apport financier.

\n

Entre 1946 et 1948, il n'y eut pas de champion du monde. Le championnat du monde de 1948, organisé par la FIDE, fut un tournoi qui opposa cinq joueurs, et fut suivi, tous les trois ans, à partir de 1951, de matchs disputés au meilleur des vingt-quatre parties. Le prétendant était le vainqueur du tournoi des candidats organisé par la FIDE. En cas de défaite, le champion déchu avait droit, à partir de 1956[17], à un match revanche disputé l'année suivante. En cas d'égalité, le champion conservait son titre.

\n
\n\n
\n

Le droit au match revanche fut aboli en 1963.

\n

En 1975, Bobby Fischer refusa de jouer le championnat du monde 1975 contre Anatoli Karpov. Les trois championnats suivants (1978, 1981 et 1984) furent disputés sans compter les parties nulles, le titre revenant au premier joueur remportant six parties.

\n

En février 1985, Le championnat du monde, commencé en septembre 1984, fut interrompu après 48 parties « pour préserver la santé des joueurs ». Le match fut rejoué en octobre-novembre 1985 en 24 parties et le droit au match revanche fut réintroduit.

\n
\n\n
\n

En 1993, Garry Kasparov provoqua une scission avec la FIDE et créa sa propre fédération, la PCA (Professional Chess Association). Il y eut alors deux champions du monde, l'un dit « classique », se revendiquant de la lignée des matchs entamée par Steinitz, l'autre dit « FIDE » vainqueur du « Championnat du monde FIDE ».

\n

Champions du monde « classiques » de 1993 à 2006 :

\n
\n
\"\"\n
\n\nLe Norvégien Magnus Carlsen, Champion du monde depuis 2013.
\n
\n
\n

Champions du monde « FIDE » de 1993 à 2006 :

\n
  • Anatoli Karpov (19931999, perdit son titre par forfait en 1999)
  • \n

À partir de 1999, contrairement à la tradition, les championnats du monde « FIDE » furent des tournois à élimination directe. Le champion du monde en titre entrait en lice dès les premiers tours, ce que Karpov n'accepta pas en 1999.

\n

Les championnats du monde 2005 et 2007 furent des tournois toutes rondes opposant huit joueurs. En 2006 eut lieu le match de réunification des deux titres. Vladimir Kramnik battit Veselin Topalov.

\n
  • Vladimir Kramnik (2006 – 2007)
  • \n
  • Viswanathan Anand (2007 - 2013)
  • \n
  • Magnus Carlsen (depuis 2013)
  • \n

À compter de 2008, le championnat du monde « unifié » se joue de nouveau sous la forme de match entre le tenant du titre et son challenger.

\n

Championnes du monde[ | modifier le code]

\n\n

Grands tournois[ | modifier le code]

\n

Depuis la saison 2004-2005, les 70 plus grands événements mondiaux sont regroupés au sein de l'ACP Tour, mise en place par l'ACP[18].

\n
Anciens grands tournois
\n

Psychologie[ | modifier le code]

\n
Article détaillé : Psychologie échiquéenne.
\n

La psychologie échiquéenne est l'objet de nombreuses études, on peut classer ces études en deux types : « ceux réalisés par les psychologues pour explorer le fonctionnement du psychisme humain et usant du jeu d'échecs comme outil, […] et, d'autre part, les analyses faites par les joueurs d'échecs […] pour améliorer leur niveau… »[19]

\n

Dans la première catégorie, Alfred Binet publie en 1894 Psychologie des grands calculateurs et joueurs d'échecs, ouvrage dans lequel il étudie les processus cognitifs nécessaires au joueur d'échecs, en particulier les représentations mentales qui permettent aux joueurs d'abstraire l'échiquier et ses pièces afin de réfléchir sans avoir à les déplacer ou jouer une partie à l'aveugle[20]. En 1946, le psychologue néerlandais (et joueur d'échecs) Adriaan de Groot publie une importante étude des mécanismes du choix des coups. Le grand maître et psychologue Reuben Fine dans son livre Psychology of the Chess Player[21] montre que la principale différence entre l'amateur et le maître réside dans la capacité à mémoriser puis reconnaître les différents schémas ou thèmes qui apparaissent lors d'une partie. Il compare cette capacité à la maîtrise d'un langage.

\n

La deuxième catégorie d'études est surtout l'œuvre de grands maîtres soviétiques, en particulier Benjamin Blumenfeld et Nikolaï Kroguious. Ils analysent la genèse des fautes commises par les joueurs et proposent divers remèdes.

\n

Histoire[ | modifier le code]

\n
Article détaillé : Histoire du jeu d'échecs.
\n
\n
\"\"\n
\n\nLe jeu d'échecs, par Charles Bargue
\n
\n
\n

De nombreux mythes et théories existent sur l'origine du jeu.

\n

Légendes[ | modifier le code]

\n

Mythe du brahmane Sissa[ | modifier le code]

\n

La légende la plus célèbre sur l'origine du jeu d'échecs[G 13] raconte l'histoire du roi Belkib (Indes, 3 000 ans avant notre ère) qui cherchait à tout prix à tromper son ennui. Il promit donc une récompense exceptionnelle à qui lui proposerait une distraction qui le satisferait. Lorsque le sage Sissa, fils du Brahmine Dahir, lui présenta le jeu d'échecs, le souverain, enthousiaste, demanda à Sissa ce que celui-ci souhaitait en échange de ce cadeau extraordinaire. Humblement, Sissa demanda au prince de déposer un grain de riz sur la première case, deux sur la deuxième, quatre sur la troisième, et ainsi de suite pour remplir l'échiquier en doublant la quantité de grain à chaque case. Le prince accorda immédiatement cette récompense en apparence modeste, mais son conseiller lui expliqua qu'il venait de signer la mort du royaume car les récoltes de l'année ne suffiraient à s'acquitter du prix du jeu. En effet, sur la dernière case de l'échiquier, il faudrait déposer 263 graines, soit plus de neuf milliards de milliards de grains (9 223 372 036 854 775 808 grains précisément), et y ajouter le total des grains déposés sur les cases précédentes, ce qui fait un total de 264-1, soit 18 446 744 073 709 551 615 grains, soit environ 4.1011 tonnes de riz décortiqué[22].

\n

Des variantes de cette légende existent, l'une suggérant que le roi accepta à condition que le sage compte les graines lui-même, une autre affirmant que Sissa eut la tête tranchée pour une telle effronterie. Certaines versions disent que Sissa ne demanda rien en échange mais que le roi insistant, Sissa aurait alors décidé de se moquer du roi en lui demandant une récompense qu'il ne pourrait donner.

\n

Légende grecque[ | modifier le code]

\n

Une autre légende place l'invention du jeu durant la guerre de Troie. Palamède, l'un des héros grecs, aurait inventé le jeu pour remonter le moral des troupes durant le siège de Troie[23], ainsi que d'autres jeux : « Les Grecs lui attribuaient [à Palamède] l'invention de plusieurs lettres de leur alphabet, de la monnaie, des dés, des osselets et du « jeu d'échecs » (sic) »[24],[25]. C'est l'origine du nom de la première revue échiquéenne, Le Palamède. Cette légende est née d'une traduction erronée du mot grec πεττεία (petteia), un terme désignant un jeu de plateau différent des échecs, l'équivalent du senet égyptien[26] et ancêtre probable du Tablut ou « Jeu des cinq lignes »[27] parfois traduit, à tort, par « dames »[28] ou « échecs »[29].

\n

Légende latine[ | modifier le code]

\n

Selon une autre légende, inventée par le poète anglais William Jones en 1763 dans un poème en latin, Euphron (frère de Vénus et dieu des sports) aurait créé les échecs pour aider Mars à séduire la belle Caïssa. Cette dernière est parfois considérée comme la déesse des échecs.

\n

Origine orientale[ | modifier le code]

\n

Les Arabes font connaissance avec le jeu. Ils s'y adonnent avec passion et étendent sa pratique au fur et à mesure de leurs conquêtes. Vers l'ouest, le jeu traverse le Maghreb et la Méditerranée pour parvenir dans l'Espagne musulmane et atteindre l'Occident chrétien à la fin du Xe siècle[30]. Il existe des jeux d'échecs différents, persans (chatrang), indiens (chaturanga), arabes (shatranj), mongols (shatar), européens, birmans (sit-tu-yin), thaïs ou cambodgiens (makruk), malais (catur), chinois ou vietnamiens (xiangqi), coréens (janggi), japonais (shogi), etc. Tous ces jeux partagent un ensemble de traits qui renvoient à une véritable préhistoire puisqu’il n’existe aucun témoignage direct et sans équivoque du supposé ancêtre commun.

\n

Si la naissance même du jeu reste encore obscure et controversée[31], on peut au moins affirmer que les échecs sont un jeu asiatique. Trois ensembles géographiques posent leur candidature au titre de berceau du roi des jeux :

\n
  • l’Inde du Nord, du Cachemire à la haute vallée du Gange, en passant par le Sind et le Pendjab, le bassin de l’Indus (aujourd’hui largement au Pakistan) ;
  • \n
  • la Chine historique, c’est-à-dire le bassin du fleuve Jaune et peut-être celui du Yangzi Jiang, plus au sud ;
  • \n
  • la grande sphère iranienne entre les deux, les pays traversés par l’antique route de la soie : la Perse mais aussi le Gandhâra, la Bactriane, le Khwarezm, la Sogdiane, la Sérinde, soit l’Asie centrale de l’Iran et de l’Afghanistan au Xinjiang. Linguistiquement et culturellement, ces régions se rattachaient à la sphère iranienne.
  • \n

L'Inde est généralement l'hypothèse la plus suivie. Elle a pour elle la tradition puisque même les premiers textes persans et arabes affirmaient que les échecs étaient venus d'Inde. Cependant, les traces historiques prouvant cette origine manquent. L'Asie centrale iranienne au contraire reste la terre des premiers témoignages comme des plus anciennes trouvailles archéologiques. Enfin la Chine revendique aussi le titre de berceau de ce jeu et s'il est vrai que les premiers témoignages confirmés sont tardifs en Chine, il existe des sources certes floues mais plus anciennes que les plus anciennes sources perses ou sanscrites (qui datent de l'époque 600 à 650 ap. J.-C.).

\n

Dans l'état actuel des connaissances, il est difficile de trancher.

\n

Une autre croyance très répandue est l'idée que les premiers échecs auraient été inventés (dans ce cas, c'est toujours en Inde) sous la forme d'un jeu se jouant à quatre joueurs et avec l'aide de dés. Vers l'an 600, des Indiens ou des Perses auraient éliminé les dés et regroupé les camps pour n'en faire que deux. Cette hypothèse est très certainement fausse. La plus ancienne mention connue du jeu à quatre date de 1030, soit quatre siècles après la mention du jeu à deux. Tout concourt à penser que ce chaturanga à quatre, appelé chaturaji, constitue une variante du chaturanga ou chatrang à deux et non le contraire[32].

\n

Le mot sanskrit chaturanga, qui a donné chatrang en pehlevi (moyen persan), signifie quatre membres et désignait à l'origine l'armée épique indienne avec infanterie, cavalerie, éléphanterie et chars de combats. Ces pièces, avec un roi et son conseiller (ministre ou général) formaient l'ensemble des pièces du jeu, très semblables à celui d'aujourd'hui. Chaque joueur maniait 16 pièces sur un tablier de 64 cases, de couleur unique.

\n

Diffusion[ | modifier le code]

\n

Lorsque les Arabes envahissent la Perse, ils l’adoptent sous le nom de shatranj. Les échecs connaissent alors un développement remarquable. C’est au cours des IXe et Xe siècles qu’apparaissent les premiers champions et les premiers traités. On retrouve alors :

\n
  • le roi (Shâh, c'est lui qui donne son nom au jeu) se déplace d’un pas dans toutes les directions ;
  • \n
  • le conseiller (Farzin ou Vizir) dont le mouvement est limité à une seule case en diagonale ;
  • \n
  • l’éléphant (Fil, cf. sanskrit pīlu qui donnera « fou ») avec un déplacement correspondant à un saut de deux cases en diagonale ;
  • \n
  • le cheval (Faras), identique au cavalier moderne ;
  • \n
  • le char (Roukh), identique à la tour actuelle ;
  • \n
  • le soldat (Baidaq, cf. sanskrit padāti : piéton, fantassin), l’équivalent du pion, mais dépourvu du double pas initial.
  • \n

Le Roukh était parfois représenté comme un char de guerre. Les Arabes y voyaient un général commandant l’armée. Mais son sens littéral reste obscur. Il semble que pour les Arabes, ce mot n’avait pas d’autre sens que celui de désigner cette pièce au Shatranj, un peu comme le mot rook pour les anglophones aujourd’hui. Le lien étymologique avec le sanskrit ratha : char est peu évident.

\n

Arrivée en Europe et évolution[ | modifier le code]

\n
\n
\"\"\n
\n\nManuscrit (c.1320)
\n
\n
\n

Les échecs arrivent en Europe sans doute aux alentours de l'an mil[G 14] par l’Espagne musulmane ou par l’Italie du Sud (Sicile)[33]. Une légende a longtemps attribué un jeu d'échecs à Charlemagne qui l'aurait reçu de la part du calife Haroun al-Rachid, on pense aujourd'hui qu'il fut fabriqué postérieurement près Salerne à la fin du XIe siècle[34]. En 1010, sa première mention écrite en Occident a été trouvée dans un testament du comte d'Urgel, en Catalogne. De nombreuses pièces d'échecs ont été retrouvées lors de fouilles sur le site des chevaliers-paysans du lac de Paladru (Isère), site qui a été abandonné au plus tard en 1040. Le Libro de los juegos écrit en Espagne entre 1251 et 1283 et illustré de nombreuses miniatures, expose les règles du jeu au XIIIe siècle.

\n
\n
\"\"\n
\n\nProblème d'échecs no 35 du Libro de los juegos
\n
\n
\n

Dès son arrivée dans la chrétienté, l’échiquier et les pièces s'occidentalisent progressivement[G 15] :

\n
  • le plateau devient bicolore avec les cases rouges et noires (qui deviendront plus tard blanches et noires) ;
  • \n
  • le vizir devient fierge (ou vierge), puis reine et/ou dame (il est difficile de déterminer lequel des deux termes prévalait — sans doute étaient-ils utilisés indifféremment) ;
  • \n
  • l'éléphant (al fil en arabe, qui reste alfil en espagnol aujourd'hui) devient aufin, puis fou (bishop : évêque en anglais) ;
  • \n
  • le roukh arabe devient roc (ce nom donnera rook en anglais, le verbe « roquer » en français et désignera la tour d'échecs en héraldique), puis tour vers la fin du XVIIe siècle (les tours de guet étant souvent placées en hauteur).
  • \n

Dans certaines régions d'Europe, le double pas initial du pion est pratiqué. Certaines règles permettent au roi ou à la reine (ou dame) d'effectuer un saut à deux cases (sans prise) à leur premier mouvement. Ceci constitue la différence principale avec les règles du Shatranj des pays musulmans[35]. Mais l’évolution la plus importante a lieu à la fin du Moyen Âge, après 1470, en Espagne ou en Italie, lorsque les mouvements limités de la reine (ou dame) et du fou sont remplacés par ceux que nous connaissons actuellement[35].

\n

Les joueurs de cette époque nomment ces nouvelles règles : « eschés de la dame » ou « jeu de la dame enragée »[36].

\n

Les plus anciens manuscrits conservés relatifs à ces évolutions sont le manuscrit de Göttingen et le Scachs d'amor. Le premier traité imprimé reflétant ces innovations est généralement attribué à Francesc Vicent, publié en 1495 à Valence, mais il est aujourd'hui perdu. Le deuxième, attribué à Lucena, nous est parvenu.

\n

Pour parer aux effets dévastateurs des pièces aux pouvoirs renforcés, le roque est inventé vers 1560 et, progressivement, il remplace le saut initial du roi ou de la reine (la dame) qui deviennent obsolètes[35]. On peut considérer que les règles du jeu moderne sont à peu près établies vers 1650. Si les premiers livres traitant des échecs remontent à l'époque arabe (dans le Kitab-al-Fihrist d'Ibn al-Nadim), la stabilisation des règles en Europe donne naissance à une littérature théorique très riche et on observe notamment l'élaboration des premiers systèmes d'ouverture[pas clair].

\n

Époque moderne[ | modifier le code]

\n
\n
\"\"\n
\n\nPièces de type Staunton
\n
\n
\n\n
\n

L’aspect physique des pièces le plus courant aujourd’hui, le style « Staunton », date de 1850. C’est également durant la seconde moitié du XIXe siècle qu’émergent les échecs modernes. Les premières compétitions internationales ont lieu, les progrès théoriques de l’art de la défense mettent un terme à l’ère romantique.

\n

Au XXe siècle, l’URSS en assure une promotion très active, le considérant comme un excellent outil de formation intellectuelle[G 16]. C’est, en outre, une vitrine de la formation intellectuelle soviétique qui leur permet de dominer largement une discipline prestigieuse.

\n

Durant la guerre froide, l'émergence de Bobby Fischer[G 17], le premier Occidental à défier les Soviétiques au plus haut niveau, puis de Viktor Kortchnoï[G 18], dissident soviétique qui parvint deux fois en finale du championnat du monde, donnent à cette compétition une véritable dimension politique. Plus tard, les tensions entre conservateurs russes et partisans de la perestroïka se cristalliseront autour de l’affrontement entre Anatoli Karpov et Garry Kasparov.

\n

À la fin du XXe siècle, la confusion concernant le titre de champion du monde amène l’attention médiatique à se concentrer sur l’opposition entre l’humain et la machine, comme en témoigne le retentissement médiatique des matchs entre Kasparov et Deep Blue[37]. Les femmes font également leur apparition au plus haut niveau dans un domaine longtemps réservé de fait aux hommes. Ainsi, depuis avril 2003, Judit Polgár figure parmi les meilleurs joueurs mondiaux du classement de la Fédération internationale des échecs[38].

\n

Depuis janvier 2000, les échecs sont devenus, en France, un sport reconnu par le Ministère de la Jeunesse et des Sports[39]. De nombreuses compétitions sportives sont organisées dans le monde entier. Depuis le début de l'année 2008, l’entrée de ce sport aux Jeux olympiques est discutée[40].

\n

L’actuel champion du monde est le Norvégien Magnus Carlsen qui a succédé à l'Indien Viswanathan Anand en 2013[41].

\n

Introduction des échecs dans le cursus scolaire[ | modifier le code]

\n

Depuis janvier 2011, en France, des études scientifiques et technologiques ont été menées sur l’intégration d’un nouveau procédé : l’apprentissage des échecs.

\n

La pratique de cet enseignement a pour origine de travailler sur la logique, la rigueur de mettre en place des stratégies. Tout ceci amène les élèves à respecter les règles du jeu et le jeu de l’adversaire. Quelques objectifs pédagogiques sont mis en avant pour les élèves :

\n
  • développer la motivation et la concentration ;
  • \n
  • encourager l’esprit d’autonomie et d’initiative ;
  • \n
  • favoriser l’apprentissage de la citoyenneté.
  • \n

Cette pratique sera effective pour les élèves des écoles, des collèges et des lycées. Ainsi, les élèves possèdent un moyen ludique pour acquérir de nouvelles aptitudes.

\n

D’autres pays nous ont précédé à la mise en place ce programme. L’Arménie est le premier pays au monde qui a, en 2011, rendu obligatoire les échecs dans le cadre scolaire. Ce fut au tour du Mexique en 2014 puis de la Chine, de l’Inde et de l’Allemagne. À la suite de ce succès l’Espagne, après adoption de la loi d’insertion des échecs comme instrument pédagogique, compte un millier d’établissements qui l’ont mis en place de manière obligatoire ou optionnelle.

\n

Composition échiquéenne[ | modifier le code]

\n

La composition échiquéenne, qui forme un monde à part dans l’univers des échecs, représente son versant artistique[G 19]. Le problème d'échecs (au sens large) se conforme à des règles de jeu aussi rigoureuses que dans le jeu d'échecs (même si elles sont parfois revisitées comme dans les problèmes féériques) mais il présente des situations très éloignées de la partie d'échecs réelle. Des considérations esthétiques, souvent géométriques, priment sur la réalité de la lutte entre deux joueurs. Cet univers comporte un certain nombre de conventions : on exige par exemple (sauf énoncé contraire) que la solution du problème soit unique, lorsqu'il s'agit d'un gain (étude) on présente le problème en donnant le trait aux blancs, on évite que le premier coup de la solution soit une prise ou un échec, etc. La composition échiquéenne est une discipline récente, au moins au sens moderne du terme (XIXe siècle).

\n

Comme dans le domaine de la partie, des compétitions sont organisées, elles sont de deux sortes :

\n
  • des concours de composition qui consistent à créer un problème, souvent sur un thème donné ;
  • \n
  • des compétitions de résolution de problèmes, dont les compétiteurs sont appelés des solutionnistes.
  • \n

Rares sont les forts joueurs d’échecs qui s’intéressent aux problèmes d’échecs, les deux univers sont très différents. Notons toutefois que les grands maîtres anglais John Nunn et Jonathan Mestel ont remporté le Championnat du monde de solutions, et que Richard Réti, Vassily Smyslov et Pal Benko sont des compositeurs d'étude réputés.

\n

Problème[ | modifier le code]

\n
Article détaillé : Problème d’échecs.
\n

Si les problèmes les plus fréquents sont les mats en deux coups[G 20], il y a une grande variété de types d'énoncé. Il y a des problèmes orthodoxes, des problèmes hétérodoxes (mats aidés et mats inverses), des problèmes féériques (où les règles et les pièces en jeu peuvent être différentes du jeu habituel), des problèmes d’analyse rétrograde, etc.

\n
Thomas Taverner\n

Dubuque Chess Journal 1889

\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\"Fou
\n
\"Tour
\n
\"Tour
\n
\"Fou
\n
\n
\n
\n
\"Cavalier
\n
\n
\n
\n
\n
\"Fou
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\"Reine
\n
\n
\n
\"Pion
\n
\n
\n
\"Roi
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\"Roi
\n
\"Tour
\n
\n
\n
\"Cavalier
\n
\n
\n
\"Tour
\n
\"Fou
\n
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Les blancs jouent et matent en deux coups.
\n
\n
\n

Ci-contre, un problème de Thomas Taverner publié en 1889 dans le Dubuque Chess Journal. C'est un mat direct en deux coups.

\n

La clé du problème est 1.Th1. Elle est difficile à trouver parce qu'elle n'introduit aucune menace. Au lieu de cela, elle évacue la case h2, qui devient utilisable pour mater ; c'est ce que les problémistes appellent le thème Bristol, en référence à un problème de Frank Healey publié en 1861 dans un tournoi de cette ville. Les noirs sont mis en zugzwang, une situation dans laquelle chacun de leur coup détériore leur position (les problémistes parlent plutôt de blocus). Mais les règles du jeu leur imposent de jouer et chacun des coups noirs entraîne un coup blanc matant. Par exemple, si les noirs jouent 1… Fxh7, la case d5 n'est plus contrôlée, et les blancs jouent 2.Cd5#. Ou bien si les noirs jouent 1… Te5, ils bloquent la case de fuite du roi, ce qui permet 2.Dg4#. Sur 1…Fg5, les blancs jouent 2.Dh2#, profitant de l'effet Bristol. Si les noirs pouvaient ne pas jouer en réponse à la clé, les blancs ne pourraient pas mater en un coup.

\n

Le thème de ce problème est appelé tuyaux d'orgues ; il se caractérise par la position des tours et des fous noirs. Si chacune de ces quatre pièces avance d'une ou de deux cases, elle intercepte une autre pièce et permet un mat. Par exemple, si les noirs jouent 1…Fe7, la case e3 n'est plus contrôlée, et cela permet 2.e3≠. Si les noirs jouent 1…Te7, c'est la case h4 qui n'est plus contrôlée et les blancs matent par 2.Th4≠. Le thème de l'interférence mutuelle de deux pièces dans deux variantes porte le nom Grimshaw, les tuyaux d'orgues présentent donc deux Grimshaw.

\n
\n

Étude[ | modifier le code]

\n
Alekseï Troïtski
\n1898\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
\nabcdefgh\n
8\n
\"Chessboard480.svg\"
\n
\n
\n
\"Fou
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Reine
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Pion
\n
\n
\n
\"Cavalier
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\"Roi
\n
\n
\n
\"Pion
\n
\"Pion
\n
\n
\n
\n
\n
\n
\n
\"Roi
\n
\"Fou
\n
\n
8
77
66
55
44
33
22
11
\nabcdefgh\n
Les blancs jouent et font nulle.
\n
\n
\n
Article détaillé : Étude d'échecs.
\n

Les études sont des compositions qui montrent un gain ou une nulle extraordinaire en fin de partie[G 21]. Si le problème d'échecs est un domaine réservé à une minorité de passionnés dans le monde des échecs, l'étude est elle-même un monde à part dans la composition échiquéenne.

\n

Ci-contre, une étude d’Alekseï Troïtski de 1898. La position est a priori facilement gagnante pour les noirs qui disposent d'un avantage matériel considérable. Toutefois une suite de coups précise (et difficile à trouver pour un débutant) permet aux blancs d’obtenir la partie nulle, quels que soient les coups des noirs. On remarque que la position bien que légale n'est pas réaliste et n'aurait aucune chance de se produire dans une partie réelle.

\n

La solution est la suivante : 1.Re1 enferme le roi noir et menace 2.Fb6 mat. (1.Re2? échoue à cause de 1…Dh5+! 2.Re1 Dd1+ 3.Rxd1 Rf2 et les noirs se libèrent de toute pression et gagnent) 1…Da7 pour empêcher Fb6, mais tout de même : 2.Fb6+ Dxb6 3.Cxb6 la position est simplifiée mais les blancs ne peuvent pas s'opposer à la promotion du pion f5 donc : 3…f4 la seule chance des blancs est d'essayer de mater le roi noir emprisonné avec leur cavalier : 4.Cd5 f3 5.Cf4 f2+ 6.Rd2! Rf1! (après 6…f1=D? les blancs gagnent avec 7.Ch3 mat tandis qu’après 6…f1=C+? 7.Re1 et les noirs ne peuvent pas empêcher 8.Ch3 mat) 7.Cd5! (si les noirs font une Dame avec 7…g1=D? les blancs gagnent avec 8.Ce3 mat) 7…Rg1 8.Cf4 Rf1 9.Cd5 et la partie est nulle par répétition de la position (nulle positionnelle). Les éléments artistiques de cette étude sont l'exploitation de l'enfermement du roi noir, une défense par sous-promotion en cavalier, deux positions de mat différentes par le cavalier blanc et une nulle positionnelle.

\n
\n

Variantes du jeu d'échecs[ | modifier le code]

\n\n

La grande popularité du jeu a encouragé l'apparition de nombreuses variantes du jeu[42], spontanément dans les clubs ou de façon volontariste par des joueurs imaginatifs. Ces variantes modifient parfois légèrement la façon de jouer comme dans le blitz ou partie rapide, dans lequel la réflexion de fond s'efface au profit de l'intuition et des réflexes des joueurs ; ou encore plus notablement dans le blitz à quatre dans lequel les pièces capturées sur un premier échiquier sont utilisables par un partenaire sur un second échiquier, la première partie gagnée faisant gagner son équipe. La partie en consultation est une autre façon de jouer en équipe : un camp, ou les deux, est tenu par plusieurs joueurs qui décident collectivement du coup à jouer.

\n

D'autres variantes ont été imaginées par des joueurs tels que José Raúl Capablanca ou Bobby Fischer, elles consistent à modifier les caractéristiques de l'échiquier ou à ajouter de nouvelles pièces afin, selon leurs auteurs, de renouveler l'intérêt du jeu en limitant l'importance des connaissances au profit de la créativité : les échecs Capablanca et les échecs aléatoires Fischer. Toutefois on ne considère pas toute invention, d'un soir ou commerciale, comme une variante du jeu, on préfère réserver ce terme (en particulier dans le cadre de cet article) aux formes du jeu qui ont trouvé leur public à travers une pratique chez les joueurs. Ainsi, les échecs de Messigny ou les échecs football ont effectivement été joués lors de réunions de problémistes à Messigny, ainsi que le Kriegspiel y compris par des champions d'échecs, le qui perd gagne étant quant à lui célèbre en club.

\n

En parallèle, les compositeurs de problème d'échecs ont élargi les possibilités de leur art en créant des problèmes basés sur des variantes connues du jeu, et ils ont eux-mêmes créé un très grand nombre de pièces nouvelles et conditions supplémentaires qui forment un domaine appelé les échecs féeriques[G 22]. On distingue donc les variantes du jeu d'échecs des échecs féeriques, sachant que des correspondances les relient souvent.

\n

Des jeux cousins tels que le chaturanga, le chatrang, le xiangqi, le makruk, le shatar et le shōgi ne sont pas des variantes du jeu d'échecs mais des jeux originaux, tous plus anciens que le jeu d'échecs moderne.

\n
\n
\n

Échecs et informatique[ | modifier le code]

\n

Programmes d'échecs[ | modifier le code]

\n
\n
\"\"\n
\n\nProgramme Open Source XBoard sous GNU/Linux
\n
\n
\n
\n
\"\"\n
\n\nDeep Blue, l'ordinateur qui a défait Garry Kasparov en 1997.
\n
\n
\n
Article détaillé : Programme d'échecs.
\n

Les échecs ont constitué l'un des premiers défis en matière d'intelligence artificielle[G 23].

\n

Le premier championnat du monde d'échecs des ordinateurs se déroula en 1974. Il fut remporté par le programme soviétique Kaissa.

\n

En 1995, IBM n'hésite pas à investir dans le projet Deep Blue, dont la seconde mouture, en 1997, sera la première machine à battre un champion du monde dans des conditions normales de jeu (à cette époque, les ordinateurs étaient déjà redoutables en partie rapide). Kasparov contestera néanmoins la valeur de cette victoire en soulignant que, contrairement aux conditions d'un match de championnat du monde contre un humain, il n'avait pas eu accès aux parties disputées par l'ordinateur auparavant pour sa préparation (la réciproque étant fausse). Il relève de plus qu'une intervention humaine a été nécessaire en cours de match afin que la machine ne reproduise pas certaines erreurs des premières parties. Kasparov exigea une revanche qui lui fut refusée par IBM. Depuis, les affrontements entre les meilleurs joueurs mondiaux et les machines (Kasparov contre Deep Junior, Kramnik contre Deep Fritz, Kasparov contre X3D Fritz) ont pris le relais d'un championnat du monde défaillant dans les médias. On peut remarquer à ce sujet que, contrairement à Deep Blue, les logiciels opposés aux humains sont des programmes commerciaux tournant sur des micro-ordinateurs standard (alors que Deep Blue fonctionnait sur une machine plus puissante).

\n

Depuis la victoire de Deep Blue, le statut des échecs en tant que défi informatique s'est amoindri, et l'attention des programmeurs s'est reportée sur le go. En effet, dans ce cas, la puissance de calcul qui fait la force des machines joue un rôle moins important face à la stratégie et la capacité d'évaluation d'une position, plus complexes à modéliser.

\n

Pourtant l'exception Hydra a refait parler des superordinateurs dédiés au jeu d'échecs en juin 2005, en battant le grand maître international et 7e mondial Michael Adams, sur un score sans appel de 5,5 points contre 0,5.

\n

En décembre 2006, le champion du monde Kramnik s'est fait battre par le nouveau logiciel Deep fritz 2006 4 à 2 (2 défaites, 4 nulles).

\n

Programmes de résolution de problèmes[ | modifier le code]

\n

De nombreux programmes ont également vu le jour pour vérifier la correction d'un problème d'échecs. Lorsqu'un problème a été vérifié par ordinateur, cela est mentionné sur le diagramme par le symbole « C+ ».

\n

Symbolique des échecs[ | modifier le code]

\n

Très rapidement après leur arrivée en Europe, les échecs acquièrent un statut particulier[43]. Divertissement de l'élite, ils représentent une activité noble au cours de laquelle s'affrontent les esprits des participants[44]. Les possibilités quasi-infinies offertes par le jeu fascinent et donnent naissance à de nombreuses interprétations ésotériques. Certains le considèrent notamment comme une représentation du monde où chaque situation peut être modélisée en une position qui peut trouver sa solution sur l'échiquier[45].

\n

Les échecs sont surnommés « le roi des jeux »[46], et ce statut particulier rend toute tentative de mécanisation extraordinaire. Si les premiers automates joueurs d'échecs comme le turc mécanique, sont des mystifications[G 24], la capacité à jouer aux échecs sera l'un des premiers objectifs des concepteurs d'ordinateurs et l'un des premiers témoignages de l'apparition de ce qui est alors considéré comme de l'intelligence artificielle[G 25]. C'est cette perception du jeu d'échecs comme expression de l'intelligence humaine qui dramatise les affrontements entre Gary Kasparov et la machine Deep Blue[37]. La défaite du champion de l'espèce humaine marque alors fortement les esprits.

\n

Le jeu d'échecs symbolise fréquemment l'affrontement de deux psychés, deux capacités intellectuelles. Cette dimension encourage l'Union soviétique à se doter d'une école d'échecs qui forme pendant un demi-siècle tous les champions du monde[47]. C'est également un aspect fréquemment utilisé dans l'art populaire pour figurer l'opposition, et parfois la séduction, entre deux personnages.

\n

Arts et culture[ | modifier le code]

\n
\n
\"\"\n
\n\nLe Joueur d'échecs d'Honoré Daumier
\n
\n
\n

De nombreux tableaux, sculptures, films et photographies mettent en scène le jeu d'échecs[48],[49],[50].

\n

Poésie[ | modifier le code]

\n
\n
\"\"\n
\n\nUne illustration ancienne de Caïssa
\n
\n
\n

Les deux plus anciens poèmes sur les échecs sont en latin :

\n

Littérature[ | modifier le code]

\n

Plusieurs livres de fiction utilisent le jeu d'échecs comme élément important de l'histoire. Parmi eux, deux se distinguent en mettant le jeu au centre de l'intrigue : Le Joueur d'échecs, de Stefan Zweig, et La Défense Loujine, de Vladimir Nabokov.

\n
\n\n
\n

Le Joueur d'échecs, nouvelle de Stefan Zweig, a pour sujet l'affrontement d'un joueur particulièrement doué, qui a appris seul à jouer aux échecs, seule façon pour lui de garder son esprit alerte alors qu'il était emprisonné en isolement total sous le régime nazi, et du champion du monde fictif de l'époque, homme particulièrement vulgaire et inculte. Le personnage principal finit par abandonner le match pour ne pas sombrer dans la folie.

\n

La Défense Loujine raconte la vie de Loujine, joueur d'échecs russe fictif qui arrive au plus haut niveau et que l'excès de jeu d'échecs conduit, lui aussi, à la folie. Le roman est particulièrement acclamé par la critique pour la façon dont il dépeint l'univers intérieur du joueur d'échecs, ce qui se passe dans son esprit pendant qu'il réfléchit[G 26].

\n

Certains romans utilisent les échecs comme élément de la trame de fond. Ainsi, l'intrigue du Tableau du maître flamand, d'Arturo Pérez-Reverte, s'explique par une analyse rétrograde, et celle de La ville est un échiquier par la liste des coups d'une partie Steinitz-Tchigorine. Dans L'Échiquier du mal, de Dan Simmons, les personnages capables de « dominer » d'autres personnages les utilisent pour jouer une partie d'échecs vivante. La nouvelle Un combat, de Patrick Süskind, relate une partie où le gagnant n'est pas celui qu'on pense, illustrant l'importance de la psychologie dans le jeu. Dans La Joueuse d'échecs, de Bertina Henrichs, une modeste femme de ménage grecque découvre la puissance du jeu d'échecs.

\n

D'autres livres entrent également dans cette catégorie, comme 5150 rue des Ormes de Patrick Senécal, La ville est un échiquier de John Brunner, Le Huit de Katherine Neville, Le Gambit des étoiles de Gérard Klein, Fous d'échecs de Serge Rezvani.

\n

En bande dessinée, le manga français Zeitnot, de Ed Tourriol et Eckyo, se déroule dans le milieu des clubs d'échecs lycéens.

\n

Le jeu d'échecs est également mentionné pour son pouvoir évocateur dans de nombreux livres, comme De l'autre côté du miroir, où Alice participe à une partie « grandeur nature » ; Le Neveu de Rameau de Denis Diderot, où, dans l'incipit, Diderot fait référence au Café de la Régence et à ses joueurs d'échecs de l'époque, notamment Legal (connu pour son mat) et Philidor (connu pour la défense du même nom). Isaac Asimov a mis en scène les échecs dans plusieurs de ses romans et nouvelles, notamment Cailloux dans le ciel où ce jeu est présenté comme une des rares choses qui n'ont pas changé au cours des millénaires. Balzac, dans Le Bal de Sceaux, décrit l'habileté aux échecs comme une qualité louable chez un gentilhomme[51].

\n

On peut également citer Fin de partie (Endgame de son titre original), pièce de théâtre écrite par Samuel Beckett, amateur d'échecs. Le titre de cette pièce renvoie au jeu d'échecs et de nombreuses références subtiles y sont faites par le biais des actes, des rôles et des positions des personnages : déplacements de Clov lors de la scène d'ouverture ; position centrale de Hamm (personnage tyrannique dont le fauteuil roulant apparait vite comme un trône), évoquant là encore la position du roi d'échecs.

\n

Dans le livre Le Trésor de la Guerre d'Espagne, Serge Pey décrit une partie d'échecs jouée par les membres d'une société secrète portant un hippocampe tatoué sur leurs poignets en hommage au déplacement du cavalier. Dans cette nouvelle, les héros procèdent à une partie aveugle s'effectuant uniquement avec le parfum de verres remplis d'alcool différents. Dans un autre chapitre du même livre Serge Pey décrit une partie en morse effectuée dans une prison chilienne, sous la dictature de Pinochet.

\n

Dans la nouvelle Strange Eden (« Étrange Eden ») de Philip K. Dick, la jeune femme extraterrestre que rencontre Brent lui propose une partie d'échecs; puis elle lui apprend que c'est son peuple qui l'aurait introduit chez les brahmanes.

\n

Dans le recueil de nouvelles Fantômes et Farfafouilles de Fredric Brown, La nouvelle l'hérésie du fou est en fait une partie d'échecs vue par un fou d'échecs (bishop en anglais). Tout le long de la narration en point de vue interne, une atmosphère de guerre moyenâgeuse s'impose à l'esprit du lecteur.

\n

Dans le roman L'Ultime Secret de Bernard Werber Isidore et Lucrèce enquêtent sur l'étrange mort de Samuel Fincher, génie du jeu d'échecs ayant vaincu le meilleur ordinateur à ce jour.

\n

Dans la nouvelle Double assassinat dans la rue Morgue d'Edgar Allan Poe, le jeu d'échecs apparaît également, mais est comparé négativement au jeu de dames anglais.

\n

Cinéma[ | modifier le code]

\n
\n
\"\"\n
\n\nJosé Raúl Capablanca dans La Fièvre des échecs en 1926.
\n
\n
\n

Le premier film réalisé autour de la thématique du jeu d'échecs est La Fièvre des échecs, de Vsevolod Poudovkine, tourné pendant le tournoi de Moscou de 1925. D'autres films sont situés dans le monde des échecs de compétition, comme La Diagonale du fou, de Richard Dembo, inspiré des matches de championnat de monde entre Karpov et Kortchnoï ; À la recherche de Bobby Fischer, de Steven Zaillian, inspiré de la vie de Josh Waitzkin ; La Partie d'échecs, d'Yves Hanchar.

\n

Le dernier film en date, Le Prodige (2015), est un film biographique dont le personnage central est Bobby Fischer, interprété par Tobey Maguire. Ce film, réalisé par Edward Zwick, est centré sur l'affrontement du champion américain avec le Soviétique Boris Spassky (joué par Liev Schreiber) et la montée de sa folie.

\n

D'autres films utilisent le jeu d'échecs de façon métaphorique, comme Le Septième Sceau, d'Ingmar Bergman, où le chevalier propose une partie d'échecs à la Mort en espérant retarder l'échéance fatidique ; Les Joueurs d'échecs, de Satyajit Ray ; ou en tant que support de l'intrigue, comme le thriller Face à face, de Carl Shenkel.

\n

Certains des romans cités ci-dessus ont également été adaptés en films, comme La Défense Loujine, de Marleen Gorris, et Joueuse, de Caroline Bottaro, dont le scénario est inspiré de La Joueuse d'échecs, transposé en Corse avec Sandrine Bonnaire.

\n

Il existe aussi des films d'animation mettant en scène les échecs, comme Geri's Game, court-métrage d'animation produit et réalisé par les studios Pixar.

\n

D'autres films sont en rapport avec les échecs, par exemple La légende de Zatoïchi: Voyage en Enfer de Kenji Misumi, L'Échiquier de la passion de Wolfgang Petersen, Jouer sa vie de Gilles Carle.

\n

On peut également noter de nombreuses apparitions du jeu d'échecs dans des films où sa présence n'est pas un ressort dramatique mais plutôt de l'ordre du symbole. Ainsi, dans Bons baisers de Russie, le méchant est un génie des échecs et de la stratégie et travaille pour le SPECTRE contre James Bond. Dans K, d'Alexandre Arcady, les deux personnages principaux sont liés par leur goût des échecs.

\n

Le jeu d'échecs comme symbole de l'intelligence humaine est repris dans Blade Runner, de Ridley Scott, où le répliquant met son créateur échec et mat, et dans 2001, l'Odyssée de l'espace, de Stanley Kubrick, grand amateur d'échecs, où le super-ordinateur CARL (HAL 9000) l'emporte sur l'astronaute David Bowman.

\n

Dans Harry Potter à l'école des sorciers, de Chris Columbus, Ronald Weasley joue avec Harry aux échecs version sorcier, avec des pièces animées par magie, puis doit diriger une partie d'échecs contre des pièces grandeur nature, l'une des épreuves à affronter avant d'accéder à la pierre philosophale.

\n

Magnéto et le professeur Charles-Xavier, les principaux antagonistes de la saga X-Men, s'affrontent régulièrement aux échecs. C'est notamment le cas dans X-Men 2, où les deux personnages jouent dans la cellule de Magnéto. Le film X-Men : L'Affrontement final se clôt sur une partie d'échecs que Magnéto joue seul.

\n

Dans L'Affaire Thomas Crown, de Norman Jewison, le suspect et celle qui le traque s'affrontent et se séduisent au cours d'une partie. Le personnage joué par Faye Dunaway fait perdre ses moyens au personnage joué par Steve McQueen en le provoquant par différents gestes et poses langoureux.

\n

Citons enfin Les Visiteurs du soir de Marcel Carné ; Revolver, de Guy Ritchie ; Whatever Works de Woody Allen, où le personnage principal, un intellectuel surdoué et misanthrope, abandonne son emploi de professeur de physique pour enseigner les échecs.

\n

Dans Sherlock Holmes : Le Jeu des Ombres de Guy Ritchie, on retrouve à plusieurs reprises un motif d'échiquier en noir et blanc afin d'illustrer la lutte intellectuelle entre Sherlock Holmes et le Professeur Moriarty. D'ailleurs, le climax mène à une partie d'échecs entre les deux personnages.

\n

Musique[ | modifier le code]

\n

Le ballet Checkmate (échec et mat) a été écrit par le compositeur britannique Arthur Bliss en 1937 et met en scène les pièces échiquéennes jusqu'à l'assaut final du roi noir.

\n

Avec son tableau Chess Piece (1944), l'américain John Cage allie peinture, musique et échecs puisqu'il s'agit d'une partition peinte sur la représentation d'un échiquier[52].

\n

L'album E2-E4 (1984) du musicien allemand Manuel Göttsching emprunte son titre à l'ouverture du pion-roi[52].

\n

La comédie musicale Chess (1986), sur une musique de Björn Ulvaeus et Benny Andersson (anciens membres d'ABBA) et des paroles de Tim Rice, met en scène un triangle amoureux entre deux participants à un championnat du monde d'échecs et une femme qui tente de séduire l'un et tombe amoureuse de l'autre.

\n
\n
\"\"\n
\n\nMarostica, Place de l'échiquier
\n
\n
\n

Tradition médiévale[ | modifier le code]

\n

La ville de Marostica, Italie, organise une partie d'échecs sur la place publique, avec des personnages vivants costumés qui tiennent lieu de pièces. Cette coutume remontre à 1454. Deux gentilshommes, Rinaldo d'Angarano et Vieri da Vallonara, étaient tous deux amoureux de Lionora, fille du seigneur de Marostica. Ils voulaient s'affronter en duel. Mais le pacifique seigneur leur proposa de s'affronter plutôt au jeu d'échecs. La place publique dallée de pierres alternativement noires et bistre tenait lieu d'échiquier. Le gagnant épouserait la belle Lionora ; le perdant, sa sœur cadette. Le spectacle se déroule au mois de septembre les années paires avec 550 figurants. Pour l'occasion, on recouvre l'échiquier de carrés de tissu[53],[54].

\n
\n

Notes et références[ | modifier le code]

\n

Références issues du Guide des Échecs[ | modifier le code]

\n

Livre dont sont issues ces références : Nicolas Giffard et Alain Biénabe, Le Guide des Échecs : Traité complet, Bouquins,‎ , 1591 p. (ISBN 978-2-221-05913-5).

\n
\n
  1. p. 3-15.
  2. \n
  3. p. 13-15.
  4. \n
  5. p. 16.
  6. \n
  7. p. 20-22.
  8. \n
  9. p. 14.
  10. \n
  11. p. 87.
  12. \n
  13. p. 87-117.
  14. \n
  15. p. 197.
  16. \n
  17. p. 161-196.
  18. \n
  19. p. 118.
  20. \n
  21. p. 36.
  22. \n
  23. p. 379-656.
  24. \n
  25. p. 333-334.
  26. \n
  27. p. 335.
  28. \n
  29. p. 337.
  30. \n
  31. p. 471.
  32. \n
  33. p. 521.
  34. \n
  35. p. 568.
  36. \n
  37. p. 915.
  38. \n
  39. p. 925.
  40. \n
  41. p. 1137.
  42. \n
  43. p. 1167.
  44. \n
  45. p. 899.
  46. \n
  47. p. 899.
  48. \n
  49. p. 900.
  50. \n
  51. p. 908.
  52. \n
\n

Autres notes et références[ | modifier le code]

\n
\n
  1. Trésor de la langue française informatisé.
  2. \n
  3. Richard Réti, Les Maîtres de l'échiquier.
  4. \n
  5. (en) « Laws of chess, article 8 », FIDE (consulté le 29 décembre 2009).
  6. \n
  7. (en) « Laws of chess — Appendices, appendice C », FIDE (consulté le 29 décembre 2009).
  8. \n
  9. Siegbert Tarrasch vs Max Euwe (1922).
  10. \n
  11. « Botvinnik-Yudovich,USSR Championship 1933 ».
  12. \n
  13. Gérard Demuydt, « A. Anderssen - L. Kieseritzky, l'Immortelle de Londres, 1851 » (consulté le 6 janvier 2010).
  14. \n
  15. (en) « Laws of chess, article 6 », FIDE (consulté le 29 décembre 2009).
  16. \n
  17. a et b (en) « Laws of Chess — Appendices », FIDE (consulté le 29 décembre 2009).
  18. \n
  19. (en) « FIDE Rating Regulations (Qualification Commission) — Rate of Play », FIDE (consulté le 29 décembre 2009).
  20. \n
  21. « Fédération Internationale des Échecs » (consulté le 29 décembre 2009).
  22. \n
  23. (en) « Laws of chess », FIDE (consulté le 29 décembre 2009).
  24. \n
  25. « FIDE Chess Ratings » (consulté le 29 décembre 2009).
  26. \n
  27. (en) « International Title Regulations (Qualification Commission) », FIDE (consulté le 29 décembre 2009).
  28. \n
  29. a et b (en) « Regulations for the Titles of Arbiters », FIDE (consulté le 23 décembre 2009).
  30. \n
  31. « Le secteur de l'Arbitrage », Fédération française des échecs (consulté le 29 décembre 2009).
  32. \n
  33. Avant 1956, le champion du monde, s'il avait été battu, aurait disputé un match-tournoi à trois avec le nouveau champion et le nouveau candidat sélectionné par la FIDE. My Great Predecessors, tome II, p. 215.
  34. \n
  35. (en) « Association of Chess Professionals » (consulté le 6 novembre 2013).
  36. \n
  37. Jacques Dexteit et Norbert Engel, Jeu d'échecs et sciences humaines, Payot page 85.
  38. \n
  39. Alfred Binet, Psychologie des grands calculateurs et joueurs d'échecs, L'Harmattan,‎ , 366 p. (ISBN 2-7475-7537-3, présentation en ligne).
  40. \n
  41. (en) Reuben Fine, Psychology of the Chess Player, Dover Publications Inc.,‎ , 74 p. (ISBN 978-0-486-21551-8).
  42. \n
  43. « Poids de 1000 grains de paddy TGR 1: 28 g » (ArchiveWikiwixArchive.isGoogleQue faire ?), consulté le 2013-11-06. « La balle constitue environ 20 pour cent du poids du paddy ».
  44. \n
  45. Homère, Odyssée, I, 107 ; Euripide, Iphigénie à Aulis, v. 195 et suiv. Voir aussi de nombreuses peintures de vases grecs.
  46. \n
  47. Larousse encyclopédique en 10 volumes, Paris, 1984, vol.VIII, p. 7747 (ISBN 978-2-03-102308-1).
  48. \n
  49. Robert Graves, « Les Mythes grecs », édition Fayard, Paris, 1967, traduit de l'anglais par Mounir Hafez, p. 497-517 édition originale : Greek myths, Cassell & c° LTD, Londres 1958.
  50. \n
  51. Pierre Monnet, Jeu d'Echecs, Jeu de Dames, histoire parallèle,‎ , p. 4.
  52. \n
  53. (en) Roland G. Austin, « Greek board games », Antiquity,‎ , p. 257-271 (lire en ligne).
  54. \n
  55. Euripide (trad. François Jouan), Iphigénie à Aulis, Belle-Lettres,‎ (ISBN 2-251-00127-1), p. 67, vers 194-198.
  56. \n
  57. Gorgias de Leontinoï (trad. Jean-Paul Dumont), Eloge de Palamède, La Pléiade, coll. « Les Présocratiques »,‎ (ISBN 978-2-07-011139-8), p. 1043, paragraphe 30.
  58. \n
  59. http://classes.bnf.fr/echecs/histoire/naissance.htm
  60. \n
  61. Jean-Louis Cazaux, L'odyssée des échecs, Praxéo,‎
  62. \n
  63. à ne pas confondre avec sa variante tardive à quatre joueurs dite chaturaji (en); voir la mise au point de Jean-Louis Cazaux, « Échecs et chaturanga : la fin d'un mythe », sur Editions Praxéo,‎ (consulté le 3 septembre 2014)
  64. \n
  65. Petite histoire d'un grand jeu : Les échecs – Développements, Moracchini Échecs Institut.
  66. \n
  67. Dossiers pédagogiques de la Bibliothèque nationale de France.
  68. \n
  69. a, b et cRéférence, Jean-Louis Cazaux, \"L'Odyssée des jeux d'échecs\", Praxéo, 2010
  70. \n
  71. Anthologie sur le jeu d'échecs sur le site de la BNF
  72. \n
  73. a et b (en) Monty Newborn, Kasparov versus Deep Blue : Computer Chess Comes of Age, Springer,‎ , 322 p. (ISBN 978-0-387-94820-1).
  74. \n
  75. (en) Tibor Karolyi, Judit Polgar : The Princess of Chess, Batsford,‎ , 288 p. (ISBN 978-0-7134-8890-6).
  76. \n
  77. Arrêté du du ministre chargé des Sports (Bulletin officiel du ministère de la jeunesse et des sports du ).
  78. \n
  79. Olympic Programme Commission au paragraphe 2.5 « Mind Sports ».
  80. \n
  81. Magnus Carlsen is World Champion.
  82. \n
  83. (en) David Pritchard, Encyclopedia of Chess Variants, Games & Puzzles Publications,‎ , 372 p. (ISBN 978-0-9524142-0-9).
  84. \n
  85. http://www.larousse.fr/encyclopedie/divers/%C3%A9checs/45405#11003855
  86. \n
  87. (la) Jacques de Cessoles, Liber de moribus hominum et officiis nobilium.
  88. \n
  89. Manouk Borzakian, « La géopolitique en échecs », Le Monde diplomatique,‎ , p. 27 (lire en ligne).
  90. \n
  91. Jean-Michel Péchiné, Les Échecs : Roi des jeux, jeu des rois, Gallimard,‎ , 128 p. (ISBN 978-2-07-053396-1).
  92. \n
  93. (en) Andrew Soltis, Soviet Chess 1917-1991, McFarland,‎ , 478 p. (ISBN 978-0-7864-0676-0).
  94. \n
  95. Yves Marek, Art, échecs et mat, Imprimerie nationale,‎ , 186 p. (ISBN 978-2-7427-7493-7).
  96. \n
  97. Jean-Marc Ricci, « Tableaux ayant pour sujet les échecs » (consulté le 29 décembre 2009).
  98. \n
  99. « Chess in the cinema » (consulté le 16 janvier 2011).
  100. \n
  101. « Je sais que ce garçon tire le pistolet admirablement, chasse très bien, joue merveilleusement au billard, aux échecs et au trictrac ; il fait des armes et monte à cheval comme feu le chevalier de Saint-Georges (…) dessine, danse et chante bien. Eh ! diantre, qu'avez-vous donc, vous autres ? Si ce n'est pas là un gentilhomme parfait, montrez-moi un bourgeois qui sache tout cela. » Honoré de Balzac, Le Bal de Sceaux, Édition Charles Furne, 1845, vol.I p. 127.
  102. \n
  103. a et b Fabricio Cardenas, Musicam scire, Echecs et musique, 7 mars 2012
  104. \n
  105. Guida rapida d'Italia, Touring Club Italiano, 1996, 5 volumes, vol.II, p. 142-143 (ISBN 978-88-365-1085-6).
  106. \n
  107. Vue du jeu d'échecs à Marostica.
  108. \n
\n

Voir aussi[ | modifier le code]

\n\n\n

Articles connexes[ | modifier le code]

\n

Bibliographie[ | modifier le code]

\n

Liens externes[ | modifier le code]

\n
\n
\n
\n\n
\n
La version du 25 novembre 2005 de cet article a été reconnue comme « article de qualité », c'est-à-dire qu'elle répond à des critères de qualité concernant le style, la clarté, la pertinence, la citation des sources et l'illustration.
\n
\n
\n
\n
\n\n\n\n\n\n\n\n
\t\t\t\t\t
\n\t\t\t\t\t\tCe document provient de « https://fr.wikipedia.org/w/index.php?title=Échecs&oldid=124883739 ».\t\t\t\t\t
\n\t\t\t\t\t\t\t\t
\n\t\t\t\t\t\t\t", + "annotations": [], + "mimetype": "text/html", + "language": "fr", + "reading_time": 85, + "domain_name": "fr.wikipedia.org", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 0, + "id": 608, + "title": "90% des dossiers médicaux des Coréens du sud vendus à des entreprises privées - ZATAZ", + "url": "http://www.zataz.com/90-des-dossiers-medicaux-des-coreens-du-sud-vendus-a-des-entreprises-privees/", + "content": "

La Corée du Sud vibre en ce moment à la lecture d’une information qui pend aux nez des Français. Une entreprise privée a récupéré 90% des dossiers médicaux des habitants du pays du matin calme au soleil levant pour les revendre.

\n

Une entreprise spécialisée dans le développement de logiciels en charge de gérer les frais médicaux, programmes utilisés dans les hôpitaux et la Korean Pharmaceutical Information Center, a offert il y a quelques mois ses logiciels de gestion d’officines. Plus de la moitié des pharmacies du pays ont utilisé l’outil. Sauf que les données sauvegardées ont été revendues à IMS Health Korea. Cette entreprise, dont le siège social est basé aux USA, a ensuite commercialisé, illégalement, les données à d’autres sociétés en Corée.

\n

La loi 2011 sur le droit de la protection des renseignements personnels interdit l’utilisation des renseignements personnels et des renseignements médicaux sans le consentement des patients. Le Pharmaceutical Information Center est actuellement jugé pour cette collecte illégale, qui date de 2013, et la distribution des informations médicales de 90% des Coréens.

\n
\n

Un cas qui pourrait toucher la France ?

\n
\n

Depuis février 2015, l’ouverture des données de santé dans l’hexagone a été décidée par le législateur. L’assurance-maladie a annoncé qu’elle proposait, en accès libre, sa base de données « Damir » sur le site data.gouv.fr. Un big data de la santé qui regroupe les informations issues de 1,2 milliard de feuilles de soins, de 500 millions d’actes médicaux et de 11 millions d’hospitalisations. Ce big data ne propose pas les identités (pas de nom, de numéro de sécurité sociale, …), uniquement des millions de chiffres et de données de santé. Cette faramineuse base de données, qui double de volume chaque année, permet d’extraire, par exemple, des statistiques liées à la santé dans les régions. L’article 47 de loi permet aux acteurs privés d’accéder aux données de la CNAMTS. C’est l’Institut national des données de santé (INDS) qui a en charge de répondre aux demandes du « privé » pour accéder aux données plus ciblées (et payantes).

\n", + "annotations": [], + "mimetype": "text/html", + "language": "fr-FR", + "reading_time": 1, + "domain_name": "www.zataz.com", + "preview_picture": "http://www.zataz.com/wp-content/uploads/HD-Virus.png", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 0, + "id": 606, + "title": "Mass Surveillance As Art", + "url": "https://www.nationaljournal.com/s/73311/mass-surveillance-art", + "content": "
\n
\n

Sept. 18, 2015, 6 a.m.

\n
\n
\n

Julian Assange in a 2012 portrait by Jacob Appelbaum. Jacob Appelbaum

\n
\n
\n

But today, the lo­qua­cious Ap­pel­baum wants to just talk about his art. Mostly.

\n

The pho­to­graph­er has as­sembled in­tim­ate por­traits of his au­thor­ity-chal­len­ging com­rades—Poitras, Har­ris­on, Wei­wei, Wikileaks cofounder Ju­li­an As­sange, Snowden-favored journ­al­ist Glenn Gre­en­wald, and former NSA ana­lyst-turned-whis­tleblower Wil­li­am Bin­ney—each bathed in an in­frared glow. The tech­nique, which res­ults in an un­mis­tak­able re­semb­lance to sur­veil­lance foot­age, was ac­com­plished us­ing ciba­chrome prints and shoot­ing with a dis­con­tin­ued Kodak Col­or In­frared cam­era—a pro­cess Ap­pel­baum likes to boast is “fully ana­log.”

\n

“A key part about this is the pro­cess and the film it­self—it is a sur­veil­lance film,” Ap­pel­baum tells me. “That said, I am par­tial to the col­or red. I really like it. and I think that it sig­ni­fies pas­sion, and I think that pas­sion is something that all the people in the show share.”

\n

Ap­pel­baum also likes black. He wears a gray but­ton-up, black jeans, black shoes, a black belt, and a con­spicu­ous black tie dur­ing our in­ter­view. His trade­mark thick horn-rimmed glasses—also black—rest eas­ily on his face, do­ing little to mask the dart­ing in­tens­ity in eyes. A met­al bar punc­tures two holes in­to the up­per car­til­age of his right ear. Even forced to dress up, he has the un­mis­tak­able look of a cy­ber­punk.

\n

His art show, which opened Sept. 11 and runs un­til Hal­loween, is titled Sam­izdata: Evid­ence of Con­spir­acy, after a Rus­si­an word re­fer­ring to the dodging of cen­sors to share il­li­cit ma­ter­i­al with­in the So­viet bloc—think Aleksandr Solzhen­it­syn’s The Gu­lag Ar­chipelago. It is hos­ted at the NOME Gal­lery, which opened earli­er this year and has a strong bend for anti-au­thor­it­ari­an—and, some might say, anti-Amer­ic­an—so­cial com­ment­ary. NOME’s pre­vi­ous two ex­hib­its, Paolo Cirio’s Over­ex­posed and James Bridle’s The Glo­m­ar Re­sponse, both took crit­ic­al aim at the U.S. in­tel­li­gence com­munity, of­fer­ing un­flinch­ing ex­am­in­a­tions of seni­or of­fi­cials like CIA Dir­ect­or John Bren­nan and FBI Dir­ect­or James Comey, and of the level of re­dac­tions present in the Sen­ate In­tel­li­gence Com­mit­tee’s land­mark tor­ture re­port.

\n

For Ap­pel­baum, though, his art­work de­veloped or­gan­ic­ally. All of the pho­tos were taken be­fore the concept of a gal­lery ma­ter­i­al­ized, ex­cept for a shot of Har­ris­on, the Brit­ish Wikileak­er. Har­ris­on’s por­trait, which finds her sit­ting on a rock and, head cocked a bit, look­ing softly in­to the cam­era, also hap­pens to be Ap­pel­baum’s fa­vor­ite, be­cause it bal­ances her qual­it­ies as both “a total ba­dass mother­fuck­er” and “the pix­ie of Wikileaks.”

\n

The pho­to­graphs “show the people in the way that I think of them,” Ap­pel­baum ex­plains. The most strik­ing demon­stra­tion of that edict rests in the por­trait of Bin­ney, which finds the former NSA of­fi­cial stand­ing, with one fist clenched, in front of a tree in Ber­lin. Sur­veil­lance nerds will be im­me­di­ately struck by the photo, be­cause Bin­ney doesn’t have legs in real life. He lost them to dia­betes years ago.

\n
\n
\n

A key part about this is the process and the film itself—it is a surveillance film. ”

\n

Jacob Appelbaum

\n
\n
\n

Later that even­ing at the gal­lery, Ap­pel­baum is giv­ing a walk-through for a small gath­er­ing of press and some friends. He seems a little less com­fort­able in front of a lar­ger group, speak­ing more de­lib­er­ately and evenly as he de­scribes each pho­to­graph.

\n

Gre­en­wald’s por­trait, taken in 2012 in Rio de Janeiro, shows the com­bat­ive journ­al­ist in a softer light. His part­ner, Dav­id Mir­anda, has his arms draped around him as the two stand be­side one of their many dogs in the rain forest.

\n

“As an artist, I think it’s really im­port­ant to be cog­niz­ant of the things you pro­mote. So I don’t take pic­tures of people smoking be­cause I think it’s dis­gust­ing. And I don’t want chil­dren to go out and smoke ci­gar­ettes. But I do want chil­dren to be ho­mo­sexu­als,” says Ap­pel­baum, who has iden­ti­fied him­self as “queer” in past in­ter­views.

\n

“Glenn Gre­en­wald and Dav­id Mir­anda are totally fierce and fant­ast­ic men; they’re beau­ti­ful,” he con­tin­ues. “They’re the hot­test gay couple alive, so if you ar­gue with me that’s fine—but they’re still go­ing to be the hot­test gay couple alive.”

\n

He stops in front of the Bin­ney por­trait, which he says is his second fa­vor­ite after Har­ris­on’s.

\n

“He’s one of the only hon­or­able people to ever work in the in­tel­li­gence com­munity,” Ap­pel­baum says. “He’s one of the very few Amer­ic­ans that makes me not ashamed to be Amer­ic­an.”

\n

Ap­ple­baum sighs deeply and pauses. He looks sud­denly vul­ner­able for a brief mo­ment be­fore re­col­lect­ing him­self and mov­ing on to Ai Wei­wei’s por­trait.

\n

Wei­wei is both a sub­ject and a bit of col­lab­or­at­or in Ap­pel­baum’s ex­hib­it, thanks to the in­clu­sion of an ad­or­able plush panda. Along with sev­er­al oth­er pan­das, its cot­ton innards were gut­ted by Ap­pel­baum and Wei­wei dur­ing a meet-up in Beijing earli­er this year—cap­tured, nat­ur­ally, on film by Poitras—and re­placed with shred­ded Snowden doc­u­ments. The pro­ject’s title, “Panda to Panda,” is a ref­er­ence to the slang term used to refer to China’s secret po­lice. It’s ab­bre­vi­ation, P2P, doubles as short­hand for peer-to-peer com­mu­nic­a­tion—a kind of de­cent­ral­ized net­work­ing di­git­al act­iv­ists like to use to avoid de­tec­tion.

\n

If the ex­hib­i­tion is an in­tim­ate win­dow in­to the lives of the world’s most fam­ous di­git­al-pri­vacy Avengers, Ap­pel­baum might best be un­der­stood as the Cap­tain Amer­ica of the group—ex­cept ob­vi­ously lack­ing in the pat­ri­ot­ism de­part­ment. While ad­ept at many things, his most po­tent con­tri­bu­tion to the team may be his rah-rah evan­gel­ism for the cause, which any­one who has listened to his con­fid­ent, long-win­ded dis­ser­ta­tions on the mor­al im­per­at­ives of pri­vacy can at­test are com­pel­ling and easy to buy in­to. It was a skill that served him well as a core de­veloper of the Tor Pro­ject, an on­line browser that keeps users an­onym­ous.

\n

Ap­pel­baum is also the com­mon link for the move­ment’s dis­par­ate mem­bers, who are spread out on sev­er­al dif­fer­ent con­tin­ents in vary­ing de­grees of ex­ile. He bridges the gap between more rad­ic­al ele­ments, like Ju­li­an As­sange, who be­lieves nearly no secret is worthy of re­dac­tion, and the more con­sid­er­ate views held by Gre­en­wald and Poitras. (An ex­ample of that ten­sion: When Gre­en­wald and Poitras, keep­ers of the Snowden trove, re­fused to pub­lish the name of a coun­try in which the NSA was re­cord­ing nearly all phone calls, Wikileaks con­demned the omis­sion in a Twit­ter rant. Not sat­is­fied to merely vent, Wikileaks an­nounced days later that “Coun­try X” was in fact Afgh­anistan.)

\n

Ap­pel­baum bristles at the no­tion that his pho­to­graphs rise to that level of na­vel-gaz­ing—that it ex­ists as cho­reo­graphed flat­tery for a team of in­ter­na­tion­al su­per-dis­sid­ents. The ex­hib­it, he says, de­picts “in­di­vidu­als that work to­geth­er for very pos­it­ive goals, very much work in tan­dem to­geth­er—but they wouldn’t call them­selves a group.”

\n

In­stead, he of­fers, “they rep­res­ent a net­work, and these are the nodes of that net­work. I’m not re­flect­ing back on our move­ment, but rather this is a trend in civil so­ci­ety, from China to the Ecuadori­an em­bassy in Lon­don to New York City to Ber­lin. It goes around the world.”

\n
\n
\n

Glenn Greenwald and David Miranda are totally fierce and fantastic men; they’re beautiful. ”

\n

Jacob Appelbaum

\n
\n
\n
\n

After I had faked my way through 20 minutes of our in­ter­view fo­cus­ing on his art—dur­ing which Ap­pel­baum seems to get an­noyed more than once at my na­iv­ete—I turned to polit­ics. I ask what he thinks of the U.S. pres­id­en­tial cam­paign and Hil­lary Clin­ton.

\n

Clin­ton would be great for ad­van­cing lots of so­cial causes and mak­ing health care more af­ford­able and could be an over­all ef­fect­ive lead­er, Ap­pel­baum con­cedes, be­fore adding that her elec­tion would also “be the worst out­come for me per­son­ally” and any­one else who tries to ex­pose gov­ern­ment secrets.

\n

“Can you ima­gine a pres­id­en­tial can­did­ate that will try to hunt down Wikileaks people more ser­i­ously?” he asks. “If Hil­lary Clin­ton be­comes pres­id­ent, it’ll be great news for my moth­er, and I think that alone is worth­while. But it will be my own death sen­tence.”

\n

Ap­pel­baum’s law­yers have ad­vised him to not re­turn to the United States. Due to a long-run­ning Justice De­part­ment in­vest­ig­a­tion in­to Wikileaks, his past af­fil­i­ation with the group could spell trouble for the thirty-something ex-pat from Cali­for­nia. The Justice De­part­ment did not re­spond to mul­tiple re­quests for com­ment re­gard­ing the in­vest­ig­a­tion.

\n

Earli­er this year, Google in­formed Ap­pel­baum that it was com­pelled to hand over his per­son­al ac­count data to the U.S. gov­ern­ment for the pur­poses of the in­vest­ig­a­tion. In a lengthy rant on Twit­ter, Ap­pel­baum pos­ted se­lect screen­shots of Google’s 306-page leg­al dis­clos­ure.

\n

“Ten pages in­to this leg­al doc­u­ment and I’m con­vinced that I’m nev­er go­ing to re­turn to my home coun­try,” Ap­pel­baum tweeted at the time. “What the ac­tu­al fuck.”

\n

Ap­pel­baum doesn’t think any of the pres­id­en­tial can­did­ates would have much sym­pathy for leak­ers—or that any would do much to rein in the NSA. Oth­er than Clin­ton, he dis­misses the rest of the pres­id­en­tial can­did­ates as “a grab bag of hil­ar­ity,” ex­pec­tedly tak­ing his time to pil­lory Don­ald Trump and his “Make Amer­ica Great Again” slo­gan. (“What a hat!,” he ex­claims with a laugh, ad­mit­ting he’d like to own one for comed­ic ef­fect.)

\n

I ask wheth­er he feels dif­fer­ently about Sen. Bernie Sanders, the self-de­scribed so­cial­ist run­ning for the Demo­crat­ic nom­in­a­tion, or Re­pub­lic­an Sen. Rand Paul, both of whom have been con­sist­ently and vo­cally op­posed to over­broad NSA data col­lec­tion.

\n

“Rand Paul might be great on the NSA, but how is he on oth­er things, like the death pen­alty?” He ad­mits a lik­ing for Sanders but quickly notes “he could do a lot bet­ter on ra­cism,” cit­ing the can­did­ate’s hand­ling of Black Lives Mat­ter pro­test­ers who in­ter­rup­ted him dur­ing a re­cent cam­paign event.

\n

Ap­pel­baum takes pains to stress that he and those fea­tured in his art are not just crit­ics of mass-sur­veil­lance re­gimes but people who be­lieve they are at the van­guard of fight­ing for civil liber­ties, of which spy­ing re­mains a cru­cially im­port­ant battle front—one that he ex­pects to rage on for dec­ades.

\n

“Rein­ing in the NSA is a really weird subis­sue,” he says. “If you look at the gay-rights move­ment, it took a really long time for that to be­come a main­stream is­sue. And I’m think­ing the NSA is­sue, it’s main­stream in a lot of ways but it’s real hard to un­der­stand.”

\n
\n

Glenn Greenwald and David Miranda in a 2012 portrait by Jacob Appelbaum. Jacob Appelbaum

\n
\n
\n
\n

If Hillary Clinton becomes president, it’ll be great news for my mother, and I think that alone is worthwhile. But it will be my own death sentence. ”

\n

Jacob Appelbaum

\n
\n
\n

The next day, the panda is wear­ing a T-shirt. “Fuck the NSA,” it reads in bold black let­ters that ad­orn it, baby-sized and powder blue. Ta­tiana Baz­zichelli, the show’s cur­at­or, ex­plains that one of Ap­pel­baum’s friends stopped by earli­er and brought it as a gal­lery-warm­ing present.

\n

I came back to the gal­lery for the pub­lic open­ing to see the big­ger crowd and be­cause Ap­pel­baum told me that Poitras—whom I’d been try­ing to get in touch with since I ar­rived in Ber­lin—would stop by.

\n

Con­nect­ing with her is no easy task. On top of be­ing in­tensely private, Poitras was keep­ing busy. I’d heard she had been spend­ing most of her time re­cently in New York, ready­ing a pre­view for the city’s an­nu­al film fest­iv­al of a new doc­u­ment­ary series she is launch­ing called Field of Vis­ion. “Asylum,” the first epis­ode of the pro­ject, is a por­trait of Wikileaks’s As­sange, fol­low­ing him as he pub­lishes the dip­lo­mat­ic cables that rocked the world and ends up ma­rooned in Lon­don’s Ecuadori­an em­bassy, where he has been holed up for the past three years.

\n

Poitras is also pre­par­ing an “im­mers­ive film en­vir­on­ment” that will de­but in Feb­ru­ary at New York’s Whit­ney Mu­seum of Amer­ic­an Art. Ap­pel­baum in­struc­ted me to pay at­ten­tion to the Whit­ney in­stall­a­tion when I asked what we might see next come out of the Snowden archive.

\n

Des­pite steady rain, the ex­hib­it’s open­ing show­ing is im­press­ive. The small gal­lery is crowded with dozens of people, and an­oth­er 20 are out­side en­joy­ing free al­co­hol and smoking ci­gar­ettes.

\n

Much of the crowd is mono­chro­mat­ic, dressed, like Ap­pel­baum, all in black. A ma­jor­ity of con­ver­sa­tions I over­hear are in Eng­lish. I spot Ap­pel­baum—now wear­ing a red shirt but still tol­er­at­ing the un­ne­ces­sary black tie—with a glass of wine in hand, laugh­ing bois­ter­ously with a couple of friends who came out for his big night. Now that it’s here, he looks re­lieved. He stops every few minutes to snap pho­tos with his smart­phone of vari­ous guests— the anti-sur­veil­lance act­iv­ist’s de­sire to doc­u­ment the mo­ment is un­res­trained. Later in the night, he will bound over to me and ju­bil­antly tell me that four of the por­traits have already been sold.

\n

True to Ap­pel­baum’s prom­ise, Poitras ar­rives, and I catch her mo­ments after she enters. She doesn’t re­cog­nize me at first, but after I jog her memory of a past in­ter­view she warms up. “Is this on the re­cord?” she asks after I’ve already put my note­book away. I tell her no, and we ex­change pleas­ant­ries briefly be­fore she is pulled away.

\n

I waited 90 minutes be­fore hav­ing an­oth­er chance to talk to her. The Oscar-win­ning film­maker who quar­ter­backs the re­lease of Snowden files in ma­jor me­dia or­gan­iz­a­tions around the world is a coveted celebrity in this room, and a nev­er-end­ing line of fans all seem to have a hug to give and a story to catch up on.

\n

Fi­nally I see Poitras alone, gaz­ing in­to the flowers that sur­round Wei­wei’s por­trait. This has been the first time all night she has had more than a mo­ment to check out the art.

\n

After agree­ing to a brief in­ter­view, she com­ments that the gal­lery is an “ex­traordin­ary doc­u­ment of a dec­ade that changed his­tory.” She said she has been ur­ging Ap­pel­baum to share his art with the world for years and is happy he is fi­nally ob­li­ging.

\n

We are in­ter­rup­ted twice by friends of Poitras who ap­proach and give her a cel­eb­rat­ory hug. I quickly ask about her law­suit against the Justice De­part­ment seek­ing re­cords re­lated to the dozens of times she was de­tained at air­ports, but she doesn’t have an up­date. On the sur­veil­lance-re­form law that Obama signed in­to law earli­er this year, which ef­fect­ively ends the NSA’s bulk col­lec­tion of do­mest­ic phone metadata, she says it is a nice start but quickly adds, “I don’t think U.S. cit­izens are the only ones who should have a right to pri­vacy.” She de­murs on tak­ing much cred­it for the law’s pas­sage, des­pite the clear line of mo­mentum that traces back to the first Snowden rev­el­a­tions. Soon my time is up, as an­oth­er friend of hers in­ter­rupts to share a quick laugh and pull her back in­to the crowd.

\n

I see Ap­pel­baum once more be­fore I leave, and he ad­mits a great sense of re­lief now that the ex­hib­it has opened. But he keeps the night in per­spect­ive. “Nev­er once dur­ing this pro­cess did I think I was go­ing to be raided,” he says when I ask how the stress com­pared to writ­ing a big ex­pose on the NSA.

\n
\n
\n

I don’t think U.S. citizens are the only ones who should have a right to privacy. ”

\n

Laura Poitras

\n
\n
\n

I don’t know if Ap­pel­baum will ever re­turn to the United States. Watch­ing him in Ber­lin, I’m not sure he really needs to. He has found a home here and just star­ted a Ph.D pro­gram at the Eind­hoven Uni­versity of Tech­no­logy in the Neth­er­lands, “primar­ily fo­cus­ing on math­em­at­ics to thwart spies for the next thou­sand years.”

\n

It is un­clear wheth­er Ap­ple­baum is be­ing sens­ible and re­act­ing to the like­li­hood of real ar­rest and in­car­cer­a­tion if he sets foot on Amer­ic­an soil, or wheth­er, like many people who in­hab­it the di­git­al-rights sphere, he is be­ing a tad para­noid.

\n

But un­like Poitras, Ap­pel­baum doesn’t have a pro­tect­ive shield that comes with the no­tori­ety of win­ning an Oscar. And he knows he’s not Snowden, an in­ter­na­tion­al celebrity he be­lieves will be able to re­turn home one day in a way that brings him home with “a tick­er-tape parade.” Former At­tor­ney Gen­er­al Eric Hold­er said this sum­mer that the “pos­sib­il­ity ex­ists” of such a scen­ario, though the Obama ad­min­is­tra­tion—which has pro­sec­uted more in­di­vidu­als un­der the Es­pi­on­age Act than all pre­vi­ous pres­id­en­cies com­bined—poured wa­ter on the idea when it re­spon­ded to an on­line pe­ti­tion call­ing for Snowden’s par­don.

\n

“What would I come home to? To what justice sys­tem?” Ap­pel­baum asks near the end of our in­ter­view. “The FBI tried to talk to me in Europe, tried to get me to go to the U.S. em­bassy to dis­cuss ‘safely re­turn­ing home’ on ‘neut­ral ground.’ It’s so ri­dicu­lous; it’s ri­dicu­lous bull­shit on so many dif­fer­ent levels.”

\n

Pon­der­ing his new life in Europe, Ap­ple­baum is still pro­cessing his ab­rupt, un­planned de­par­ture from the United States. Ber­lin, he says, “is a won­der­ful place. It’s won­der­ful on so many levels.’’

\n

But the sep­ar­a­tion is clearly pain­ful too.

\n

“I kind of wish I had said good­bye to my moth­er, if I ever see her again in my life. That stuff weighs very heav­ily on me,” he says. “It would have been nice to pack my house, get some ex­tra un­der­wear, and take some pho­tos of my dead fath­er with me.”

\n
\n

Laura Poitras in a 2013 a portrait by Jacob Appelbaum. Jacob Appelbaum

\n
\n
\n

Dustin Volz is cur­rently on as­sign­ment in Ber­lin through the Ar­thur F. Burns Fel­low­ship, a two-month re­port­ing pro­gram in Ger­many run by the In­ter­na­tion­al Cen­ter for Journ­al­ists. A ver­sion of this story was also pub­lished in Han­dels­blatt Glob­al Edi­tion.

\n
\n

Share Tweet Email

\n", + "annotations": [], + "mimetype": "text/html", + "reading_time": 20, + "domain_name": "www.nationaljournal.com", + "preview_picture": "https://www.nationaljournal.com/media/media/2015/09/17/06Julian-Assange.jpg", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 0, + "id": 605, + "title": "What David Cameron did to the pig, his party is now doing to the country", + "url": "http://www.newstatesman.com/2015/09/what-david-cameron-did-pig-his-party-now-doing-country", + "content": "

Whatever you do, don’t think about David Cameron and a dead pig. I know, I know it’s like trying not to think of an elephant, but the fact is that the allegations that the Prime Minister may have put a 'private part of his anatomy\" into a dead pig's mouth as part of an initiation ritual for an elite drinking society at Oxford University are actually a very serious matter, and it’s all about corruption and the nature of elected power, and it would help if we could all just calm down for a second and stop giggling. Don’t think I don’t see you at the back there.

\n

You know, I feel for David Cameron today, I really do. Politicians’ private sex lives should never be used against them - unless their particular proclivities implicate them in gross hypocrisy or they have harmed another human being. If the rumours are true, it’s unlikely that the pig in question was hurt by the Prime Minister’s ministrations, given that it was already missing its limbs and torso.

\n

Sniggering aside, this is unlikely to hurt David Cameron in the long run. He’s not looking for re-election, and besides, everyone knows posh people get up to weird sex stuff. Weird sex stuff is as British as weak tea and racism. When I was at Oxford, it was an open secret that the posh kids had naughty parties, and, of course, so did the rest of us - the difference was the much lower budget, and the fact that the posh kids didn’t seem to enjoy it as much as we did. It all seemed to be more about getting on than getting off. You didn’t shag or not shag the pig’s face because that was what you were into, you did it because you had your eye on a safe seat in Dorset in 20 years’ time and you needed to make the right friends.

\n

There is a reason that David Cameron is allowed to hold office when everyone assumes he spent the 1980s taking drugs and getting up to weird things with his Eton mates, but Jeremy Corbyn is considered unelectable because he didn’t sing the national anthem last week. Cameron is part of a select group of people to whom different rules apply, and he knows it, and his friends know it, and the tabloids know it, and the whole cosy British political machine knows it. This is why Corbyn will spend the next five years being savaged for having a slightly rumpled tie by the same newspapers that reported on the dead pig allegations under the title \"the making of an extraordinary Prime Minister\".

\n

The thing that's really horrifying about what has already been dubbed the 'Hameron' scandal is that it demonstrates what entitlement of this kind actually means, and how embarrassing it all is. There are people out there who can spend their early twenties in close proximity to cocaine and popping their peckers in offal and not even consider for a second that there might be anyone better placed to run the country. These are people who know the rules don’t apply to them, who know they can do whatever they want and still end up in charge.

\n

I don't honestly care whether or not David Cameron shagged a dead pig. I've been to enough house parties in Bethnal Green that this sort of thing doesn't shock me. Come back to me when there’s video evidence of Cameron dressed in a leather gimp-suit tanned from the flayed skins of the former shadow cabinet, leaping into an entire Shropshire field full of pigs and screaming that his name is Legion. Then we’ll talk. There are a lot of things that David Cameron has definitely done that I do find disgusting, though. Taking away benefits from sick and disabled people, pricing poor kids out of higher education, and forcing millions of families to rely on food banks. That, to me, is shocking and grotesque. I don't give a damn about what he did or didn’t do to that pig, and whether there was mood-lighting involved.

\n

But the fact is that a lot of people do, and they're precisely the sort of people whose votes Cameron has relied on to shore up the power he clearly feels is his by right, might and various dodgy initiation rituals involving sex workers, smashing up pubs and knobbing bits of meat. Cameron clearly believes those people are there to be manipulated, and that’s the reason this story actually matters, beyond the immediate risk that a handful of pearl-clutchers in the Home Counties might splutter themselves to death.

\n

I was explaining all this to an American friend who asked, not unreasonably, why I'd spent all morning scrolling through Twitter and cackling like a toddler with a nerf gun. I did my best to describe seriously what had happened, and my friend, who does not follow British politics, asked me, 'so this guy, was he elected or appointed?'

\n

The answer, of course, is both. David Cameron is not just prime minister because a quarter of the country voted for him. That's not how power works in Britain, or anywhere, and it's moments like this that show it plainly, which is why we're all vaguely embarrassed today. Cameron's route to the office he clearly believes himself born to began much earlier, possibly even on a balmy Oxford night, just Dave, a dead pig and a select group of wide-eyed, gurning future business leaders, all whooping and cheering.

\n

It would surely have been a moment more important to Cameron's career than any number of photoshoots with builders in Totnes. Power and money are accessed through the back door, or, as it may be, the pig's mouth, and as with any kink, the eroticism isn't about the act, but about what the act symbolises. It's about humiliation, about control, about power play. What might the young swain have been thinking as he unzipped? What went through his head? If you ask me, I'll bet he was thinking: Soon. Someday soon, I will do this to the whole bloody country.

\n", + "annotations": [], + "mimetype": "text/html", + "language": "en", + "reading_time": 5, + "domain_name": "www.newstatesman.com", + "preview_picture": "http://www.newstatesman.com/sites/default/files/styles/thumb_730/public/blogs_2015/09/gettyimages-464604046.jpg?itok=EaABrZda", + "tags": [] + }, + { + "is_archived": 1, + "is_starred": 0, + "id": 604, + "title": "CLICK HERE to support 2016 CES Winner, Revolutionary Auto-Tracking Robot", + "url": "https://www.indiegogo.com/projects/2016-ces-winner-revolutionary-auto-tracking-robot", + "content": "
\n
\n

Sign Up for Inspiration

\n

Private, secure, spam-free

\n

Follow us

\n
\n
\n

About Indiegogo

\n
\n

Language

\n
\n
\n
\n
\n
\n

Campaigning

\n
\n
\n

Contributing

\n
\n
\n

Sign Up for Inspiration

\n

Private, secure, spam-free

\n
\n
\n

About Indiegogo

\n
\n
\n

Follow us

\n
\n
\n

Language

\n
\n
\n
\n
", + "annotations": [], + "mimetype": "text/html", + "reading_time": 0, + "domain_name": "www.indiegogo.com", + "preview_picture": "https://c1.iggcdn.com/indiegogo-media-prod-cld/image/upload/c_fill,f_auto,h_630,w_1200/v1447395263/d6ckex9ynild6ica1xdk.jpg", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 1, + "id": 603, + "title": "No title found", + "url": "http://carnetdevol.shost.ca/wordpress/aide-memoire-sur-les-commandes-associees-a-systemd/", + "content": "wallabag can't retrieve contents for this article. Please report this issue to us.", + "annotations": [], + "mimetype": "text/html", + "reading_time": 0, + "domain_name": "carnetdevol.shost.ca", + "tags": [] + }, + { + "is_archived": 1, + "is_starred": 0, + "id": 602, + "title": "Présentation d'Arduino - Tuto Arduino - Le blog d'Eskimon", + "url": "http://eskimon.fr/73-arduino-101-presentation", + "content": "
\n

Comment faire de l’électronique en utilisant un langage de programmation ? La réponse, c’est le projet Arduino qui l’apporte. Vous allez le voir, celui-ci a été conçu pour être accessible à tous par sa simplicité. Mais il peut également être d’usage professionnel, tant les possibilités d’applications sont nombreuses.

\n\n
\n
\n
\n

Qu’est-ce que c’est ?

\n
Une équipe de développeurs composée de Massimo Banzi, David Cuartielles, Tom Igoe, Gianluca Martino, David Mellis et Nicholas Zambetti a imaginé un projet répondant au doux nom de Arduino et mettant en œuvre une petite carte électronique programmable et un logiciel multiplateforme, qui puisse être accessible à tout un chacun dans le but de créer facilement des systèmes électroniques. Étant donné qu’il y a des débutants parmi nous, commençons par voir un peu le vocabulaire commun propre au domaine de l’électronique et de l’informatique.\n

Une carte électronique

\n

Une carte électronique est un support plan, flexible ou rigide, généralement composé d’epoxy ou de fibre de verre. Elle possède des pistes électriques disposées sur une, deux ou plusieurs couches (en surface et/ou en interne) qui permettent la mise en relation électrique des composants électroniques. Chaque piste relie tel composant à tel autre, de façon à créer un système électronique qui fonctionne et qui réalise les opérations demandées.

\n
\"\"\"\"
Exemples de cartes électroniques
\n

Évidemment, tous les composants d’une carte électronique ne sont pas forcément reliés entre eux. Le câblage des composants suit un plan spécifique à chaque carte électronique, qui se nomme le schéma électronique.

\n
\"\"
Exemple de schéma électronique – carte Arduino Uno
\n

Enfin, avant de passer à la réalisation d’un carte électronique, il est nécessaire de transformer le schéma électronique en un schéma de câblage, appelé typon.

\n
\"\"
Exemple de typon – carte Arduino
\n

Une fois que l’on a une carte électronique, on fait quoi avec ?

\n

Eh bien une fois que la carte électronique est faite, nous n’avons plus qu’à la tester et l’utiliser ! Dans notre cas, avec Arduino, nous n’aurons pas à fabriquer la carte et encore moins à la concevoir. Elle existe, elle est déjà prête à l’emploi et nous n’avons plus qu’à l’utiliser. Et pour cela, vous allez devoir apprendre comment l’utiliser, ce que je vais vous montrer dans ce tutoriel.

\n

Programmable ?

\n

J’ai parlé de carte électronique programmable au début de ce chapitre. Mais savez-vous ce que c’est exactement ? Non pas vraiment. Alors voyons ensemble de quoi il s’agit. La carte Arduino est une carte électronique qui ne sait rien faire sans qu’on lui dise quoi faire. Pourquoi ? Eh bien c’est du au fait qu’elle est programmable. Cela signifie qu’elle a besoin d’un programme pour fonctionner.

\n

Un programme

\n

Un programme est une liste d’instructions qui est exécuté par un système. Par exemple votre navigateur internet, avec lequel vous lisez probablement ce cours, est un programme. On peut analogiquement faire référence à une liste de course :

\n
\"\"
\n

Chaque élément de cette liste est une instruction qui vous dis : « Va chercher le lait » ou « Va chercher le pain », etc. Dans un programme le fonctionnement est similaire :

\n
  • Attendre que l’utilisateur rentre un site internet à consulter
  • \n
  • Rechercher sur internet la page demandée
  • \n
  • Afficher le résultat
  • \n

Tel pourrait être le fonctionnement de votre navigateur internet. Il va attendre que vous lui demandiez quelque chose pour aller le chercher et ensuite vous le montrer. Eh bien, tout aussi simplement que ces deux cas, une carte électronique programmable suit une liste d’instructions pour effectuer les opérations demandées par le programme.

\n

Et on les trouves où ces programmes ? Comment on fait pour le mettre dans la carte ? o_O

\n

Des programmes, on peut en trouver de partout. Mais restons concentré sur Arduino. Le programme que nous allons mettre dans la carte Arduino, c’est nous qui allons le réaliser. Oui, vous avez bien lu. Nous allons programmer cette carte Arduino. Bien sûr, ce ne sera pas aussi simple qu’une liste de course, mais rassurez-vous cependant car nous allons réussir quand même ! Je vous montrerais comment y parvenir, puisque avant tout c’est un des objectifs de ce tutoriel. Voici un exemple de programme : \"\" Vous le voyez comme moi, il s’agit de plusieurs lignes de texte, chacune étant une instruction. Ce langage ressemble à un véritable baragouin et ne semble vouloir à priori rien dire du tout… Et pourtant, c’est ce que nous saurons faire dans quelques temps ! Car nous apprendrons le langage informatique utilisé pour programmer la carte Arduino. Je ne m’attarde pas sur les détails, nous aurons amplement le temps de revenir sur le sujet plus tard. Pour répondre à la deuxième question, nous allons avoir besoin d’un logiciel…

\n

Et un logiciel ?

\n

Bon, je ne vais pas vous faire le détail de ce qu’est un logiciel, vous savez sans aucun doute de quoi il s’agit. Ce n’est autre qu’un programme informatique exécuté sur un ordinateur. Oui, pour programmer la carte Arduino, nous allons utiliser un programme ! En fait, il va s’agir d’un compilateur. Alors qu’est-ce que c’est exactement ?

\n

Un compilateur

\n

En informatique, ce terme désigne un logiciel qui est capable de traduire un langage informatique, ou plutôt un programme utilisant un langage informatique, vers un langage plus approprié afin que la machine qui va le lire puisse le comprendre. C’est un peu comme si le patron anglais d’une firme Chinoise donnait des instructions en anglais à un de ses ouvriers chinois. L’ouvrier ne pourrait comprendre ce qu’il doit faire. Pour cela, il a besoin que l’on traduise ce que lui dit son patron. C’est le rôle du traducteur. Le compilateur va donc traduire les instructions du programme précédent, écrites en langage texte, vers un langage dit « machine ». Ce langage utilise uniquement des 0 et des 1. Nous verrons plus tard pourquoi. Cela pourrait être imagé de la façon suivante :

\n
\"\"
\n

Donc, pour traduire le langage texte vers le langage machine (avec des 0 et des 1), nous aurons besoin de ce fameux compilateur. Et pas n’importe lequel, il faut celui qui soit capable de traduire le langage texte Arduino vers le langage machine Arduino. Et oui, sinon rien ne va fonctionner. Si vous mettez un traducteur Français vers Allemand entre notre patron anglais et son ouvrier chinois, ça ne fonctionnera pas mieux que s’ils discutaient directement. Vous comprenez ?

\n

Et pourquoi on doit utiliser un traducteur, on peut pas simplement apprendre le langage machine directement ?

\n

Comment dire… non ! Non parce que le langage machine est quasiment impossible à utiliser tel quel. Par exemple, comme il est composé de 0 et de 1, si je vous montre ça : « 0001011100111010101000111 », vous serez incapable, tout comme moi, de dire ce que cela signifie ! Et même si je vous dis que la suite « 01000001 » correspond à la lettre « A », je vous donne bien du courage pour coder rien qu’une phrase ! 😛 Bref, oubliez cette idée. C’est quand même plus facile d’utiliser des mots anglais (car oui nous allons être obligé de faire un peu d’anglais pour programmer, mais rien de bien compliqué rassure-vous) que des suites de 0 et de 1. Vous ne croyez pas ?

\n

Envoyer le programme dans la carte

\n

Là, je ne vais pas vous dire grand chose car c’est l’environnement de développement qui va gérer tout ça. Nous n’aurons qu’à apprendre comment utiliser ce dernier et il se débrouillera tout seul pour envoyer le programme dans la carte. Nah ! Nous n’aurons donc qu’à créer le programme sans nous soucier du reste.

\n
\n
\n
\n

Pourquoi choisir Arduino ?

\n
\n

Que va-t-on faire avec ?

\n

Avec Arduino, nous allons commencer par apprendre à programmer puis à utiliser des composants électroniques. Au final, nous saurons créer des systèmes électroniques plus ou moins complexes. Mais ce n’est pas tout…

\n

D’abord, Arduino c’est…

\n

… une carte électronique programmable et un logiciel gratuit :

\n
\"\"
\n

Mais aussi

\n

– Un prix dérisoire étant donné l’étendue des applications possibles. On comptera 20 euros pour la carte que l’on va utiliser dans le cours. Le logiciel est fournit gratuitement ! – Une compatibilité sous toutes les plateformes, à savoir : Windows, Linux et Mac OS. – Une communauté ultra développée ! Des milliers de forums d’entre-aide, de présentations de projets, de propositions de programmes et de bibliothèques, … – Un site en anglais arduino.cc et un autre en français arduino.cc où vous trouverez tout de la référence Arduino, le matériel, des exemples d’utilisations, de l’aide pour débuter, des explications sur le logiciel et le matériel, etc. – Une liberté quasi absolue. Elle constitue en elle même deux choses :

\n
  • Le logiciel : gratuit et open source, développé en Java, dont la simplicité d’utilisation relève du savoir cliquer sur la souris
  • \n
  • Le matériel : cartes électroniques dont les schémas sont en libre circulation sur internet
  • \n

Cette liberté a une condition : le nom « Arduino » ne doit être employé que pour les cartes « officielles ». En somme, vous ne pouvez pas fabriquer votre propre carte sur le modèle Arduino et lui assigner le nom « Arduino ».

\n

Et enfin, les applications possibles

\n

Voici une liste non exhaustive des applications possible réalisées grâce à Arduino :

\n
  • contrôler des appareils domestiques
  • \n
  • donner une « intelligence » à un robot
  • \n
  • réaliser des jeux de lumières
  • \n
  • permettre à un ordinateur de communiquer avec une carte électronique et différents capteurs
  • \n
  • télécommander un appareil mobile (modélisme)
  • \n
  • etc.
  • \n

Il y a tellement d’autres infinités d’utilisations, vous pouvez simplement chercher sur votre moteur de recherche préféré ou sur Youtube le mot « Arduino » pour découvrir les milliers de projets réalisés avec !

\n

Arduino dans ce tutoriel

\n

Je vais quand même rappeler les principaux objectifs de ce cours. Nous allons avant tout découvrir Arduino dans son ensemble et apprendre à l’utiliser. Dans un premier temps, il s’agira de vous présenter ce qu’est Arduino, comment cela fonctionne globalement, pour ensuite entrer un peu plus dans le détail. Nous allons alors apprendre à utiliser le langage Arduino pour pouvoir créer des programmes très simple pour débuter. Nous enchainerons ensuite avec les différentes fonctionnalités de la carte et ferons de petits TP qui vous permettront d’assimiler chaque notion abordée. Dès lors que vous serez plutôt à l’aise avec toutes les bases, nous nous rapprocherons de l’utilisation de composants électroniques plus ou moins complexes et finirons par un plus « gros » TP alliant la programmation et l’électronique. De quoi vous mettre de l’eau à la bouche ! 😛

\n

Arduino à l’école ?

\n

Pédagogiquement, Arduino a aussi pas mal d’atout. En effet, ses créateurs ont d’abord pensé ce projet pour qu’il soit facile d’accès. Il permet ainsi une très bonne approche de nombreux domaines et ainsi d’apprendre plein de choses assez simplement.

\n

Des exemples

\n

Voici quelques exemples d’utilisation possible :

\n
  • Simuler le fonctionnement des portes logiques
  • \n
  • Permettre l’utilisation de différents capteurs
  • \n
  • Mettre en œuvre et faciliter la compréhension d’un réseau informatique
  • \n
  • Se servir d’Arduino pour créer des maquettes animées montrant le fonctionnement des collisions entres les plaques de la croute terrestre, par exemple \":mrgreen:\"
  • \n
  • Donner un exemple concret d’utilisation des matrices avec un clavier alphanumérique 16 touches ou plus
  • \n
  • Être la base pour des élèves ayant un TPE à faire pour le BAC
  • \n
  • \n

De plus, énormément de ressources et tutoriels (mais souvent en anglais) se trouvent sur internet, ce qui offre un autonomie particulière à l’apprenant.

\n

Des outils existant !

\n

Enfin, pour terminer de vous convaincre d’utiliser Arduino pour découvrir le monde merveilleux de l’embarqué, il existe différents outils qui puissent être utilisé avec Arduino. Je vais en citer deux qui me semble être les principaux : Ardublock est un outil qui se greffe au logiciel Arduino et qui permet de programmer avec des blocs. Chaque bloc est une instruction. On peut aisément faire des programmes avec cet outil et mêmes des plutôt complexes. Cela permet par exemple de se concentrer sur ce que l’on doit faire avec Arduino et non se concentrer sur Arduino pour ensuite ce que l’on doit comprendre avec. Citons entre autre la simulation de porte logique : plutôt créer des programmes rapidement sans connaitre le langage pour comprendre plus facilement comment fonctionne une porte logique. Et ce n’est qu’un exemple. Car cela permet aussi de permettre à de jeunes enfants de commencer à programmer sans de trop grandes complications.

\n
\"\"\"\"\"\"
Exemple de programmes réalisés avec Ardublock
\n

Processing est une autre plateforme en lien avec Arduino. Là il n’y a pas de matériel, uniquement un logiciel. Il permet entre autre de créer des interfaces graphiques avec un langage de programmation très similaire à celui d’Arduino. Par contre, cela demande un niveau un peu plus élevé pour pouvoir l’utiliser, même si cela reste simple dans l’ensemble.

\n
\"\"
Voilà un exemple de ce que j’avais réalisé avec Processing pour faire communiquer mon ordinateur avec ma carte Arduino
\n

J’espère avoir été assez convaincant afin que vous franchissiez le pas et ayez du plaisir à apprendre ! \":)\"

\n
\n
\n
\n

Les cartes Arduino

\n
Le matériel que j’ai choisi d’utiliser tout au long de ce cours n’a pas un prix excessif et, je l’ai dit, tourne aux alentours de 25 € TTC. Il existe plusieurs magasins en lignes et en boutiques qui vendent des cartes Arduino. Je vais vous en donner quelques-uns, mais avant, il va falloir différencier certaines choses.\n

Les fabricants

\n

Le projet Arduino est libre et les schémas des cartes circulent librement sur internet. D’où la mise en garde que je vais faire : il se peut qu’un illustre inconnu fabrique lui même ses cartes Arduino. Cela n’a rien de mal en soi, s’il veut les commercialiser, il peut. Mais s’il est malhonnête, il peut vous vendre un produit défectueux. Bien sûr, tout le monde ne cherchera pas à vous arnaquer. Mais la prudence est de rigueur. Faites donc attention où vous achetez vos cartes.

\n

Les types de cartes

\n
\"\"
\n

Il y a trois types de cartes :

\n
  • Lesdites « officielles » qui sont fabriquées en Italie par le fabricant officiel : Smart Projects
  • \n
  • Lesdits « compatibles » qui ne sont pas fabriqués par Smart Projects, mais qui sont totalement compatibles avec les Arduino officielles.
  • \n
  • Les « autres » fabriquées par diverse entreprise et commercialisées sous un nom différent (Freeduino, Seeduino, Femtoduino, …).
  • \n

Les différentes cartes

\n

Des cartes Arduino il en existe beaucoup ! Voyons celles qui nous intéressent… La carte Uno et Duemilanove Nous choisirons d’utiliser la carte portant le nom de « Uno » ou « Duemilanove ». Ces deux versions sont presque identiques.

\n
\"\"\"\"
Carte Arduino « Duemilavove » et « Uno » avec laquelle nous allons travailler
\n

La carte Mega La carte Arduino Mega est une autre carte qui offre toutes les fonctionnalités de la carte précédente, mais avec des fonctionnalités supplémentaires. On retrouve notamment un nombre d’entrées et de sorties plus important ainsi que plusieurs liaisons séries. Bien sûr, le prix est plus élevé : > 40 € !

\n
\"\"
Carte Arduino « Mega »
\n

Les autres cartes Il existe encore beaucoup d’autres cartes, je vous laisse vous débrouiller pour trouver celle qui conviendra à vos projets. Cela dit, je vous conseil dans un premier temps d’utiliser la carte Arduino Uno ou Duemilanove d’une part car elle vous sera largement suffisante pour débuter et d’autre part car c’est avec celle-ci que nous présentons le cours.

\n

Où acheter ?

\n

Il existe sur le net une multitude de magasins qui proposent des cartes Arduino. Pour consulter la liste de ces magasins, rien de plus simple, il suffit de vous rendre sur le forum dédié :

\n\n

J’ai vu des cartes officielles « édition SMD/CMS ». Ca à l’air bien aussi, c’est quoi la différence ? Je peux m’en servir ?

\n

Il n’y a pas de différence ! enfin presque… « SMD » signifie Surface Mount Device, en français on appelle ça des « CMS » pour Composants Montés en Surface. Ces composants sont soudés directement sur le cuivre de la carte, il ne la traverse pas comme les autres. Pour les cartes Arduino, on retrouve le composant principal en édition SMD dans ces cartes. La carte est donc la même, aucune différence pour le tuto. Les composants sont les mêmes, seule l’allure « physique » est différente. Par exemple, ci-dessus la « Mega » est en SMD et la Uno est « classique ».

\n
\n
\n
\n

Liste d’achat

\n
Tout au long du cours, nous allons utiliser du matériel en supplément de la carte. Rassurez-vous le prix est bien moindre. Je vous donne cette liste, cela vous évitera d’acheter en plusieurs fois. Vous allez devoir me croire sur parole sur leur intérêt. Nous découvrirons comment chaque composant fonctionne et comment les utiliser tout au long du tutoriel. \":)\"

Attention, cette liste ne contient que les composants en quantités minimales strictes. Libre à vous de prendre plus de LED et de résistances par exemple (au cas où vous en perdriez ou détruisiez…). Pour ce qui est des prix, j’ai regardé sur différents sites grands publics (donc pas Farnell par exemple), ils peuvent donc paraître plus élevé que la normale dans la mesure où ces sites amortissent moins sur des ventes à des clients fidèles qui prennent tout en grande quantité…

\n

Avant que j’oublie, quatres éléments n’apparaitront pas dans la liste et sont indispensables :

\n\n\n\n\n
Une Arduino Uno ou DuemilanoveUn câble USB A mâle/B mâle
\"\"\"\"
\n\n\n\n
Une BreadBoard (plaque d’essai)Un lot de fils pour brancher le tout !
\"\"\"\"

Liste Globale

\n

Voici donc la liste du matériel nécessaire pour suivre le cours. Libre à vous de tout acheter ou non.

\n
Liste incomplète, le tutoriel n’est pas terminé ! Mais elle suffit pour suivre les chapitres en ligne.
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
DésignationQuantitéPhotoDescription
LED rouge7\"\"Ce composant est une sorte de lampe un peu spécial. Nous nous en servirons principalement pour faire de la signalisation.
LED verte3
LED jaune (ou orange)2
Résistance (entre 220 et 470 Ohm)10\"\"La résistance est un composant de base qui s’oppose au passage du courant. On s’en sert pour limiter des courants maximums mais aussi pour d’autres choses.
Résistance (entre 2.2 et 4.7 kOhm)2
Résistance (10 kOhm)2
Bouton Poussoir2\"\"Un bouton poussoir sert à faire passer le courant lorsqu’on appuie dessus ou au contraire garder le circuit « éteint » lorsqu’il est relâché.
Transistor (2N2222 ou BC547)2\"\"Le transistor sert à plein de chose. Il peut être utilisé pour faire de l’amplification (de courant ou de tension) mais aussi comme un interrupteur commandé électriquement.
Afficheur 7 segments (anode commune)2\"\"Un afficheur 7 segments est un ensemble de LEDs (cf. ci-dessus) disposées géométriquement pour afficher des chiffres.
Décodeur BCD (CD4543BE)1\"\"Le décodeur BCD (Binaire Codé Décimal) permet piloter des afficheurs 7 segments en limitant le nombre de fils de données (4 au lieu de 7).
Condensateur (10nF/5V ou plus)2\"\"Le condensateur est un composant de base. Il sert à plein de chose. On peut se le représenter comme un petit réservoir à électricité.
Condensateur (1000µF 15V ou plus1\"\"Celui-ci est un plus gros réservoir que le précédent
Potentiomètre linéaire (10 kOhm)1\"\"Le potentiomètre est une résistance que l’on peut faire varier manuellement.
LED RVB1\"\"Une LED RVB (Rouge Vert Bleu) est une LED permettant de mélanger les couleurs de bases pour en créer d’autres.
Écran LCD alphanumérique1\"\"L’écran LCD alphanumérique permet d’afficher des caractères tels que les chiffres et les lettres. Il va apporter de l’interactivité à vos projets les plus fous !

Les revendeurs

\n

Je vous ai déjà donné le lien, vous pourrez trouver ces composants chez les revendeurs listés dans ce sujet du forum :

\n\n

Les kits

\n

Enfin, il existe des kits tout prêts chez certains revendeurs. Nous n’en conseillerons aucun pour plusieurs raisons. Tout d’abord, pour ne pas faire trop de publicité et rester conforme avec la charte du site. Ensuite, car il est difficile de trouver un kit « complet ». Ils ont tous des avantages et des inconvénients mais aucun (au moment de la publication de ces lignes) ne propose absolument tous les composants que nous allons utiliser. Nous ne voulons donc pas que vous reveniez vous plaindre sur les forums car nous vous aurions fait dépenser votre argent inutilement !

\n
Cela étant dit, merci de ne pas nous spammer de MP pour que l’on donne notre avis sur tel ou tel kit ! Usez des forums pour cela, il y a toujours quelqu’un qui sera là pour vous aider. Et puis nous n’avons pas les moyens de tous les acheter et tester leur qualité !
\n
\n
\n
\n
\n
\n\n\n", + "annotations": [], + "mimetype": "text/html", + "language": "fr-FR", + "reading_time": 17, + "domain_name": "eskimon.fr", + "preview_picture": "http://eskimon.fr/wp-content/uploads/tuto/385800.jpg", + "tags": [] + }, + { + "is_archived": 1, + "is_starred": 0, + "id": 543, + "title": "Lenovo ThinkPad X1 Carbon Ultrabook Review", + "url": "http://www.notebookcheck.net/Lenovo-ThinkPad-X1-Carbon-Ultrabook-Review.138033.0.html", + "content": "

The Lenovo X1 Carbon 3rd Gen is a beautiful machine. Much like the Dell XPS 13 took the initiative to cram a 13-inch screen into an 11-inch form factor, the X1 Carbon sports dimensions that are more comparable to a typical 13-inch machine—and that includes its weight and thinness, both of which are indisputably manageable. It’s also practically designed; the matte black surfaces that comprise the majority of the case are minimalist and attractive, but they’re simultaneously haptically comfortable, with an unmistakably cool metal feel and a comfortable fit for use on both lap and desk. The case also feels fairly solid, though the incidence of flex and relative lack of torsion resistance in some regards gave us pause.

\n

Perhaps more exciting, however, is what has improved over the X1 Carbon 2nd Gen. Criticism of the 2nd Gen’s radical (and arguably illogical) keyboard design and polarizing full-depression clickpad scared away many prospective buyers—as such fearlessly progressive and experimental design decisions generally don’t fit well with the business market, where practicality rules supreme. The Gen 3 wholeheartedly acknowledges these complaints and implements a complete reversal of those decisions. As a result, the keyboard—immediately familiar and accessible—is one of the absolute best we have ever used on an Ultrabook. Meanwhile, the three classic top-mounted physical buttons for use with the Trackpoint have returned, and the touchpad itself ditches the controversial full-click design in favor of a far more comfortable (and, in our judgment, reliable) clickpad approach. The end result is that the X1 Carbon Gen 3 features some of the best input devices we’ve tested on an Ultrabook.

\n

What about performance? CPU performance differences between the 2nd Gen and 3rd Gen X1 Carbon models were essentially nil in our testing—for all intents and purposes, the machines are identical in this regard. However in GPU testing, we witnessed a notable speed boost—in some cases up to 19% better. As compared with other modern notebooks of its class, apart from some multi-core synthetic performance hiccups, the X1 Carbon 3rd Gen holds its own, both in terms of CPU/GPU and general system performance. The only final niggle here is the Samsung PM851 SSD, whose write speeds are conspicuously capped at around 250 MB/s.

\n

While the leap to a Broadwell chipset and slightly larger battery seemed sure to promise improved battery runtimes, we were surprised to find throughout our testing that there wasn’t much of a difference at all. Our classic Wi-Fi Surfing Test produced a result only slightly better than that of the 2nd Gen, and the revised Wi-Fi test we just recently implemented—which is more broad and aggressive and arguably closer to actual real-world usage patterns—recorded under five hours before the machine shut down. That’s hardly an impressive number on one hand given the 50 Wh battery and supposed enhanced efficiency, though it’s still likely to get most users through a typical trip unplugged, especially if more restrictive power savings options are employed. If longer battery life is a priority, we’d suggest taking a look instead at the Dell XPS 13-9343 or the MacBook Air 13.

\n

Rounding out the list of considerations is an underwhelming screen, at least in terms of brightness, contrast, and color saturation—though we do most certainly appreciate the anti-glare display filter for both its diffusion of reflections and relative ease of cleaning. The X1 Carbon Gen 3 is also invariably cool and quiet, clearly favoring comfort over top-end performance (as we discovered during our stress testing of the device).

\n

Summing up, the X1 Carbon Gen 3 is indisputably superior to its predecessor. Although some of these improvements come in the form of better GPU performance, cooler temperatures, and lower average system noise levels, the vast majority of them center on the thankful retreat from the experimental (and finicky) input devices of the Gen 2 design and back to sanity. This isn’t just a return to form, either; to reiterate, by our judgment, they are some of the best input devices on any Ultrabook we’ve tested to date. But in spite of this self-improvement, how does the X1 Carbon compare with its modern competitors? In truth, though it’s a compelling option, it’s lost some ground since our last encounter. While notebooks such as the MacBook Air 13 and (especially) the Dell XPS 13-9343 have sprinted forward with such massive improvements in portability, battery life, and LCD quality, the X1 Carbon 3rd Gen has only marginally improved, mostly regaining footing it’s lost elsewhere. It’s still a strong contender, and it’s certainly the best Carbon to date, but especially at a pricey $1,574, we fear that its inability to innovate further may relegate it to the shadows of these more aggressive contenders.

\n", + "annotations": [], + "mimetype": "text/html", + "language": "en", + "reading_time": 3, + "domain_name": "www.notebookcheck.net", + "preview_picture": "http://www.notebookcheck.net/fileadmin/Notebooks/Lenovo/ThinkPad_X1_Carbon_2015/x12015.jpg", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 0, + "id": 541, + "title": "Visitons le Château de Landsberg !", + "url": "http://autour-du-mont-sainte-odile.overblog.com/2016/01/visitons-le-chateau-de-landsberg.html", + "content": "

Situé au dessus de Heiligenstein et de Barr, à mi-hauteur de la montée vers plateau de la Bloss, le château de Landsberg est un but de promenade apprécié des randonneurs et des amateurs de vieilles pierres. Posé sur un éperon de granite, visible de loin, la forteresse médiévale domine fièrement la plaine d’Alsace.

\n

\"Visitons

\n

Le château primitif

\n

Fin du onzième siècle, Philippe de Souabe et Otton de Brunswick se disputent le trône impérial. Philippe, tenant des Hohenstaufen, veut conforter la puissance des couvents du Mont Sainte Odile. Il s’appuie sur un de ses représentants pour construire une nouvelle forteresse sur le Mont. Conrad de Vinhege érige le château de Landsberg dont sa famille prendra bientôt le nom. Nous sommes en 1197-1198. Le burg est construit sur les terres de l’abbaye de Niedermunster, le Couvent du Bas du Mont Sainte Odile. Un acte est signé de l’Abbesse Edelinde, il est daté du 23 juin 1200, il confirme la cession du terrain au chevalier Conrad.
Nous avons publié le texte de l’abbesse Edelinde, l’an passé. (cliquez sur le lien)
Ce premier château n’avait pas l’importance des ruines que nous pouvons admirer aujourd’hui. Il ne comportait qu’un donjon et un corps de logis, complété par une basse-cour. Commençons la visite des ruines par cette partie. Le donjon est construit sur un carré de dix mètres de côté. Puissant, placé en diagonale par rapport au corps de logis, il formait bouclier pour défendre celui-ci du côté de la plaine, au nord-est. En levant la tête, le visiteur découvre l’accès : une porte haute, qui était accessible par un pont volant à partir de l’étage du logis seigneurial. Tout en haut du donjon, on distingue encore les corbeaux qui portaient les hourds. Les hourds, constructions de bois, ancêtres des mâchicoulis, permettaient aux défenseurs de surplomber les assaillants du château.
\"VisitonsLe logis seigneurial comportait un rez-de-chaussée dédié à l’usage domestique : cuisines, citerne, salle des gardes. Ces pièces basses n’étaient éclairées que par quelques meurtrières. Le premier étage était l’habitation de Conrad et de ses successeurs : la façade nord-est est largement éclairée par quatre baies géminées en plein-cintre. Les fenêtres doubles sont séparées par de fines colonnes et surmontées d’élégants occuli. Au sud-est, deux doubles baies et un magnifique oriel agrémentent la façade. Situé sur le coté de la porte du château, cet élément de décor est le plus frappant du site du Landsberg, un genre d’échauguette semi-cylindrique, portée par un cul de lampe conique. Nous serions là dans la chapelle castrale du Landsberg. Coté extérieur, deux petites sculptures ornent la base de l’oriel : une fleur de lys et un petit personnage accroupi. A l’intérieur, une frise d’arceaux délimite les petites fenêtres du chœur de la chapelle. L’une d’elles a la forme d’une croix. Admirez la finesse des dessins du chapiteau de la colonne toujours en place au droite de l’oriel !
Le château de Landsberg est situé à la limite géologique des grès, prédominants au nord, et des granites d’Andlau, côté sud. Les bases et les assises de la forteresse sont en granite, roche dure extraite du fossé creusé au nord, par les carriers du moyen âge. Les parties hautes sont en grès rose du Maennelstein, plus facile à travailler.

\n

Le château neuf

\n

Ce premier château était, somme toute, de dimensions modestes. Quelques dizaines d’ années plus tard, preuve de l’importance des couvents du Mont Sainte Odile et signe de la puissance croissante de la famille de Landsberg, le site fut agrandi de façon conséquente. Nous sommes alors en 1240-1250.
L’extension se fait au nord-ouest. Les deux tours rondes et l’imposante courtine délimitent l’emprise du Château Neuf. A l’origine, cette nouvelle enceinte était détachée du vieux burg, les deux sites ne furent rattachés que postérieurement. Les tours circulaires mesurent environ sept mètres de diamètre. Leur sommet portaient des créneaux. Un chemin de ronde surmontait la courtine : il était garni de hourds et traversait les deux tours. Archères et bretèches sont toutes orientées au nord, côté montagne, là où le risque d’attaque était le plus grand.
Deux corps de logis se partageaient le site. Au sud, le bâtiment ne comportait qu’un seul étage, éclairé au sud-ouest par des fenêtres à banquettes, aujourd’hui murées. Le deuxième bâtiment, côté nord, fut construit plus tard. Composé de deux étages, il communiquait directement avec la tour nord.
L’ensemble a connu plusieurs remaniements au cours des siècles.

\n

\"Visitons

\n

Les extensions du XVème et du XVIème siècles

\n

L’apparition des armes à feu modifie l’art de la guerre. La géométrie des châteaux doit s’adapter aux armes nouvelles : résister aux boulets et accueillir couleuvrines et canons. Tous les châteaux des Vosges n’ont pas connu cette évolution, certains, trop solitaires, trop isolés, ont été abandonnés dés cette époque. Au Landsberg, les seigneurs ont respecté le vieux burg dans sa forme initiale, mais ils l’ont adapté tout d’abord en renforçant ses murs, puis en l’entourant de nouvelles murailles. Au nord, le château neuf se voit délaissé, ses nombreuses ouvertures extérieures sont murées, à part les archères, bien entendu. Ses courtines sont prolongées pour assurer la continuité avec celles du vieux burg. Un puissant bastion est construit au sud-est. Les armes nouvelles sont mises en place : couleuvrines, poivrière portant une canonnière protégée par des vastes vantaux. Ainsi, un vaste glacis s’étend devant la forteresse. C’est en faisant le tour du château dans les fossés que le promeneur se rendra le mieux compte de la force de la place et de son adaptation aux débuts de l’artillerie.
\"VisitonsQuelques temps, le château est passé des Landsberg au Comte Palatin, avant de revenir à la famille qui resta maître des lieux jusqu’à la Révolution.
On ne trouve guère de textes relatant l’histoire du château de Landsberg dans les temps troublés que connut l’Alsace. Lors de la Guerre du Bundschuh, les paysans révoltés s’étaient installés dans la prévôté de Truttenhausen, toute proche. Cependant, rien ne dit que les Rustauds aient attaqué le château. Lors de la Guerre de Trente Ans, il semble que les troupes de Mansfeld, puis les Suédois se soient plutôt attaqués aux riches villes de la plaine, comme Obernai, oubliant les forteresses de montagne, déjà devenues inutiles. Cependant, le château est décrit comme ruiné au milieu du XVIIème siècle.
Depuis longtemps, les Landsberg avaient délaissé le burg de leurs ancêtres pour lui préférer leur résidence de Niedernai, située en plaine. Le château fut confié à des gardes, puis servit de ferme. Aujourd’hui, les ruines du Landsberg sont une propriété privée. Merci à vous de respecter le site.

\n

Dans la basse cour du château fleurit au printemps, l’éranthe d’hiver. Sa floraison ne dure que quelques jours. Cette petite fleur jaune est rare sous nos climats. De son nom latin ‘Eranthis hyemalis’, l’éranthe est originaire d’Italie ou de Turquie. La légende nous dit qu’elle aurait été rapportée des croisades par un sire de Landsberg. Elle serait un remède à la mélancolie.

\n

Promenade et accès au château de Landsberg\"Visitons

\n

Les automobilistes pourront garer leur véhicule sur le petit parking situé à l’ouest d’Heiligenstein. Ils gagneront Truttenhausen sur la petite route à travers prés. Un sentier du Club Vosgien ( disque bleu ) monte vers les ruines.
Les marcheurs préfèreront effectuer une boucle à partir de Saint-Jacques, par exemple. Saint-Jacques, Kapellenhausfelsen, Ameisenberg, Landsberg ( balisage : triangle bleu ) avec un retour sur Saint-Jacques par le sentier sans dénivelé ( rectangle : bicolore rouge et blanc).
A moins qu’ils ne préfèrent rechercher les pierres sculptées du Chemin des Chameaux. (cliquez sur le lien ), ou découvrir la Chapelle du Frère Léon dans la vallée de la Kirneck.

\n

Illustrations

\n
  • Photographies du château de Landsberg ( BrR, FrP et PiP)
  • \n
  • Schéma des ruines ( PiP)
  • \n
  • L’Orgueil, Herrade de landsberg, Hortus Deliciarum
  • \n
  • Les Ruines du Landsberg, aquarelle de Osterwald, 1873
  • \n
\n", + "annotations": [], + "mimetype": "text/html", + "language": "fr", + "reading_time": 7, + "domain_name": "autour-du-mont-sainte-odile.overblog.com", + "preview_picture": "http://img.over-blog-kiwi.com/0/28/39/78/20160128/ob_68f59d_12.jpg", + "tags": [] + }, + { + "is_archived": 1, + "is_starred": 0, + "id": 454, + "title": "Contrer les stéréotypes par les livres : “C'est dès l'enfance qu'ils se construisent”", + "url": "https://www.actualitte.com/article/monde-edition/contrer-les-stereotypes-par-les-livres-c-est-des-l-enfance-qu-ils-se-construisent/64058", + "content": "

Même dans l'espace jeunesse d'un immense salon littéraire, difficile de passer à côté du livre de coloriage féministe — et fier de l'être — de la maison d'édition Goater. Cette structure atypique, adossée à un bar de Rennes, Le Papier-Timbré, propose des titres tout aussi uniques, de ces livres que l'on devine importants avant même de les ouvrir.

\n\n

\"Livre

\n

Judikael et Jean-Marie Goater (ActuaLitté, CC BY SA 2.0)

\n\n\n

Le Papier-Timbré et les éditions Goater avancent main dans la main, dans une même structure, depuis 2009. En plus de la licence IV, permis de publier : « On est très occupé par les soirées étudiantes et festives, mais on développe en plus des projets autour des livres qui, parfois, émergent d'ailleurs avec la clientèle. C'est surtout un motif supplémentaire pour se retrouver et partager des moments de convivialité, partager des goûts, des envies, de la littérature, des essais et de la jeunesse », explique Jean-Marie Goater.

\n

\n

En plus de la production maison, le café-librairie propose celle de maisons soeurs : les éditions de juillet, L'Oeuf, les Éditions Pontcerq, essentiellement des petits éditeurs de Rennes et de la Bretagne. La maison est diffusée et distribuée en Bretagne par Coop Breizh, diffusé par Hobo Diffusion, distribué par Makassar pour la France (comme les éditions surréalistes Prairial). Les tirages vont de 500 exemplaires à 3000 sauf exceptions et coéditions comme Détachez vos ceintures, projet collectif des éditions du Kyste contre l'aéroport de Notre-Dame-des-Landes, ou Galette-saucisse, je t’aime ! de Benjamin Keltz, avec les Éditions du coin de la rue.

\n

\n

Mon premier cahier de coloriage féministe ! sera en librairie d'ici quelques jours, et la maison Goater y croit dur comme fer : « Nous avons commencé par traduire et adapter C'est quoi ton genre ?, un livre écrit par Jacinta Bunnell et publié par l'éditeur anarchiste américain PM Press », explique Jean-Marie Goater. Dans les pages du livre, on croise des monstres qui aiment les petits sacs à main et les chaussures, des princesses qui ne suivent pas vraiment le dress code, ou des enfants en fauteuil roulant, encore rares dans les livres jeunesse.

\n

\n

\"Livre

\n

(ActuaLitté, CC BY SA 2.0)

\n

\n

\n

Après cette publication, la maison a souhaité développer un projet en France, en réunissant 16 dessinatrices dont 4 dessinateurs pour leur proposer d'expliquer le féminisme aux enfants, à travers un dessin. « Le livre aborde la vie à l'école, les habillements, les métiers, le sport, mais aussi les quelques femmes féministes importantes de l'histoire... Ça reste ludique et sans prétention encyclopédique sur le féminisme, mais il est plus simple d'aborder le sujet avec un support comme celui-ci à la maison, à l'école ou au centre de loisirs. » Comme le précédent, l'ouvrage présente d'une nouvelle manière les situations traditionnelles des livres de coloriage ou jeunesse.

\n

\n

Des ressources rares, des besoins importants

\n

\n

Pas la peine d'insister pour que Jean-Marie Goater partage son avis : « La production majoritaire est quand même très caricaturale et stéréotypée, cependant on remarque depuis quelques années des éditeurs intéressants qui essaient de bousculer un peu ces stéréotypes comme La Ville Brûle, ou encore l'édition LGBT qui commence à arriver avec Des ailes sur un tracteur qui a publié un cahier de coloriage avec Sophie Labelle, plutôt sur les questions trans. »

\n

\n

Contrairement à ce que les détracteurs des livres jeunesse atypiques prétendent (coucou, Jean-François Copé), lutter contre les stéréotypes n'a rien d'une guerre de civilisation ou autre affabulation du genre. Il s'agit simplement de montrer que chacun doit être fier de ce qu'il est, respecter ce que l'autre est, et ne pas chercher l'assentiment des uns ou des autres. « Fuck the world », comme dirait 2Pac... « Se poser ces questions est indispensable, il faut qu'il y ait ce débat : les enfants ne sont pas si naïfs que ça, ils ont besoin de se poser ce genre de questions. »

\n

\n

\"Livre

\n

(ActuaLitté, CC BY SA 2.0)

\n

\n

\n

Judikael, venu aider son père, acquiesce : il a répondu à l'appel à dessins et proposé une activité dans le cahier de coloriage féministe. « Ce genre de ressources pour enfant est important, parce que c'est dès l'enfance que se construisent certains préjugés, certains stéréotypes qui restent ensuite. Quand on voit que 90 % des personnes présentes dans les manuels scolaires sont des hommes par exemple, ce genre d'ouvrages permet à certaines personnes de se reconnaître davantage dans certains rôles, qu'on ne leur attribue pas forcément de base. »

\n

\n

Là est la lutte, résumée par Judikael : « On parle souvent de “déconstruire” dans le féminisme, les préjugés ou autre, ces livres peuvent permettre d'éviter de les construire. » En Terminale L, Judikael confirme que les préjugés sont toujours présents, forcément surtout en sport ou vis-à-vis de « la filière homme » (comprendre, la filière scientifique) et de « la filière femme » (comprendre, la filière littéraire). Si l'histoire du féminisme est désormais abordée en classe, certains sujets restent touchy : le journal du lycée s'est vu censurer un article sur la culture du viol, et la ségrégation hommes-femmes, « au prétexte que c'était trop hard, que les lycéens n'allaient pas comprendre »...

\n

\n

« L'édition, c'est un milieu qui est quand même très hypocrite »

\n

\n

À votre avis, comment réagit l'éditeur de Mon premier cahier de coloriage féministe lorsqu'on lui parle des différences entre les salaires des hommes et des femmes dans l'édition, ou dans les aides attribuées par le CNL ? Sans langue de bois : « C'est pas trop surprenant malheureusement, parce que c'est à l'image des autres professions. En tant que bar-maison d'édition, de toute façon, je ne rentre pas dans la case du CNL, je me tourne plutôt vers la région », explique Jean-Marie Goater.

\n

\n

Haut les coeurs : « C'est pas grave, je m'en passe très bien. L'édition, c'est un milieu qui a quand même une grande dimension d'hypocrisie sur pas mal d'aspects, on le voit sur certaines pratiques... Il y a des cons dans ce métier-là comme dans d'autres métiers, mais je pense que ce serait bien de faire le ménage, comme à Angoulême, c'est quand même criant. Dans certains secteurs du livre, la majorité des lecteurs sont des lectrices, très clairement, ce serait quand même la moindre des choses qu'il n'y ait pas des inégalités de ce type qui existent dans le monde de l'édition. »

\n

\n

\"Livre

\n

Un album jeunesse bilingue français-langue des signes (ActuaLitté, CC BY SA 2.0)

\n

\n

\n

Même sans aide du CNL, les éditions Goater produisent de quoi lire : Les Joyeux Punks, un album à compter, mais aussi une collection d'albums bilingues français-langue des signes, des livres en breton, dont une traduction du Persepolis de Marjane Satrapi. Pour les amateurs de polar, Goater noir, une collection de 14 titres qui a notamment fait revivre Le Soviet, la série culte des années 89-90 d'abord publiée chez Fleuve Noir et Série Noire. Pour les amateurs des écrits du Colonel Durruti, un inédit est prévu pour le mois d'octobre prochain.

\n

\n

Perdez-vous sans hésiter dans le catalogue de la maison, qui propose aussi de la littérature blanche \"classique\", des essais sur l'écologie et le convivialisme ou la convivialité, l'écologie sociale, et l'Histoire, surtout XXe.

\n\n
", + "annotations": [], + "mimetype": "text/html", + "language": "en", + "reading_time": 6, + "domain_name": "www.actualitte.com", + "preview_picture": "//cdn.actualitte.com/images/facebook/25284788384-ea234db7b9-z-56eeff88c23a0.jpg", + "tags": [] + }, + { + "is_archived": 1, + "is_starred": 0, + "id": 99, + "title": "[ROM][6.0.1][Layers][N5] TipsyOS official builds {UBER TCs}", + "url": "http://forum.xda-developers.com/google-nexus-5/development/rom-tipsyos-official-builds-uber-tcs-t3325989", + "content": "
\n
\"\"

*Welcome to Tipsy OS*

\n
\"\"
\n
This is something Martin Coulon (@martinusbe) started in his 'free' time when waiting for GZR Validus or Tesla builds and having some drinks.
He just started with a slim base; on lp and early mm, and now using AOSPB as a base (wich Martin is also a part of)....then added what we thought useful.
It grew up to be a fully functional Rom
The main goal is to keep it AOSPB/Slim based and will not add any cm features unless AOSPB/Slim does.
\n

TipsyOS is Black and Yellow default themed, we don't like white UI, so it may look a bit weird on light switch,
but heh! use layers to suit it your needs =)

\n

We are a team wich is constantly learning and ...drinking because both gets along so well hahaha \"\"

\n

Just report the bugs and request features, we'll see what we can do!.......

\n

And don't get it twisted, its not because we're Tipsy that this project is not a serious one, try it and feel speed and stability \"\"

\n

My builds are compiled with UBER toolchains 4.9 on both rom and kernel code.
We can go over with 5.2/3, 6 ...but to be really honest, 4.9 is the most battery friendly in my opinion, and is still super smooth and snappy!

\n

Im using a cm device tree base a bit modified...
This rom includes a custom kernel, cm was my base, and i have decided to go a bit wild and cherry-picked up some stuff from:
Francisco Franco, Chet Kener, Faux123, Flar2, Hellsgod and some others
So a huge thanks for them and their AWESOME open sources work!
I will continue to work on this kernel because its fun \"\"

\n
\"\"

For cool wallpapers, test builds, requests; join us on our community!

\n
\nGoogle+ Community\n
Features:\n
\n

Code:

\n
\nWe slowly adding only what we need, we don't want a 1236544547 features rom...\n\"The Tavern\":\nPower menu customisations\nToasts/ListView/System animations\nBattery bar\nLCD Density\nGesture Anywhere\nExpanded Desktop\nStatus bar customisations\nAosp Recents and OmniSwitch \nBuilt in:\nDashboard (settings) columns selector\nDashboard lines remover\nSlim Navbar customisation\nVolume steps\nNotification led changer\nHeadsUp switch\nKernel Adiutor app\nLayers manager app\nLayers backup/restore app\nAdaway\nNova launcher\nViper4Android\nES Manager\nSnapCam\nChangelog generator in about phone menu....\nand prolly some other stuff that i can't remember but heh,\n just flash dat sh#t to figure out by yourself =)\n
\n
\n


Installation Instructions

\n

1. Make sure you have a working recovery and a nandroid backup

\n

2. Download the ROM.

\n

3. Reboot to recovery.

\n

4.Wipe everything! system/data/cache partitions (except internal storage indeed!)

\n

7. Flash the ROM.

\n

Optional- Flash 6.x GApps.

\n

8. Reboot and feel Tipsy!

\n

Download:

\n

Tipsy HammerHead downloads folder

\n

STAY TIPSY \"\"\"\"
\"\"

\n

TipsyOs, a ROM for Nexus 5 aka Hammerhead

\n
\n
Kernel features:
Kexec patch for multirom support, intelliplug, hellsactive governor, extra io schedulers, intellithermal v2, etc etc and growing, just check commits on history on the link below\n

Known bugs:
quick tiles may be a bit messy while re arranging them....
u tell me then.

\n

don't even think to report a bug with:
a dirty flashed rom,
xposed frameworks installed,
all of your apps installed, if u have a bug, clean flash the build, (flash the gapps if needed) and reproduce your bug without any data restore.
BRING BACK LOGCAT or u will be simply ignored....

\n

Contributors
martinusbe
@Alx31
ROM OS Version: 6.0.1 Marshmallow
ROM Kernel: Linux 3.4.x
Based On: AOSPB

\n

Version Information
Status: Stable
Created 2016_29_02

\n

Credits:
AOSPB/Slim team for an amazing base, Google, CyanogenMod for device trees and some other repos, Dirty Unicorns/CrDroid/AICP and other roms with their open sources i may have forgotten...

\n

Rom code: https://github.com/TipsyOs
Device: Hammerhead commits history
Kernel code: Kernel commits history
My github: https://github.com/Alx31

\n

\"\"

\n

Last edited by Alx31; 29th February 2016 at 10:48 PM.

", + "annotations": [], + "mimetype": "text/html", + "language": "en", + "reading_time": 3, + "domain_name": "forum.xda-developers.com", + "preview_picture": "http://cdn3.xda-developers.com/images/xda-facebook-default.jpg", + "tags": [] + }, + { + "is_archived": 0, + "is_starred": 0, + "id": 98, + "title": "Top 15 Podcasts All Web Developers Should Follow - Envato Tuts+ Code Article", + "url": "http://code.tutsplus.com/articles/top-15-podcasts-all-web-developers-should-follow--net-14461", + "content": "

As web developers, we’re always trying to get better at what we do. One of the best ways to do that is to listen to what other developers have to share. And even if you’re not learning, it’s still fun to hear what other devs are talking about. Today, I’ll share 15 podcasts that you should definitely check out.

\n
\n
\n

It seems that as often as a few times a month, Yahoo! brings in developers on the cutting edge of web technology to keep their employees up to date. For the benefit of the rest of us, these talks are recorded and published. You’ll find well-known devs like Douglas Crockford and NNicolas Zakas, and talks on everything from performance and accessibility to JavaScript and the DOM.

\n
\n
\n

This may be my favourite show from this list. The Dev Show, hosted weekly by Dan Benjamin and Jason Seifer, will give you a carefully curated set of development-related links (usually with a web dev slant) to enjoy. As an added bonus, you can watch the show live on Tuesdays at 1pm EST.

\n
\n
\n

The tagline for the Changelog says it all: “Open Source moves fast. Keep up.” This podcast, and the accompanying blog, is all about keeping you updated with the latest in Open Source Technology. It’s hosted by Adam Stacoviak and Wynn Netherland, and seems to be the official Github podcast.

\n
\n
\n

If you’re familiar with jQuery (and you probably are), you know there’s a podcast to go with it. Each week, hosts Ralph Whitbeck and Rey Bango bring you the latest in jQuery news, as well as great interviews with important people in the jQuery community. You can listen in to the jQuery wisdom of people like Remy Sharp, Yehuda Katz, Cody Lindley, and our own Jeffrey Way, among so many others.

\n
\n
\n

Sitepoint is a great resource for anyone interesting in technology, design, and even business. Books, courses, forums, blogs, articles, they’ve got it all. Of course, there’s a podcast too: check it out to find out what’s going on in the web industry.

\n
\n
\n

According to the site, WebPulp is “a podcast about technology that powers the web.” In each podcast, host Josh Owens interviews someone from behind the scenes of a well-known webapp; you’ll find out what hardware and software it takes to run apps like the 37signals apps, or GitHub.

\n
\n
\n

It’s pretty apparent that both Nettuts+ readers and writers are big fans of WordPress. If you can’t get enough WordPress goodness, you’ll probably want to sign up for the WordPress Podcast, “a weekly podcast with news, interviews and plugin tips.” There’s a bonus here: one of the most recent interviewees was none other than Collis Ta’eed, CEO of Envato.

\n
\n
\n

If you’re a user of Ellis Lab’s Expression Engine, you’ll enjoy the EE Podcast (Ellis Lab is the company behind CodeIgniter; in fact, EE is build completely on CI). Each week, Ryan Irelan and Lea Alcantara will fill in you on a certain aspect of of EE deveopment.

\n
\n
\n

If you’re a web developer, you’re probably pretty familiar with Chris Coyier’s website CSS Tricks. Besides his excellent articles, Chris occasionally puts out a screencast every few weeks. With his relaxed style, you’ll learn about a random—but always practical—part of web development in each episode.

\n
\n
\n

Hosted by Jeffrey Zeldman and Dan Benjamin, the Big Web Show “features special guests and topics like web publishing, art direction, content strategy, typography, web technology, and more. It’s everything web that matters.” You’ll listen to interviews in which famous web personalities like Eric Meyer, Jason Fried, Nicole Sullivan, Ethan Marcotte, and other professionals you should know open their minds and let you learn from the best. You can catch this show live on Thursdays a 1PM EST. Just like the Dev Show and the EE Podcast, the Big Web Show is part of Dan Benjamin’s incredible 5by5 podcast network.

\n
\n
\n

You may be familiar with Carsonified, the company behind many web dev / design training initiatives (including the Future of Web Design and Future of Web Apps confernces). On Think Vitamin, Carsonified’s “blog about the web”, you can catch Think Vitamin Radio, “a bi-weekly chat about web design, development, and entrepreneurship.”

\n
\n
\n

User Interface Engineering “is a leading research, training, and consulting firm specializing in web site and product usability.” You can take advantage of some of the free usability training they offer in their podcast, the Userability Podcast.

\n
\n
\n

This is a great resource for any beginner (and even intermediate) jQuery developers. In each episode, Remy Sharp will explain how to create an popular web effect using jQuery. You’ll learn how to build pop-up bubbles, sliding headers, and simple tabs.

\n
\n
\n

Of course, Nettuts+ publishes screencasts, too! For your convenience, you can get these in an iTunes feed, or subscribe to them on YouTube.

\n
\n
\n

HuffDuffer is a site created by Jeremy Keith; it allows you to easily create your own podcasts. From the tag cloud above, you can see that a lot of the content being collected is related to web development. Check it out!

\n
\n

I’m sure most developers listen to the occasional podcast. Have I missed your favourite podcast? Let us all know in the comments!

\n", + "annotations": [], + "mimetype": "text/html", + "reading_time": 4, + "domain_name": "code.tutsplus.com", + "preview_picture": "https://cdn.tutsplus.com/net/uploads/legacy/793_podcasts/preview.jpg", + "tags": [] + }, + { + "is_archived": 1, + "is_starred": 0, + "id": 97, + "title": "University of Mississippi", + "url": "http://olemiss.edu", + "content": "
\n
\n
\n
\n

UNIVERSITY OF MISSISSIPPI SCHOOLS AND COLLEGES

\n

The Schools of Nursing and Pharmacy operate on both the Oxford and Jackson campuses. The Schools of Dentistry, Health Related Professionals and Medicine, and the Health Sciences Graduate School, are based in Jackson only. (Additional healthcare programs are available through the School of Applied Sciences on the Oxford campus.) Other than these exceptions, the schools above are on the Oxford campus.

\n
\n
\n

Public Service Announcement: UM Health Center Seeing Increase in Flu Cases MORE INFO

\n
\n
  • \n
    \"\"
    Hotty Toddy
    \n

    Martavious Newby celebrates with fans after Ole Miss' 86-78 win over Mississippi State.

    \n
  • \n
  • \n
    \"\"
    Moody's Amazing Night
    \n

    Stefan Moody celebrates at the end of Ole Miss' 86-78 win over Mississippi State. Moody finished his final game at The Pavilion on Senior Night with a career-high 43 points.

    \n
  • \n
  • \n
    \"\"
    Career Expo
    \n

    Students talk to prospective employers during an all-majors career fair held at The Inn at Ole Miss.

    \n
  • \n
  • \n
    \"\"
    Choir Rehearsal
    \n

    Members of the University of Mississippi Concert Singers prepare for a performance at the American Choral Directors Association convention March 10 in Chattanooga, Tenn.

    \n
  • \n
  • \n
    \"\"
    Dentist's Pledge
    \n

    Kendra Clark (right) and other students in the UMMC School of Dentistry Class of 2018 recite the Dentist's Pledge at the American College of Dentists White Coat Ceremony held Feb. 26 in Jackson.

    \n
  • \n
\n

45°

\n

Oxford, MS

\n
\n
\n
\n
\n
ALL NEWS\n

Latest News

\n
\n

UM Honors 150 Students with Who's Who Distinction

\n
\n

UM Lazarus Project Attracts International Collaborations

\n
\n

Schedule Set for BancorpSouth Rebel Road Trip

\n
ALL ANNOUNCEMENTS\n

Announcements

\n
\n

olemisssports.com\n

UM Athletics

\n
\n
\n

Men's Basketball

\n
\"team Ole Miss 86
\n
\"team Miss. St. 78
\n

Wednesday, Mar. 2

\n
\n
\n

Men's Baseball

\n
\"team Ole Miss 9
\n
\"team Memphis 7
\n

Wednesday, Mar. 2

\n
\n
\n

Women's Basketball

\n
\"team Ole Miss 59
\n
\"team Vanderbilt 74
\n

Wednesday, Mar. 2

\n
\n
\n