Passed
Push — master ( bf1cbc...93a2f9 )
by Kevin
03:05
created

MakeFactory   A

Complexity

Total Complexity 28

Size/Duplication

Total Lines 197
Duplicated Lines 0 %

Test Coverage

Coverage 69.23%

Importance

Changes 4
Bugs 0 Features 0
Metric Value
wmc 28
eloc 101
c 4
b 0
f 0
dl 0
loc 197
ccs 54
cts 78
cp 0.6923
rs 10

9 Methods

Rating   Name   Duplication   Size   Complexity  
A getCommandDescription() 0 3 1
A __construct() 0 8 1
A getCommandName() 0 3 1
A configureCommand() 0 11 1
A configureDependencies() 0 2 1
B generate() 0 53 7
A interact() 0 20 4
B defaultPropertiesFor() 0 25 7
A entityChoices() 0 19 5
1
<?php
2
3
namespace Zenstruck\Foundry\Bundle\Maker;
4
5
use Doctrine\ORM\EntityManagerInterface;
6
use Doctrine\Persistence\ManagerRegistry;
7
use Symfony\Bundle\MakerBundle\ConsoleStyle;
8
use Symfony\Bundle\MakerBundle\DependencyBuilder;
9
use Symfony\Bundle\MakerBundle\Exception\RuntimeCommandException;
10
use Symfony\Bundle\MakerBundle\Generator;
11
use Symfony\Bundle\MakerBundle\InputConfiguration;
12
use Symfony\Bundle\MakerBundle\Maker\AbstractMaker;
13
use Symfony\Component\Console\Command\Command;
14
use Symfony\Component\Console\Input\InputArgument;
15
use Symfony\Component\Console\Input\InputInterface;
16
use Symfony\Component\Console\Input\InputOption;
17
use Zenstruck\Foundry\ModelFactory;
18
19
/**
20
 * @author Kevin Bond <[email protected]>
21
 */
