Completed
Push — master ( 90e1f1...e696e7 )
by Tim
06:16
created

AbstractSubject::getEmitter()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 4

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 1

Importance

Changes 0
Metric Value
dl 0
loc 4
ccs 2
cts 2
cp 1
rs 10
c 0
b 0
f 0
cc 1
nc 1
nop 0
crap 1
1
<?php
2
3
/**
4
 * TechDivision\Import\Subjects\AbstractSubject
5
 *
6
 * NOTICE OF LICENSE
7
 *
8
 * This source file is subject to the Open Software License (OSL 3.0)
9
 * that is available through the world-wide-web at this URL:
10
 * http://opensource.org/licenses/osl-3.0.php
11
 *
12
 * PHP version 5
13
 *
14
 * @author    Tim Wagner <[email protected]>
15
 * @copyright 2016 TechDivision GmbH <[email protected]>
16
 * @license   http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0)
17
 * @link      https://github.com/techdivision/import
18
 * @link      http://www.techdivision.com
19
 */
20
21
namespace TechDivision\Import\Subjects;
22
23
use Psr\Log\LogLevel;
24
use Ramsey\Uuid\Uuid;
25
use League\Event\EmitterInterface;
26
use Doctrine\Common\Collections\Collection;
27
use TechDivision\Import\RowTrait;
28
use TechDivision\Import\HeaderTrait;
29
use TechDivision\Import\SystemLoggerTrait;
30
use TechDivision\Import\Utils\ScopeKeys;
31
use TechDivision\Import\Utils\ColumnKeys;
32
use TechDivision\Import\Utils\EventNames;
33
use TechDivision\Import\Utils\MemberNames;
34
use TechDivision\Import\Utils\RegistryKeys;
35
use TechDivision\Import\Utils\EntityTypeCodes;
36
use TechDivision\Import\Utils\Generators\GeneratorInterface;
37
use TechDivision\Import\Callbacks\CallbackInterface;
38
use TechDivision\Import\Observers\ObserverInterface;
39
use TechDivision\Import\Adapter\ImportAdapterInterface;
40
use TechDivision\Import\Exceptions\WrappedColumnException;
41
use TechDivision\Import\Services\RegistryProcessorInterface;
42
use TechDivision\Import\Configuration\SubjectConfigurationInterface;
43
44
/**
45
 * An abstract subject implementation.
46
 *
47
 * @author    Tim Wagner <[email protected]>
48
 * @copyright 2016 TechDivision GmbH <[email protected]>
49
 * @license   http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0)
50
 * @link      https://github.com/techdivision/import
51
 * @link      http://www.techdivision.com
52
 */
