3 namespace Wallabag\ImportBundle\Import
;
5 use Psr\Log\LoggerInterface
;
6 use Psr\Log\NullLogger
;
7 use Doctrine\ORM\EntityManager
;
9 use GuzzleHttp\Exception\RequestException
;
10 use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface
;
11 use Wallabag\CoreBundle\Entity\Entry
;
12 use Wallabag\CoreBundle\Helper\ContentProxy
;
13 use Craue\ConfigBundle\Util\Config
;
15 class PocketImport
implements ImportInterface
19 private $contentProxy;
23 private $skippedEntries = 0;
24 private $importedEntries = 0;
26 protected $accessToken;
28 public function __construct(TokenStorageInterface
$tokenStorage, EntityManager
$em, ContentProxy
$contentProxy, Config
$craueConfig)
30 $this->user
= $tokenStorage->getToken()->getUser();
32 $this->contentProxy
= $contentProxy;
33 $this->consumerKey
= $craueConfig->get('pocket_consumer_key');
34 $this->logger
= new NullLogger();
37 public function setLogger(LoggerInterface
$logger)
39 $this->logger
= $logger;
45 public function getName()
53 public function getUrl()
55 return 'import_pocket';
61 public function getDescription()
63 return 'import.pocket.description';
67 * Return the oauth url to authenticate the client.
69 * @param string $redirectUri Redirect url in case of error
71 * @return string|false request_token for callback method
73 public function getRequestToken($redirectUri)
75 $request = $this->client
->createRequest('POST', 'https://getpocket.com/v3/oauth/request',
77 'body' => json_encode([
78 'consumer_key' => $this->consumerKey
,
79 'redirect_uri' => $redirectUri,
85 $response = $this->client
->send($request);
86 } catch (RequestException
$e) {
87 $this->logger
->error(sprintf('PocketImport: Failed to request token: %s', $e->getMessage()), ['exception' => $e]);
92 return $response->json()['code'];
96 * Usually called by the previous callback to authorize the client.
97 * Then it return a token that can be used for next requests.
99 * @param string $code request_token from getRequestToken
103 public function authorize($code)
105 $request = $this->client
->createRequest('POST', 'https://getpocket.com/v3/oauth/authorize',
107 'body' => json_encode([
108 'consumer_key' => $this->consumerKey
,
115 $response = $this->client
->send($request);
116 } catch (RequestException
$e) {
117 $this->logger
->error(sprintf('PocketImport: Failed to authorize client: %s', $e->getMessage()), ['exception' => $e]);
122 $this->accessToken
= $response->json()['access_token'];
128 * Set whether articles must be all marked as read.
130 * @param bool $markAsRead
132 public function setMarkAsRead($markAsRead)
134 $this->markAsRead
= $markAsRead;
140 * Get whether articles must be all marked as read.
142 public function getMarkAsRead()
144 return $this->markAsRead
;
150 public function import()
152 $request = $this->client
->createRequest('POST', 'https://getpocket.com/v3/get',
154 'body' => json_encode([
155 'consumer_key' => $this->consumerKey
,
156 'access_token' => $this->accessToken
,
157 'detailType' => 'complete',
165 $response = $this->client
->send($request);
166 } catch (RequestException
$e) {
167 $this->logger
->error(sprintf('PocketImport: Failed to import: %s', $e->getMessage()), ['exception' => $e]);
172 $entries = $response->json();
174 $this->parseEntries($entries['list']);
182 public function getSummary()
185 'skipped' => $this->skippedEntries
,
186 'imported' => $this->importedEntries
,
191 * Set the Guzzle client.
193 * @param Client $client
195 public function setClient(Client
$client)
197 $this->client
= $client;
201 * @see https://getpocket.com/developer/docs/v3/retrieve
205 private function parseEntries($entries)
209 foreach ($entries as $pocketEntry) {
210 $url = isset($pocketEntry['resolved_url']) && $pocketEntry['resolved_url'] != '' ? $pocketEntry['resolved_url'] : $pocketEntry['given_url'];
212 $existingEntry = $this->em
213 ->getRepository('WallabagCoreBundle:Entry')
214 ->findByUrlAndUserId($url, $this->user
->getId());
216 if (false !== $existingEntry) {
217 ++
$this->skippedEntries
;
221 $entry = new Entry($this->user
);
222 $entry = $this->contentProxy
->updateEntry($entry, $url);
224 // 0, 1, 2 - 1 if the item is archived - 2 if the item should be deleted
225 if ($pocketEntry['status'] == 1 || $this->markAsRead
) {
226 $entry->setArchived(true);
229 // 0 or 1 - 1 If the item is favorited
230 if ($pocketEntry['favorite'] == 1) {
231 $entry->setStarred(true);
235 if (isset($pocketEntry['resolved_title']) && $pocketEntry['resolved_title'] != '') {
236 $title = $pocketEntry['resolved_title'];
237 } elseif (isset($pocketEntry['given_title']) && $pocketEntry['given_title'] != '') {
238 $title = $pocketEntry['given_title'];
241 $entry->setTitle($title);
243 // 0, 1, or 2 - 1 if the item has images in it - 2 if the item is an image
244 if (isset($pocketEntry['has_image']) && $pocketEntry['has_image'] > 0 && isset($pocketEntry['images'][1])) {
245 $entry->setPreviewPicture($pocketEntry['images'][1]['src']);
248 if (isset($pocketEntry['tags']) && !empty($pocketEntry['tags'])) {
249 $this->contentProxy
->assignTagsToEntry(
251 array_keys($pocketEntry['tags'])
255 $this->em
->persist($entry);
256 ++
$this->importedEntries
;
258 // flush every 20 entries
259 if (($i %
20) === 0) {
261 $this->em
->clear($entry);