mirror of
https://github.com/wallabag/wallabag.git
synced 2024-11-27 11:31:05 +00:00
Merge pull request #2308 from wallabag/tags-duplicate
Fix duplicate tags on import
This commit is contained in:
commit
7e98ad9626
11 changed files with 135 additions and 57 deletions
|
@ -322,9 +322,7 @@ class WallabagRestController extends FOSRestController
|
||||||
|
|
||||||
$tags = $this->getDoctrine()
|
$tags = $this->getDoctrine()
|
||||||
->getRepository('WallabagCoreBundle:Tag')
|
->getRepository('WallabagCoreBundle:Tag')
|
||||||
->findAllTags($this->getUser()->getId())
|
->findAllTagsWithEntries($this->getUser()->getId());
|
||||||
->getQuery()
|
|
||||||
->getResult();
|
|
||||||
|
|
||||||
$json = $this->get('serializer')->serialize($tags, 'json');
|
$json = $this->get('serializer')->serialize($tags, 'json');
|
||||||
|
|
||||||
|
|
|
@ -84,16 +84,11 @@ class TagController extends Controller
|
||||||
{
|
{
|
||||||
$tags = $this->getDoctrine()
|
$tags = $this->getDoctrine()
|
||||||
->getRepository('WallabagCoreBundle:Tag')
|
->getRepository('WallabagCoreBundle:Tag')
|
||||||
->findAllTags($this->getUser()->getId())
|
->findAllTagsWithEntries($this->getUser()->getId());
|
||||||
->getQuery()
|
|
||||||
->getResult();
|
|
||||||
|
|
||||||
return $this->render(
|
return $this->render('WallabagCoreBundle:Tag:tags.html.twig', [
|
||||||
'WallabagCoreBundle:Tag:tags.html.twig',
|
|
||||||
[
|
|
||||||
'tags' => $tags,
|
'tags' => $tags,
|
||||||
]
|
]);
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -127,13 +122,10 @@ class TagController extends Controller
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render(
|
return $this->render('WallabagCoreBundle:Entry:entries.html.twig', [
|
||||||
'WallabagCoreBundle:Entry:entries.html.twig',
|
|
||||||
[
|
|
||||||
'form' => null,
|
'form' => null,
|
||||||
'entries' => $entries,
|
'entries' => $entries,
|
||||||
'currentPage' => $page,
|
'currentPage' => $page,
|
||||||
]
|
]);
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -96,13 +96,23 @@ class ContentProxy
|
||||||
*
|
*
|
||||||
* @param Entry $entry
|
* @param Entry $entry
|
||||||
* @param array|string $tags An array of tag or a string coma separated of tag
|
* @param array|string $tags An array of tag or a string coma separated of tag
|
||||||
|
* @param array $entitiesReady Entities from the EntityManager which are persisted but not yet flushed
|
||||||
|
* It is mostly to fix duplicate tag on import @see http://stackoverflow.com/a/7879164/569101
|
||||||
*/
|
*/
|
||||||
public function assignTagsToEntry(Entry $entry, $tags)
|
public function assignTagsToEntry(Entry $entry, $tags, array $entitiesReady = [])
|
||||||
{
|
{
|
||||||
if (!is_array($tags)) {
|
if (!is_array($tags)) {
|
||||||
$tags = explode(',', $tags);
|
$tags = explode(',', $tags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// keeps only Tag entity from the "not yet flushed entities"
|
||||||
|
$tagsNotYetFlushed = [];
|
||||||
|
foreach ($entitiesReady as $entity) {
|
||||||
|
if ($entity instanceof Tag) {
|
||||||
|
$tagsNotYetFlushed[$entity->getLabel()] = $entity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
foreach ($tags as $label) {
|
foreach ($tags as $label) {
|
||||||
$label = trim($label);
|
$label = trim($label);
|
||||||
|
|
||||||
|
@ -111,12 +121,16 @@ class ContentProxy
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isset($tagsNotYetFlushed[$label])) {
|
||||||
|
$tagEntity = $tagsNotYetFlushed[$label];
|
||||||
|
} else {
|
||||||
$tagEntity = $this->tagRepository->findOneByLabel($label);
|
$tagEntity = $this->tagRepository->findOneByLabel($label);
|
||||||
|
|
||||||
if (is_null($tagEntity)) {
|
if (is_null($tagEntity)) {
|
||||||
$tagEntity = new Tag();
|
$tagEntity = new Tag();
|
||||||
$tagEntity->setLabel($label);
|
$tagEntity->setLabel($label);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// only add the tag on the entry if the relation doesn't exist
|
// only add the tag on the entry if the relation doesn't exist
|
||||||
if (false === $entry->getTags()->contains($tagEntity)) {
|
if (false === $entry->getTags()->contains($tagEntity)) {
|
||||||
|
|
|
@ -7,17 +7,45 @@ use Doctrine\ORM\EntityRepository;
|
||||||
class TagRepository extends EntityRepository
|
class TagRepository extends EntityRepository
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Find Tags.
|
* Count all tags per user.
|
||||||
|
*
|
||||||
|
* @param int $userId
|
||||||
|
* @param int $cacheLifeTime Duration of the cache for this query
|
||||||
|
*
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
public function countAllTags($userId, $cacheLifeTime = null)
|
||||||
|
{
|
||||||
|
$query = $this->createQueryBuilder('t')
|
||||||
|
->select('t.slug')
|
||||||
|
->leftJoin('t.entries', 'e')
|
||||||
|
->where('e.user = :userId')->setParameter('userId', $userId)
|
||||||
|
->groupBy('t.slug')
|
||||||
|
->getQuery();
|
||||||
|
|
||||||
|
if (null !== $cacheLifeTime) {
|
||||||
|
$query->useQueryCache(true);
|
||||||
|
$query->useResultCache(true);
|
||||||
|
$query->setResultCacheLifetime($cacheLifeTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
return count($query->getArrayResult());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find all tags with associated entries per user.
|
||||||
*
|
*
|
||||||
* @param int $userId
|
* @param int $userId
|
||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function findAllTags($userId)
|
public function findAllTagsWithEntries($userId)
|
||||||
{
|
{
|
||||||
return $this->createQueryBuilder('t')
|
return $this->createQueryBuilder('t')
|
||||||
->leftJoin('t.entries', 'e')
|
->leftJoin('t.entries', 'e')
|
||||||
->where('e.user = :userId')->setParameter('userId', $userId);
|
->where('e.user = :userId')->setParameter('userId', $userId)
|
||||||
|
->getQuery()
|
||||||
|
->getResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
|
|
||||||
namespace Wallabag\CoreBundle\Twig;
|
namespace Wallabag\CoreBundle\Twig;
|
||||||
|
|
||||||
use Doctrine\ORM\Query;
|
|
||||||
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
|
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
|
||||||
use Wallabag\CoreBundle\Repository\EntryRepository;
|
use Wallabag\CoreBundle\Repository\EntryRepository;
|
||||||
use Wallabag\CoreBundle\Repository\TagRepository;
|
use Wallabag\CoreBundle\Repository\TagRepository;
|
||||||
|
@ -85,10 +84,11 @@ class WallabagExtension extends \Twig_Extension implements \Twig_Extension_Globa
|
||||||
->groupBy('e.id')
|
->groupBy('e.id')
|
||||||
->getQuery();
|
->getQuery();
|
||||||
|
|
||||||
$data = $this->enableCache($query)
|
$query->useQueryCache(true);
|
||||||
->getArrayResult();
|
$query->useResultCache(true);
|
||||||
|
$query->setResultCacheLifetime($this->lifeTime);
|
||||||
|
|
||||||
return count($data);
|
return count($query->getArrayResult());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -104,28 +104,7 @@ class WallabagExtension extends \Twig_Extension implements \Twig_Extension_Globa
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$qb = $this->tagRepository->findAllTags($user->getId());
|
return $this->tagRepository->countAllTags($user->getId());
|
||||||
|
|
||||||
$data = $this->enableCache($qb->getQuery())
|
|
||||||
->getArrayResult();
|
|
||||||
|
|
||||||
return count($data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Enable cache for a query.
|
|
||||||
*
|
|
||||||
* @param Query $query
|
|
||||||
*
|
|
||||||
* @return Query
|
|
||||||
*/
|
|
||||||
private function enableCache(Query $query)
|
|
||||||
{
|
|
||||||
$query->useQueryCache(true);
|
|
||||||
$query->useResultCache(true);
|
|
||||||
$query->setResultCacheLifetime($this->lifeTime);
|
|
||||||
|
|
||||||
return $query;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getName()
|
public function getName()
|
||||||
|
|
|
@ -227,7 +227,8 @@ class PocketImport extends AbstractImport
|
||||||
if (isset($importedEntry['tags']) && !empty($importedEntry['tags'])) {
|
if (isset($importedEntry['tags']) && !empty($importedEntry['tags'])) {
|
||||||
$this->contentProxy->assignTagsToEntry(
|
$this->contentProxy->assignTagsToEntry(
|
||||||
$entry,
|
$entry,
|
||||||
array_keys($importedEntry['tags'])
|
array_keys($importedEntry['tags']),
|
||||||
|
$this->em->getUnitOfWork()->getScheduledEntityInsertions()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -111,7 +111,8 @@ abstract class WallabagImport extends AbstractImport
|
||||||
if (array_key_exists('tags', $data)) {
|
if (array_key_exists('tags', $data)) {
|
||||||
$this->contentProxy->assignTagsToEntry(
|
$this->contentProxy->assignTagsToEntry(
|
||||||
$entry,
|
$entry,
|
||||||
$data['tags']
|
$data['tags'],
|
||||||
|
$this->em->getUnitOfWork()->getScheduledEntityInsertions()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -296,6 +296,29 @@ class ContentProxyTest extends \PHPUnit_Framework_TestCase
|
||||||
$this->assertEquals('tag2', $entry->getTags()[1]->getLabel());
|
$this->assertEquals('tag2', $entry->getTags()[1]->getLabel());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testAssignTagsNotFlushed()
|
||||||
|
{
|
||||||
|
$graby = $this->getMockBuilder('Graby\Graby')
|
||||||
|
->disableOriginalConstructor()
|
||||||
|
->getMock();
|
||||||
|
|
||||||
|
$tagRepo = $this->getTagRepositoryMock();
|
||||||
|
$tagRepo->expects($this->never())
|
||||||
|
->method('__call');
|
||||||
|
|
||||||
|
$proxy = new ContentProxy($graby, $this->getTaggerMock(), $tagRepo, $this->getLogger());
|
||||||
|
|
||||||
|
$tagEntity = new Tag();
|
||||||
|
$tagEntity->setLabel('tag1');
|
||||||
|
|
||||||
|
$entry = new Entry(new User());
|
||||||
|
|
||||||
|
$proxy->assignTagsToEntry($entry, 'tag1', [$tagEntity]);
|
||||||
|
|
||||||
|
$this->assertCount(1, $entry->getTags());
|
||||||
|
$this->assertEquals('tag1', $entry->getTags()[0]->getLabel());
|
||||||
|
}
|
||||||
|
|
||||||
private function getTaggerMock()
|
private function getTaggerMock()
|
||||||
{
|
{
|
||||||
return $this->getMockBuilder('Wallabag\CoreBundle\Helper\RuleBasedTagger')
|
return $this->getMockBuilder('Wallabag\CoreBundle\Helper\RuleBasedTagger')
|
||||||
|
|
|
@ -41,6 +41,20 @@ class PocketImportTest extends \PHPUnit_Framework_TestCase
|
||||||
->disableOriginalConstructor()
|
->disableOriginalConstructor()
|
||||||
->getMock();
|
->getMock();
|
||||||
|
|
||||||
|
$this->uow = $this->getMockBuilder('Doctrine\ORM\UnitOfWork')
|
||||||
|
->disableOriginalConstructor()
|
||||||
|
->getMock();
|
||||||
|
|
||||||
|
$this->em
|
||||||
|
->expects($this->any())
|
||||||
|
->method('getUnitOfWork')
|
||||||
|
->willReturn($this->uow);
|
||||||
|
|
||||||
|
$this->uow
|
||||||
|
->expects($this->any())
|
||||||
|
->method('getScheduledEntityInsertions')
|
||||||
|
->willReturn([]);
|
||||||
|
|
||||||
$pocket = new PocketImport(
|
$pocket = new PocketImport(
|
||||||
$this->em,
|
$this->em,
|
||||||
$this->contentProxy
|
$this->contentProxy
|
||||||
|
|
|
@ -26,6 +26,20 @@ class WallabagV1ImportTest extends \PHPUnit_Framework_TestCase
|
||||||
->disableOriginalConstructor()
|
->disableOriginalConstructor()
|
||||||
->getMock();
|
->getMock();
|
||||||
|
|
||||||
|
$this->uow = $this->getMockBuilder('Doctrine\ORM\UnitOfWork')
|
||||||
|
->disableOriginalConstructor()
|
||||||
|
->getMock();
|
||||||
|
|
||||||
|
$this->em
|
||||||
|
->expects($this->any())
|
||||||
|
->method('getUnitOfWork')
|
||||||
|
->willReturn($this->uow);
|
||||||
|
|
||||||
|
$this->uow
|
||||||
|
->expects($this->any())
|
||||||
|
->method('getScheduledEntityInsertions')
|
||||||
|
->willReturn([]);
|
||||||
|
|
||||||
$this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy')
|
$this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy')
|
||||||
->disableOriginalConstructor()
|
->disableOriginalConstructor()
|
||||||
->getMock();
|
->getMock();
|
||||||
|
|
|
@ -26,6 +26,20 @@ class WallabagV2ImportTest extends \PHPUnit_Framework_TestCase
|
||||||
->disableOriginalConstructor()
|
->disableOriginalConstructor()
|
||||||
->getMock();
|
->getMock();
|
||||||
|
|
||||||
|
$this->uow = $this->getMockBuilder('Doctrine\ORM\UnitOfWork')
|
||||||
|
->disableOriginalConstructor()
|
||||||
|
->getMock();
|
||||||
|
|
||||||
|
$this->em
|
||||||
|
->expects($this->any())
|
||||||
|
->method('getUnitOfWork')
|
||||||
|
->willReturn($this->uow);
|
||||||
|
|
||||||
|
$this->uow
|
||||||
|
->expects($this->any())
|
||||||
|
->method('getScheduledEntityInsertions')
|
||||||
|
->willReturn([]);
|
||||||
|
|
||||||
$this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy')
|
$this->contentProxy = $this->getMockBuilder('Wallabag\CoreBundle\Helper\ContentProxy')
|
||||||
->disableOriginalConstructor()
|
->disableOriginalConstructor()
|
||||||
->getMock();
|
->getMock();
|
||||||
|
|
Loading…
Reference in a new issue