22
final class MakeFactory extends AbstractMaker
23
{
24
    private const ORM_DEFAULTS = [
25 20
        'ARRAY' => '[],',
26
        'ASCII_STRING' => 'self::faker()->text(),',
27 20
        'BIGINT' => 'self::faker()->randomNumber(),',
28 20
        'BLOB' => 'self::faker()->text(),',
29
        'BOOLEAN' => 'self::faker()->boolean(),',
30 4
        'DATE' => 'self::faker()->datetime(),',
31
        'DATE_MUTABLE' => 'self::faker()->datetime(),',
32 4
        'DATE_IMMUTABLE' => 'self::faker()->datetime(),',
33
        'DATETIME_MUTABLE' => 'self::faker()->datetime(),',
34
        'DATETIME_IMMUTABLE' => 'self::faker()->datetime(),',
35 20
        'DATETIMETZ_MUTABLE' => 'self::faker()->datetime(),',
36
        'DATETIMETZ_IMMUTABLE' => 'self::faker()->datetime(),',
37
        'DECIMAL' => 'self::faker()->randomFloat(),',
38 20
        'FLOAT' => 'self::faker()->randomFloat(),',
39 20
        'INTEGER' => 'self::faker()->randomNumber(),',
40 20
        'JSON' => '[],',
41
        'JSON_ARRAY' => '[],',
42
        'SIMPLE_ARRAY' => '[],',
43 20
        'SMALLINT' => 'self::faker()->randomNumber(1, 32767),',
44 20
        'STRING' => 'self::faker()->text(),',
45
        'TEXT' => 'self::faker()->text(),',
46 20
        'TIME_MUTABLE' => 'self::faker()->datetime(),',
47
        'TIME_IMMUTABLE' => 'self::faker()->datetime(),',
48 20
    ];
49 12
50
    /** @var ManagerRegistry */
51
    private $managerRegistry;
52 8
53 4
    /** @var string[] */
54 4
    private $entitiesWithFactories;
55
56
    public function __construct(ManagerRegistry $managerRegistry, \Traversable $factories)
57 8
    {
58 8
        $this->managerRegistry = $managerRegistry;
59
        $this->entitiesWithFactories = \array_map(
60 8
            static function(ModelFactory $factory) {
61 8
                return $factory::getEntityClass();
62
            },
63 20
            \iterator_to_array($factories)
64
        );
65 20
    }
66
67 20
    public static function getCommandName(): string
68 4
    {
69
        return 'make:factory';
70
    }
71 20
72 4
    public static function getCommandDescription(): string
73
    {
74
        return 'Creates a Foundry model factory for a Doctrine entity class';
75 16
    }
76 16
77 16
    public function configureCommand(Command $command, InputConfiguration $inputConfig): void
78 16
    {
79 16
        $command
80
            ->setDescription(self::getCommandDescription())
81
            ->addArgument('entity', InputArgument::OPTIONAL, 'Entity class to create a factory for')
82 16
            ->addOption('namespace', null, InputOption::VALUE_REQUIRED, 'Customize the namespace for generated factories', 'Factory')
83
            ->addOption('test', null, InputOption::VALUE_NONE, 'Create in <fg=yellow>tests/</> instead of <fg=yellow>src/</>')
84 16
            ->addOption('all-fields', null, InputOption::VALUE_NONE, 'Create defaults for all entity fields, not only required fields')
85
        ;
86 16
87
        $inputConfig->setArgumentAsNonInteractive('entity');
88
    }
89 16
90 16
    public function interact(InputInterface $input, ConsoleStyle $io, Command $command): void
91 16
    {
92
        if ($input->getArgument('entity')) {
93 16
            return;
94 16
        }
95
96
        if (!$input->getOption('test')) {
97
            $io->text('// Note: pass <fg=yellow>--test</> if you want to generate factories in your <fg=yellow>tests/</> directory');
98 16
            $io->newLine();
99
        }
100 16
101
        if (!$input->getOption('all-fields')) {
102 16
            $io->text('// Note: pass <fg=yellow>--all-fields</> if you want to generate default values for all fields, not only required fields');
103 16
            $io->newLine();
104
        }
105
106 16
        $argument = $command->getDefinition()->getArgument('entity');
107
        $entity = $io->choice($argument->getDescription(), $this->entityChoices());
108 20
109
        $input->setArgument('entity', $entity);
110
    }
111 20
112
    public function generate(InputInterface $input, ConsoleStyle $io, Generator $generator): void
113 8
    {
114
        $class = $input->getArgument('entity');
115 8
116
        if (!\class_exists($class)) {
117 8
            $class = $generator->createClassNameDetails($class, 'Entity\\')->getFullName();
118 8
        }
119 8
120
        if (!\class_exists($class)) {
121
            throw new RuntimeCommandException(\sprintf('Entity "%s" not found.', $input->getArgument('entity')));
122
        }
123 8
124
        $namespace = $input->getOption('namespace');
125 8
126
        // strip maker's root namespace if set
127
        if (0 === \mb_strpos($namespace, $generator->getRootNamespace())) {
128
            $namespace = \mb_substr($namespace, \mb_strlen($generator->getRootNamespace()));
129
        }
130
131
        $namespace = \trim($namespace, '\\');
132
133
        // if creating in tests dir, ensure namespace prefixed with Tests\
134
        if ($input->getOption('test') && 0 !== \mb_strpos($namespace, 'Tests\\')) {
135
            $namespace = 'Tests\\'.$namespace;
136
        }
137
138
        $entity = new \ReflectionClass($class);
139
        $factory = $generator->createClassNameDetails($entity->getShortName(), $namespace, 'Factory');
140
141
        $repository = new \ReflectionClass($this->managerRegistry->getRepository($entity->getName()));
142
143
        if (0 !== \mb_strpos($repository->getName(), $generator->getRootNamespace())) {
144
            // not using a custom repository
145
            $repository = null;
146
        }
147
148
        $generator->generateClass(
149
            $factory->getFullName(),
150
            __DIR__.'/../Resources/skeleton/Factory.tpl.php',
151
            [
152
                'entity' => $entity,
153
                'defaultProperties' => $this->defaultPropertiesFor($entity->getName(), $input->getOption('all-fields')),
154
                'repository' => $repository,
155
            ]
156
        );
157
158
        $generator->writeChanges();
159
160
        $this->writeSuccessMessage($io);
161
162
        $io->text([
163
            'Next: Open your new factory and set default values/states.',
164
            'Find the documentation at https://symfony.com/bundles/ZenstruckFoundryBundle/current/index.html#model-factories',
165
        ]);
166
    }
167
168
    public function configureDependencies(DependencyBuilder $dependencies): void
169
    {
170
        // noop
171
    }
172
173
    private function entityChoices(): array
174
    {
175
        $choices = [];
176
177
        foreach ($this->managerRegistry->getManagers() as $manager) {
178
            foreach ($manager->getMetadataFactory()->getAllMetadata() as $metadata) {
179
                if (!\in_array($metadata->getName(), $this->entitiesWithFactories, true)) {
180
                    $choices[] = $metadata->getName();
181
                }
182
            }
183
        }
184
185
        \sort($choices);
186
187
        if (empty($choices)) {
188
            throw new RuntimeCommandException('No entities found.');
189
        }
190
191
        return $choices;
192
    }
193
194
    private function defaultPropertiesFor(string $class, bool $allFields): iterable
195
    {
196
        $em = $this->managerRegistry->getManagerForClass($class);
197
198
        if (!$em instanceof EntityManagerInterface) {
199
            return [];
200
        }
201
202
        $metadata = $em->getClassMetadata($class);
203
        $ids = $metadata->getIdentifierFieldNames();
204
205
        foreach ($metadata->fieldMappings as $property) {
206
            // ignore identifiers and nullable fields
207
            if ((!$allFields && ($property['nullable'] ?? false)) || \in_array($property['fieldName'], $ids, true)) {
208
                continue;
209
            }
210
211
            $type = \mb_strtoupper($property['type']);
212
            $value = "null, // TODO add {$type} ORM type manually";
213
214
            if (\array_key_exists($type, self::ORM_DEFAULTS)) {
215
                $value = self::ORM_DEFAULTS[$type];
216
            }
217
218
            yield $property['fieldName'] => $value;
219
        }
220
    }
221
}
222