Я пытаюсь разделить один большой service.yaml на несколько файлов меньшего размера. В origin service.yaml у меня было
Сервисы:
_defaults:
autowire: true
autoconfigure: true
public: false
App\Domain\Country\Infrastructure\Repository\CountryRepository:
public: true
class: App\Domain\Country\Infrastructure\Repository\CountryRepository
factory: ["@doctrine.orm.default_entity_manager", getRepository]
arguments: [App\Domain\Country\Entity\Country]
Затем я добавил импорт в начало service.yam
imports:
- {resource: services/repositories.yaml}
repositories.yaml
services:
_defaults:
autowire: true
autoconfigure: true
public: true
App\Domain\Country\Infrastructure\Repository\CountryRepository:
factory: ["@doctrine.orm.default_entity_manager", getRepository]
arguments: [App\Domain\Country\Entity\Country]
После этого я начал получать ошибку
Cannot autowire service "App\Domain\Country\Infrastructure\Repository\Count
ryRepository": argument "$class" of method "Doctrine\ORM\EntityRepository::
__construct()" references class "Doctrine\ORM\Mapping\ClassMetadata" but no
such service exists.
Что там не так?
спасибо, но то же самое :(
Отредактируйте свой вопрос и поместите новые файлы
Я добавил ссылку на pastebin
Версия Symfony?
последний. Symfony 4
Если вы используете автоматическое подключение, вам вообще нужно предоставить определение репо? Достаточно просто иметь его в качестве аргумента конструктора ... Просто убедитесь, что у вашего объекта Country есть аннотация репозитория для класса CountryRepository
потому что я использую одну стороннюю библиотеку, куда я отправлял контейнер напрямую для таких запросов. Наверное, можно по-другому сделать? $ repository = $ this-> container-> get ('App \ Domain \ Billing \ Counterparty \ Infra structure \ Repository \ CounterpartyReposit ory'); return $ repository-> getById ($ input);
Вам следует избегать отправки всего контейнера целиком. Но это, вероятно, сработает. В какой-то момент вы можете столкнуться с рекурсивными проблемами, особенно если вы начнете использовать запрос и т. д. Я бы постарался избежать отправки всего контейнера в какой-либо класс.
Да, я знаю, что это плохая практика :( Но я понятия не имею, как обернуть стороннюю библиотеку для поддержки DI
Внедрить стороннюю библиотеку в службу вместе с любыми другими зависимостями и использовать ее изолированно, а не пытаться подключить ее извне.
позволь мне проверить, но это не быстро
@akor - все файлы yaml в config / services.yaml автоматически загружаются Kernel.php, поэтому нет необходимости в явном импорте. Сообщение об ошибке исходит не из вашей конфигурации. Скорее у вас есть что-то еще, которое сканирует ваш каталог инфраструктуры и пытается создать службу из всего, что находит. Проверьте остальную часть настройки autowire и выясните, как исключить каталог. Или сделайте так, как предлагали другие, и просто расширите ServiceEntityRepository.
@akor ты нашел какое-нибудь решение?




Вместо этого используйте именованные аргументы:
repositories.yaml
services:
App\Domain\Country\Infrastructure\Repository\CountryRepository:
factory: ["@doctrine.orm.default_entity_manager", getRepository]
arguments:
$class: '@App\Domain\Country\Entity\Country'
Внутри repositories.yaml. Когда у меня есть все объявления репозитория в services.yaml, он работает нормально. Но как только я разбиваю его, я получаю ошибку.
попробуйте переместить imports в самую верхнюю строку над параметрами
Вам не нужно определять репозиторий для целей автоматического подключения.
services.yaml:
services:
_defaults:
autowire: true
autoconfigure: true
public: false
App\:
resource: '../src/*'
exclude: '../src/{DependencyInjection,Entity,Migrations,Tests,Kernel.php}'
App\Controller\:
resource: '../src/Controller'
tags: ['controller.service_arguments']
Организация \ Страна:
<?php
namespace App\Entity;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Entity(repositoryClass = "App\Repository\CountryRepository")
*/
class Country
{
/**
* @ORM\Id()
* @ORM\GeneratedValue()
* @ORM\Column(type = "integer")
*/
private $id;
/**
* @ORM\Column(type = "string", length=255)
*/
private $name;
public function getId(): ?int
{
return $this->id;
}
public function getName(): ?string
{
return $this->name;
}
}
Репозиторий \ CountryRepository:
<?php
namespace App\Repository;
use App\Entity\Country;
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Symfony\Bridge\Doctrine\RegistryInterface;
/**
* @method Country|null find($id, $lockMode = null, $lockVersion = null)
* @method Country|null findOneBy(array $criteria, array $orderBy = null)
* @method Country[] findAll()
* @method Country[] findBy(array $criteria, array $orderBy = null, $limit = null, $offset = null)
*/
class CountryRepository extends ServiceEntityRepository
{
public function __construct(RegistryInterface $registry)
{
parent::__construct($registry, Country::class);
}
}
И напоследок ваш сервис:
<?php
namespace App\Service;
use App\Repository\CountryRepository;
class ExampleService
{
/**
* @var CountryRepository
*/
private $repository;
/**
* @param CountryRepository $repository
*/
public function __construct(CountryRepository $repository)
{
$this->repository = $repository;
}
}
Автоматическое подключение увидит, что вы внедрили этот CountryRepository в конструктор ExampleService, и обработает все остальное.
Ваш вариант попробую чуть позже. Но поправьте меня, будет ли это работать с $ repository = $ this-> container-> get ('App \ Domain \ Billing \ Counterparty \ Infra structure \ Repository \ CounterpartyReposit ory'); ? Как в таком случае получить репозиторий из контейнера?
Это должно работать (или использовать префикс @ и использовать в определении службы), но, как уже упоминалось, звучит неприятно, потому что контейнер вводится. Трудно дать совет, не зная точной проблемы, но вы должны иметь возможность настроить свою стороннюю библиотеку и передать то, что вам нужно, вместо того, чтобы делать это.
удалить
_defaultsизrepositories/yaml. Также сделайте отступApp\Domain\Country\Infrastructure\Repository\CountryRepository