53
abstract class AbstractSubject implements SubjectInterface, FilesystemSubjectInterface, DateConverterSubjectInterface
54
{
55
56
    /**
57
     * The trait that provides basic filesystem handling functionality.
58
     *
59
     * @var \TechDivision\Import\Subjects\FilesystemTrait
60
     */
61
    use FilesystemTrait;
62
63
    /**
64
     * The trait that provides basic filesystem handling functionality.
65
     *
66
     * @var \TechDivision\Import\SystemLoggerTrait
67
     */
68
    use SystemLoggerTrait;
69
70
    /**
71
     * The trait that provides date converting functionality.
72
     *
73
     * @var \TechDivision\Import\DateConverterTrait
74
     */
75
    use DateConverterTrait;
76
77
    /**
78
     * The trait that provides header handling functionality.
79
     *
80
     * @var \TechDivision\Import\HeaderTrait
81
     */
82
    use HeaderTrait;
83
84
    /**
85
     * The trait that provides row handling functionality.
86
     *
87
     * @var \TechDivision\Import\RowTrait
88
     */
89
    use RowTrait;
90
91
    /**
92
     * The unique identifier for the actual invocation.
93
     *
94
     * @var string
95
     */
96
    protected $uniqueId;
97
98
    /**
99
     * The name of the file to be imported.
100
     *
101
     * @var string
102
     */
103
    protected $filename;
104
105
    /**
106
     * The actual line number.
107
     *
108
     * @var integer
109
     */
110
    protected $lineNumber = 0;
111
112
    /**
113
     * The import adapter instance.
114
     *
115
     * @var \TechDivision\Import\Adapter\ImportAdapterInterface
116
     */
117
    protected $importAdapter;
118
119
    /**
120
     * The subject configuration.
121
     *
122
     * @var \TechDivision\Import\Configuration\SubjectConfigurationInterface
123
     */
124
    protected $configuration;
125
126
    /**
127
     * The plugin configuration.
128
     *
129
     * @var \TechDivision\Import\Configuration\PluginConfigurationInterface
130
     */
131
    protected $pluginConfiguration;
132
133
    /**
134
     * The RegistryProcessor instance to handle running threads.
135
     *
136
     * @var \TechDivision\Import\Services\RegistryProcessorInterface
137
     */
138
    protected $registryProcessor;
139
140
    /**
141
     * The actions unique serial.
142
     *
143
     * @var string
144
     */
145
    protected $serial;
146
147
    /**
148
     * Array with the subject's observers.
149
     *
150
     * @var array
151
     */
152
    protected $observers = array();
153
154
    /**
155
     * Array with the subject's callbacks.
156
     *
157
     * @var array
158
     */
159
    protected $callbacks = array();
160
161
    /**
162
     * The subject's callback mappings.
163
     *
164
     * @var array
165
     */
166
    protected $callbackMappings = array();
167
168
    /**
169
     * The available root categories.
170
     *
171
     * @var array
172
     */
173
    protected $rootCategories = array();
174
175
    /**
176
     * The Magento configuration.
177
     *
178
     * @var array
179
     */
180
    protected $coreConfigData = array();
181
182
    /**
183
     * The available stores.
184
     *
185
     * @var array
186
     */
187
    protected $stores = array();
188
189
    /**
190
     * The available websites.
191
     *
192
     * @var array
193
     */
194
    protected $storeWebsites = array();
195
196
    /**
197
     * The default store.
198
     *
199
     * @var array
200
     */
201
    protected $defaultStore;
202
203
    /**
204
     * The store view code the create the product/attributes for.
205
     *
206
     * @var string
207
     */
208
    protected $storeViewCode;
209
210
    /**
211
     * The UID generator for the core config data.
212
     *
213
     * @var \TechDivision\Import\Utils\Generators\GeneratorInterface
214
     */
215
    protected $coreConfigDataUidGenerator;
216
217
    /**
218
     * UNIX timestamp with the date the last row counter has been logged.
219
     *
220
     * @var integer
221
     */
222
    protected $lastLog = 0;
223
224
    /**
225
     * The number of the last line that has been logged with the row counter
226
     * @var integer
227
     */
228
    protected $lastLineNumber = 0;
229
230
    /**
231
     * The event emitter instance.
232
     *
233
     * @var \League\Event\EmitterInterface
234
     */
235
    protected $emitter;
236
237
    /**
238
     * The status of the file (0 = not processed, 1 = successfully processed, 2 = processed with failure)
239
     *
240
     * @var array
241
     */
242
    protected $status = array();
243
244
    /**
245
     * The default values for the columns from the configuration.
246
     *
247
     * @var array
248
     */
249
    protected $defaultColumnValues = array();
250
251
    /**
252
     * The values of the actual column, pre-initialized with the default values.
253
     *
254
     * @var array
255
     */
256
    protected $columnValues = array();
257
258
    /**
259
     * Mapping for the virtual entity type code to the real Magento 2 EAV entity type code.
260
     *
261
     * @var array
262
     */
263
    protected $entityTypeCodeMappings = array(
264
        EntityTypeCodes::EAV_ATTRIBUTE                 => EntityTypeCodes::CATALOG_PRODUCT,
265
        EntityTypeCodes::EAV_ATTRIBUTE_SET             => EntityTypeCodes::CATALOG_PRODUCT,
266
        EntityTypeCodes::CATALOG_PRODUCT_URL           => EntityTypeCodes::CATALOG_PRODUCT,
267
        EntityTypeCodes::CATALOG_PRODUCT_PRICE         => EntityTypeCodes::CATALOG_PRODUCT,
268
        EntityTypeCodes::CATALOG_PRODUCT_INVENTORY     => EntityTypeCodes::CATALOG_PRODUCT,
269
        EntityTypeCodes::CATALOG_PRODUCT_INVENTORY_MSI => EntityTypeCodes::CATALOG_PRODUCT,
270
        EntityTypeCodes::CATALOG_PRODUCT_TIER_PRICE    => EntityTypeCodes::CATALOG_PRODUCT
271
    );
272
273
    /**
274
     * Initialize the subject instance.
275
     *
276
     * @param \TechDivision\Import\Services\RegistryProcessorInterface $registryProcessor          The registry processor instance
277
     * @param \TechDivision\Import\Utils\Generators\GeneratorInterface $coreConfigDataUidGenerator The UID generator for the core config data
278
     * @param \Doctrine\Common\Collections\Collection                  $systemLoggers              The array with the system loggers instances
279
     * @param \League\Event\EmitterInterface                           $emitter                    The event emitter instance
280
     */
281 81
    public function __construct(
282
        RegistryProcessorInterface $registryProcessor,
283
        GeneratorInterface $coreConfigDataUidGenerator,
284
        Collection $systemLoggers,
285
        EmitterInterface $emitter
286
    ) {
287 81
        $this->emitter = $emitter;
288 81
        $this->systemLoggers = $systemLoggers;
0 ignored issues
show
Documentation Bug introduced by
It seems like $systemLoggers of type object<Doctrine\Common\Collections\Collection> is incompatible with the declared type array of property $systemLoggers.

Our type inference engine has found an assignment to a property that is incompatible with the declared type of that property.

Either this assignment is in error or the assigned type should be added to the documentation/type hint for that property..

Loading history...
289 81
        $this->registryProcessor = $registryProcessor;
290 81
        $this->coreConfigDataUidGenerator = $coreConfigDataUidGenerator;
291 81
    }
292
293
    /**
294
     * Return's the event emitter instance.
295
     *
296
     * @return \League\Event\EmitterInterface The event emitter instance
297
     */
298 9
    public function getEmitter()
299
    {
300 9
        return $this->emitter;
301
    }
302
303
    /**
304
     * Set's the name of the file to import
305
     *
306
     * @param string $filename The filename
307
     *
308
     * @return void
309
     */
310 13
    public function setFilename($filename)
311
    {
312 13
        $this->filename = $filename;
313 13
    }
314
315
    /**
316
     * Return's the name of the file to import.
317
     *
318
     * @return string The filename
319
     */
320 12
    public function getFilename()
321
    {
322 12
        return $this->filename;
323
    }
324
325
    /**
326
     * Set's the actual line number.
327
     *
328
     * @param integer $lineNumber The line number
329
     *
330
     * @return void
331
     */
332 1
    public function setLineNumber($lineNumber)
333
    {
334 1
        $this->lineNumber = $lineNumber;
335 1
    }
336
337
    /**
338
     * Return's the actual line number.
339
     *
340
     * @return integer The line number
341
     */
342 17
    public function getLineNumber()
343
    {
344 17
        return $this->lineNumber;
345
    }
346
347
    /**
348
     * Return's the default callback mappings.
349
     *
350
     * @return array The default callback mappings
351
     */
352 1
    public function getDefaultCallbackMappings()
353
    {
354 1
        return array();
355
    }
356
357
    /**
358
     * Load the default column values from the configuration.
359
     *
360
     * @return array The array with the default column values
361
     */
362 81 View Code Duplication
    public function getDefaultColumnValues()
0 ignored issues
show
Duplication introduced by
This method seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
363
    {
364
365
        // initialize the array for the default column values
366 81
        $defaultColumnValues = array();
367
368
        // load the entity type from the execution context
369 81
        $entityTypeCode = $this->getExecutionContext()->getEntityTypeCode();
370
371
        // load the column values from the configuration
372 81
        $columnValues = $this->getConfiguration()->getDefaultValues();
373
374
        // query whether or not default column values for the entity type are available
375 81
        if (isset($columnValues[$entityTypeCode])) {
376
            $defaultColumnValues = $columnValues[$entityTypeCode];
377
        }
378
379
        // return the default column values
380 81
        return $defaultColumnValues;
381
    }
382
383
    /**
384
     * Load the default header mappings from the configuration.
385
     *
386
     * @return array The array with the default header mappings
387
     */
388 81 View Code Duplication
    public function getDefaultHeaderMappings()
0 ignored issues
show
Duplication introduced by
This method seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
389
    {
390
391
        // initialize the array for the default header mappings
392 81
        $defaultHeaderMappings = array();
393
394
        // load the entity type from the execution context
395 81
        $entityTypeCode = $this->getExecutionContext()->getEntityTypeCode();
396
397
        // load the header mappings from the configuration
398 81
        $headerMappings = $this->getConfiguration()->getHeaderMappings();
399
400
        // query whether or not header mappings for the entity type are available
401 81
        if (isset($headerMappings[$entityTypeCode])) {
402
            $defaultHeaderMappings = $headerMappings[$entityTypeCode];
403
        }
404
405
        // return the default header mappings
406 81
        return $defaultHeaderMappings;
407
    }
408
409
    /**
410
     * Tries to format the passed value to a valid date with format 'Y-m-d H:i:s'.
411
     * If the passed value is NOT a valid date, NULL will be returned.
412
     *
413
     * @param string $value The value to format
414
     *
415
     * @return string|null The formatted date or NULL if the date is not valid
416
     * @throws \InvalidArgumentException Is thrown, if the passed can not be formatted according to the configured date format
417
     */
418
    public function formatDate($value)
419
    {
420
421
        // try to format the date according to the configured date format
422
        $formattedDate = $this->getDateConverter()->convert($value);
423
424
        // query whether or not the formatting was successufull
425
        if ($formattedDate === null) {
426
            throw new \InvalidArgumentException(
427
                sprintf('Can\'t format date "%s" according given format "%s"', $value, $this->getSourceDateFormat())
428
            );
429
        }
430
431
        // return the formatted date
432
        return $formattedDate;
433
    }
434
435
    /**
436
     * Extracts the elements of the passed value by exploding them
437
     * with the also passed delimiter.
438
     *
439
     * @param string      $value     The value to extract
440
     * @param string|null $delimiter The delimiter used to extrace the elements
441
     *
442
     * @return array The exploded values
443
     */
444
    public function explode($value, $delimiter = null)
445
    {
446
        return $this->getImportAdapter()->explode($value, $delimiter);
447
    }
448
449
    /**
450
     * Queries whether or not debug mode is enabled or not, default is FALSE.
451
     *
452
     * @return boolean TRUE if debug mode is enabled, else FALSE
453
     */
454 1
    public function isDebugMode()
455
    {
456 1
        return $this->getConfiguration()->isDebugMode();
457
    }
458
459
    /**
460
     * Queries whether or not strict mode is enabled or not, default is FALSE.
461
     *
462
     * @return boolean TRUE if strict mode is enabled, else FALSE
463
     */
464
    public function isStrictMode()
465
    {
466
        return $this->getConfiguration()->isStrictMode();
467
    }
468
469
    /**
470
     * Return's the subject's execution context configuration.
471
     *
472
     * @return \TechDivision\Import\ExecutionContextInterface The execution context configuration to use
473
     */
474
    public function getExecutionContext()
475
    {
476 81
        return $this->getConfiguration()->getPluginConfiguration()->getExecutionContext();
477
    }
478 81
479 81
    /**
480
     * Set's the subject configuration.
481
     *
482
     * @param \TechDivision\Import\Configuration\SubjectConfigurationInterface $configuration The subject configuration
483
     *
484
     * @return void
485
     */
486 81
    public function setConfiguration(SubjectConfigurationInterface $configuration)
487
    {
488 81
        $this->configuration = $configuration;
489
    }
490
491
    /**
492
     * Return's the subject configuration.
493
     *
494
     * @return \TechDivision\Import\Configuration\SubjectConfigurationInterface The subject configuration
495
     */
496
    public function getConfiguration()
497
    {
498 1
        return $this->configuration;
499
    }
500 1
501 1
    /**
502
     * Set's the import adapter instance.
503
     *
504
     * @param \TechDivision\Import\Adapter\ImportAdapterInterface $importAdapter The import adapter instance
505
     *
506
     * @return void
507
     */
508 1
    public function setImportAdapter(ImportAdapterInterface $importAdapter)
509
    {
510 1
        $this->importAdapter = $importAdapter;
511
    }
512
513
    /**
514
     * Return's the import adapter instance.
515
     *
516
     * @return \TechDivision\Import\Adapter\ImportAdapterInterface The import adapter instance
517
     */
518 81
    public function getImportAdapter()
519
    {
520 81
        return $this->importAdapter;
521
    }
522
523
    /**
524
     * Return's the RegistryProcessor instance to handle the running threads.
525
     *
526
     * @return \TechDivision\Import\Services\RegistryProcessorInterface The registry processor instance
527
     */
528
    public function getRegistryProcessor()
529
    {
530 8
        return $this->registryProcessor;
531
    }
532 8
533 8
    /**
534
     * Set's the unique serial for this import process.
535
     *
536
     * @param string $serial The unique serial
537
     *
538
     * @return void
539
     */
540 1
    public function setSerial($serial)
541
    {
542 1
        $this->serial = $serial;
543
    }
544
545
    /**
546
     * Return's the unique serial for this import process.
547
     *
548
     * @return string The unique serial
549
     */
550
    public function getSerial()
551
    {
552 4
        return $this->serial;
553
    }
554 4
555 4
    /**
556
     * Merge's the passed status into the actual one.
557
     *
558
     * @param array $status The status to MergeBuilder
559
     *
560
     * @return void
561
     */
562
    public function mergeStatus(array $status)
563
    {
564
        $this->status = array_replace_recursive($this->status, $status);
565
    }
566
567
    /**
568
     * Retur's the actual status.
569
     *
570
     * @return array The actual status
571
     */
572 4
    public function getStatus()
573
    {
574 4
        return $this->status;
575
    }
576
577
    /**
578
     * Return's the unique identifier for the actual invocation.
579
     *
580
     * @return string The unique identifier
581
     */
582
    public function getUniqueId()
583
    {
584
        return $this->uniqueId;
585
    }
586
587
    /**
588
     * Return's the source date format to use.
589
     *
590
     * @return string The source date format
591
     */
592 1
    public function getSourceDateFormat()
593
    {
594 1
        return $this->getConfiguration()->getSourceDateFormat();
0 ignored issues
show
Bug introduced by
The method getSourceDateFormat() does not seem to exist on object<TechDivision\Impo...ConfigurationInterface>.

This check looks for calls to methods that do not seem to exist on a given type. It looks for the method on the type itself as well as in inherited classes or implemented interfaces.

This is most likely a typographical error or the method has been renamed.

Loading history...
595
    }
596
597
    /**
598
     * Return's the multiple field delimiter character to use, default value is comma (,).
599
     *
600
     * @return string The multiple field delimiter character
601
     */
602 1
    public function getMultipleFieldDelimiter()
603
    {
604 1
        return $this->getConfiguration()->getMultipleFieldDelimiter();
605
    }
606
607
    /**
608
     * Return's the multiple value delimiter character to use, default value is comma (|).
609
     *
610
     * @return string The multiple value delimiter character
611
     */
612
    public function getMultipleValueDelimiter()
613
    {
614 81
        return $this->getConfiguration()->getMultipleValueDelimiter();
615
    }
616
617
    /**
618 81
     * Intializes the previously loaded global data for exactly one bunch.
619
     *
620
     * @param string $serial The serial of the actual import
621 81
     *
622
     * @return void
623
     */
624 81
    public function setUp($serial)
625 81
    {
626 81
627 81
        // initialize the unique ID for the actual invocation
628 81
        $this->uniqueId = Uuid::uuid4()->toString();
629 81
630
        // load the status of the actual import
631
        $status = $this->getRegistryProcessor()->getAttribute(RegistryKeys::STATUS);
632
633 81
        // load the global data, if prepared initially
634
        if (isset($status[RegistryKeys::GLOBAL_DATA])) {
635
            $this->stores = $status[RegistryKeys::GLOBAL_DATA][RegistryKeys::STORES];
636 81
            $this->defaultStore = $status[RegistryKeys::GLOBAL_DATA][RegistryKeys::DEFAULT_STORE];
637
            $this->storeWebsites  = $status[RegistryKeys::GLOBAL_DATA][RegistryKeys::STORE_WEBSITES];
638
            $this->rootCategories = $status[RegistryKeys::GLOBAL_DATA][RegistryKeys::ROOT_CATEGORIES];
639 81
            $this->coreConfigData = $status[RegistryKeys::GLOBAL_DATA][RegistryKeys::CORE_CONFIG_DATA];
640
        }
641
642 81
        // merge the header mappings with the values found in the configuration
643
        $this->headerMappings = array_merge($this->headerMappings, $this->getDefaultHeaderMappings());
644
645 81
        // merge the callback mappings with the mappings from the child instance
646 81
        $this->callbackMappings = array_merge($this->callbackMappings, $this->getDefaultCallbackMappings());
647
648
        // merge the default column values with the values found in the configuration
649 81
        $this->defaultColumnValues = array_merge($this->defaultColumnValues, $this->getDefaultColumnValues());
650 81
651 81
        // load the available callbacks from the configuration
652
        $availableCallbacks = $this->getConfiguration()->getCallbacks();
653
654
        // merge the callback mappings the the one from the configuration file
655
        foreach ($availableCallbacks as $callbackMappings) {
656 81
            foreach ($callbackMappings as $attributeCode => $mappings) {
657
                // write a log message, that default callback configuration will
658
                // be overwritten with the one from the configuration file
659 81
                if (isset($this->callbackMappings[$attributeCode])) {
660
                    $this->getSystemLogger()->notice(
661
                        sprintf('Now override callback mappings for attribute %s with values found in configuration file', $attributeCode)
662
                    );
663
                }
664
665
                // override the attributes callbacks
666
                $this->callbackMappings[$attributeCode] = $mappings;
667
            }
668 1
        }
669
    }
670
671
    /**
672 1
     * Clean up the global data after importing the variants.
673
     *
674
     * @param string $serial The serial of the actual import
675 1
     *
676 1
     * @return void
677
     */
678
    public function tearDown($serial)
679
    {
680 1
681 1
        // load the registry processor
682
        $registryProcessor = $this->getRegistryProcessor();
683 1
684
        // update the source directory for the next subject
685
        foreach ($this->getStatus() as $key => $status) {
686
            $registryProcessor->mergeAttributesRecursive($key, $status);
687
        }
688
689
        // log a debug message with the new source directory
690 1
        $this->getSystemLogger()->debug(
691
            sprintf('Subject %s successfully updated status data for import %s', get_class($this), $serial)
692
        );
693
    }
694 1
695
    /**
696
     * Return's the target directory for the artefact export.
697 1
     *
698 1
     * @return string The target directory for the artefact export
699
     */
700 View Code Duplication
    public function getTargetDir()
0 ignored issues
show
Duplication introduced by
This method seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
701
    {
702
703
        // load the status from the registry processor
704
        $status = $this->getRegistryProcessor()->getAttribute(RegistryKeys::STATUS);
705
706
        // query whether or not a target directory (mandatory) has been configured
707
        if (isset($status[RegistryKeys::TARGET_DIRECTORY])) {
708
            return $status[RegistryKeys::TARGET_DIRECTORY];
709
        }
710
711
        // throw an exception if the root category is NOT available
712
        throw new \Exception(sprintf('Can\'t find a target directory in status data for import %s', $this->getSerial()));
713 6
    }
714
715
    /**
716
     * Register the passed observer with the specific type.
717
     *
718 6
     * @param \TechDivision\Import\Observers\ObserverInterface $observer The observer to register
719 6
     * @param string                                           $type     The type to register the observer with
720
     *
721
     * @return void
722
     */
723 6
    public function registerObserver(ObserverInterface $observer, $type)
724 6
    {
725
726
        // query whether or not the array with the callbacks for the
727
        // passed type has already been initialized, or not
728
        if (!isset($this->observers[$type])) {
729
            $this->observers[$type] = array();
730
        }
731
732
        // append the callback with the instance of the passed type
733
        $this->observers[$type][] = $observer;
734 2
    }
735
736
    /**
737
     * Register the passed callback with the specific type.
738
     *
739 2
     * @param \TechDivision\Import\Callbacks\CallbackInterface $callback The subject to register the callbacks for
740 2
     * @param string                                           $type     The type to register the callback with
741
     *
742
     * @return void
743
     */
744 2
    public function registerCallback(CallbackInterface $callback, $type)
745 2
    {
746
747
        // query whether or not the array with the callbacks for the
748
        // passed type has already been initialized, or not
749
        if (!isset($this->callbacks[$type])) {
750
            $this->callbacks[$type] = array();
751
        }
752
753
        // append the callback with the instance of the passed type
754 1
        $this->callbacks[$type][] = $callback;
755
    }
756
757
    /**
758 1
     * Return's the array with callbacks for the passed type.
759
     *
760
     * @param string $type The type of the callbacks to return
761 1
     *
762 1
     * @return array The callbacks
763
     */
764
    public function getCallbacksByType($type)
765
    {
766 1
767
        // initialize the array for the callbacks
768
        $callbacks = array();
769
770
        // query whether or not callbacks for the type are available
771
        if (isset($this->callbacks[$type])) {
772
            $callbacks = $this->callbacks[$type];
773
        }
774 6
775
        // return the array with the type's callbacks
776 6
        return $callbacks;
777
    }
778
779
    /**
780
     * Return's the array with the available observers.
781
     *
782
     * @return array The observers
783
     */
784 1
    public function getObservers()
785
    {
786 1
        return $this->observers;
787
    }
788
789
    /**
790
     * Return's the array with the available callbacks.
791
     *
792
     * @return array The callbacks
793
     */
794 2
    public function getCallbacks()
795
    {
796 2
        return $this->callbacks;
797
    }
798
799
    /**
800
     * Return's the callback mappings for this subject.
801
     *
802
     * @return array The array with the subject's callback mappings
803
     */
804
    public function getCallbackMappings()
805
    {
806
        return $this->callbackMappings;
807
    }
808
809 2
    /**
810
     * Imports the content of the file with the passed filename.
811
     *
812
     *
813
     * @param string $serial   The serial of the actual import
814 2
     * @param string $filename The filename to process
815 2
     *
816
     * @return void
817
     * @throws \Exception Is thrown, if the import can't be processed
818 2
     */
819 2
    public function import($serial, $filename)
820
    {
821
822 2
        try {
823
            // initialize the serial/filename
824
            $this->setSerial($serial);
825 2
            $this->setFilename($filename);
826 2
827 2
            // invoke the events that has to be fired before the artfact will be processed
828
            $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_PROCESS_START, $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
829
            $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_PROCESS_START), $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
830 2
831 1
            // load the system logger instance
832 2
            $systemLogger = $this->getSystemLogger();
833
834
            // prepare the flag filenames
835 1
            $inProgressFilename = sprintf('%s.inProgress', $filename);
836 1
            $importedFilename = sprintf('%s.imported', $filename);
837
            $failedFilename = sprintf('%s.failed', $filename);
838
839
            // query whether or not the file has already been imported
840 1
            if ($this->isFile($failedFilename) ||
841
                $this->isFile($importedFilename) ||
842
                $this->isFile($inProgressFilename)
843 1
            ) {
844
                // log a debug message and exit
845
                $systemLogger->debug(sprintf('Import running, found inProgress file "%s"', $inProgressFilename));
846 1
                return;
847
            }
848
849 1
            // flag file as in progress
850 1
            $this->touch($inProgressFilename);
851 1
852
            // track the start time
853
            $startTime = microtime(true);
854
855 1
            // initialize the last time we've logged the counter with the processed rows per minute
856
            $this->lastLog = time();
857
858 1
            // log a message that the file has to be imported
859
            $systemLogger->info(
860
                sprintf('Now start processing file "%s"', basename($filename)),
861
                array('operation-name' => $operationName = $this->getFullOperationName())
862 1
            );
863 1
864 1
            // let the adapter process the file
865 1
            $this->getImportAdapter()->import(array($this, 'importRow'), $filename);
866 1
867 1
            // track the time needed for the import in seconds
868 1
            $endTime = microtime(true) - $startTime;
869 1
870
            // log a message that the file has successfully been imported,
871 1
            // use log level warning ONLY if rows have been skipped
872
            $systemLogger->log(
873
                $skippedRows = $this->getSkippedRows() > 0 ? LogLevel::WARNING : LogLevel::INFO,
874
                sprintf(
875 1
                    'Successfully processed file "%s" with "%d" lines (skipping "%d") in "%f" s',
876 1
                    basename($filename),
877
                    $this->getLineNumber() - 1,
878
                    $skippedRows,
879
                    $endTime
880
                ),
881
                array('operation-name' => $operationName)
882 1
            );
883
884
            // rename flag file, because import has been successfull
885
            if ($this->getConfiguration()->isCreatingImportedFile()) {
886
                $this->rename($inProgressFilename, $importedFilename);
887 1
            } else {
888 1
                $this->getFilesystemAdapter()->delete($inProgressFilename);
889 1
            }
890 1
891
            // update the status
892
            $this->mergeStatus(
893
                array(
894
                    RegistryKeys::STATUS => array(
895
                        RegistryKeys::FILES => array(
896
                            $filename => array(
897
                                $this->getUniqueId() => array(
898
                                    RegistryKeys::STATUS => 1,
899 1
                                    RegistryKeys::SKIPPED_ROWS => $this->getSkippedRows(),
900 1
                                    RegistryKeys::PROCESSED_ROWS => $this->getLineNumber() - 1
901
                                )
902
                            )
903
                        )
904
                    )
905
                )
906
            );
907
908
            // invoke the events that has to be fired when the artfact has been successfully processed
909
            $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_PROCESS_SUCCESS, $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
910
            $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_PROCESS_SUCCESS), $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
911
        } catch (\Exception $e) {
912
            // rename the flag file, because import failed and write the stack trace
913
            $this->rename($inProgressFilename, $failedFilename);
0 ignored issues
show
Bug introduced by
The variable $inProgressFilename does not seem to be defined for all execution paths leading up to this point.

If you define a variable conditionally, it can happen that it is not defined for all execution paths.

Let’s take a look at an example:

function myFunction($a) {
    switch ($a) {
        case 'foo':
            $x = 1;
            break;

        case 'bar':
            $x = 2;
            break;
    }

    // $x is potentially undefined here.
    echo $x;
}

In the above example, the variable $x is defined if you pass “foo” or “bar” as argument for $a. However, since the switch statement has no default case statement, if you pass any other value, the variable $x would be undefined.

Available Fixes

  1. Check for existence of the variable explicitly:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        if (isset($x)) { // Make sure it's always set.
            echo $x;
        }
    }
    
  2. Define a default value for the variable:

    function myFunction($a) {
        $x = ''; // Set a default which gets overridden for certain paths.
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        echo $x;
    }
    
  3. Add a value for the missing path:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
    
            // We add support for the missing case.
            default:
                $x = '';
                break;
        }
    
        echo $x;
    }
    
