Passed
Push — master ( 8b1641...595372 )
by Aimeos
03:00
created

Standard   A

Complexity

Total Complexity 35

Size/Duplication

Total Lines 424
Duplicated Lines 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 89
dl 0
loc 424
rs 9.6
c 1
b 0
f 0
wmc 35

10 Methods

Rating   Name   Duplication   Size   Complexity  
A getDescription() 0 3 1
A getName() 0 3 1
A domains() 0 18 1
A importNode() 0 28 4
A run() 0 36 4
A location() 0 17 1
A import() 0 23 5
B importTree() 0 32 10
B process() 0 29 7
A backup() 0 30 1
1
<?php
2
3
/**
4
 * @license LGPLv3, http://opensource.org/licenses/LGPL-3.0
5
 * @copyright Aimeos (aimeos.org), 2019-2022
6
 * @package Controller
7
 * @subpackage Jobs
8
 */
9
10
11
namespace Aimeos\Controller\Jobs\Catalog\Import\Xml;
12
13
14
/**
15
 * Job controller for XML catalog imports
16
 *
17
 * @package Controller
18
 * @subpackage Jobs
19
 */
20
class Standard
21
	extends \Aimeos\Controller\Jobs\Base
22
	implements \Aimeos\Controller\Jobs\Iface
