Completed
Pull Request — master (#58)
by Greg
01:57
created

CommandProcessor::dataCanBeFormatted()   A

Complexity

Conditions 3
Paths 3

Size

Total Lines 9
Code Lines 6

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
c 0
b 0
f 0
dl 0
loc 9
rs 9.6666
cc 3
eloc 6
nc 3
nop 1
1
<?php
2
namespace Consolidation\AnnotatedCommand;
3
4
use Symfony\Component\Console\Command\Command;
5
use Symfony\Component\Console\Input\InputInterface;
6
use Symfony\Component\Console\Output\OutputInterface;
7
use Symfony\Component\Console\Output\ConsoleOutputInterface;
8
9
use Consolidation\OutputFormatters\FormatterManager;
10
use Consolidation\OutputFormatters\Options\FormatterOptions;
11
use Consolidation\AnnotatedCommand\Hooks\HookManager;
12
use Consolidation\AnnotatedCommand\Options\PrepareFormatter;
13
14
/**
15
 * Process a command, including hooks and other callbacks.
16
 * There should only be one command processor per application.
17
 * Provide your command processor to the AnnotatedCommandFactory
18
 * via AnnotatedCommandFactory::setCommandProcessor().
19
 */
20
class CommandProcessor
21
{
22
    /** var HookManager */
23
    protected $hookManager;
24
    /** var FormatterManager */
25
    protected $formatterManager;
26
    /** var callable */
27
    protected $displayErrorFunction;
28
    /** var PrepareFormatterOptions[] */
29
    protected $prepareOptionsList = [];
30
31
    public function __construct(HookManager $hookManager)
32
    {
33
        $this->hookManager = $hookManager;
34
    }
35
36
    /**
37
     * Return the hook manager
38
     * @return HookManager
39
     */
40
    public function hookManager()
41
    {
42
        return $this->hookManager;
43
    }
44
45
    public function addPrepareFormatter(PrepareFormatter $preparer)
46
    {
47
        $this->prepareOptionsList[] = $preparer;
48
    }
49
50
    public function setFormatterManager(FormatterManager $formatterManager)
51
    {
52
        $this->formatterManager = $formatterManager;
53
        return $this;
54
    }
55
56
    public function setDisplayErrorFunction(callable $fn)
57
    {
58
        $this->displayErrorFunction = $fn;
59
        return $this;
60
    }
61
62
    /**
63
     * Return the formatter manager
64
     * @return FormatterManager
65
     */
66
    public function formatterManager()
67
    {
68
        return $this->formatterManager;
69
    }
70
71
    public function initializeHook(
72
        InputInterface $input,
73
        $names,
74
        AnnotationData $annotationData
75
    ) {
76
        return $this->hookManager()->initializeHook($input, $names, $annotationData);
77
    }
78
79
    public function optionsHook(
80
        AnnotatedCommand $command,
81
        $names,
82
        AnnotationData $annotationData
83
    ) {
84
        $this->hookManager()->optionsHook($command, $names, $annotationData);
85
    }
86
87
    public function interact(
88
        InputInterface $input,
89
        OutputInterface $output,
90
        $names,
91
        AnnotationData $annotationData
92
    ) {
93
        return $this->hookManager()->interact($input, $output, $names, $annotationData);
94
    }
95
96
    public function process(
97
        OutputInterface $output,
98
        $names,
99
        $commandCallback,
100
        CommandData $commandData
101
    ) {
102
        $result = [];
103
        try {
104
            $result = $this->validateRunAndAlter(
105
                $names,
106
                $commandCallback,
107
                $commandData
108
            );
109
            return $this->handleResults($output, $names, $result, $commandData);
110
        } catch (\Exception $e) {
111
            $result = new CommandError($e->getMessage(), $e->getCode());
112
            return $this->handleResults($output, $names, $result, $commandData);
113
        }
114
    }
115
116
    public function validateRunAndAlter(
117
        $names,
118
        $commandCallback,
119
        CommandData $commandData
120
    ) {
121
        // Validators return any object to signal a validation error;
122
        // if the return an array, it replaces the arguments.
123
        $validated = $this->hookManager()->validateArguments($names, $commandData);
124
        if (is_object($validated)) {
125
            return $validated;
126
        }
127
128
        // Run the command, alter the results, and then handle output and status
129
        $result = $this->runCommandCallback($commandCallback, $commandData);
130
        return $this->processResults($names, $result, $commandData);
131
    }
132
133
    public function processResults($names, $result, CommandData $commandData)
134
    {
135
        return $this->hookManager()->alterResult($names, $result, $commandData);
136
    }
137
138
    /**
139
     * Handle the result output and status code calculation.
140
     */
141
    public function handleResults(OutputInterface $output, $names, $result, CommandData $commandData)
142
    {
143
        $status = $this->hookManager()->determineStatusCode($names, $result);
144
        // If the result is an integer and no separate status code was provided, then use the result as the status and do no output.
145
        if (is_integer($result) && !isset($status)) {
146
            return $result;
147
        }
148
        $status = $this->interpretStatusCode($status);
149
150
        // Get the structured output, the output stream and the formatter
151
        $structuredOutput = $this->hookManager()->extractOutput($names, $result);
152
        $output = $this->chooseOutputStream($output, $status);
153
        if ($status != 0) {
154
            return $this->writeErrorMessage($output, $status, $structuredOutput, $result);
155
        }
156
        if ($this->dataCanBeFormatted($structuredOutput) && isset($this->formatterManager)) {
157
            return $this->writeUsingFormatter($output, $structuredOutput, $commandData);
158
        }
159
        return $this->writeCommandOutput($output, $structuredOutput);
160
    }
161
162
    protected function dataCanBeFormatted($structuredOutput)
163
    {
164
        if (!isset($this->formatterManager)) {
165
            return false;
166
        }
167
        return
168
            is_object($structuredOutput) ||
169
            is_array($structuredOutput);
170
    }
171
172
    /**
173
     * Run the main command callback
174
     */
175
    protected function runCommandCallback($commandCallback, CommandData $commandData)
176
    {
177
        $result = false;
178
        try {
179
            $args = $commandData->getArgsAndOptions();
180
            $result = call_user_func_array($commandCallback, $args);
181
        } catch (\Exception $e) {
182
            $result = new CommandError($e->getMessage(), $e->getCode());
183
        }
184
        return $result;
185
    }
186
187
    /**
188
     * Determine the formatter that should be used to render
189
     * output.
190
     *
191
     * If the user specified a format via the --format option,
192
     * then always return that.  Otherwise, return the default
193
     * format, unless --pipe was specified, in which case
194
     * return the default pipe format, format-pipe.
195
     *
196
     * n.b. --pipe is a handy option introduced in Drush 2
197
     * (or perhaps even Drush 1) that indicates that the command
198
     * should select the output format that is most appropriate
199
     * for use in scripts (e.g. to pipe to another command).
200
     *
201
     * @return string
202
     */
203
    protected function getFormat(FormatterOptions $options)
204
    {
205
        // In Symfony Console, there is no way for us to differentiate
206
        // between the user specifying '--format=table', and the user
207
        // not specifying --format when the default value is 'table'.
208
        // Therefore, we must make --field always override --format; it
209
        // cannot become the default value for --format.
210
        if ($options->get('field')) {
211
            return 'string';
212
        }
213
        $defaults = [];
214
        if ($options->get('pipe')) {
215
            $defaults[FormatterOptions::DEFAULT_FORMAT] = $options->get('pipe-format');
216
        }
217
        return $options->getFormat($defaults);
218
    }
219
220
    /**
221
     * Determine whether we should use stdout or stderr.
222
     */
223
    protected function chooseOutputStream(OutputInterface $output, $status)
224
    {
225
        // If the status code indicates an error, then print the
226
        // result to stderr rather than stdout
227
        if ($status && ($output instanceof ConsoleOutputInterface)) {
228
            return $output->getErrorOutput();
229
        }
230
        return $output;
231
    }
232
233
    /**
234
     * Call the formatter to output the provided data.
235
     */
236
    protected function writeUsingFormatter(OutputInterface $output, $structuredOutput, CommandData $commandData)
237
    {
238
        $formatterOptions = $this->createFormatterOptions($commandData);
239
        $format = $this->getFormat($formatterOptions);
240
        $this->formatterManager->write(
241
            $output,
242
            $format,
243
            $structuredOutput,
244
            $formatterOptions
245
        );
246
        return 0;
247
    }
248
249
    /**
250
     * Create a FormatterOptions object for use in writing the formatted output.
251
     * @param CommandData $commandData
252
     * @return FormatterOptions
253
     */
254
    protected function createFormatterOptions($commandData)
255
    {
256
        $options = $commandData->input()->getOptions();
257
        $formatterOptions = new FormatterOptions($commandData->annotationData()->getArrayCopy(), $options);
258
        foreach ($this->prepareOptionsList as $preparer) {
259
            $preparer->prepare($commandData, $formatterOptions);
260
        }
261
        return $formatterOptions;
262
    }
263
264
    /**
265
     * Description
266
     * @param OutputInterface $output
267
     * @param int $status
268
     * @param string $structuredOutput
269
     * @param mixed $originalResult
270
     * @return type
271
     */
272
    protected function writeErrorMessage($output, $status, $structuredOutput, $originalResult)
273
    {
274
        if (isset($this->displayErrorFunction)) {
275
            call_user_func($this->displayErrorFunction, $output, $structuredOutput, $status, $originalResult);
276
        } else {
277
            $this->writeCommandOutput($output, $structuredOutput);
278
        }
279
        return $status;
280
    }
281
282
    /**
283
     * If the result object is a string, then print it.
284
     */
285
    protected function writeCommandOutput(
286
        OutputInterface $output,
287
        $structuredOutput
288
    ) {
289
        // If there is no formatter, we will print strings,
290
        // but can do no more than that.
291
        if (is_string($structuredOutput)) {
292
            $output->writeln($structuredOutput);
293
        }
294
        return 0;
295
    }
296
297
    /**
298
     * If a status code was set, then return it; otherwise,
299
     * presume success.
300
     */
301
    protected function interpretStatusCode($status)
302
    {
303
        if (isset($status)) {
304
            return $status;
305
        }
306
        return 0;
307
    }
308
}
309