Loading history...
Bug introduced by
The variable $failedFilename does not seem to be defined for all execution paths leading up to this point.

If you define a variable conditionally, it can happen that it is not defined for all execution paths.

Let’s take a look at an example:

function myFunction($a) {
    switch ($a) {
        case 'foo':
            $x = 1;
            break;

        case 'bar':
            $x = 2;
            break;
    }

    // $x is potentially undefined here.
    echo $x;
}

In the above example, the variable $x is defined if you pass “foo” or “bar” as argument for $a. However, since the switch statement has no default case statement, if you pass any other value, the variable $x would be undefined.

Available Fixes

  1. Check for existence of the variable explicitly:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        if (isset($x)) { // Make sure it's always set.
            echo $x;
        }
    }
    
  2. Define a default value for the variable:

    function myFunction($a) {
        $x = ''; // Set a default which gets overridden for certain paths.
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        echo $x;
    }
    
  3. Add a value for the missing path:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
    
            // We add support for the missing case.
            default:
                $x = '';
                break;
        }
    
        echo $x;
    }
    
Loading history...
914
            $this->write($failedFilename, $e->__toString());
915
916
            // update the status with the error message
917
            $this->mergeStatus(
918
                array(
919
                    RegistryKeys::STATUS => array(
920
                        RegistryKeys::FILES => array(
921
                            $filename => array(
922
                                $this->getUniqueId() => array(
923
                                    RegistryKeys::STATUS         => 2,
924
                                    RegistryKeys::ERROR_MESSAGE  => $e->getMessage(),
925
                                    RegistryKeys::SKIPPED_ROWS => $this->getSkippedRows(),
926
                                    RegistryKeys::PROCESSED_ROWS => $this->getLineNumber() - 1
927
                                )
928
                            )
929
                        )
930
                    )
931
                )
932
            );
