Passed
Push — master ( 39fa8e...01ebfa )
by Kevin
03:53
created

MakeFactory::generateFactory()   B

Complexity

Conditions 7
Paths 18

Size

Total Lines 51
Code Lines 26

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 0
CRAP Score 56

Importance

Changes 0
Metric Value
eloc 26
c 0
b 0
f 0
dl 0
loc 51
ccs 0
cts 0
cp 0
rs 8.5706
cc 7
nc 18
nop 4
crap 56

How to fix   Long Method   

Long Method

Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.

For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.

Commonly applied refactorings include:

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()->numberBetween(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 configureDependencies(DependencyBuilder $dependencies): void
78 16
    {
79 16
        // noop
80
    }
81
82 16
    public function configureCommand(Command $command, InputConfiguration $inputConfig): void
83
    {
84 16
        $command
85
            ->setDescription(self::getCommandDescription())
86 16
            ->addArgument('entity', InputArgument::OPTIONAL, 'Entity class to create a factory for')
87
            ->addOption('namespace', null, InputOption::VALUE_REQUIRED, 'Customize the namespace for generated factories', 'Factory')
88
            ->addOption('test', null, InputOption::VALUE_NONE, 'Create in <fg=yellow>tests/</> instead of <fg=yellow>src/</>')
89 16
            ->addOption('all-fields', null, InputOption::VALUE_NONE, 'Create defaults for all entity fields, not only required fields')
90 16
        ;
91 16
92
        $inputConfig->setArgumentAsNonInteractive('entity');
93 16
    }
94 16
95
    public function interact(InputInterface $input, ConsoleStyle $io, Command $command): void
96
    {
97
        if ($input->getArgument('entity')) {
98 16
            return;
99
        }
100 16
101
        if (!$input->getOption('test')) {
102 16
            $io->text('// Note: pass <fg=yellow>--test</> if you want to generate factories in your <fg=yellow>tests/</> directory');
103 16
            $io->newLine();
104
        }
105
106 16
        if (!$input->getOption('all-fields')) {
107
            $io->text('// Note: pass <fg=yellow>--all-fields</> if you want to generate default values for all fields, not only required fields');
108 20
            $io->newLine();
109
        }
110
111 20
        $argument = $command->getDefinition()->getArgument('entity');
112
        $entity = $io->choice($argument->getDescription(), \array_merge($this->entityChoices(), ['All']));
113 8
114
        $input->setArgument('entity', $entity);
115 8
    }
116
117 8
    public function generate(InputInterface $input, ConsoleStyle $io, Generator $generator): void
118 8
    {
119 8
        $entity = $input->getArgument('entity');
120
        $classes = 'All' === $entity ? $this->entityChoices() : [$entity];
121
122
        foreach ($classes as $class) {
123 8
            $this->generateFactory($class, $input, $io, $generator);
124
        }
125 8
    }
126
127
    /**
128
     * Generates a single entity factory.
129
     */
130
    private function generateFactory(string $class, InputInterface $input, ConsoleStyle $io, Generator $generator)
131
    {
132
        if (!\class_exists($class)) {
133
            $class = $generator->createClassNameDetails($class, 'Entity\\')->getFullName();
134
        }
135
136
        if (!\class_exists($class)) {
137
            throw new RuntimeCommandException(\sprintf('Entity "%s" not found.', $input->getArgument('entity')));
138
        }
139
140
        $namespace = $input->getOption('namespace');
141
142
        // strip maker's root namespace if set
143
        if (0 === \mb_strpos($namespace, $generator->getRootNamespace())) {
144
            $namespace = \mb_substr($namespace, \mb_strlen($generator->getRootNamespace()));
145
        }
146
147
        $namespace = \trim($namespace, '\\');
148
149
        // if creating in tests dir, ensure namespace prefixed with Tests\
150
        if ($input->getOption('test') && 0 !== \mb_strpos($namespace, 'Tests\\')) {
151
            $namespace = 'Tests\\'.$namespace;
152
        }
153
154
        $entity = new \ReflectionClass($class);
155
        $factory = $generator->createClassNameDetails($entity->getShortName(), $namespace, 'Factory');
156
157
        $repository = new \ReflectionClass($this->managerRegistry->getRepository($entity->getName()));
158
159
        if (0 !== \mb_strpos($repository->getName(), $generator->getRootNamespace())) {
160
            // not using a custom repository
161
            $repository = null;
162
        }
163
164
        $generator->generateClass(
165
            $factory->getFullName(),
166
            __DIR__.'/../Resources/skeleton/Factory.tpl.php',
167
            [
168
                'entity' => $entity,
169
                'defaultProperties' => $this->defaultPropertiesFor($entity->getName(), $input->getOption('all-fields')),
170
                'repository' => $repository,
171
            ]
172
        );
173
174
        $generator->writeChanges();
175
176
        $this->writeSuccessMessage($io);
177
178
        $io->text([
179
            'Next: Open your new factory and set default values/states.',
180
            'Find the documentation at https://symfony.com/bundles/ZenstruckFoundryBundle/current/index.html#model-factories',
181
        ]);
182
    }
183
184
    private function entityChoices(): array
185
    {
186
        $choices = [];
187
188
        foreach ($this->managerRegistry->getManagers() as $manager) {
189
            foreach ($manager->getMetadataFactory()->getAllMetadata() as $metadata) {
190
                if ($metadata->getReflectionClass()->isAbstract()) {
191
                    continue;
192
                }
193
                if (!\in_array($metadata->getName(), $this->entitiesWithFactories, true)) {
194
                    $choices[] = $metadata->getName();
195
                }
196
            }
197
        }
198
199
        \sort($choices);
200
201
        if (empty($choices)) {
202
            throw new RuntimeCommandException('No entities or documents found, or none left to make factories for.');
203
        }
204
205
        return $choices;
206
    }
207
208
    private function defaultPropertiesFor(string $class, bool $allFields): iterable
209
    {
210
        $em = $this->managerRegistry->getManagerForClass($class);
211
212
        if (!$em instanceof EntityManagerInterface) {
213
            return [];
214
        }
215
216
        $metadata = $em->getClassMetadata($class);
217
        $ids = $metadata->getIdentifierFieldNames();
218
219
        foreach ($metadata->fieldMappings as $property) {
220
            // ignore identifiers and nullable fields
221
            if ((!$allFields && ($property['nullable'] ?? false)) || \in_array($property['fieldName'], $ids, true)) {
222
                continue;
223
            }
224
225
            $type = \mb_strtoupper($property['type']);
226
            $value = "null, // TODO add {$type} ORM type manually";
227
228
            if (\array_key_exists($type, self::ORM_DEFAULTS)) {
229
                $value = self::ORM_DEFAULTS[$type];
230
            }
231
232
            yield $property['fieldName'] => $value;
233
        }
234
    }
235
}
236