szurubooru/src/Dao/PostDao.php
2014-10-18 18:48:31 +02:00

244 lines
6.6 KiB
PHP

<?php
namespace Szurubooru\Dao;
class PostDao extends AbstractDao implements ICrudDao
{
private $tagDao;
private $userDao;
private $fileService;
private $thumbnailService;
public function __construct(
\Szurubooru\DatabaseConnection $databaseConnection,
\Szurubooru\Dao\TagDao $tagDao,
\Szurubooru\Dao\UserDao $userDao,
\Szurubooru\Services\FileService $fileService,
\Szurubooru\Services\ThumbnailService $thumbnailService)
{
parent::__construct(
$databaseConnection,
'posts',
new \Szurubooru\Dao\EntityConverters\PostEntityConverter());
$this->tagDao = $tagDao;
$this->userDao = $userDao;
$this->fileService = $fileService;
$this->thumbnailService = $thumbnailService;
}
public function getCount()
{
return count($this->fpdo->from($this->tableName));
}
public function getTotalFileSize()
{
$query = $this->fpdo->from($this->tableName)->select('SUM(originalFileSize) AS __sum');
return intval(iterator_to_array($query)[0]['__sum']);
}
public function findByName($name)
{
return $this->findOneBy('name', $name);
}
public function findByContentChecksum($checksum)
{
return $this->findOneBy('contentChecksum', $checksum);
}
protected function afterLoad(\Szurubooru\Entities\Entity $post)
{
$post->setLazyLoader(
\Szurubooru\Entities\Post::LAZY_LOADER_CONTENT,
function (\Szurubooru\Entities\Post $post)
{
return $this->fileService->load($post->getContentPath());
});
$post->setLazyLoader(
\Szurubooru\Entities\Post::LAZY_LOADER_THUMBNAIL_SOURCE_CONTENT,
function (\Szurubooru\Entities\Post $post)
{
return $this->fileService->load($post->getThumbnailSourceContentPath());
});
$post->setLazyLoader(
\Szurubooru\Entities\Post::LAZY_LOADER_USER,
function (\Szurubooru\Entities\Post $post)
{
return $this->getUser($post);
});
$post->setLazyLoader(
\Szurubooru\Entities\Post::LAZY_LOADER_TAGS,
function (\Szurubooru\Entities\Post $post)
{
return $this->getTags($post);
});
$post->setLazyLoader(
\Szurubooru\Entities\Post::LAZY_LOADER_RELATED_POSTS,
function (\Szurubooru\Entities\Post $post)
{
return $this->getRelatedPosts($post);
});
}
protected function afterSave(\Szurubooru\Entities\Entity $post)
{
$this->syncContent($post);
$this->syncThumbnailSourceContent($post);
$this->syncTags($post);
$this->syncPostRelations($post);
}
protected function decorateQueryFromRequirement($query, \Szurubooru\SearchServices\Requirements\Requirement $requirement)
{
if ($requirement->getType() === \Szurubooru\SearchServices\Filters\PostFilter::REQUIREMENT_TAG)
{
$sql = 'EXISTS (
SELECT 1 FROM postTags
INNER JOIN tags ON postTags.tagId = tags.id
WHERE postTags.postId = posts.id
AND LOWER(tags.name) = LOWER(?))';
if ($requirement->isNegated())
$sql = 'NOT ' . $sql;
$query->where($sql, $requirement->getValue()->getValue());
return;
}
elseif ($requirement->getType() === \Szurubooru\SearchServices\Filters\PostFilter::REQUIREMENT_FAVORITE)
{
$query->innerJoin('favorites _fav ON _fav.postId = posts.id INNER JOIN users favorite ON favorite.id = _fav.userId');
}
elseif ($requirement->getType() === \Szurubooru\SearchServices\Filters\PostFilter::REQUIREMENT_UPLOADER)
{
$query->innerJoin('users uploader ON uploader.id = userId');
}
parent::decorateQueryFromRequirement($query, $requirement);
}
private function getTags(\Szurubooru\Entities\Post $post)
{
return $this->tagDao->findByPostId($post->getId());
}
private function getUser(\Szurubooru\Entities\Post $post)
{
return $this->userDao->findById($post->getUserId());
}
private function getRelatedPosts(\Szurubooru\Entities\Post $post)
{
$query = $this->fpdo
->from('postRelations')
->where('post1id = :post1id OR post2id = :post2id', [
':post1id' => $post->getId(),
':post2id' => $post->getId()]);
$relatedPostIds = [];
foreach ($query as $arrayEntity)
{
$post1id = intval($arrayEntity['post1id']);
$post2id = intval($arrayEntity['post2id']);
if ($post1id !== $post->getId())
$relatedPostIds[] = $post1id;
if ($post2id !== $post->getId())
$relatedPostIds[] = $post2id;
}
return $this->findByIds($relatedPostIds);
}
private function syncContent(\Szurubooru\Entities\Post $post)
{
$targetPath = $post->getContentPath();
$content = $post->getContent();
if ($content)
$this->fileService->save($targetPath, $content);
else
$this->fileService->delete($targetPath, $content);
$this->thumbnailService->deleteUsedThumbnails($targetPath);
}
private function syncThumbnailSourceContent(\Szurubooru\Entities\Post $post)
{
$targetPath = $post->getThumbnailSourceContentPath();
$content = $post->getThumbnailSourceContent();
if ($content)
$this->fileService->save($targetPath, $content);
else
$this->fileService->delete($targetPath);
$this->thumbnailService->deleteUsedThumbnails($targetPath);
}
private function syncTags(\Szurubooru\Entities\Post $post)
{
$tagNames = array_filter(array_unique(array_map(
function ($tag)
{
return $tag->getName();
},
$post->getTags())));
$this->tagDao->createMissingTags($tagNames);
$tagIds = array_map(
function ($tag)
{
return $tag->getId();
},
$this->tagDao->findByNames($tagNames));
$existingTagRelationIds = array_map(
function ($arrayEntity)
{
return $arrayEntity['tagId'];
},
iterator_to_array($this->fpdo->from('postTags')->where('postId', $post->getId())));
$tagRelationsToInsert = array_diff($tagIds, $existingTagRelationIds);
$tagRelationsToDelete = array_diff($existingTagRelationIds, $tagIds);
foreach ($tagRelationsToInsert as $tagId)
{
$this->fpdo->insertInto('postTags')->values(['postId' => $post->getId(), 'tagId' => $tagId])->execute();
}
foreach ($tagRelationsToDelete as $tagId)
{
$this->fpdo->deleteFrom('postTags')->where('postId', $post->getId())->where('tagId', $tagId)->execute();
}
$this->tagDao->exportJson();
}
private function syncPostRelations(\Szurubooru\Entities\Post $post)
{
$this->fpdo->deleteFrom('postRelations')->where('post1id', $post->getId())->execute();
$this->fpdo->deleteFrom('postRelations')->where('post2id', $post->getId())->execute();
$relatedPostIds = array_filter(array_unique(array_map(
function ($post)
{
if (!$post->getId())
throw new \RuntimeException('Unsaved entities found');
return $post->getId();
},
$post->getRelatedPosts())));
foreach ($relatedPostIds as $postId)
{
$this->fpdo
->insertInto('postRelations')
->values([
'post1id' => $post->getId(),
'post2id' => $postId])
->execute();
}
}
}