933
934
            // invoke the events that has to be fired when the artfact can't be processed
935
            $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_PROCESS_FAILURE, $this, $e);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
936 1
            $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_PROCESS_FAILURE), $this, $e);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
937
938
            // do not wrap the exception if not already done
939
            if ($e instanceof WrappedColumnException) {
940
                throw $e;
941
            }
942
943
            // else wrap and throw the exception
944
            throw $this->wrapException(array(), $e);
945
        }
946 7
    }
947
948
    /**
949
     * Imports the passed row into the database. If the import failed, the exception
950 7
     * will be catched and logged, but the import process will be continued.
951
     *
952
     * @param array $row The row with the data to be imported
953 7
     *
954 7
     * @return void
955
     */
956
    public function importRow(array $row)
957 7
    {
958 7
959
        // initialize the row
960
        $this->row = $row;
961 7
962
        // raise the line number and reset the skip row flag
963 1
        $this->lineNumber++;
964 1
        $this->skipRow = false;
965
966
        // invoke the events that has to be fired before the artfact's row will be processed
967 1
        $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_ROW_PROCESS_START, $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
968 1
        $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_ROW_PROCESS_START), $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
969
970
        // initialize the headers with the columns from the first line
971
        if (sizeof($this->headers) === 0) {
972 1
            // invoke the events that has to be fired before the artfact's header row will be processed
973
            $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_HEADER_ROW_PROCESS_START, $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
974
            $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_HEADER_ROW_PROCESS_START), $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
