tagDao = $tagDao; $this->userDao = $userDao; $this->fileDao = $fileDao; $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 findByTagName($tagName) { $query = $this->fpdo->from('posts') ->disableSmartJoin() ->innerJoin('postTags ON postTags.postId = posts.id') ->innerJoin('tags ON postTags.tagId = tags.id') ->where('tags.name', $tagName); $arrayEntities = iterator_to_array($query); return $this->arrayToEntities($arrayEntities); } public function findByContentChecksum($checksum) { return $this->findOneBy('contentChecksum', $checksum); } protected function afterLoad(Entity $post) { $post->setLazyLoader( Post::LAZY_LOADER_CONTENT, function (Post $post) { return $this->fileDao->load($post->getContentPath()); }); $post->setLazyLoader( Post::LAZY_LOADER_THUMBNAIL_SOURCE_CONTENT, function (Post $post) { return $this->fileDao->load($post->getThumbnailSourceContentPath()); }); $post->setLazyLoader( Post::LAZY_LOADER_USER, function (Post $post) { return $this->getUser($post); }); $post->setLazyLoader( Post::LAZY_LOADER_TAGS, function (Post $post) { return $this->getTags($post); }); $post->setLazyLoader( Post::LAZY_LOADER_RELATED_POSTS, function (Post $post) { return $this->getRelatedPosts($post); }); } protected function afterSave(Entity $post) { $this->syncContent($post); $this->syncThumbnailSourceContent($post); $this->syncTags($post); $this->syncPostRelations($post); } protected function decorateQueryFromRequirement($query, Requirement $requirement) { if ($requirement->getType() === 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() === PostFilter::REQUIREMENT_FAVORITE) { $query->innerJoin('favorites _fav ON _fav.postId = posts.id'); $query->innerJoin('users favoritedBy ON favoritedBy.id = _fav.userId'); } elseif ($requirement->getType() === PostFilter::REQUIREMENT_COMMENT) { $query->innerJoin('comments _comment ON _comment.postId = posts.id'); $query->innerJoin('users commentedBy ON commentedBy.id = _comment.userId'); } elseif ($requirement->getType() === PostFilter::REQUIREMENT_UPLOADER) { $query->innerJoin('users uploader ON uploader.id = userId'); } elseif ($requirement->getType() === PostFilter::REQUIREMENT_USER_SCORE) { $values = $requirement->getValue()->getValues(); $userName = $values[0]; $score = $values[1]; $sql = 'EXISTS ( SELECT 1 FROM scores INNER JOIN users ON scores.userId = users.id WHERE scores.postId = posts.id AND LOWER(users.name) = LOWER(?) AND scores.score = ?)'; if ($requirement->isnegated()) $sql = 'NOT ' . $sql; $query->where($sql, $userName, $score); return; } parent::decorateQueryFromRequirement($query, $requirement); } private function getTags(Post $post) { return $this->tagDao->findByPostId($post->getId()); } private function getUser(Post $post) { return $this->userDao->findById($post->getUserId()); } private function getRelatedPosts(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(Post $post) { $targetPath = $post->getContentPath(); $content = $post->getContent(); if ($content) $this->fileDao->save($targetPath, $content); else $this->fileDao->delete($targetPath, $content); $this->thumbnailService->deleteUsedThumbnails($targetPath); } private function syncThumbnailSourceContent(Post $post) { $targetPath = $post->getThumbnailSourceContentPath(); $content = $post->getThumbnailSourceContent(); if ($content) $this->fileDao->save($targetPath, $content); else $this->fileDao->delete($targetPath); $this->thumbnailService->deleteUsedThumbnails($targetPath); } private function syncTags(Post $post) { $tagIds = array_map( function ($tag) { if (!$tag->getId()) throw new \RuntimeException('Unsaved entities found'); return $tag->getId(); }, $post->getTags()); $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(); } } private function syncPostRelations(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(); } } }