commit c63206aa71c269d0afba3de199b4fce95fb1f963 Author: Christopher Vagnetoft Date: Sat Aug 30 23:12:22 2025 +0200 Initial commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..cbaa689 --- /dev/null +++ b/.gitignore @@ -0,0 +1,3 @@ +/vendor/ +/.phpunit.* +/composer.lock diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..322e434 --- /dev/null +++ b/LICENSE @@ -0,0 +1,309 @@ +GNU GENERAL PUBLIC LICENSE +Version 2, June 1991 + +Copyright (C) 1989, 1991 Free Software Foundation, Inc. +51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA + +Everyone is permitted to copy and distribute verbatim copies of this license +document, but changing it is not allowed. + +Preamble + +The licenses for most software are designed to take away your freedom to share +and change it. By contrast, the GNU General Public License is intended to +guarantee your freedom to share and change free software--to make sure the +software is free for all its users. This General Public License applies to most +of the Free Software Foundation's software and to any other program whose +authors commit to using it. (Some other Free Software Foundation software is +covered by the GNU Lesser General Public License instead.) You can apply it to +your programs, too. + +When we speak of free software, we are referring to freedom, not price. Our +General Public Licenses are designed to make sure that you have the freedom to +distribute copies of free software (and charge for this service if you wish), +that you receive source code or can get it if you want it, that you can change +the software or use pieces of it in new free programs; and that you know you can +do these things. + +To protect your rights, we need to make restrictions that forbid anyone to deny +you these rights or to ask you to surrender the rights. These restrictions +translate to certain responsibilities for you if you distribute copies of the +software, or if you modify it. + +For example, if you distribute copies of such a program, whether gratis or for a +fee, you must give the recipients all the rights that you have. You must make +sure that they, too, receive or can get the source code. And you must show them +these terms so they know their rights. + +We protect your rights with two steps: (1) copyright the software, and (2) offer +you this license which gives you legal permission to copy, distribute and/or +modify the software. + +Also, for each author's protection and ours, we want to make certain that +everyone understands that there is no warranty for this free software. If the +software is modified by someone else and passed on, we want its recipients to +know that what they have is not the original, so that any problems introduced by +others will not reflect on the original authors' reputations. + +Finally, any free program is threatened constantly by software patents. We wish +to avoid the danger that redistributors of a free program will individually +obtain patent licenses, in effect making the program proprietary. To prevent +this, we have made it clear that any patent must be licensed for everyone's free +use or not licensed at all. + +The precise terms and conditions for copying, distribution and modification +follow. + +TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + +0. This License applies to any program or other work which contains a notice +placed by the copyright holder saying it may be distributed under the terms of +this General Public License. The "Program", below, refers to any such program or +work, and a "work based on the Program" means either the Program or any +derivative work under copyright law: that is to say, a work containing the +Program or a portion of it, either verbatim or with modifications and/or +translated into another language. (Hereinafter, translation is included without +limitation in the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not covered by +this License; they are outside its scope. The act of running the Program is not +restricted, and the output from the Program is covered only if its contents +constitute a work based on the Program (independent of having been made by +running the Program). Whether that is true depends on what the Program does. + +1. You may copy and distribute verbatim copies of the Program's source code as +you receive it, in any medium, provided that you conspicuously and appropriately +publish on each copy an appropriate copyright notice and disclaimer of warranty; +keep intact all the notices that refer to this License and to the absence of any +warranty; and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and you may at +your option offer warranty protection in exchange for a fee. + +2. You may modify your copy or copies of the Program or any portion of it, thus +forming a work based on the Program, and copy and distribute such modifications +or work under the terms of Section 1 above, provided that you also meet all of +these conditions: + + a) You must cause the modified files to carry prominent notices stating +that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in whole or +in part contains or is derived from the Program or any part thereof, to be +licensed as a whole at no charge to all third parties under the terms of this +License. + + c) If the modified program normally reads commands interactively when run, +you must cause it, when started running for such interactive use in the most +ordinary way, to print or display an announcement including an appropriate +copyright notice and a notice that there is no warranty (or else, saying that +you provide a warranty) and that users may redistribute the program under these +conditions, and telling the user how to view a copy of this License. (Exception: +if the Program itself is interactive but does not normally print such an +announcement, your work based on the Program is not required to print an +announcement.) + +These requirements apply to the modified work as a whole. If identifiable +sections of that work are not derived from the Program, and can be reasonably +considered independent and separate works in themselves, then this License, and +its terms, do not apply to those sections when you distribute them as separate +works. But when you distribute the same sections as part of a whole which is a +work based on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the entire whole, +and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest your +rights to work written entirely by you; rather, the intent is to exercise the +right to control the distribution of derivative or collective works based on the +Program. + +In addition, mere aggregation of another work not based on the Program with the +Program (or with a work based on the Program) on a volume of a storage or +distribution medium does not bring the other work under the scope of this +License. + +3. You may copy and distribute the Program (or a work based on it, under Section +2) in object code or executable form under the terms of Sections 1 and 2 above +provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable source +code, which must be distributed under the terms of Sections 1 and 2 above on a +medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three years, to +give any third party, for a charge no more than your cost of physically +performing source distribution, a complete machine-readable copy of the +corresponding source code, to be distributed under the terms of Sections 1 and 2 +above on a medium customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer to +distribute corresponding source code. (This alternative is allowed only for +noncommercial distribution and only if you received the program in object code +or executable form with such an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for making +modifications to it. For an executable work, complete source code means all the +source code for all modules it contains, plus any associated interface +definition files, plus the scripts used to control compilation and installation +of the executable. However, as a special exception, the source code distributed +need not include anything that is normally distributed (in either source or +binary form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component itself +accompanies the executable. + +If distribution of executable or object code is made by offering access to copy +from a designated place, then offering equivalent access to copy the source code +from the same place counts as distribution of the source code, even though third +parties are not compelled to copy the source along with the object code. + +4. You may not copy, modify, sublicense, or distribute the Program except as +expressly provided under this License. Any attempt otherwise to copy, modify, +sublicense or distribute the Program is void, and will automatically terminate +your rights under this License. However, parties who have received copies, or +rights, from you under this License will not have their licenses terminated so +long as such parties remain in full compliance. + +5. You are not required to accept this License, since you have not signed it. +However, nothing else grants you permission to modify or distribute the Program +or its derivative works. These actions are prohibited by law if you do not +accept this License. Therefore, by modifying or distributing the Program (or any +work based on the Program), you indicate your acceptance of this License to do +so, and all its terms and conditions for copying, distributing or modifying the +Program or works based on it. + +6. Each time you redistribute the Program (or any work based on the Program), +the recipient automatically receives a license from the original licensor to +copy, distribute or modify the Program subject to these terms and conditions. +You may not impose any further restrictions on the recipients' exercise of the +rights granted herein. You are not responsible for enforcing compliance by third +parties to this License. + +7. If, as a consequence of a court judgment or allegation of patent infringement +or for any other reason (not limited to patent issues), conditions are imposed +on you (whether by court order, agreement or otherwise) that contradict the +conditions of this License, they do not excuse you from the conditions of this +License. If you cannot distribute so as to satisfy simultaneously your +obligations under this License and any other pertinent obligations, then as a +consequence you may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by all those +who receive copies directly or indirectly through you, then the only way you +could satisfy both it and this License would be to refrain entirely from +distribution of the Program. + +If any portion of this section is held invalid or unenforceable under any +particular circumstance, the balance of the section is intended to apply and the +section as a whole is intended to apply in other circumstances. + +It is not the purpose of this section to induce you to infringe any patents or +other property right claims or to contest validity of any such claims; this +section has the sole purpose of protecting the integrity of the free software +distribution system, which is implemented by public license practices. Many +people have made generous contributions to the wide range of software +distributed through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing to +distribute software through any other system and a licensee cannot impose that +choice. + +This section is intended to make thoroughly clear what is believed to be a +consequence of the rest of this License. + +8. If the distribution and/or use of the Program is restricted in certain +countries either by patents or by copyrighted interfaces, the original copyright +holder who places the Program under this License may add an explicit +geographical distribution limitation excluding those countries, so that +distribution is permitted only in or among countries not thus excluded. In such +case, this License incorporates the limitation as if written in the body of this +License. + +9. The Free Software Foundation may publish revised and/or new versions of the +General Public License from time to time. Such new versions will be similar in +spirit to the present version, but may differ in detail to address new problems +or concerns. + +Each version is given a distinguishing version number. If the Program specifies +a version number of this License which applies to it and "any later version", +you have the option of following the terms and conditions either of that version +or of any later version published by the Free Software Foundation. If the +Program does not specify a version number of this License, you may choose any +version ever published by the Free Software Foundation. + +10. If you wish to incorporate parts of the Program into other free programs +whose distribution conditions are different, write to the author to ask for +permission. For software which is copyrighted by the Free Software Foundation, +write to the Free Software Foundation; we sometimes make exceptions for this. +Our decision will be guided by the two goals of preserving the free status of +all derivatives of our free software and of promoting the sharing and reuse of +software generally. + +NO WARRANTY + +11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY FOR THE +PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED +IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS +IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT +NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A +PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE +PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + +12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL +ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE +PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, +SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY +TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING +RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF +THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER +PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + +END OF TERMS AND CONDITIONS + +How to Apply These Terms to Your New Programs + +If you develop a new program, and you want it to be of the greatest possible use +to the public, the best way to achieve this is to make it free software which +everyone can redistribute and change under these terms. + +To do so, attach the following notices to the program. It is safest to attach +them to the start of each source file to most effectively convey the exclusion +of warranty; and each file should have at least the "copyright" line and a +pointer to where the full notice is found. + + one line to give the program's name and an idea of what it does. Copyright +(C) yyyy name of author + + This program is free software; you can redistribute it and/or modify it +under the terms of the GNU General Public License as published by the Free +Software Foundation; either version 2 of the License, or (at your option) any +later version. + + This program is distributed in the hope that it will be useful, but WITHOUT +ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS +FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. + + You should have received a copy of the GNU General Public License along +with this program; if not, write to the Free Software Foundation, Inc., 51 +Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. Also add information +on how to contact you by electronic and paper mail. + +If the program is interactive, make it output a short notice like this when it +starts in an interactive mode: + + Gnomovision version 69, Copyright (C) year name of author Gnomovision comes +with ABSOLUTELY NO WARRANTY; for details type `show w'. This is free software, +and you are welcome to redistribute it under certain conditions; type `show c' +for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, the commands you use may be +called something other than `show w' and `show c'; they could even be +mouse-clicks or menu items--whatever suits your program. + +You should also get your employer (if you work as a programmer) or your school, +if any, to sign a "copyright disclaimer" for the program, if necessary. Here is +a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the program +`Gnomovision' (which makes passes at compilers) written by James Hacker. + +signature of Ty Coon, 1 April 1989 Ty Coon, President of Vice diff --git a/README.md b/README.md new file mode 100644 index 0000000..4997492 --- /dev/null +++ b/README.md @@ -0,0 +1,5 @@ +# NoccyLabs Hotwire + +This is an autowiring DI container library. It is not zero-conf; you still have +to specify the directories containing classes to add to the container, and +annotate the classes with the `#[AsService]` attribute. diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..fb4fd0b --- /dev/null +++ b/composer.json @@ -0,0 +1,28 @@ +{ + "name": "noccylabs/hotwire", + "description": "A flexible container for long running applications and daemons", + "type": "library", + "license": "GPL-2.0-or-later", + "autoload": { + "psr-4": { + "NoccyLabs\\Hotwire\\": "src/" + } + }, + "autoload-dev": { + "psr-4": { + "": "tests/classes/" + } + }, + "authors": [ + { + "name": "Christopher Vagnetoft", + "email": "labs@noccy.com" + } + ], + "require": { + "psr/container": "^2.0" + }, + "require-dev": { + "phpunit/phpunit": "^12.3" + } +} diff --git a/phpunit.xml b/phpunit.xml new file mode 100644 index 0000000..ce0df9c --- /dev/null +++ b/phpunit.xml @@ -0,0 +1,25 @@ + + + + + tests + + + + + + src + + + diff --git a/src/Attributes/AsService.php b/src/Attributes/AsService.php new file mode 100644 index 0000000..de8dd93 --- /dev/null +++ b/src/Attributes/AsService.php @@ -0,0 +1,33 @@ +alias ? $this->alias : $className; + if ($this->shared) { + $container->addShared($name, $className); + } else { + $container->add($name, $className); + } + } +} diff --git a/src/BuilderInterface.php b/src/BuilderInterface.php new file mode 100644 index 0000000..e472673 --- /dev/null +++ b/src/BuilderInterface.php @@ -0,0 +1,12 @@ + */ + private array $builders = []; + + /** @var array */ + private array $services = []; + + /** @var array Parameters to bind to variables in constructors */ + private array $parameters = []; + + /** @var list The order in which to load services to keep dependencies from breaking */ + private array $loadOrder = []; + + private DependencyResolver $resolver; + + public function __construct( + // ?string $magic = null + ) + { + $this->resolver = new DependencyResolver(); + } + + public function has(string $id): bool + { + return array_key_exists($id, $this->services); + } + + public function get(string $id): object + { + $def = $this->services[$id]; + if ($def->shared && $def->instance) return $def->instance; + $inst = $def->newInstance($this, $this->resolver); + if ($def->shared) { + $def->instance = $inst; + } + return $inst; + } + + public function add(string $id, ?string $alias = null, ?object $instance = null): Definition + { + // printf("container:add{id=%s,instance=%s}\n", $id, is_object($instance) ? spl_object_hash($instance) : $instance); + $this->services[$id] = new Definition( + id: $id, + instance: $instance, + shared: $instance ? true : false, + ); + return $this->services[$id]; + } + + public function addShared(string $id, ?string $alias = null, ?object $instance = null): Definition + { + // printf("container:add{shared=true,id=%s,instance=%s}\n", $id, is_object($instance) ? spl_object_hash($instance) : $instance); + $this->services[$id] = new Definition( + id: $id, + instance: $instance, + shared: true, + ); + return $this->services[$id]; + } + + public function addBuilder(BuilderInterface $builder): void + { + $this->builders[] = $builder; + } + + public function setParameter(string $name, mixed $value): void + { + $this->parameters[$name] = $value; + } + + public function setParameters(array $parameters): void + { + $this->parameters = $parameters; + } + + public function hasParameter(string $name): bool + { + return array_key_exists($name, $this->parameters); + } + + public function getParameter(string $name, mixed $default = null): mixed + { + return $this->parameters[$name] ?? $default; + } + + public function build(): void + { + foreach ($this->builders as $builder) { + $builder->buildServices($this); + } + $this->loadEarlyServices(); + } + + private function loadEarlyServices(): void + { + foreach ($this->services as $service) { + if ($service->early && $service->shared && !$service->instance) { + $service->instance = $service->newInstance($this, $this->resolver); + } + } + } +} + diff --git a/src/Definition.php b/src/Definition.php new file mode 100644 index 0000000..1c25099 --- /dev/null +++ b/src/Definition.php @@ -0,0 +1,61 @@ +shared; + } + + public function hasInstance(): bool + { + return null !== $this->instance; + } + + public function getConstructorArguments(): array + { + $rc = new ReflectionClass($this->id); + if (!$rc->hasMethod('__construct')) { + return []; + } + $rm = $rc->getMethod('__construct'); + return $rm->getParameters(); + } + + public function newInstance(Container $container, DependencyResolver $resolver): object + { + + $ctargs = $resolver->autowireConstructorArguments($container, $this->id); + // printf("new: %s (%s)\n", $this->id, join(", ", array_map(fn($v)=>var_export($v,true), $ctargs))); + return new $this->id(...$ctargs); + } + + public function getArguments(): array + { + return $this->arguments; + } + + public function addArguments(array $args): self + { + $this->arguments = [ ...$this->arguments, ...$args ]; + return $this; + } + +} diff --git a/src/DependencyResolver.php b/src/DependencyResolver.php new file mode 100644 index 0000000..a681c11 --- /dev/null +++ b/src/DependencyResolver.php @@ -0,0 +1,126 @@ +autowire($container, [$controller,$method])( + * request: $request + * ); + * + * @param Container $container + * @param callable $callable + * @return callable + */ + public function autowire(Container $container, callable $callable): callable + { + return function(...$args) use ($container, $callable) { + $callableArgs = $this->autowireArguments($container, $callable); + foreach ($callableArgs as $name => $value) { + if (isset($args[$name])) { + $callableArgs[$name] = $args[$name]; + } + } + return call_user_func($callable, ...$callableArgs); + }; + } + + /** + * Autowire the arguments for a provided callable + * + * @param Container $container + * @param callable $callable + * @return array + */ + public function autowireArguments(Container $container, callable $callable): array + { + $arguments = []; + + $rc = new ReflectionFunction($callable); + $params = $rc->getParameters(); + + foreach ($params as $param) { + $name = $param->getName(); + $type = $param->getType(); + if ($type->isBuiltin()) { + if ($container->hasParameter($name)) { + // TODO match type? (intParameter, stringParameter, ..) + $arguments[$name] = $container->getParameter($name); + } else { + $arguments[$name] = $param->isDefaultValueAvailable() + ? $param->getDefaultValue() + : null + ; + } + } else { + if ($type instanceof ReflectionNamedType) { + $fqcn = $type->getName(); + if ($container->has($fqcn)) { + $arguments[$name] = $container->get($fqcn); + } else { + $arguments[$name] = null; + } + } else { + $arguments[$name] = null; + } + } + } + + return $arguments; + } + + /** + * Autowire the arguments for a provided callable + * + * @param Container $container + * @param callable $callable + * @return array + */ + public function autowireConstructorArguments(Container $container, string $class): array + { + $arguments = []; + + $rc = (new ReflectionClass($class))->getMethod('__construct'); + $params = $rc->getParameters(); + + foreach ($params as $param) { + $name = $param->getName(); + $type = $param->getType(); + if ($type->isBuiltin()) { + if ($container->hasParameter($name)) { + // TODO match type? (intParameter, stringParameter, ..) + $arguments[$name] = $container->getParameter($name); + } else { + $arguments[$name] = $param->isDefaultValueAvailable() + ? $param->getDefaultValue() + : null + ; + } + } else { + if ($type instanceof ReflectionNamedType) { + $fqcn = $type->getName(); + if ($container->has($fqcn)) { + $arguments[$name] = $container->get($fqcn); + } else { + $arguments[$name] = null; + } + } else { + $arguments[$name] = null; + } + } + } + + return $arguments; + } + +} diff --git a/src/DirectoryBuilder.php b/src/DirectoryBuilder.php new file mode 100644 index 0000000..fc56283 --- /dev/null +++ b/src/DirectoryBuilder.php @@ -0,0 +1,54 @@ +$info) { + if (fnmatch("*.php", $filename)) { + $relFile = str_replace($directory, "", $filename); + $expect = $nsroot . strtr(substr($relFile, 0, -4), [ "/" => "\\" ]); + $this->scanFile($filename, $expect); + } + } + } + + private function scanFile(string $filename, string $expect): void + { + // echo $expect."\n"; + if (class_exists($expect)) { + $rc = new \ReflectionClass($expect); + $ra = $rc->getAttributes(Attributes\AsService::class); + if (count($ra) == 0) return; + $this->foundClasses[$expect] = (object)[ + 'class' => $rc->getName(), + 'service' => $ra[0]->newInstance() + ]; + } + } + + public function buildServices(Container $container): void + { + foreach ($this->foundClasses as $s) { + $s->service->define($s->class, $container); + } + } + +} + diff --git a/src/StaticBuilder.php b/src/StaticBuilder.php new file mode 100644 index 0000000..56aab11 --- /dev/null +++ b/src/StaticBuilder.php @@ -0,0 +1,29 @@ +services[$id] = $arguments; + } + + public function buildServices(Container $container): void + { + foreach ($this->services as $id=>$args) { + $container->addShared($id)->addArguments($args); + } + } +} + diff --git a/tests/classes/ClassA.php b/tests/classes/ClassA.php new file mode 100644 index 0000000..ca61c28 --- /dev/null +++ b/tests/classes/ClassA.php @@ -0,0 +1,12 @@ +add(\ClassA::class); + $container->add(\ClassB::class); + $container->add(\ClassC::class); + + $resolver = new DependencyResolver(); + + $args = $resolver->autowireArguments($container, function (\ClassC $a, string $b, string $c = "42") { }); + + $this->assertInstanceOf(\ClassC::class, $args['a']); + $this->assertNull($args['b']); + $this->assertEquals("42", $args['c']); + + } + + public function testAutowiringCallback(): void + { + $container = new Container(); + $container->add(\ClassA::class); + $container->add(\ClassB::class); + $container->add(\ClassC::class); + + $resolver = new DependencyResolver(); + + $hit = false; + $cb = $resolver->autowire($container, function (\ClassC $a, string $b, string $c = "42") use (&$hit) { + $hit = true; + $this->assertEquals('hello', $b); + $this->assertEquals('42', $c); + return true; + }); + $result = $cb(b: 'hello'); + + $this->assertTrue($hit); + $this->assertTrue($result); + + } + + public function testAutowiringConstructor(): void + { + $container = new Container(); + $container->add(\ClassA::class); + $container->add(\ClassB::class); + $container->add(\ClassC::class); + + $container->setParameter("b", "bazinga"); + + $resolver = new DependencyResolver(); + + $args = $resolver->autowireConstructorArguments($container, TestClass::class); + $this->assertInstanceOf(\ClassC::class, $args['a']); + $this->assertEquals("bazinga", $args['b']); + $this->assertEquals("42", $args['c']); + + } + + public function testAutowiringConstructorWithParameters(): void + { + $container = new Container(); + $container->add(\ClassA::class); + $container->add(\ClassB::class); + $container->add(\ClassC::class); + + $resolver = new DependencyResolver(); + + $args = $resolver->autowireConstructorArguments($container, TestClass::class); + $this->assertInstanceOf(\ClassC::class, $args['a']); + $this->assertEquals(null, $args['b']); + $this->assertEquals("42", $args['c']); + $this->assertInstanceOf(\ClassA::class, $args['d']); + + } + +} + +class TestClass { + public function __construct(\ClassC $a, string $b, string $c = "42", \ClassA $d) { + } +} \ No newline at end of file