975
976
            // iterate over the column name => key an map the header names, if necessary
977
            foreach ($this->row as $value => $key) {
978
                $this->headers[$this->mapAttributeCodeByHeaderMapping($key)] = $value;
979
            }
980
981
            // iterate over the default column values to figure out whether or not the column exists
982
            foreach ($this->defaultColumnValues as $name => $value) {
983 1
                // do nothing, if the column already exists
984 1
                if (array_key_exists($key = $this->mapAttributeCodeByHeaderMapping($name), $this->headers)) {
985
                    continue;
986
                }
987 6
                // add the header and the default value for the column
988
                $this->headers[$key] = $columnKey = sizeof($this->headers);
989
                $this->columnValues[$columnKey] = $value;
990 6
            }
991
992
            // invoke the events that has to be fired when the artfact's header row has been successfully processed
993 6
            $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_HEADER_ROW_PROCESS_SUCCESS, $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
994
            $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_HEADER_ROW_PROCESS_SUCCESS), $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
995
        } else {
996 6
            // merge the default column value into the actual row
997
            $this->row = array_replace($this->row, $this->columnValues);
998 6
999
            // load the available observers
1000 1
            $availableObservers = $this->getObservers();
1001 1
1002 1
            // process the observers
1003 1
            foreach ($availableObservers as $observers) {
1004 1
                // invoke the pre-import/import and post-import observers
1005 1
                /** @var \TechDivision\Import\Observers\ObserverInterface $observer */
1006
                foreach ($observers as $observer) {
1007
                    // query whether or not we have to skip the row
1008
                    if ($this->skipRow) {
1009
                        // log a debug message with the actual line nr/file information
1010
                        $this->getSystemLogger()->debug(
1011 1
                            $this->appendExceptionSuffix(
1012
                                sprintf(
1013
                                    'Skip processing operation "%s" after observer "%s"',
1014
                                    $this->getFullOperationName(),
1015 6
                                    get_class($observer)
1016 6
                                )
1017
                            )
1018
                        );
1019
1020
                        // skip the row
1021
                        break 2;
1022
                    }
1023 7
1024
                    // if not, set the subject and process the observer
1025 7
                    if ($observer instanceof ObserverInterface) {
1026 7
                        $this->row = $observer->handle($this);
1027 7
                    }
1028 7
                }
1029 7
            }
1030 7
        }
1031 7
1032
        // query whether or not a minute has been passed
1033
        if ($this->lastLog < time() - 59) {
1034
            // log the number processed rows per minute
1035
            $this->getSystemLogger()->info(
1036 7
                sprintf(
1037 7
                    'Operation "%s" successfully processed "%d (%d)" rows per minute of file "%s"',
1038
                    $this->getFullOperationName(),
1039
                    $this->lineNumber - $this->lastLineNumber,
1040
                    $this->lineNumber,
1041 7
                    basename($this->getFilename())
1042 7
                )
1043 7
            );
1044 7
1045 7
            // reset the last log time and the line number
1046
            $this->lastLog = time();
1047
            $this->lastLineNumber = $this->lineNumber;
1048
        }
