Test implementation of crawler API. NOT SECURED BY TOKEN YET.

This commit is contained in:
Alexey Skobkin 2016-03-24 01:49:22 +03:00
parent d539cd18c5
commit a1b982d891
5 changed files with 181 additions and 108 deletions

View file

@ -2,6 +2,7 @@
namespace Skobkin\Bundle\PointToolsBundle\Controller\Api; namespace Skobkin\Bundle\PointToolsBundle\Controller\Api;
use Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs\PostFactory;
use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Request;
class CrawlerController extends AbstractApiController class CrawlerController extends AbstractApiController
@ -15,8 +16,13 @@ class CrawlerController extends AbstractApiController
$page = $serializer->deserialize($json, 'Skobkin\Bundle\PointToolsBundle\DTO\Api\Crawler\PostsPage', 'json'); $page = $serializer->deserialize($json, 'Skobkin\Bundle\PointToolsBundle\DTO\Api\Crawler\PostsPage', 'json');
/** @var PostFactory $factory */
$factory = $this->get('skobkin__point_tools.service_factory.post_factory');
$continue = $factory->createFromPageDTO($page);
return $this->createSuccessResponse([ return $this->createSuccessResponse([
'continue' => false, 'continue' => $continue,
]); ]);
} }
} }

View file

@ -46,7 +46,7 @@ class Post
/** /**
* @var \DateTime * @var \DateTime
* *
* @ORM\Column(name="updated_at", type="datetime") * @ORM\Column(name="updated_at", type="datetime", nullable=true)
*/ */
private $updatedAt; private $updatedAt;
@ -55,7 +55,7 @@ class Post
* *
* @ORM\Column(name="type", type="string", length=6) * @ORM\Column(name="type", type="string", length=6)
*/ */
private $type; private $type = self::TYPE_POST;
/** /**
* @var bool * @var bool
@ -227,6 +227,7 @@ class Post
*/ */
public function addPostTag(PostTag $tag) public function addPostTag(PostTag $tag)
{ {
$tag->setPost($this);
$this->postTags[] = $tag; $this->postTags[] = $tag;
return $this; return $this;

View file

@ -35,8 +35,8 @@ services:
skobkin__point_tools.service_factory.tag_factory: skobkin__point_tools.service_factory.tag_factory:
class: Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs\TagFactory class: Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs\TagFactory
arguments: [ @doctrine.orm.entity_manager ] arguments: [ @logger, @doctrine.orm.entity_manager ]
skobkin__point_tools.service_factory.post_factory: skobkin__point_tools.service_factory.post_factory:
class: Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs\PostFactory class: Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs\PostFactory
arguments: [ @doctrine.orm.entity_manager, @skobkin__point_tools.service_factory.user_factory, @skobkin__point_tools.service_factory.comment_factory, @skobkin__point_tools.service_factory.tag_factory ] arguments: [ @logger, @doctrine.orm.entity_manager, @skobkin__point_tools.service_factory.user_factory, @skobkin__point_tools.service_factory.comment_factory, @skobkin__point_tools.service_factory.tag_factory ]

View file

@ -5,14 +5,25 @@ namespace Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs;
use Doctrine\ORM\EntityManager; use Doctrine\ORM\EntityManager;
use Doctrine\ORM\EntityManagerInterface; use Doctrine\ORM\EntityManagerInterface;
use Doctrine\ORM\EntityRepository; use Doctrine\ORM\EntityRepository;
use Psr\Log\LoggerInterface;
use Skobkin\Bundle\PointToolsBundle\DTO\Api\Crawler\MetaPost;
use Skobkin\Bundle\PointToolsBundle\DTO\Api\Crawler\PostsPage;
use Skobkin\Bundle\PointToolsBundle\Entity\Blogs\Post; use Skobkin\Bundle\PointToolsBundle\Entity\Blogs\Post;
use Skobkin\Bundle\PointToolsBundle\Entity\Blogs\PostTag;
use Skobkin\Bundle\PointToolsBundle\Entity\Blogs\Tag;
use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\ApiException; use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\ApiException;
use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\Factory\Blogs\InvalidPostDataException;
use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\InvalidResponseException; use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\InvalidResponseException;
use Skobkin\Bundle\PointToolsBundle\Service\Factory\UserFactory; use Skobkin\Bundle\PointToolsBundle\Service\Factory\UserFactory;
class PostFactory class PostFactory
{ {
/**
* @var LoggerInterface
*/
private $log;
/** /**
* @var EntityManager * @var EntityManager
*/ */
@ -41,8 +52,9 @@ class PostFactory
/** /**
* @param EntityManager $em * @param EntityManager $em
*/ */
public function __construct(EntityManagerInterface $em, UserFactory $userFactory, CommentFactory $commentFactory, TagFactory $tagFactory) public function __construct(LoggerInterface $log, EntityManagerInterface $em, UserFactory $userFactory, CommentFactory $commentFactory, TagFactory $tagFactory)
{ {
$this->log = $log;
$this->userFactory = $userFactory; $this->userFactory = $userFactory;
$this->commentFactory = $commentFactory; $this->commentFactory = $commentFactory;
$this->tagFactory = $tagFactory; $this->tagFactory = $tagFactory;
@ -51,91 +63,145 @@ class PostFactory
} }
/** /**
* @param array $data * Creates posts and return status of new insertions
* *
* @return Post * @param PostsPage $data
*
* @return bool
* @throws ApiException * @throws ApiException
* @throws InvalidResponseException * @throws InvalidResponseException
*/ */
public function createFromArray(array $data) public function createFromPageDTO(PostsPage $page)
{ {
$this->validateData($data); $posts = [];
if (null === ($post = $this->postRepository->find($data['post']['id']))) { $hasNew = false;
$createdAt = new \DateTime($data['post']['created']);
$author = $this->userFactory->createFromArray($data['post']['author']);
$post = new Post($data['post']['id'], $data['post']['type'], $data['post']['text'], $createdAt, $author); foreach ($page->getPosts() as $postData) {
try {
if (null === $this->postRepository->find($postData->getPost()->getId())) {
$hasNew = true;
}
$post = $this->createFromDTO($postData);
$posts[] = $post;
} catch (\Exception $e) {
$this->log->error('Error while processing post DTO', ['post_id' => $postData->getPost()->getId()]);
}
}
foreach ($posts as $post) {
if ($this->em->getUnitOfWork()->isScheduledForInsert($post)) {
$hasNew = true;
}
}
$this->em->flush();
return $hasNew;
}
/**
* @todo Implement full post with comments processing
*
* @param MetaPost $postData
*
* @return Post
* @throws ApiException
* @throws InvalidPostDataException
*/
private function createFromDTO(MetaPost $postData)
{
if (!$this->validateMetaPost($postData)) {
throw new InvalidPostDataException('Invalid post data', $postData);
}
if (!$postData->getPost()->getAuthor()->getId()) {
$this->log->error('Post author does not contain id', ['post_id' => $postData->getPost()->getId()]);
throw new InvalidPostDataException('Post author does not contain id', $postData);
}
$user = $this->userFactory->createFromDTO($postData->getPost()->getAuthor());
if (null === ($post = $this->postRepository->find($postData->getPost()->getId()))) {
// Creating new post
$post = new Post($postData->getPost()->getId());
$this->em->persist($post); $this->em->persist($post);
} }
$post->setText($data['post']['text']); // Updating data
$post
->setAuthor($user)
->setCreatedAt((new \DateTime($postData->getPost()->getCreated())) ?: null)
// @fixme Need bugfix for point API (type is not showing now)
//->setType($postData->getPost()->getType())
->setText($postData->getPost()->getText())
->setPrivate($postData->getPost()->getPrivate())
;
// Tags $this->updatePostTags($post, $postData->getPost()->getTags());
$tags = $this->tagFactory->createFromListArray($data['post']['tags']);
// Removing deleted tags
foreach ($post->getTags() as $tag) {
if (false === in_array($tag, $tags, true)) {
$post->removeTag($tag);
}
}
// Adding new tags
foreach ($tags as $tag) {
if (!$post->getTags()->contains($tag)) {
$post->addTag($tag);
}
}
// Flushing post before linking comments
try {
$this->em->flush($post); $this->em->flush($post);
} catch (\Exception $e) {
throw new ApiException(sprintf('Error while flushing changes for #%s: %s', $data['post']['id'], $e->getMessage()), 0, $e);
}
// Comments
$comments = $this->commentFactory->createFromListArray($data['comments']);
// Marking removed comments
foreach ($post->getComments() as $comment) {
if (false === in_array($comment, $comments, true)) {
$comment->setDeleted(true);
}
}
// Adding comments
foreach ($comments as $comment) {
$post->addComment($comment);
}
return $post; return $post;
} }
/** /**
* @param array $data * @param Post $post
* * @param Tag[] $tags
* @throws InvalidResponseException
*/ */
private function validateData(array $data) private function updatePostTags(Post $post, array $tagsStrings)
{ {
if (!array_key_exists('post', $data)) { $tags = $this->tagFactory->createFromStringsArray($tagsStrings);
throw new InvalidResponseException('Post data not found in API response');
// Hashing tags strings
$tagStringsHash = [];
foreach ($tagsStrings as $tagsString) {
$tagStringsHash[mb_strtolower($tagsString)] = $tagsString;
} }
if (!array_key_exists('comments', $data)) { // Hashing current post tags
throw new InvalidResponseException('Comments data not found in API response'); $newTagsHash = [];
foreach ($tags as $tag) {
$newTagsHash[mb_strtolower($tag->getText())] = $tag;
} }
if (!( // Hashing old post tags (from DB)
array_key_exists('id', $data['post']) && $oldTagsHash = [];
array_key_exists('type', $data['post']) && foreach ($post->getPostTags() as $postTag) {
array_key_exists('text', $data['post']) && $oldTagsHash[mb_strtolower($postTag->getOriginalTagText())] = $postTag;
array_key_exists('tags', $data['post']) &&
array_key_exists('author', $data['post'])
)) {
throw new InvalidResponseException('Post content not found in API response');
} }
// Adding missing tags
foreach ($tags as $tag) {
if (!array_key_exists(mb_strtolower($tag->getText()), $oldTagsHash)) {
$tmpPostTag = (new PostTag($tag))
->setText($tagStringsHash[mb_strtolower($tag->getText())])
;
$post->addPostTag($tmpPostTag);
}
}
// Removing deleted tags
foreach ($post->getPostTags() as $postTag) {
if (!array_key_exists(mb_strtolower($postTag->getOriginalTagText()), $newTagsHash)) {
$post->removePostTag($postTag);
}
}
}
private function validateMetaPost(MetaPost $post)
{
if (!$post->getPost()->getId()) {
$this->log->error('Post DTO contains no id');
return false;
}
if (null === $post->getPost()->getAuthor()->getId()) {
$this->log->error('Post DTO contains no valid User DTO.', ['post_id' => $post->getPost()->getId()]);
return false;
}
return true;
} }
} }

View file

@ -5,8 +5,8 @@ namespace Skobkin\Bundle\PointToolsBundle\Service\Factory\Blogs;
use Doctrine\ORM\EntityManager; use Doctrine\ORM\EntityManager;
use Doctrine\ORM\EntityManagerInterface; use Doctrine\ORM\EntityManagerInterface;
use Doctrine\ORM\EntityRepository; use Doctrine\ORM\EntityRepository;
use Psr\Log\LoggerInterface;
use Skobkin\Bundle\PointToolsBundle\Entity\Blogs\Tag; use Skobkin\Bundle\PointToolsBundle\Entity\Blogs\Tag;
use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\ApiException;
use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\InvalidResponseException; use Skobkin\Bundle\PointToolsBundle\Service\Exceptions\InvalidResponseException;
@ -17,6 +17,11 @@ class TagFactory
*/ */
private $em; private $em;
/**
* @var LoggerInterface
*/
private $log;
/** /**
* @var EntityRepository * @var EntityRepository
*/ */
@ -25,62 +30,56 @@ class TagFactory
/** /**
* @param EntityManager $em * @param EntityManager $em
*/ */
public function __construct(EntityManagerInterface $em) public function __construct(LoggerInterface $log, EntityManagerInterface $em)
{ {
$this->log = $log;
$this->em = $em; $this->em = $em;
$this->tagRepository = $em->getRepository('SkobkinPointToolsBundle:Blogs\Tag'); $this->tagRepository = $em->getRepository('SkobkinPointToolsBundle:Blogs\Tag');
} }
/** /**
* @param $data * @param string[] $tagStrings
*
* @return Tag
* @throws ApiException
* @throws InvalidResponseException
* @throws \Doctrine\ORM\NonUniqueResultException
*/
public function createFromArray($data)
{
$this->validateData($data);
$qb = $this->tagRepository->createQueryBuilder('t');
$qb
->select()
->where($qb->expr()->eq('lower(t.text)', 'lower(:text)'))
->setParameter('text', $data)
;
if (null === ($tag = $qb->getQuery()->getOneOrNullResult())) {
$tag = new Tag($data);
$this->em->persist($tag);
}
try {
$this->em->flush($tag);
} catch (\Exception $e) {
throw new ApiException(sprintf('Error while flushing changes for [%d] %s: %s', $tag->getId(), $tag->getText(), $e->getMessage()), 0, $e);
}
return $tag;
}
/**
* @param array $data
* *
* @return Tag[] * @return Tag[]
* @throws ApiException
*/ */
public function createFromListArray(array $data) public function createFromStringsArray(array $tagStrings)
{ {
$tags = []; $tags = [];
foreach ($data as $text) { foreach ($tagStrings as $string) {
$tags[] = $this->createFromArray($text); try {
$tag = $this->createFromString($string);
$tags[] = $tag;
} catch (\Exception $e) {
$this->log->error('Error while creating tag from DTO', ['tag' => $string, 'message' => $e->getMessage()]);
continue;
}
} }
return $tags; return $tags;
} }
/**
* @param $text
*
* @return Tag
* @throws InvalidResponseException
*/
public function createFromString($text)
{
$this->validateData($text);
if (null === ($tag = $this->tagRepository->findOneByLowerText($text))) {
// Creating new tag
$tag = new Tag($text);
$this->em->persist($tag);
}
$this->em->flush($tag);
return $tag;
}
/** /**
* @param $data * @param $data
* *
@ -89,6 +88,7 @@ class TagFactory
private function validateData($data) private function validateData($data)
{ {
if (!is_string($data)) { if (!is_string($data)) {
// @todo Change exception
throw new InvalidResponseException('Tag data must be a string'); throw new InvalidResponseException('Tag data must be a string');
} }
} }