提问者:小点点

如何删除Symfony服务?(奏鸣曲分类)


我使用SonataAdminBundle与MediaBundle有一个依赖分类包。默认情况下,分类包会在后端管理员管理中添加类别、标签、集合和上下文,但由于我的应用程序不使用它们,所以我想从菜单和管理面板中删除它们。

我以前从未删除过服务,所以我不知道如何删除。

必须有办法从SonataClassificationBundle/Resources/config/admin中删除这些服务。xml,显然没有修改文件本身,因为它是一个供应商文件。

 <services>
        <service id="sonata.classification.admin.category" class="%sonata.classification.admin.category.class%">
            <tag name="sonata.admin" manager_type="orm" group="sonata_classification" label="label_categories"  label_catalogue="%sonata.classification.admin.category.translation_domain%" label_translator_strategy="sonata.admin.label.strategy.underscore" />
            <argument />
            <argument>%sonata.classification.admin.category.entity%</argument>
            <argument>%sonata.classification.admin.category.controller%</argument>
            <argument type="service" id="sonata.classification.manager.context" />

            <call method="setTranslationDomain">
                <argument>%sonata.classification.admin.category.translation_domain%</argument>
            </call>

            <call method="setTemplates">
                <argument type="collection">
                    <argument key="list">SonataClassificationBundle:CategoryAdmin:list.html.twig</argument>
                </argument>
            </call>
        </service>

        <service id="sonata.classification.admin.tag" class="%sonata.classification.admin.tag.class%">
            <tag name="sonata.admin" manager_type="orm" group="sonata_classification" label="label_tags"  label_catalogue="%sonata.classification.admin.tag.translation_domain%" label_translator_strategy="sonata.admin.label.strategy.underscore" />
            <argument />
            <argument>%sonata.classification.admin.tag.entity%</argument>
            <argument>%sonata.classification.admin.tag.controller%</argument>

            <call method="setTranslationDomain">
                <argument>%sonata.classification.admin.tag.translation_domain%</argument>
            </call>
        </service>

        <service id="sonata.classification.admin.collection" class="%sonata.classification.admin.collection.class%">
            <tag name="sonata.admin" manager_type="orm" group="sonata_classification" label="label_collections"  label_catalogue="%sonata.classification.admin.collection.translation_domain%" label_translator_strategy="sonata.admin.label.strategy.underscore" />
            <argument />
            <argument>%sonata.classification.admin.collection.entity%</argument>
            <argument>%sonata.classification.admin.collection.controller%</argument>

            <call method="setTranslationDomain">
                <argument>%sonata.classification.admin.collection.translation_domain%</argument>
            </call>
        </service>

        <service id="sonata.classification.admin.context" class="%sonata.classification.admin.context.class%">
            <tag name="sonata.admin" manager_type="orm" group="sonata_classification" label="label_contexts"  label_catalogue="%sonata.classification.admin.context.translation_domain%" label_translator_strategy="sonata.admin.label.strategy.underscore" />
            <argument />
            <argument>%sonata.classification.admin.context.entity%</argument>
            <argument>%sonata.classification.admin.context.controller%</argument>

            <call method="setTranslationDomain">
                <argument>%sonata.classification.admin.context.translation_domain%</argument>
            </call>
        </service>
    </services>

或者也许有办法将它们从索纳塔管理池中删除?因为它们被标记为sonata.admin

编辑

使用Sonata Easy Extends,我扩展了包并添加了一个编译器过程:

class ApplicationSonataClassificationBundle extends Bundle
{
    /**
     * {@inheritdoc}
     */
    public function getParent()
    {
        return 'SonataClassificationBundle';
    }

    public function build(ContainerBuilder $container)
    {
        parent::build($container);

        $container->addCompilerPass(new CustomCompilerPass());
    }

}

编译器过程如下所示

class CustomCompilerPass implements CompilerPassInterface
{
    public function process(ContainerBuilder $container)
    {
        $container->removeDefinition('sonata.classification.admin.category');
    }
}

但是我越来越

  You have requested a non-existent service "sonata.classification.admin.cate  
  gory" in . (which is being imported from "E:\svn\parkresort\app/config\rout  
  ing/sonata.yml").

那个文件正在导入整个奏鸣曲包的路由

admin:
    resource: '@SonataAdminBundle/Resources/config/routing/sonata_admin.xml'
    prefix: /admin

_sonata_admin:
    resource: .
    type: sonata_admin
    prefix: /admin

#sonata media
media:
    resource: '@SonataMediaBundle/Resources/config/routing/media.xml'
    prefix: /media

我猜sonata管理员甚至在从容器中删除该服务后也在使用该服务。我怎样才能改变这一点?

编辑2

我做到了!我必须将编译器传递放在Sonata管理扩展(及其名称空间)中,而不是放在Sonata媒体中。显然,还扩展了管理包。后来一切都很好。

我真正不明白的是,当原始包在我的扩展包之后加载时,为什么它会工作:

//AppKernel.php
new ApplicationSonataAdminBundle(),//extended
new Sonata\AdminBundle\SonataAdminBundle(),

这是奇怪的。


共3个答案

匿名用户

由于我们讨论的是一个包,它是应用程序的依赖项,因此您无法控制它定义了哪些服务并将其注册到应用程序中。

我相信可以使用ContainerBuilder::removeDefinition()删除它们。它也适用于其他捆绑包中定义的服务,因此也适用于Sonata捆绑包。

您可以在Symfony的文档中看到一个示例,该示例具体说明了如何放置此代码以及如何访问ContainerBuilder对象。

但是,我建议你不要这样做。即使您不使用某些服务,它们也不会打扰您,而且考虑到Symfony处理服务的方式,它们不会在生产中造成任何性能问题,我保证。

匿名用户

您需要创建一个编译器过程来删除定义。为了能够做到这一点,你必须确保你的捆绑声明后奏鸣曲一。如果您不能控制它,那么扩展sonatabundle并在其中定义编译器传递。

匿名用户

半年后,我发现了一种更干净的方法,可以将服务保存在容器中(因为它们在某些地方是需要的),但不会在管理面板上显示它们。

   $definitionsNames = array('sonata.media.admin.media', 'sonata.media.admin.gallery_has_media', 'sonata.media.admin.gallery',
        'sonata.classification.admin.category','sonata.classification.admin.tag','sonata.classification.admin.context','sonata.classification.admin.collection');

    foreach ($definitionsNames as $definitionName) {
        $definition = $container->getDefinition($definitionName);

        $tags = $definition->getTags();

        $tags['sonata.admin'][0]['show_in_dashboard'] = false;
        $definition->setTags($tags);

    }

不幸的是,管理路由仍然可用。对我来说不是问题,但我相信有办法消除它们。问题是media_widget包含到管理媒体编辑路由的链接,因此需要覆盖以不再显示它。然后需要重写Media、Gallery和GHM管理员,并重写configure路线()函数并删除所有路由。然后我相信你不能通过管理员访问任何东西,但是应用程序仍然可以使用管理员服务,如果它们在任何地方都需要的话。

这样,我仍然遵循Radu的建议,不要从容器中删除服务。