1049
1050
        // log a debug message with the actual line nr/file information
1051 7
        $this->getSystemLogger()->debug(
1052 7
            $this->appendExceptionSuffix(
1053 7
                sprintf(
1054
                    'Successfully processed operation "%s"',
1055
                    implode(' > ', $this->getConfiguration()->getConfiguration()->getOperationNames())
1056
                )
1057
            )
1058
        );
1059
1060 1
        // invoke the events that has to be fired when the artfact's row has been successfully processed
1061
        $this->getEmitter()->emit(EventNames::SUBJECT_ARTEFACT_ROW_PROCESS_SUCCESS, $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
1062 1
        $this->getEmitter()->emit($this->getEventName(EventNames::SUBJECT_ARTEFACT_ROW_PROCESS_SUCCESS), $this);
0 ignored issues
show
Unused Code introduced by
The call to EmitterInterface::emit() has too many arguments starting with $this.

This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.

If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.

In this case you can add the @ignore PhpDoc annotation to the duplicate definition and it will be ignored.

Loading history...
1063
    }
1064
1065
    /**
1066
     * Queries whether or not that the subject needs an OK file to be processed.
1067
     *
1068
     * @return boolean TRUE if the subject needs an OK file, else FALSE
1069
     */
1070
    public function isOkFileNeeded()
1071
    {
1072
        return $this->getConfiguration()->isOkFileNeeded();
1073
    }
1074
1075
    /**
1076
     * Return's the default store.
1077
     *
1078
     * @return array The default store
1079
     */
1080 5
    public function getDefaultStore()
1081
    {
1082 5
        return $this->defaultStore;
1083
    }
1084
1085
    /**
1086
     * Return's the default store view code.
1087
     *
1088
     * @return array The default store view code
1089
     */
1090
    public function getDefaultStoreViewCode()
1091
    {
1092 4
        return $this->defaultStore[MemberNames::CODE];
1093
    }
1094 4
1095 4
    /**
1096
     * Set's the store view code the create the product/attributes for.
1097
     *
1098
     * @param string $storeViewCode The store view code
1099
     *
1100
     * @return void
1101
     */
1102
    public function setStoreViewCode($storeViewCode)
1103
    {
1104 8
        $this->storeViewCode = $storeViewCode;
1105
    }
1106
1107
    /**
1108 8
     * Return's the store view code the create the product/attributes for.
1109 4
     *
1110
     * @param string|null $default The default value to return, if the store view code has not been set
1111
     *
1112
     * @return string The store view code
1113 4
     */
1114
    public function getStoreViewCode($default = null)
1115 3
    {
1116
1117
        // return the store view code, if available
1118
        if ($this->storeViewCode !== null) {
1119 1
            return $this->storeViewCode;
1120
        }
1121
1122
        // if NOT and a default code is available
1123
        if ($default !== null) {
1124
            // return the default value
1125
            return $default;
1126
        }
1127
1128 2
        // return the default store view code
1129
        return $this->getDefaultStoreViewCode();
0 ignored issues
show
Bug Best Practice introduced by
The return type of return $this->getDefaultStoreViewCode(); (array) is incompatible with the return type declared by the interface TechDivision\Import\Subj...rface::getStoreViewCode of type string.

If you return a value from a function or method, it should be a sub-type of the type that is given by the parent type f.e. an interface, or abstract method. This is more formally defined by the Lizkov substitution principle, and guarantees that classes that depend on the parent type can use any instance of a child type interchangably. This principle also belongs to the SOLID principles for object oriented design.

Let’s take a look at an example:

class Author {
    private $name;

    public function __construct($name) {
        $this->name = $name;
    }

    public function getName() {
        return $this->name;
    }
}

abstract class Post {
    public function getAuthor() {
        return 'Johannes';
    }
}

class BlogPost extends Post {
    public function getAuthor() {
        return new Author('Johannes');
    }
}

class ForumPost extends Post { /* ... */ }

function my_function(Post $post) {
    echo strtoupper($post->getAuthor());
}

Our function my_function expects a Post object, and outputs the author of the post. The base class Post returns a simple string and outputting a simple string will work just fine. However, the child class BlogPost which is a sub-type of Post instead decided to return an object, and is therefore violating the SOLID principles. If a BlogPost were passed to my_function, PHP would not complain, but ultimately fail when executing the strtoupper call in its body.

Loading history...
1130
    }
1131
1132 2
    /**
1133
     * Prepare's the store view code in the subject. If the store_view_code row doesn't contain
1134
     * any value, the default code of the default store view will be set.
1135 2
     *
1136 2
     * @return void
1137
     */
1138 2
    public function prepareStoreViewCode()
1139
    {
1140
1141
        // re-set the store view code
1142
        $this->setStoreViewCode(null);
1143
1144
        // initialize the store view code
1145
        if ($storeViewCode = $this->getValue(ColumnKeys::STORE_VIEW_CODE)) {
1146
            $this->setStoreViewCode($storeViewCode);
1147
        }
1148 4
    }
1149
1150
    /**
1151
     * Return's the store ID of the store with the passed store view code
1152 4
     *
1153 3
     * @param string $storeViewCode The store view code to return the store ID for
1154
     *
1155
     * @return integer The ID of the store with the passed ID
1156
     * @throws \Exception Is thrown, if the store with the actual code is not available
1157 1
     */
1158 1
    public function getStoreId($storeViewCode)
1159 1
    {
1160 1
1161 1
        // query whether or not, the requested store is available
1162 1
        if (isset($this->stores[$storeViewCode])) {
1163
            return (integer) $this->stores[$storeViewCode][MemberNames::STORE_ID];
1164
        }
1165
1166
        // throw an exception, if not
1167
        throw new \Exception(
1168
            sprintf(
1169
                'Found invalid store view code %s in file %s on line %d',
1170
                $storeViewCode,
1171
                $this->getFilename(),
1172
                $this->getLineNumber()
1173
            )
1174
        );
1175
    }
1176 2
1177
    /**
1178
     * Return's the store ID of the actual row, or of the default store
1179
     * if no store view code is set in the CSV file.
1180 2
     *
1181 2
     * @param string|null $default The default store view code to use, if no store view code is set in the CSV file
1182
     *
1183
     * @return integer The ID of the actual store
1184
     * @throws \Exception Is thrown, if the store with the actual code is not available
1185 2
     */
1186
    public function getRowStoreId($default = null)
1187
    {
1188
1189
        // initialize the default store view code, if not passed
1190
        if ($default === null) {
1191
            $default = $this->getDefaultStoreViewCode();
1192
        }
1193
1194 2
        // load the store view code the create the product/attributes for
1195
        return $this->getStoreId($this->getStoreViewCode($default));
0 ignored issues
show
Bug introduced by
It seems like $default defined by $this->getDefaultStoreViewCode() on line 1191 can also be of type array; however, TechDivision\Import\Subj...ect::getStoreViewCode() does only seem to accept string|null, maybe add an additional type check?

If a method or function can return multiple different values and unless you are sure that you only can receive a single value in this context, we recommend to add an additional type check:

/**
 * @return array|string
 */
function returnsDifferentValues($x) {
    if ($x) {
        return 'foo';
    }

    return array();
}

$x = returnsDifferentValues($y);
if (is_array($x)) {
    // $x is an array.
}

If this a common case that PHP Analyzer should handle natively, please let us know by opening an issue.

Loading history...
Bug introduced by
It seems like $this->getStoreViewCode($default) targeting TechDivision\Import\Subj...ect::getStoreViewCode() can also be of type array; however, TechDivision\Import\Subj...ctSubject::getStoreId() does only seem to accept string, maybe add an additional type check?

This check looks at variables that are passed out again to other methods.

If the outgoing method call has stricter type requirements than the method itself, an issue is raised.

An additional type check may prevent trouble.

Loading history...
1196
    }