23
{
24
	/** controller/jobs/catalog/import/xml/name
25
	 * Class name of the used catalog suggestions scheduler controller implementation
26
	 *
27
	 * Each default job controller can be replace by an alternative imlementation.
28
	 * To use this implementation, you have to set the last part of the class
29
	 * name as configuration value so the controller factory knows which class it
30
	 * has to instantiate.
31
	 *
32
	 * For example, if the name of the default class is
33
	 *
34
	 *  \Aimeos\Controller\Jobs\Catalog\Import\Xml\Standard
35
	 *
36
	 * and you want to replace it with your own version named
37
	 *
38
	 *  \Aimeos\Controller\Jobs\Catalog\Import\Xml\Myxml
39
	 *
40
	 * then you have to set the this configuration option:
41
	 *
42
	 *  controller/jobs/catalog/import/xml/name = Myxml
43
	 *
44
	 * The value is the last part of your own class name and it's case sensitive,
45
	 * so take care that the configuration value is exactly named like the last
46
	 * part of the class name.
47
	 *
48
	 * The allowed characters of the class name are A-Z, a-z and 0-9. No other
49
	 * characters are possible! You should always start the last part of the class
50
	 * name with an upper case character and continue only with lower case characters
51
	 * or numbers. Avoid chamel case names like "MyXml"!
52
	 *
53
	 * @param string Last part of the class name
54
	 * @since 2019.04
55
	 */
56
57
	/** controller/jobs/catalog/import/xml/decorators/excludes
58
	 * Excludes decorators added by the "common" option from the catalog import CSV job controller
59
	 *
60
	 * Decorators extend the functionality of a class by adding new aspects
61
	 * (e.g. log what is currently done), executing the methods of the underlying
62
	 * class only in certain conditions (e.g. only for logged in users) or
63
	 * modify what is returned to the caller.
64
	 *
65
	 * This option allows you to remove a decorator added via
66
	 * "controller/jobs/common/decorators/default" before they are wrapped
67
	 * around the job controller.
68
	 *
69
	 *  controller/jobs/catalog/import/xml/decorators/excludes = array( 'decorator1' )
70
	 *
71
	 * This would remove the decorator named "decorator1" from the list of
72
	 * common decorators ("\Aimeos\Controller\Jobs\Common\Decorator\*") added via
73
	 * "controller/jobs/common/decorators/default" to the job controller.
74
	 *
75
	 * @param array List of decorator names
76
	 * @since 2019.04
77
	 * @see controller/jobs/common/decorators/default
78
	 * @see controller/jobs/catalog/import/xml/decorators/global
79
	 * @see controller/jobs/catalog/import/xml/decorators/local
80
	 */
81
82
	/** controller/jobs/catalog/import/xml/decorators/global
83
	 * Adds a list of globally available decorators only to the catalog import CSV job controller
84
	 *
85
	 * Decorators extend the functionality of a class by adding new aspects
86
	 * (e.g. log what is currently done), executing the methods of the underlying
87
	 * class only in certain conditions (e.g. only for logged in users) or
88
	 * modify what is returned to the caller.
89
	 *
90
	 * This option allows you to wrap global decorators
91
	 * ("\Aimeos\Controller\Jobs\Common\Decorator\*") around the job controller.
92
	 *
93
	 *  controller/jobs/catalog/import/xml/decorators/global = array( 'decorator1' )
94
	 *
95
	 * This would add the decorator named "decorator1" defined by
96
	 * "\Aimeos\Controller\Jobs\Common\Decorator\Decorator1" only to the job controller.
97
	 *
98
	 * @param array List of decorator names
99
	 * @since 2019.04
100
	 * @see controller/jobs/common/decorators/default
101
	 * @see controller/jobs/catalog/import/xml/decorators/excludes
102
	 * @see controller/jobs/catalog/import/xml/decorators/local
103
	 */
104
105
	/** controller/jobs/catalog/import/xml/decorators/local
106
	 * Adds a list of local decorators only to the catalog import CSV job controller
107
	 *
108
	 * Decorators extend the functionality of a class by adding new aspects
109
	 * (e.g. log what is currently done), executing the methods of the underlying
110
	 * class only in certain conditions (e.g. only for logged in users) or
111
	 * modify what is returned to the caller.
112
	 *
113
	 * This option allows you to wrap local decorators
114
	 * ("\Aimeos\Controller\Jobs\Catalog\Import\Xml\Decorator\*") around the job
115
	 * controller.
116
	 *
117
	 *  controller/jobs/catalog/import/xml/decorators/local = array( 'decorator2' )
118
	 *
119
	 * This would add the decorator named "decorator2" defined by
120
	 * "\Aimeos\Controller\Jobs\Catalog\Import\Xml\Decorator\Decorator2"
121
	 * only to the job controller.
122
	 *
123
	 * @param array List of decorator names
124
	 * @since 2019.04
125
	 * @see controller/jobs/common/decorators/default
126
	 * @see controller/jobs/catalog/import/xml/decorators/excludes
127
	 * @see controller/jobs/catalog/import/xml/decorators/global
128
	 */
129
130
131
	use \Aimeos\Controller\Common\Common\Import\Xml\Traits;
132
133
134
	/**
135
	 * Returns the localized name of the job.
136
	 *
137
	 * @return string Name of the job
138
	 */
139
	public function getName() : string
140
	{
141
		return $this->context()->translate( 'controller/jobs', 'Catalog import XML' );
142
	}
143
144
145
	/**
146
	 * Returns the localized description of the job.
147
	 *
148
	 * @return string Description of the job
149
	 */
150
	public function getDescription() : string
151
	{
152
		return $this->context()->translate( 'controller/jobs', 'Imports new and updates existing categories from XML files' );
153
	}
154
155
156
	/**
157
	 * Executes the job.
158
	 *
159
	 * @throws \Aimeos\Controller\Jobs\Exception If an error occurs
160
	 */
161
	public function run()
162
	{
163
		$context = $this->context();
164
		$logger = $context->logger();
165
		$process = $context->process();
166
167
		$location = $this->location();
168
		$fs = $context->fs( 'fs-import' );
169
170
		if( $fs->isDir( $location ) === false ) {
171
			return;
172
		}
173
174
		try
175
		{
176
			$logger->info( sprintf( 'Started catalog import from "%1$s"', $location ), 'import/xml/catalog' );
177
178
			$fcn = function( \Aimeos\MShop\ContextIface $context, string $path ) {
179
				$this->import( $context, $path );
180
			};
181
182
			foreach( map( $fs->scan( $location ) )->sort() as $filename ) {
183
				$process->start( $fcn, [$context, $fs->readf( $location . '/' . $filename )] );
184
			}
185
186
			$process->wait();
187
188
			$context->process()->wait();
189
190
			$logger->info( sprintf( 'Finished catalog import from "%1$s"', $location ), 'import/xml/catalog' );
191
		}
192
		catch( \Exception $e )
193
		{
194
			$logger->error( 'Catalog import error: ' . $e->getMessage() . "\n" . $e->getTraceAsString(), 'import/xml/catalog' );
195
			$this->mail( 'Catalog XML import error', $e->getMessage() . "\n" . $e->getTraceAsString() );
196
			throw $e;
197
		}
198
	}
199
200
201
	/**
202
	 * Returns the directory for storing imported files
203
	 *
204
	 * @return string Directory for storing imported files
205
	 */
206
	protected function backup() : string
207
	{
208
		/** controller/jobs/catalog/import/xml/backup
209
		 * Name of the backup for sucessfully imported files
210
		 *
211
		 * After a XML file was imported successfully, you can move it to another
212
		 * location, so it won't be imported again and isn't overwritten by the
213
		 * next file that is stored at the same location in the file system.
214
		 *
215
		 * You should use an absolute path to be sure but can be relative path
216
		 * if you absolutely know from where the job will be executed from. The
217
		 * name of the new backup location can contain placeholders understood
218
		 * by the PHP DateTime::format() method (with percent signs prefix) to
219
		 * create dynamic paths, e.g. "backup/%Y-%m-%d" which would create
220
		 * "backup/2000-01-01". For more information about the date() placeholders,
221
		 * please have a look  into the PHP documentation of the
222
		 * {@link https://www.php.net/manual/en/datetime.format.php format() method}.
223
		 *
224
		 * **Note:** If no backup name is configured, the file or directory
225
		 * won't be moved away. Please make also sure that the parent directory
226
		 * and the new directory are writable so the file or directory could be
227
		 * moved.
228
		 *
229
		 * @param integer Name of the backup file, optionally with date/time placeholders
230
		 * @since 2019.04
231
		 * @see controller/jobs/catalog/import/xml/domains
232
		 * @see controller/jobs/catalog/import/xml/location
233
		 * @see controller/jobs/catalog/import/xml/max-query
234
		 */
235
		return (string) $this->context()->config()->get( 'controller/jobs/catalog/import/xml/backup' );
236
	}
237
238
239
	/**
240
	 * Returns the list of domain names that should be retrieved along with the catalog items
241
	 *
242
	 * @return array List of domain names
243
	 */
244
	protected function domains() : array
245
	{
246
		/** controller/jobs/catalog/import/xml/domains
247
		 * List of item domain names that should be retrieved along with the catalog items
248
		 *
249
		 * For efficient processing, the items associated to the products can be
250
		 * fetched to, minimizing the number of database queries required. To be
251
		 * most effective, the list of item domain names should be used in the
252
		 * mapping configuration too, so the retrieved items will be used during
253
		 * the import.
254
		 *
255
		 * @param array Associative list of MShop item domain names
256
		 * @since 2019.04
257
		 * @see controller/jobs/catalog/import/xml/backup
258
		 * @see controller/jobs/catalog/import/xml/location
259
		 * @see controller/jobs/catalog/import/xml/max-query
260
		 */
261
		return $this->context()->config()->get( 'controller/jobs/catalog/import/xml/domains', ['media', 'text'] );
262
	}
263
264
265
	/**
266
	 * Imports the XML file given by its path
267
	 *
268
	 * @param \Aimeos\MShop\ContextIface $context Context object
269
	 * @param string $filename Absolute or relative path to the XML file
270
	 */
271
	protected function import( \Aimeos\MShop\ContextIface $context, string $filename )
272
	{
273
		$logger = $context->logger();
274
		$xml = new \XMLReader();
275
276
		if( $xml->open( $filename, LIBXML_COMPACT | LIBXML_PARSEHUGE ) === false ) {
277
			throw new \Aimeos\Controller\Jobs\Exception( sprintf( 'No XML file "%1$s" found', $filename ) );
278
		}
279
280
		$logger->info( sprintf( 'Started catalog import from file "%1$s"', $filename ), 'import/xml/catalog' );
281
282
		$this->importTree( $xml, $this->domains() );
283
284
		foreach( $this->getProcessors() as $proc ) {
285
			$proc->finish();
286
		}
287
288
		$logger->info( sprintf( 'Finished catalog import from file "%1$s"', $filename ), 'import/xml/catalog' );
289
290
		if( !empty( $backup = $this->backup() ) && @rename( $filename, $backup = \Aimeos\Base\Str::strtime( $backup ) ) === false )
291
		{
292
			$msg = sprintf( 'Unable to move imported file "%1$s" to "%2$s"', $filename, $backup );
293
			throw new \Aimeos\Controller\Jobs\Exception( $msg );
294
		}
295
	}
296
297
298
	/**
299
	 * Imports a single category node
300
	 *
301
	 * @param \DomElement $node DOM node of "catalogitem" element
302
	 * @param string[] $domains List of domain names whose referenced items will be updated in the catalog items
303
	 * @param string|null $parentid ID of the parent catalog node
304
	 * @param array &$map Will contain the associative list of code/ID pairs of the child categories
305
	 * @return string Catalog ID of the imported category
306
	 */
307
	protected function importNode( \DomElement $node, array $domains, string $parentid = null, array &$map ) : string
308
	{
309
		$manager = \Aimeos\MShop::create( $this->context(), 'catalog' );
310
311
		if( ( $attr = $node->attributes->getNamedItem( 'ref' ) ) !== null )
0 ignored issues
show
Bug introduced by
The method getNamedItem() does not exist on null. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

311
		if( ( $attr = $node->attributes->/** @scrutinizer ignore-call */ getNamedItem( 'ref' ) ) !== null )

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...
312
		{
313
			try
314
			{
315
				$item = $manager->find( $attr->nodeValue, $domains );
316
				$manager->move( $item->getId(), $item->getParentId(), $parentid );
317
318
				$item = $this->process( $item, $node );
319
				$currentid = $manager->save( $item )->getId();
320
				unset( $item );
321
322
				$tree = $manager->getTree( $currentid, [], \Aimeos\MW\Tree\Manager\Base::LEVEL_LIST );
323
324
				foreach( $tree->getChildren() as $child ) {
325
					$map[$child->getCode()] = $child->getId();
326
				}
327
328
				return $currentid;
329
			}
330
			catch( \Aimeos\MShop\Exception $e ) {} // not found, create new
331
		}
332
333
		$item = $this->process( $manager->create(), $node );
334
		return $manager->insert( $item, $parentid )->getId();
335
	}
336
337
338
	/**
339
	 * Imports the catalog document
340
	 *
341
	 * @param \XMLReader $xml Catalog document to import
342
	 * @param string[] $domains List of domain names whose referenced items will be updated in the catalog items
343
	 * @param string|null $parentid ID of the parent catalog node
344
	 * @param array $map Associative list of catalog code as keys and category ID as values
345
	 */
346
	protected function importTree( \XMLReader $xml, array $domains, string $parentid = null, array $map = [] )
347
	{
348
		$total = 0;
349
		$childMap = [];
350
		$currentid = $parentid;
351
352
		while( $xml->read() === true )
353
		{
354
			if( $xml->nodeType === \XMLReader::ELEMENT && $xml->name === 'catalogitem' )
355
			{
356
				if( ( $node = $xml->expand() ) === false )
357
				{
358
					$msg = sprintf( 'Expanding "%1$s" node failed', 'catalogitem' );
359
					throw new \Aimeos\Controller\Jobs\Exception( $msg );
360
				}
361
362
				if( ( $attr = $node->attributes->getNamedItem( 'ref' ) ) !== null ) {
363
					unset( $map[$attr->nodeValue] );
364
				}
365
366
				$currentid = $this->importNode( $node, $domains, $parentid, $childMap );
367
				$total++;
368
			}
369
			elseif( $xml->nodeType === \XMLReader::ELEMENT && $xml->name === 'catalog' )
370
			{
371
				$this->importTree( $xml, $domains, $currentid, $childMap );
372
				$childMap = [];
373
			}
374
			elseif( $xml->nodeType === \XMLReader::END_ELEMENT && $xml->name === 'catalog' )
375
			{
376
				\Aimeos\MShop::create( $this->context(), 'catalog' )->delete( $map );
377
				break;
378
			}
379
		}
380
	}
381
382
383
	/**
384
	 * Returns the path to the directory with the XML file
385
	 *
386
	 * @return string Path to the directory with the XML file
387
	 */
388
	protected function location() : string
389
	{
390
		/** controller/jobs/catalog/import/xml/location
391
		 * File or directory where the content is stored which should be imported
392
		 *
393
		 * You need to configure the XML file or directory with the XML files that
394
		 * should be imported. It should be an absolute path to be sure but can be
395
		 * relative path if you absolutely know from where the job will be executed
396
		 * from.
397
		 *
398
		 * @param string Relative path to the XML files
399
		 * @since 2019.04
400
		 * @see controller/jobs/catalog/import/xml/backup
401
		 * @see controller/jobs/catalog/import/xml/domains
402
		 * @see controller/jobs/catalog/import/xml/max-query
403
		 */
404
		return (string) $this->context()->config()->get( 'controller/jobs/catalog/import/xml/location', 'catalog' );
405
	}
406
407
408
	/**
409
	 * Updates the catalog item and its referenced items using the given DOM node
410
	 *
411
	 * @param \Aimeos\MShop\Catalog\Item\Iface $item Catalog item object to update
412
	 * @param \DomElement $node DOM node used for updateding the catalog item
413
	 * @return \Aimeos\MShop\Catalog\Item\Iface $item Updated catalog item object
414
	 */
415
	protected function process( \Aimeos\MShop\Catalog\Item\Iface $item, \DomElement $node ) : \Aimeos\MShop\Catalog\Item\Iface
416
	{
417
		try
418
		{
419
			$list = [];
420
421
			foreach( $node->attributes as $attr ) {
422
				$list[$attr->nodeName] = $attr->nodeValue;
423
			}
424
425
			foreach( $node->childNodes as $tag )
426
			{
427
				if( $tag->nodeName === 'lists' ) {
428
					$item = $this->getProcessor( $tag->nodeName )->process( $item, $tag );
429
				} elseif( $tag->nodeName[0] !== '#' ) {
430
					$list[$tag->nodeName] = $tag->nodeValue;
431
				}
432
			}
433
434
			$list['catalog.config'] = isset( $list['catalog.config'] ) ? json_decode( $list['catalog.config'], true ) : [];
435
			$item->fromArray( $list, true );
436
		}
437
		catch( \Exception $e )
438
		{
439
			$msg = 'Catalog import error: ' . $e->getMessage() . "\n" . $e->getTraceAsString();
440
			$this->context()->logger()->error( $msg, 'import/xml/catalog' );
441
		}
442
443
		return $item;
444
	}
445
}
446