Multifactory with parameters
- rp
- Member | 20
tldr;
It seems it is possible to pass parameters to generated factories (i.e.
create($param)
) but not to generated multifactories (i.e.
createFoo($param)
).
It is possible to pass parameters to factory method definitions like this:
In presenter I can create FooControl
object like this:
So far so good. The problem comes when I try to group more factories in one interface like this
In presenter I would be able to get both FooControl
and
BarControl
objects by using only one injected
factory, like this:
However, Nette says: Service ‘XXX’: Method MultiFactory::createFooControl() does not meet the requirements: is create($name), get($name), create*() or get*() and is non-static.
I have this neon configuration in place for MultiFactory:
Does it mean that parameters cannot be passed in multifactory methods? Or do I have an error in my neon config and params have to be passed in another way? Thanks.
Last edited by rp (2023-07-28 09:40)
- dakur
- Member | 493
@rp The problem with setting ID after creating the object is that the object is in invalid state. You can render it but having no ID it will crash. So it's a workaround rather than proper solution to the problem.
I would go with rewriting the interface into usual class. It doesn't pollute presenters either and you have freedom to express yourself. It's a little bit more writing but my experience says it's worth it.
Last edited by dakur (2023-07-27 10:53)
- Marek Bartoš
- Nette Blogger | 1297
Or just use the regular generated factories and inject them into hand-written multi factory.
Last edited by Marek Bartoš (2023-07-28 13:50)
- rp
- Member | 20
@MarekBartoš
Let me describe an issue that I experience using your solution.
I move $id
parameter into constructor:
Let's create FooControl:
I receive an Nette\DI\ServiceCreationException
:
Service of type FooControl: Parameter $id in FooControl::__construct() has no
class type or default value, so its value must be specified.
I could assign a value to $id in constructor to make it work:
However, that is far from ideal. What do you think?
Edit:
There maybe a problem with my config.neon
:
When ContainerBuilder
tries to build a container it bumps into
FooControl
. It looks at its constructor and finds out there are
three parameters he should inject:
- DummyService1
- DummyService2
- $id
It can find and inject the first two. Third one is unknown. That might be the source of the problem.
Last edited by rp (2023-07-28 12:57)
- rp
- Member | 20
@dakur
That solution could get problematic very fast. Imagine you have 100 controls, each with its own set of dependencies:
In that case, you will ask for 200 dependencies in your constructor, 198 of which are useless for each control (as each control only needs its 2 specific dependencies).
It might look far fetched, however, there is a very same problem with the example you provided:
You might rely on a person who removed createFooControl
that he
will also check for related services in the constructor and remove them.
Therefore I made up that far fetched example to demonstrate that it most likely
won't happen.
What do you think?
- Marek Bartoš
- Nette Blogger | 1297
You are supposed to create a factory which creates control and register it as a service. You are registering control instead of factory. Check my example, I updated it a bit.
- Marek Bartoš
- Nette Blogger | 1297
You might rely on a person who removed createFooControl that he will also check for related services in the constructor and remove them.
That's problem of the programmer and not the actual code. However, unused services are easily checked by tools – PHPStorm and PHPStan report unused private properties.
- rp
- Member | 20
@MarekBartoš
I see where the misunderstanding is coming from.
Constructor of FooControl
in my example:
Constructor of FooControl
in your example:
Your example omits Dummy services, therefore FooControl
does not
have to be registered in config.neon.
My FooControl
requires Dummy services to be injected so
I register it in config.neon (as well as FooControlFactory
) and
receive Nette\DI\ServiceCreationException
. I am almost sure that
I receive that exception because DI tries to inject $id parameter that cannot
be found.
One way around it is @dakur suggestion to have services injected in MultiFactory constructor and pass them to factory methods:
At this point I should let you know that I'm more than happy with my solution and there is no need, unless you want to discuss it in more detail, to spend time on it. Thank you @dakur and @MarekBartoš for your time and your suggestions.
- Marek Bartoš
- Nette Blogger | 1297
Services are automatically passed to control from generated factory, you don't have to do anything except defining them in control constructor.
- rp
- Member | 20
@MarekBartoš
Yes, you are correct!
The mistake was in registering FooControl
in DI. That threw that
Exception. I thought it has to be registered otherwise nothing will be
injected. But it will be automatically, exactly as you said.
So your suggestion really works and the advatage is that I can pass $id in constructor.
Presenter.php
MultiFactory.php
FooControlFactory.php
FooControl.php
config.neon
Thanks @MarekBartoš for your time and effort.