1197
1198 2
    /**
1199
     * Return's the root category for the actual view store.
1200
     *
1201 2
     * @return array The store's root category
1202 1
     * @throws \Exception Is thrown if the root category for the passed store code is NOT available
1203
     */
1204
    public function getRootCategory()
1205
    {
1206 1
1207
        // load the actual store view code
1208
        $storeViewCode = $this->getStoreViewCode($this->getDefaultStoreViewCode());
0 ignored issues
show
Documentation introduced by
$this->getDefaultStoreViewCode() is of type array, but the function expects a string|null.

It seems like the type of the argument is not accepted by the function/method which you are calling.

In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.

We suggest to add an explicit type cast like in the following example:

function acceptsInteger($int) { }

$x = '123'; // string "123"

// Instead of
acceptsInteger($x);

// we recommend to use
acceptsInteger((integer) $x);
Loading history...
1209
1210
        // query weather or not we've a root category or not
1211
        if (isset($this->rootCategories[$storeViewCode])) {
1212
            return $this->rootCategories[$storeViewCode];
1213
        }
1214
1215
        // throw an exception if the root category is NOT available
1216
        throw new \Exception(sprintf('Root category for %s is not available', $storeViewCode));
1217
    }
1218
1219
    /**
1220 5
     * Return's the Magento configuration value.
1221
     *
1222
     * @param string  $path    The Magento path of the requested configuration value
1223
     * @param mixed   $default The default value that has to be returned, if the requested configuration value is not set
1224
     * @param string  $scope   The scope the configuration value has been set
1225 5
     * @param integer $scopeId The scope ID the configuration value has been set
1226 5
     *
1227 5
     * @return mixed The configuration value
1228
     * @throws \Exception Is thrown, if nor a value can be found or a default value has been passed
1229
     */
1230
    public function getCoreConfigData($path, $default = null, $scope = ScopeKeys::SCOPE_DEFAULT, $scopeId = 0)
1231 5
    {
1232
1233
        // initialize the core config data
1234 5
        $coreConfigData = array(
1235 1
            MemberNames::PATH => $path,
1236
            MemberNames::SCOPE => $scope,
1237
            MemberNames::SCOPE_ID => $scopeId
1238
        );
1239 4
1240
        // generate the UID from the passed data
1241 2
        $uniqueIdentifier = $this->coreConfigDataUidGenerator->generate($coreConfigData);
1242 2
1243
        // iterate over the core config data and try to find the requested configuration value
1244 2
        if (isset($this->coreConfigData[$uniqueIdentifier])) {
1245 2
            return $this->coreConfigData[$uniqueIdentifier][MemberNames::VALUE];
1246
        }
1247 2
1248
        // query whether or not we've to query for the configuration value on fallback level 'websites' also
1249
        if ($scope === ScopeKeys::SCOPE_STORES) {
1250
            // query whether or not the website with the passed ID is available
1251
            foreach ($this->storeWebsites as $storeWebsite) {
1252
                if ($storeWebsite[MemberNames::WEBSITE_ID] === $scopeId) {
1253 2
                    // replace scope with 'websites' and website ID
1254
                    $coreConfigData = array_merge(
1255
                        $coreConfigData,
1256 2
                        array(
1257 2
                            MemberNames::SCOPE    => ScopeKeys::SCOPE_WEBSITES,
1258
                            MemberNames::SCOPE_ID => $storeWebsite[MemberNames::WEBSITE_ID]
1259
                        )
1260
                    );
1261
1262
                    // generate the UID from the passed data, merged with the 'websites' scope and ID
1263
                    $uniqueIdentifier = $this->coreConfigDataUidGenerator->generate($coreConfigData);
1264 3
1265 3
                    // query whether or not, the configuration value on 'websites' level
1266 View Code Duplication
                    if (isset($this->coreConfigData[$uniqueIdentifier][MemberNames::VALUE])) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
1267 3
                        return $this->coreConfigData[$uniqueIdentifier][MemberNames::VALUE];
1268
                    }
1269
                }
1270
            }
1271
        }
1272
1273 3
        // replace scope with 'default' and scope ID '0'
1274
        $coreConfigData = array_merge(
1275
            $coreConfigData,
1276 3
            array(
1277 1
                MemberNames::SCOPE    => ScopeKeys::SCOPE_DEFAULT,
1278
                MemberNames::SCOPE_ID => 0
1279
            )
1280
        );
1281 2
1282 1
        // generate the UID from the passed data, merged with the 'default' scope and ID 0
1283
        $uniqueIdentifier = $this->coreConfigDataUidGenerator->generate($coreConfigData);
1284
1285
        // query whether or not, the configuration value on 'default' level
1286 View Code Duplication
        if (isset($this->coreConfigData[$uniqueIdentifier][MemberNames::VALUE])) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
1287 1
            return $this->coreConfigData[$uniqueIdentifier][MemberNames::VALUE];
1288 1
        }
1289 1
1290 1
        // if not, return the passed default value
1291 1
        if ($default !== null) {
1292 1
            return $default;
1293
        }
1294
1295
        // throw an exception if no value can be found
1296
        // in the Magento configuration
1297
        throw new \Exception(
1298
            sprintf(
1299
                'Can\'t find a value for configuration "%s-%s-%d" in "core_config_data"',
1300
                $path,
1301
                $scope,
1302
                $scopeId
1303
            )
1304 2
        );
1305
    }
1306
1307
    /**
1308 2
     * Resolve the original column name for the passed one.
1309
     *
1310
     * @param string $columnName The column name that has to be resolved
1311 2
     *
1312
     * @return string|null The original column name
1313 1
     */
1314 1
    public function resolveOriginalColumnName($columnName)
1315
    {
1316
1317
        // try to load the original data
1318 1
        $originalData = $this->getOriginalData();
1319 1
1320
        // query whether or not original data is available
1321
        if (isset($originalData[ColumnKeys::ORIGINAL_COLUMN_NAMES])) {
1322
            // query whether or not the original column name is available
1323
            if (isset($originalData[ColumnKeys::ORIGINAL_COLUMN_NAMES][$columnName])) {
1324 1
                return $originalData[ColumnKeys::ORIGINAL_COLUMN_NAMES][$columnName];
1325
            }
1326
1327
            // query whether or a wildcard column name is available
1328
            if (isset($originalData[ColumnKeys::ORIGINAL_COLUMN_NAMES]['*'])) {
1329
                return $originalData[ColumnKeys::ORIGINAL_COLUMN_NAMES]['*'];
1330
            }
1331
        }
1332 2
1333
        // return the original column name
1334
        return $columnName;
1335
    }
1336 2
1337
    /**
1338
     * Return's the original data if available, or an empty array.
1339 2
     *
1340
     * @return array The original data
1341 1
     */
1342
    public function getOriginalData()
1343
    {
1344
1345 2
        // initialize the array for the original data
1346
        $originalData = array();
1347
1348
        // query whether or not the column contains original data
1349
        if ($this->hasOriginalData()) {
1350
            // unerialize the original data from the column
1351
            $originalData = unserialize($this->row[$this->headers[ColumnKeys::ORIGINAL_DATA]]);
1352
        }
1353
1354 2
        // return an empty array, if not
1355
        return $originalData;
1356 2
    }
