Symfony 4.2在vendor / acme / demo-bundle中创建存储库作为服务

问题描述 投票:0回答:1

我正在处理供应商/目录中的第三方软件包。

我有一个Entity类,看起来像这样:

/**
 * @ORM\Entity(repositoryClass="Acme\DemoBundle\Repository\ArticleRepository")
 * @ORM\Table(name="acme_demo_article")
 */
class Article

还有一个像这样的Repository类:

class ArticleRepository extends ServiceEntityRepository
{
    public function __construct(RegistryInterface $registry)
    {
        parent::__construct($registry, Article::class);
    }
}

这会生成以下错误:

“Acme \ DemoBundle \ Repository \ ArticleRepository”实体存储库实现“Doctrine \ Bundle \ DoctrineBundle \ Repository \ ServiceEntityRepositoryInterface”,但找不到其服务。确保服务存在并标记为“doctrine.repository_service”。

如果我从实体定义中删除repositoryClass,我不再有错误,我可以从我的控制器使用doctrine:

this->getDoctrine()->getRepository(Article::class)->findBy([], null, $limit, ($page - 1) * $limit);

我尝试将存储库添加为捆绑服务定义中的服务,但它不会更改任何内容:

vendor/Acme/demo-bundle/Resources/config/services.yaml

services:
  Acme\DemoBundle\Repository\:
    resource: '../../Repository/ArticleRepository.php'
    autoconfigure: true
    tags: ['doctrine.repository_service']

bin / console debug:autowire或debug:container不会显示服务。

我也尝试添加扩展名:

namespace Acme\BlogBundle\DependencyInjection;


use Symfony\Component\Config\FileLocator;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Extension\Extension;
use Symfony\Component\DependencyInjection\Loader\YamlFileLoader;

class AcmeBlogExtension extends Extension
{
    public function load(array $configs, ContainerBuilder $container)
    {
        $loader = new YamlFileLoader(
            $container,
            new FileLocator(__DIR__.'/../Resources/config')
        );
        $loader->load('services.xml');
    }
}

也没用。我没有被称为扩展名的印象。我尝试添加一个构造函数并在构造函数中转储,但是没有转储的结果。

所以我的问题是我如何从供应商目录中将我的存储库定义为服务?

源代码是完整的:https://github.com/khalid-s/sf4-bundle-test

service doctrine repository symfony4
1个回答
0
投票

经过多次努力,我成功完成了我的任务。我不认为应该这样做,但如果这可以帮助某人......

我在我的DependencyInjection文件夹中添加了:

class AcmeBlogExtension extends Extension
{
    public function load(array $configs, ContainerBuilder $container)
    {
        $loader = new YamlFileLoader(
            $container,
            new FileLocator(__DIR__.'/../Resources/config')
        );
        $loader->load('services.yaml');
    }
}

我创建了一个编译器(这是我努力弄清楚的部分)来注册我的服务

class RepositoryCompiler implements CompilerPassInterface
{
    /**
     * @inheritdoc
     */
    public function process(ContainerBuilder $container)
    {
        $container->register('acme_blog.repository', ArticleRepository::class);
    }
}

我在Bundle类中添加了:

class AcmeBlogBundle extends Bundle
{
    /** @info this function normally is useless */
    public function getContainerExtension()
    {
        // This is only useful if the naming convention is not used
        return new AcmeBlogExtension();
    }

    /**
     * @inheritDoc
     */
    public function build(ContainerBuilder $container)
    {
        $container->addCompilerPass(new RepositoryCompiler());

        parent::build($container);
    }
}

最后服务本身:

services:
  Acme\BlogBundle\Repository\:
    resource: '../../Repository/*Repository.php'
    autoconfigure: true
    autowire: true
    tags: ['doctrine.repository_service']

autoconfigure和autowire是无用的,因为我调试时不考虑它们:容器看起来像这样:

php bin/console debug:container acme

Information for Service "acme_blog.article.repository"
=======================================================

 ---------------- -----------------------------------------------
  Option           Value
 ---------------- -----------------------------------------------
  Service ID       acme_blog.article.repository
  Class            Acme\BlogBundle\Repository\ArticleRepository
  Tags             doctrine.repository_service
  Public           yes
  Synthetic        no
  Lazy             no
  Shared           yes
  Abstract         no
  Autowired        no
  Autoconfigured   no
 ---------------- -----------------------------------------------

一个非常重要的说明让我失去了很多时间:

每次更改服务后都要清除缓存。即使在开发模式下,每次刷新后都不会重新加载

© www.soinside.com 2019 - 2024. All rights reserved.