2008-08-09 04:53:34 +00:00
|
|
|
<?php
|
2016-06-15 16:03:16 +12:00
|
|
|
|
2016-08-19 10:51:35 +12:00
|
|
|
namespace SilverStripe\Dev;
|
|
|
|
|
2020-08-30 16:01:27 +12:00
|
|
|
use League\Csv\MapIterator;
|
2018-02-21 20:22:37 +00:00
|
|
|
use League\Csv\Reader;
|
2016-08-19 10:51:35 +12:00
|
|
|
use SilverStripe\Control\Director;
|
2016-06-15 16:03:16 +12:00
|
|
|
use SilverStripe\ORM\DataObject;
|
2016-08-19 10:51:35 +12:00
|
|
|
|
2008-08-09 04:53:34 +00:00
|
|
|
/**
|
2014-08-15 18:53:05 +12:00
|
|
|
* Utility class to facilitate complex CSV-imports by defining column-mappings
|
|
|
|
* and custom converters.
|
2013-05-14 22:01:15 +12:00
|
|
|
*
|
2014-08-15 18:53:05 +12:00
|
|
|
* Uses the fgetcsv() function to process CSV input. Accepts a file-handler as
|
2013-05-14 22:01:15 +12:00
|
|
|
* input.
|
2014-08-15 18:53:05 +12:00
|
|
|
*
|
2014-02-05 14:42:27 +13:00
|
|
|
* @see http://tools.ietf.org/html/rfc4180
|
2013-05-14 22:01:15 +12:00
|
|
|
*
|
2014-08-15 18:53:05 +12:00
|
|
|
* @todo Support for deleting existing records not matched in the import
|
2013-05-14 22:01:15 +12:00
|
|
|
* (through relation checks)
|
2008-08-09 04:53:34 +00:00
|
|
|
*/
|
2016-11-29 12:31:16 +13:00
|
|
|
class CsvBulkLoader extends BulkLoader
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delimiter character (Default: comma).
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public $delimiter = ',';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enclosure character (Default: doublequote)
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public $enclosure = '"';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Identifies if csv the has a header row.
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
public $hasHeaderRow = true;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of lines to split large CSV files into.
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*
|
|
|
|
* @config
|
|
|
|
*/
|
|
|
|
private static $lines = 1000;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
public function preview($filepath)
|
|
|
|
{
|
|
|
|
return $this->processAll($filepath, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $filepath
|
|
|
|
* @param boolean $preview
|
|
|
|
*
|
|
|
|
* @return null|BulkLoader_Result
|
|
|
|
*/
|
|
|
|
protected function processAll($filepath, $preview = false)
|
|
|
|
{
|
2018-02-21 20:22:37 +00:00
|
|
|
$previousDetectLE = ini_get('auto_detect_line_endings');
|
|
|
|
ini_set('auto_detect_line_endings', true);
|
2021-05-14 08:51:21 +12:00
|
|
|
|
|
|
|
$this->extend('onBeforeProcessAll', $filepath, $preview);
|
|
|
|
|
2019-04-05 11:33:01 +13:00
|
|
|
$result = BulkLoader_Result::create();
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
try {
|
2018-02-21 20:22:37 +00:00
|
|
|
$filepath = Director::getAbsFile($filepath);
|
|
|
|
$csvReader = Reader::createFromPath($filepath, 'r');
|
2019-06-20 00:59:42 +02:00
|
|
|
$csvReader->setDelimiter($this->delimiter);
|
2018-02-21 20:22:37 +00:00
|
|
|
|
2020-08-30 16:01:27 +12:00
|
|
|
// league/csv 9
|
|
|
|
if (method_exists($csvReader, 'skipInputBOM')) {
|
|
|
|
$csvReader->skipInputBOM();
|
|
|
|
// league/csv 8
|
|
|
|
} else {
|
|
|
|
$csvReader->stripBom(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
$tabExtractor = function ($row, $rowOffset) {
|
2018-02-21 20:22:37 +00:00
|
|
|
foreach ($row as &$item) {
|
|
|
|
// [SS-2017-007] Ensure all cells with leading tab and then [@=+] have the tab removed on import
|
|
|
|
if (preg_match("/^\t[\-@=\+]+.*/", $item)) {
|
|
|
|
$item = ltrim($item, "\t");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $row;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($this->columnMap) {
|
|
|
|
$headerMap = $this->getNormalisedColumnMap();
|
2020-08-30 16:01:27 +12:00
|
|
|
|
|
|
|
$remapper = function ($row, $rowOffset) use ($headerMap, $tabExtractor) {
|
|
|
|
$row = $tabExtractor($row, $rowOffset);
|
2018-02-21 20:22:37 +00:00
|
|
|
foreach ($headerMap as $column => $renamedColumn) {
|
|
|
|
if ($column == $renamedColumn) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (array_key_exists($column, $row)) {
|
|
|
|
if (strpos($renamedColumn, '_ignore_') !== 0) {
|
|
|
|
$row[$renamedColumn] = $row[$column];
|
|
|
|
}
|
|
|
|
unset($row[$column]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $row;
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
$remapper = $tabExtractor;
|
|
|
|
}
|
2016-11-29 12:31:16 +13:00
|
|
|
|
2018-02-21 20:22:37 +00:00
|
|
|
if ($this->hasHeaderRow) {
|
2020-08-30 16:01:27 +12:00
|
|
|
if (method_exists($csvReader, 'fetchAssoc')) {
|
|
|
|
$rows = $csvReader->fetchAssoc(0, $remapper);
|
|
|
|
} else {
|
|
|
|
$csvReader->setHeaderOffset(0);
|
|
|
|
$rows = new MapIterator($csvReader->getRecords(), $remapper);
|
|
|
|
}
|
2018-02-21 20:22:37 +00:00
|
|
|
} elseif ($this->columnMap) {
|
2020-08-30 16:01:27 +12:00
|
|
|
if (method_exists($csvReader, 'fetchAssoc')) {
|
|
|
|
$rows = $csvReader->fetchAssoc($headerMap, $remapper);
|
|
|
|
} else {
|
|
|
|
$rows = new MapIterator($csvReader->getRecords($headerMap), $remapper);
|
|
|
|
}
|
2018-02-21 20:22:37 +00:00
|
|
|
}
|
2016-11-29 12:31:16 +13:00
|
|
|
|
2018-02-21 20:22:37 +00:00
|
|
|
foreach ($rows as $row) {
|
|
|
|
$this->processRecord($row, $this->columnMap, $result, $preview);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
2018-02-21 20:22:37 +00:00
|
|
|
} catch (\Exception $e) {
|
|
|
|
$failedMessage = sprintf("Failed to parse %s", $filepath);
|
2017-01-31 15:17:29 +00:00
|
|
|
if (Director::isDev()) {
|
|
|
|
$failedMessage = sprintf($failedMessage . " because %s", $e->getMessage());
|
|
|
|
}
|
|
|
|
print $failedMessage . PHP_EOL;
|
2018-02-21 20:22:37 +00:00
|
|
|
} finally {
|
|
|
|
ini_set('auto_detect_line_endings', $previousDetectLE);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
2021-05-14 08:51:21 +12:00
|
|
|
|
|
|
|
$this->extend('onAfterProcessAll', $result, $preview);
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2018-02-21 20:22:37 +00:00
|
|
|
protected function getNormalisedColumnMap()
|
|
|
|
{
|
|
|
|
$map = [];
|
|
|
|
foreach ($this->columnMap as $column => $newColumn) {
|
|
|
|
if (strpos($newColumn, "->") === 0) {
|
|
|
|
$map[$column] = $column;
|
|
|
|
} elseif (is_null($newColumn)) {
|
|
|
|
// the column map must consist of unique scalar values
|
|
|
|
// `null` can be present multiple times and is not scalar
|
|
|
|
// so we name it in a standard way so we can remove it later
|
|
|
|
$map[$column] = '_ignore_' . $column;
|
|
|
|
} else {
|
|
|
|
$map[$column] = $newColumn;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $map;
|
|
|
|
}
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Splits a large file up into many smaller files.
|
|
|
|
*
|
|
|
|
* @param string $path Path to large file to split
|
|
|
|
* @param int $lines Number of lines per file
|
|
|
|
*
|
|
|
|
* @return array List of file paths
|
|
|
|
*/
|
|
|
|
protected function splitFile($path, $lines = null)
|
|
|
|
{
|
2018-02-21 20:22:37 +00:00
|
|
|
Deprecation::notice('5.0', 'splitFile is deprecated, please process files using a stream');
|
2016-11-29 12:31:16 +13:00
|
|
|
$previous = ini_get('auto_detect_line_endings');
|
|
|
|
|
|
|
|
ini_set('auto_detect_line_endings', true);
|
|
|
|
|
|
|
|
if (!is_int($lines)) {
|
|
|
|
$lines = $this->config()->get("lines");
|
|
|
|
}
|
|
|
|
|
|
|
|
$new = $this->getNewSplitFileName();
|
|
|
|
|
|
|
|
$to = fopen($new, 'w+');
|
|
|
|
$from = fopen($path, 'r');
|
|
|
|
|
|
|
|
$header = null;
|
|
|
|
|
|
|
|
if ($this->hasHeaderRow) {
|
|
|
|
$header = fgets($from);
|
|
|
|
fwrite($to, $header);
|
|
|
|
}
|
|
|
|
|
2020-04-20 18:58:09 +01:00
|
|
|
$files = [];
|
2016-11-29 12:31:16 +13:00
|
|
|
$files[] = $new;
|
|
|
|
|
|
|
|
$count = 0;
|
|
|
|
|
|
|
|
while (!feof($from)) {
|
|
|
|
fwrite($to, fgets($from));
|
|
|
|
|
|
|
|
$count++;
|
|
|
|
|
|
|
|
if ($count >= $lines) {
|
|
|
|
fclose($to);
|
|
|
|
|
|
|
|
// get a new temporary file name, to write the next lines to
|
|
|
|
$new = $this->getNewSplitFileName();
|
|
|
|
|
|
|
|
$to = fopen($new, 'w+');
|
|
|
|
|
|
|
|
if ($this->hasHeaderRow) {
|
|
|
|
// add the headers to the new file
|
|
|
|
fwrite($to, $header);
|
|
|
|
}
|
|
|
|
|
|
|
|
$files[] = $new;
|
|
|
|
|
|
|
|
$count = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose($to);
|
|
|
|
|
|
|
|
ini_set('auto_detect_line_endings', $previous);
|
|
|
|
|
|
|
|
return $files;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function getNewSplitFileName()
|
|
|
|
{
|
2018-02-21 20:22:37 +00:00
|
|
|
Deprecation::notice('5.0', 'getNewSplitFileName is deprecated, please name your files yourself');
|
2017-10-09 12:41:34 +13:00
|
|
|
return TEMP_PATH . DIRECTORY_SEPARATOR . uniqid(str_replace('\\', '_', static::class), true) . '.csv';
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $filepath
|
|
|
|
* @param boolean $preview
|
|
|
|
*
|
|
|
|
* @return BulkLoader_Result
|
|
|
|
*/
|
|
|
|
protected function processChunk($filepath, $preview = false)
|
|
|
|
{
|
2018-02-21 20:22:37 +00:00
|
|
|
Deprecation::notice('5.0', 'processChunk is deprecated, please process rows individually');
|
2017-05-17 17:40:13 +12:00
|
|
|
$results = BulkLoader_Result::create();
|
2016-11-29 12:31:16 +13:00
|
|
|
|
|
|
|
$csv = new CSVParser(
|
|
|
|
$filepath,
|
|
|
|
$this->delimiter,
|
|
|
|
$this->enclosure
|
|
|
|
);
|
|
|
|
|
|
|
|
// ColumnMap has two uses, depending on whether hasHeaderRow is set
|
|
|
|
if ($this->columnMap) {
|
|
|
|
// if the map goes to a callback, use the same key value as the map
|
|
|
|
// value, rather than function name as multiple keys may use the
|
|
|
|
// same callback
|
|
|
|
$map = [];
|
|
|
|
foreach ($this->columnMap as $k => $v) {
|
|
|
|
if (strpos($v, "->") === 0) {
|
|
|
|
$map[$k] = $k;
|
|
|
|
} else {
|
|
|
|
$map[$k] = $v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->hasHeaderRow) {
|
|
|
|
$csv->mapColumns($map);
|
|
|
|
} else {
|
|
|
|
$csv->provideHeaderRow($map);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($csv as $row) {
|
|
|
|
$this->processRecord($row, $this->columnMap, $results, $preview);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $results;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @todo Better messages for relation checks and duplicate detection
|
|
|
|
* Note that columnMap isn't used.
|
|
|
|
*
|
|
|
|
* @param array $record
|
|
|
|
* @param array $columnMap
|
|
|
|
* @param BulkLoader_Result $results
|
|
|
|
* @param boolean $preview
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
protected function processRecord($record, $columnMap, &$results, $preview = false)
|
|
|
|
{
|
|
|
|
$class = $this->objectClass;
|
|
|
|
|
|
|
|
// find existing object, or create new one
|
|
|
|
$existingObj = $this->findExistingObject($record, $columnMap);
|
|
|
|
/** @var DataObject $obj */
|
|
|
|
$obj = ($existingObj) ? $existingObj : new $class();
|
|
|
|
$schema = DataObject::getSchema();
|
|
|
|
|
|
|
|
// first run: find/create any relations and store them on the object
|
|
|
|
// we can't combine runs, as other columns might rely on the relation being present
|
|
|
|
foreach ($record as $fieldName => $val) {
|
|
|
|
// don't bother querying of value is not set
|
|
|
|
if ($this->isNullValue($val)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// checking for existing relations
|
|
|
|
if (isset($this->relationCallbacks[$fieldName])) {
|
|
|
|
// trigger custom search method for finding a relation based on the given value
|
|
|
|
// and write it back to the relation (or create a new object)
|
|
|
|
$relationName = $this->relationCallbacks[$fieldName]['relationname'];
|
|
|
|
/** @var DataObject $relationObj */
|
|
|
|
$relationObj = null;
|
|
|
|
if ($this->hasMethod($this->relationCallbacks[$fieldName]['callback'])) {
|
|
|
|
$relationObj = $this->{$this->relationCallbacks[$fieldName]['callback']}($obj, $val, $record);
|
|
|
|
} elseif ($obj->hasMethod($this->relationCallbacks[$fieldName]['callback'])) {
|
|
|
|
$relationObj = $obj->{$this->relationCallbacks[$fieldName]['callback']}($val, $record);
|
|
|
|
}
|
|
|
|
if (!$relationObj || !$relationObj->exists()) {
|
|
|
|
$relationClass = $schema->hasOneComponent(get_class($obj), $relationName);
|
|
|
|
$relationObj = new $relationClass();
|
|
|
|
//write if we aren't previewing
|
|
|
|
if (!$preview) {
|
|
|
|
$relationObj->write();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$obj->{"{$relationName}ID"} = $relationObj->ID;
|
|
|
|
//write if we are not previewing
|
|
|
|
if (!$preview) {
|
|
|
|
$obj->write();
|
|
|
|
$obj->flushCache(); // avoid relation caching confusion
|
|
|
|
}
|
|
|
|
} elseif (strpos($fieldName, '.') !== false) {
|
|
|
|
// we have a relation column with dot notation
|
2020-09-24 17:09:37 -07:00
|
|
|
[$relationName, $columnName] = explode('.', $fieldName);
|
2016-11-29 12:31:16 +13:00
|
|
|
// always gives us an component (either empty or existing)
|
|
|
|
$relationObj = $obj->getComponent($relationName);
|
|
|
|
if (!$preview) {
|
|
|
|
$relationObj->write();
|
|
|
|
}
|
|
|
|
$obj->{"{$relationName}ID"} = $relationObj->ID;
|
|
|
|
|
|
|
|
//write if we are not previewing
|
|
|
|
if (!$preview) {
|
|
|
|
$obj->write();
|
|
|
|
$obj->flushCache(); // avoid relation caching confusion
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// second run: save data
|
|
|
|
|
|
|
|
foreach ($record as $fieldName => $val) {
|
|
|
|
// break out of the loop if we are previewing
|
|
|
|
if ($preview) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// look up the mapping to see if this needs to map to callback
|
|
|
|
$mapped = $this->columnMap && isset($this->columnMap[$fieldName]);
|
|
|
|
|
|
|
|
if ($mapped && strpos($this->columnMap[$fieldName], '->') === 0) {
|
|
|
|
$funcName = substr($this->columnMap[$fieldName], 2);
|
|
|
|
|
|
|
|
$this->$funcName($obj, $val, $record);
|
|
|
|
} elseif ($obj->hasMethod("import{$fieldName}")) {
|
|
|
|
$obj->{"import{$fieldName}"}($val, $record);
|
|
|
|
} else {
|
2020-04-20 18:58:09 +01:00
|
|
|
$obj->update([$fieldName => $val]);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-14 08:51:21 +12:00
|
|
|
$isChanged = $obj->isChanged();
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
// write record
|
|
|
|
if (!$preview) {
|
|
|
|
$obj->write();
|
|
|
|
}
|
|
|
|
|
|
|
|
// @todo better message support
|
|
|
|
$message = '';
|
|
|
|
|
|
|
|
// save to results
|
|
|
|
if ($existingObj) {
|
2021-05-14 08:51:21 +12:00
|
|
|
// We mark as updated regardless of isChanged, since custom formatters and importers
|
|
|
|
// might have affected relationships and other records.
|
2016-11-29 12:31:16 +13:00
|
|
|
$results->addUpdated($obj, $message);
|
|
|
|
} else {
|
|
|
|
$results->addCreated($obj, $message);
|
|
|
|
}
|
|
|
|
|
2021-05-14 08:51:21 +12:00
|
|
|
$this->extend('onAfterProcessRecord', $obj, $preview, $isChanged);
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
$objID = $obj->ID;
|
|
|
|
|
|
|
|
$obj->destroy();
|
|
|
|
|
|
|
|
// memory usage
|
2018-02-21 20:22:37 +00:00
|
|
|
unset($existingObj, $obj);
|
2016-11-29 12:31:16 +13:00
|
|
|
|
|
|
|
return $objID;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find an existing objects based on one or more uniqueness columns
|
|
|
|
* specified via {@link self::$duplicateChecks}.
|
|
|
|
*
|
|
|
|
* @todo support $columnMap
|
|
|
|
*
|
|
|
|
* @param array $record CSV data column
|
|
|
|
* @param array $columnMap
|
|
|
|
* @return DataObject
|
|
|
|
*/
|
|
|
|
public function findExistingObject($record, $columnMap = [])
|
|
|
|
{
|
|
|
|
$SNG_objectClass = singleton($this->objectClass);
|
|
|
|
// checking for existing records (only if not already found)
|
|
|
|
|
|
|
|
foreach ($this->duplicateChecks as $fieldName => $duplicateCheck) {
|
|
|
|
$existingRecord = null;
|
|
|
|
if (is_string($duplicateCheck)) {
|
|
|
|
// Skip current duplicate check if field value is empty
|
|
|
|
if (empty($record[$duplicateCheck])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check existing record with this value
|
|
|
|
$dbFieldValue = $record[$duplicateCheck];
|
|
|
|
$existingRecord = DataObject::get($this->objectClass)
|
|
|
|
->filter($duplicateCheck, $dbFieldValue)
|
|
|
|
->first();
|
|
|
|
|
|
|
|
if ($existingRecord) {
|
|
|
|
return $existingRecord;
|
|
|
|
}
|
|
|
|
} elseif (is_array($duplicateCheck) && isset($duplicateCheck['callback'])) {
|
|
|
|
if ($this->hasMethod($duplicateCheck['callback'])) {
|
|
|
|
$existingRecord = $this->{$duplicateCheck['callback']}($record[$fieldName], $record);
|
|
|
|
} elseif ($SNG_objectClass->hasMethod($duplicateCheck['callback'])) {
|
|
|
|
$existingRecord = $SNG_objectClass->{$duplicateCheck['callback']}($record[$fieldName], $record);
|
|
|
|
} else {
|
2020-09-24 17:09:37 -07:00
|
|
|
throw new \RuntimeException(
|
|
|
|
"CsvBulkLoader::processRecord():"
|
|
|
|
. " {$duplicateCheck['callback']} not found on importer or object class."
|
|
|
|
);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($existingRecord) {
|
|
|
|
return $existingRecord;
|
|
|
|
}
|
|
|
|
} else {
|
2020-09-24 17:09:37 -07:00
|
|
|
throw new \InvalidArgumentException(
|
|
|
|
'CsvBulkLoader::processRecord(): Wrong format for $duplicateChecks'
|
|
|
|
);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine whether any loaded files should be parsed with a
|
|
|
|
* header-row (otherwise we rely on {@link self::$columnMap}.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function hasHeaderRow()
|
|
|
|
{
|
|
|
|
return ($this->hasHeaderRow || isset($this->columnMap));
|
|
|
|
}
|
2008-08-09 04:53:34 +00:00
|
|
|
}
|