1357
1358
    /**
1359
     * Query's whether or not the actual column contains original data like
1360
     * filename, line number and column names.
1361
     *
1362
     * @return boolean TRUE if the actual column contains origin data, else FALSE
1363
     */
1364
    public function hasOriginalData()
1365
    {
1366
        return isset($this->headers[ColumnKeys::ORIGINAL_DATA]) && isset($this->row[$this->headers[ColumnKeys::ORIGINAL_DATA]]);
1367
    }
1368
1369 1
    /**
1370
     * Wraps the passed exeception into a new one by trying to resolve the original filname,
1371
     * line number and column names and use it for a detailed exception message.
1372
     *
1373
     * @param array      $columnNames The column names that should be resolved and wrapped
1374
     * @param \Exception $parent      The exception we want to wrap
1375
     * @param string     $className   The class name of the exception type we want to wrap the parent one
1376 1
     *
1377
     * @return \Exception the wrapped exception
1378
     */
1379 1
    public function wrapException(
1380
        array $columnNames = array(),
1381 1
        \Exception $parent = null,
1382
        $className = '\TechDivision\Import\Exceptions\WrappedColumnException'
1383
    ) {
1384 1
1385 1
        // initialize the message
1386 1
        $message = $parent->getMessage();
0 ignored issues
show
Bug introduced by
It seems like $parent is not always an object, but can also be of type null. Maybe add an additional type check?

If a variable is not always an object, we recommend to add an additional type check to ensure your method call is safe:

function someFunction(A $objectMaybe = null)
{
    if ($objectMaybe instanceof A) {
        $objectMaybe->doSomething();
    }
}
Loading history...
1387 1
1388
        // query whether or not has been a result of invalid data of a previous column of a CSV file
1389
        if ($this->hasOriginalData()) {
1390
            // load the original data
1391
            $originalData = $this->getOriginalData();
1392
1393
            // replace old filename and line number of the original message
1394
            $message = $this->appendExceptionSuffix(
1395
                $this->stripExceptionSuffix($message),
1396
                $originalData[ColumnKeys::ORIGINAL_FILENAME],
1397
                $originalData[ColumnKeys::ORIGINAL_LINE_NUMBER]
1398
            );
1399 1
        } else {
1400
            // append filename and line number to the original message
1401 1
            $message = $this->appendExceptionSuffix(
1402 1
                $this->stripExceptionSuffix($message),
1403 1
                $this->filename,
1404
                $this->lineNumber
1405
            );
1406
        }
1407 1
1408
        // query whether or not, column names has been passed
1409
        if (sizeof($columnNames) > 0) {
1410
            // prepare the original column names
1411 1
            $originalColumnNames = array();
1412
            foreach ($columnNames as $columnName) {
1413
                $originalColumnNames[] = $this->resolveOriginalColumnName($columnName);
1414
            }
1415
1416
            // append the column information
1417
            $message = sprintf('%s in column(s) %s', $message, implode(', ', $originalColumnNames));
1418
        }
1419
1420
        // create a new exception and wrap the parent one
1421
        return new $className($message, null, $parent);
1422 1
    }
1423
1424 1
    /**
1425
     * Strip's the exception suffix containing filename and line number from the
1426
     * passed message.
1427
     *
1428
     * @param string $message The message to strip the exception suffix from
1429
     *
1430
     * @return mixed The message without the exception suffix
1431
     */
1432
    public function stripExceptionSuffix($message)
1433
    {
1434
        return str_replace($this->appendExceptionSuffix(), '', $message);
1435
    }
1436
1437
    /**
1438 12
     * Append's the exception suffix containing filename and line number to the
1439
     * passed message. If no message has been passed, only the suffix will be
1440
     * returned
1441
     *
1442 12
     * @param string|null $message    The message to append the exception suffix to
1443 12
     * @param string|null $filename   The filename used to create the suffix
1444
     * @param string|null $lineNumber The line number used to create the suffx
1445
     *
1446
     * @return string The message with the appended exception suffix
1447 12
     */
1448 12
    public function appendExceptionSuffix($message = null, $filename = null, $lineNumber = null)
1449
    {
1450
1451
        // query whether or not a filename has been passed
1452 12
        if ($filename === null) {
1453 1
            $filename = $this->getFilename();
1454
        }
1455
1456
        // query whether or not a line number has been passed
1457 12
        if ($lineNumber === null) {
1458
            $lineNumber = $this->getLineNumber();
1459
        }
1460
1461
        // if no message has been passed, only return the suffix
1462
        if ($message === null) {
1463
            return sprintf(' in file %s on line %d', basename($filename), $lineNumber);
1464
        }
1465
1466
        // concatenate the message with the suffix and return it
1467 1
        return sprintf('%s in file %s on line %d', $message, basename($filename), $lineNumber);
1468
    }
1469
1470
    /**
1471 1
     * Raises the value for the counter with the passed key by one.
1472 1
     *
1473 1
     * @param mixed $counterName The name of the counter to raise
1474
     *
1475
     * @return integer The counter's new value
1476
     */
1477
    public function raiseCounter($counterName)
1478
    {
1479
1480
        // raise the counter with the passed name
1481
        return $this->getRegistryProcessor()->raiseCounter(
1482
            RegistryKeys::COUNTERS,
1483
            $counterName
1484 1
        );
1485
    }
1486
1487
    /**
1488 1
     * Merge the passed array into the status of the actual import.
1489 1
     *
1490 1
     * @param array $status The status information to be merged
1491
     *
1492
     * @return void
1493
     */
1494
    public function mergeAttributesRecursive(array $status)
1495
    {
1496
1497
        // merge the passed status
1498
        return $this->getRegistryProcessor()->mergeAttributesRecursive(
1499
            RegistryKeys::STATUS,
1500
            $status
1501
        );
1502
    }
1503
1504
    /**
1505
     * Return's the entity type code to be used.
1506
     *
1507
     * @return string The entity type code to be used
1508
     */
1509
    public function getEntityTypeCode()
1510
    {
1511
1512
        // load the configuration specific entity type code from the plugin configuration
1513
        $entityTypeCode = $this->getExecutionContext()->getEntityTypeCode();
1514
1515
        // try to map the entity type code
1516
        if (isset($this->entityTypeCodeMappings[$entityTypeCode])) {
1517
            $entityTypeCode = $this->entityTypeCodeMappings[$entityTypeCode];
1518
        }
1519
1520
        // return the (mapped) entity type code
1521 9
        return $entityTypeCode;
1522
    }
1523 9
1524 9
    /**
1525 9
     * Concatenates and returns the event name for the actual plugin and subject context.
1526 9
     *
1527 9
     * @param string $eventName The event name to concatenate
1528
     *
1529
     * @return string The concatenated event name
1530
     */
1531
    protected function getEventName($eventName)
1532
    {
1533
        return  sprintf(
1534
            '%s.%s.%s',
1535
            $this->getConfiguration()->getPluginConfiguration()->getId(),
1536
            $this->getConfiguration()->getId(),
1537
            $eventName
1538
        );
1539
    }
1540
1541
    /**
1542
     * Return's the full opration name, which consists of the Magento edition, the entity type code and the operation name.
1543
     *
1544
     * @param string $separator The separator used to seperate the elements
1545
     *
1546
     * @return string The full operation name
1547
     */
1548
    public function getFullOperationName($separator = '/')
1549
    {
1550
        return $this->getConfiguration()->getFullOperationName($separator);
1551
    }
1552
}
1553