-
Notifications
You must be signed in to change notification settings - Fork 257
Expand file tree
/
Copy pathSaveMultiple.php
More file actions
executable file
·129 lines (118 loc) · 3.68 KB
/
SaveMultiple.php
File metadata and controls
executable file
·129 lines (118 loc) · 3.68 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
<?php
/**
* Copyright © Magento, Inc. All rights reserved.
* See COPYING.txt for license details.
*/
declare(strict_types=1);
namespace Magento\Inventory\Model\ResourceModel\SourceItem;
use Magento\Framework\App\ResourceConnection;
use Magento\Inventory\Model\ResourceModel\SourceItem as SourceItemResourceModel;
use Magento\InventoryApi\Api\Data\SourceItemInterface;
/**
* Implementation of SourceItem save multiple operation for specific db layer
*
* Save Multiple used here for performance efficient purposes over single save operation
*/
class SaveMultiple
{
/**
* @var ResourceConnection
*/
private $resourceConnection;
/**
* @param ResourceConnection $resourceConnection
*/
public function __construct(
ResourceConnection $resourceConnection
) {
$this->resourceConnection = $resourceConnection;
}
/**
* Multiple save source items
*
* @param SourceItemInterface[] $sourceItems
* @return void
*/
public function execute(array $sourceItems)
{
if (!count($sourceItems)) {
return;
}
$connection = $this->resourceConnection->getConnection();
$tableName = $this->resourceConnection->getTableName(SourceItemResourceModel::TABLE_NAME_SOURCE_ITEM);
$columnsSql = $this->buildColumnsSqlPart([
SourceItemInterface::SOURCE_CODE,
SourceItemInterface::SKU,
SourceItemInterface::QUANTITY,
SourceItemInterface::STATUS
]);
$valuesSql = $this->buildValuesSqlPart($sourceItems);
$bind = $this->getSqlBindData($sourceItems);
/* can not use "INSERT ... ON DUPLICATE KEY UPDATE" statement against a table having more than one unique
or primary key because it is unsafe */
$insertSql = sprintf(
'INSERT INTO `%s` (%s) VALUES %s',
$tableName,
$columnsSql,
$valuesSql
);
$connection->query($insertSql, $bind);
}
/**
* Build column sql part
*
* @param array $columns
* @return string
*/
private function buildColumnsSqlPart(array $columns): string
{
$connection = $this->resourceConnection->getConnection();
$processedColumns = array_map([$connection, 'quoteIdentifier'], $columns);
$sql = implode(', ', $processedColumns);
return $sql;
}
/**
* Build sql query for values
*
* @param SourceItemInterface[] $sourceItems
* @return string
*/
private function buildValuesSqlPart(array $sourceItems): string
{
$sql = rtrim(str_repeat('(?, ?, ?, ?), ', count($sourceItems)), ', ');
return $sql;
}
/**
* Get Sql bind data
*
* @param SourceItemInterface[] $sourceItems
* @return array
*/
private function getSqlBindData(array $sourceItems): array
{
$bind = [];
foreach ($sourceItems as $sourceItem) {
$bind[] = $sourceItem->getSourceCode();
$bind[] = $sourceItem->getSku();
$bind[] = $sourceItem->getQuantity();
$bind[] = $sourceItem->getStatus();
}
return $bind;
}
/**
* Build sql query for on duplicate event
*
* @param array $fields
* @return string
*/
private function buildOnDuplicateSqlPart(array $fields): string
{
$connection = $this->resourceConnection->getConnection();
$processedFields = [];
foreach ($fields as $field) {
$processedFields[] = sprintf('%1$s = VALUES(%1$s)', $connection->quoteIdentifier($field));
}
$sql = 'ON DUPLICATE KEY UPDATE ' . implode(', ', $processedFields);
return $sql;
}
}