Add CSV formatter
nouveau fichier: core/lib/Thelia/Core/FileFormat/Formatting/Formatter/CSVFormatter.php nouveau fichier: core/lib/Thelia/Tests/FileFormat/Formatting/Formatter/CSVFormatterTest.php
This commit is contained in:
@@ -0,0 +1,172 @@
|
|||||||
|
<?php
|
||||||
|
/*************************************************************************************/
|
||||||
|
/* This file is part of the Thelia package. */
|
||||||
|
/* */
|
||||||
|
/* Copyright (c) OpenStudio */
|
||||||
|
/* email : dev@thelia.net */
|
||||||
|
/* web : http://www.thelia.net */
|
||||||
|
/* */
|
||||||
|
/* For the full copyright and license information, please view the LICENSE.txt */
|
||||||
|
/* file that was distributed with this source code. */
|
||||||
|
/*************************************************************************************/
|
||||||
|
|
||||||
|
namespace Thelia\Core\FileFormat\Formatting\Formatter;
|
||||||
|
use Thelia\Core\FileFormat\Formatting\AbstractFormatter;
|
||||||
|
use Thelia\Core\FileFormat\Formatting\Exception\BadFormattedStringException;
|
||||||
|
use Thelia\Core\FileFormat\Formatting\FormatterData;
|
||||||
|
use Thelia\Core\FileFormat\FormatType;
|
||||||
|
use Thelia\Core\Translation\Translator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class CSVFormatter
|
||||||
|
* @package Thelia\Core\FileFormat\Formatting\Formatter
|
||||||
|
* @author Benjamin Perche <bperche@openstudio.fr>
|
||||||
|
*/
|
||||||
|
class CSVFormatter extends AbstractFormatter
|
||||||
|
{
|
||||||
|
public $delimiter = ";";
|
||||||
|
public $lineReturn = "\r\n";
|
||||||
|
public $stringDelimiter = "\"";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*
|
||||||
|
* This method must return a string, the name of the format.
|
||||||
|
*
|
||||||
|
* example:
|
||||||
|
* return "XML";
|
||||||
|
*/
|
||||||
|
public function getName()
|
||||||
|
{
|
||||||
|
return "CSV";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*
|
||||||
|
* This method must return a string, the extension of the file format, without the ".".
|
||||||
|
* The string should be lowercase.
|
||||||
|
*
|
||||||
|
* example:
|
||||||
|
* return "xml";
|
||||||
|
*/
|
||||||
|
public function getExtension()
|
||||||
|
{
|
||||||
|
return "csv";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*
|
||||||
|
* This method must return a string, the mime type of the file format.
|
||||||
|
*
|
||||||
|
* example:
|
||||||
|
* return "application/json";
|
||||||
|
*/
|
||||||
|
public function getMimeType()
|
||||||
|
{
|
||||||
|
return "text/csv";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param FormatterData $data
|
||||||
|
* @return mixed
|
||||||
|
*
|
||||||
|
* This method must use a FormatterData object and output
|
||||||
|
* a formatted value.
|
||||||
|
*/
|
||||||
|
public function encode(FormatterData $data)
|
||||||
|
{
|
||||||
|
$string = "";
|
||||||
|
$firstRow = $data->getRow();
|
||||||
|
$delimiterLength = strlen($this->delimiter);
|
||||||
|
$lineReturnLength = strlen($this->lineReturn);
|
||||||
|
|
||||||
|
|
||||||
|
if (false !== $firstRow) {
|
||||||
|
$rawKeys = array_keys($firstRow);
|
||||||
|
$keys = [];
|
||||||
|
|
||||||
|
foreach ($rawKeys as $key) {
|
||||||
|
$keys[$key] = $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
$values = $data->getData();
|
||||||
|
array_unshift($values, $keys);
|
||||||
|
|
||||||
|
while (null !== $row = array_shift($values)) {
|
||||||
|
foreach ($keys as $key) {
|
||||||
|
if (!is_scalar($row[$key])) {
|
||||||
|
$row[$key] = serialize($row[$key]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$string .= $this->stringDelimiter . addslashes($row[$key]) . $this->stringDelimiter . $this->delimiter;
|
||||||
|
}
|
||||||
|
|
||||||
|
$string = substr($string,0, -$delimiterLength) . $this->lineReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return substr($string,0, -$lineReturnLength);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $rawData
|
||||||
|
* @return FormatterData
|
||||||
|
*
|
||||||
|
* This must takes raw data as argument and outputs
|
||||||
|
* a FormatterData object.
|
||||||
|
*/
|
||||||
|
public function decode($rawData)
|
||||||
|
{
|
||||||
|
$raw = explode($this->lineReturn, $rawData);
|
||||||
|
$decoded = [];
|
||||||
|
|
||||||
|
if (count($raw) > 0) {
|
||||||
|
$keys = explode($this->delimiter, array_shift($raw));
|
||||||
|
|
||||||
|
foreach ($keys as &$key) {
|
||||||
|
$key = trim($key, $this->stringDelimiter);
|
||||||
|
}
|
||||||
|
|
||||||
|
$columns = count ($keys);
|
||||||
|
|
||||||
|
while (null !== $row = array_shift($raw)) {
|
||||||
|
$newRow = [];
|
||||||
|
$row = explode($this->delimiter, $row);
|
||||||
|
|
||||||
|
for ($i = 0; $i < $columns; ++$i) {
|
||||||
|
$value = trim($row[$i], $this->stringDelimiter);
|
||||||
|
|
||||||
|
if (false !== $unserialized = @unserialize($row[$i])) {
|
||||||
|
$value = $unserialized;
|
||||||
|
}
|
||||||
|
|
||||||
|
$newRow[$keys[$i]] = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
$decoded[] = $newRow;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (new FormatterData())->setData($decoded);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*
|
||||||
|
* return a string that defines the handled format type.
|
||||||
|
*
|
||||||
|
* Thelia types are defined in \Thelia\Core\FileFormat\FormatType
|
||||||
|
*
|
||||||
|
* examples:
|
||||||
|
* return FormatType::TABLE;
|
||||||
|
* return FormatType::UNBOUNDED;
|
||||||
|
*/
|
||||||
|
public function getHandledType()
|
||||||
|
{
|
||||||
|
return FormatType::TABLE;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,134 @@
|
|||||||
|
<?php
|
||||||
|
/*************************************************************************************/
|
||||||
|
/* This file is part of the Thelia package. */
|
||||||
|
/* */
|
||||||
|
/* Copyright (c) OpenStudio */
|
||||||
|
/* email : dev@thelia.net */
|
||||||
|
/* web : http://www.thelia.net */
|
||||||
|
/* */
|
||||||
|
/* For the full copyright and license information, please view the LICENSE.txt */
|
||||||
|
/* file that was distributed with this source code. */
|
||||||
|
/*************************************************************************************/
|
||||||
|
|
||||||
|
namespace Thelia\Tests\FileFormat\Formatting\Formatter;
|
||||||
|
use Symfony\Component\DependencyInjection\Container;
|
||||||
|
use Thelia\Core\FileFormat\Formatting\Formatter\CSVFormatter;
|
||||||
|
use Thelia\Core\FileFormat\Formatting\FormatterData;
|
||||||
|
use Thelia\Core\Translation\Translator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class CSVFormatterTest
|
||||||
|
* @package Thelia\Tests\FileFormat\Formatting\Formatter
|
||||||
|
* @author Benjamin Perche <bperche@openstudio.fr>
|
||||||
|
*/
|
||||||
|
class CSVFormatterTest extends \PHPUnit_Framework_TestCase
|
||||||
|
{
|
||||||
|
/** @var CSVFormatter */
|
||||||
|
protected $formatter;
|
||||||
|
|
||||||
|
public function setUp()
|
||||||
|
{
|
||||||
|
new Translator(new Container());
|
||||||
|
|
||||||
|
$this->formatter = new CSVFormatter();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testSimpleEncode()
|
||||||
|
{
|
||||||
|
$expected = "\"ref\";\"stock\"\r\n\"foo\";\"bar\"";
|
||||||
|
|
||||||
|
$data = [
|
||||||
|
[
|
||||||
|
"ref" => "foo",
|
||||||
|
"stock" => "bar",
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$data = (new FormatterData())->setData($data);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
$this->formatter->encode($data)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testComplexEncode()
|
||||||
|
{
|
||||||
|
$this->formatter->lineReturn = "\n";
|
||||||
|
$this->formatter->delimiter = ",";
|
||||||
|
$expected = "\"foo\",\"bar\",\"baz\"\n\"1\",\"2\",\"3\"\n\"4\",\"5\",\"6\"\n\"1\",\"2\",\"3\"";
|
||||||
|
|
||||||
|
$data = [
|
||||||
|
[
|
||||||
|
"foo" => "1",
|
||||||
|
"bar" => "2",
|
||||||
|
"baz" => "3",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"foo" => "4",
|
||||||
|
"bar" => "5",
|
||||||
|
"baz" => "6",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"foo" => "1",
|
||||||
|
"bar" => "2",
|
||||||
|
"baz" => "3",
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$data = (new FormatterData())->setData($data);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
$this->formatter->encode($data)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testSimpleDecode()
|
||||||
|
{
|
||||||
|
$data = "\"ref\";\"stock\"\r\n\"foo\";\"bar\"";
|
||||||
|
|
||||||
|
$expected = [
|
||||||
|
[
|
||||||
|
"ref" => "foo",
|
||||||
|
"stock" => "bar",
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
$this->formatter->decode($data)->getData()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testComplexDecode()
|
||||||
|
{
|
||||||
|
$this->formatter->lineReturn = "\n";
|
||||||
|
$this->formatter->delimiter = ",";
|
||||||
|
$data = "\"foo\",\"bar\",\"baz\"\n\"1\",\"2\",\"3\"\n\"4\",\"5\",\"6\"\n\"1\",\"2\",\"3\"";
|
||||||
|
|
||||||
|
$expected = [
|
||||||
|
[
|
||||||
|
"foo" => "1",
|
||||||
|
"bar" => "2",
|
||||||
|
"baz" => "3",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"foo" => "4",
|
||||||
|
"bar" => "5",
|
||||||
|
"baz" => "6",
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"foo" => "1",
|
||||||
|
"bar" => "2",
|
||||||
|
"baz" => "3",
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
$expected,
|
||||||
|
$this->formatter->decode($data)->getData()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user