192 lines
6.6 KiB
PHP
192 lines
6.6 KiB
PHP
|
<?php
|
||
|
namespace nur\io\csv;
|
||
|
|
||
|
use nur\A;
|
||
|
use nur\b\io\Tfilter;
|
||
|
use nur\b\params\Parametrable;
|
||
|
use nur\b\params\Tparametrable;
|
||
|
use nur\data\types\Metadata;
|
||
|
use nur\io\Tencoding;
|
||
|
use nur\io\Ttmpwriter;
|
||
|
use nur\ref\ref_csv;
|
||
|
|
||
|
/**
|
||
|
* Class CsvWriter: écrire un flux de données au format CSV dans un fichier
|
||
|
* destination
|
||
|
*
|
||
|
* --autogen-properties-and-methods--
|
||
|
* @method string|null getInputEncoding()
|
||
|
* @method string|null getOutputEncoding()
|
||
|
* @method setOutput($value)
|
||
|
* @method string|null setInputEncoding(?string $value)
|
||
|
* @method string|null setOutputEncoding(?string $value)
|
||
|
* @method string|null setFlavour(?string $value)
|
||
|
* @method bool setMultiSchema(bool $value)
|
||
|
* @method array|null setHeaders(?array $value)
|
||
|
* @method bool setOutputHeaders(bool $value)
|
||
|
*/
|
||
|
class CsvWriter extends Parametrable {
|
||
|
use Tparametrable, Ttmpwriter, Tencoding, Tfilter;
|
||
|
|
||
|
const FLAVOUR = ref_csv::OO_FLAVOUR;
|
||
|
|
||
|
function __construct($output=null, ?array $params=null) {
|
||
|
A::set_nn($params, "output", $output);
|
||
|
self::set_parametrable_params_defaults($params, [
|
||
|
"flavour" => static::FLAVOUR,
|
||
|
]);
|
||
|
$this->helper = new Assoc2CsvHelper();
|
||
|
parent::__construct($params);
|
||
|
}
|
||
|
|
||
|
function setEncodingFilter(string $to, string $from="utf-8"): void {
|
||
|
$this->_setEncodingFilter($from, $to);
|
||
|
}
|
||
|
|
||
|
const PARAMETRABLE_PARAMS_SCHEMA = [
|
||
|
"output" => [null, null, "fichier en sortie"],
|
||
|
"input_encoding" => ["?string", null, "encoding en entrée"],
|
||
|
"output_encoding" => ["?string", null, "encoding en sortie"],
|
||
|
"flavour" => ["?string", null, "type de fichier CSV"],
|
||
|
"multi_schema" => ["bool", false, "les flux multi-schémas sont-ils supportés?"],
|
||
|
"headers" => ["?array", null, "liste et ordre des en-têtes en entrée"],
|
||
|
"output_headers" => ["bool", true, "faut-il afficher les en-têtes en sortie?"],
|
||
|
"header_mappings" => [null /*string|array*/, null, "mappings des en-têtes",
|
||
|
"desc" => "le mapping peut-être fourni comme un tableau [include, exclude => null, dest => source] ou une chaine 'include,=exclude,dest=source'"
|
||
|
],
|
||
|
];
|
||
|
|
||
|
protected $ppFlavour;
|
||
|
|
||
|
function pp_setFlavour(string $flavour): void {
|
||
|
$this->ppFlavour = flavours::verifix($flavour);
|
||
|
}
|
||
|
|
||
|
/** @var Assoc2CsvHelper */
|
||
|
protected $helper;
|
||
|
function pp_setMultiSchema(bool $multiSchema) { $this->helper->setMultiSchema($multiSchema); }
|
||
|
function pp_setHeaders(array $headers) { $this->helper->setHeaders($headers); }
|
||
|
function pp_setOutputHeaders(bool $outputHeaders) { $this->helper->setOutputHeaders($outputHeaders); }
|
||
|
function pp_setHeaderMappings($headerMappings) { $this->helper->setHeaderMappings($headerMappings); }
|
||
|
|
||
|
protected function afterSetParametrableParams(array $modifiedKeys, ?Metadata $md=null): void {
|
||
|
if (!in_array("output_encoding", $modifiedKeys)
|
||
|
&& in_array("flavour", $modifiedKeys)) {
|
||
|
$this->ppOutputEncoding = flavours::get_encoding($this->ppFlavour);
|
||
|
$modifiedKeys[] = "output_encoding";
|
||
|
}
|
||
|
$this->encodingOutput__afterSetParametrableParams($modifiedKeys);
|
||
|
}
|
||
|
|
||
|
private $inSession;
|
||
|
|
||
|
function write(array $row, bool $flush=false): self {
|
||
|
$flavour = $this->ppFlavour;
|
||
|
$helper = $this->helper;
|
||
|
$writer = $this->ensureWriter(true);
|
||
|
$resource = $writer->getResource();
|
||
|
if ($resource !== null) {
|
||
|
[$separator, $enclosure, $escape] = flavours::get_params($flavour);
|
||
|
if ($this->inSession == null) {
|
||
|
$this->inSession = true;
|
||
|
[$skipLine, $outputHeaders, $headers] = $helper->checkHeaders(null);
|
||
|
if ($skipLine) fwrite($resource, "\n");
|
||
|
if ($outputHeaders) {
|
||
|
$cols = $helper->cookHeaders($headers);
|
||
|
fputcsv($resource, $cols, $separator, $enclosure, $escape);
|
||
|
$helper->resetOutputHeaders($headers);
|
||
|
}
|
||
|
}
|
||
|
[$skipLine, $outputHeaders, $headers] = $helper->checkHeaders($row);
|
||
|
if ($skipLine) fwrite($resource, "\n");
|
||
|
if ($outputHeaders) {
|
||
|
$cols = $helper->cookHeaders($headers);
|
||
|
fputcsv($resource, $cols, $separator, $enclosure, $escape);
|
||
|
$helper->resetOutputHeaders($headers);
|
||
|
}
|
||
|
$cols = $helper->cookValues($headers, $row);
|
||
|
fputcsv($resource, $cols, $separator, $enclosure, $escape);
|
||
|
if ($flush) fflush($resource);
|
||
|
} else {
|
||
|
if ($this->inSession == null) {
|
||
|
$this->inSession = true;
|
||
|
[$skipLine, $outputHeaders, $headers] = $helper->checkHeaders(null);
|
||
|
if ($skipLine) $writer->wnl();
|
||
|
if ($outputHeaders) {
|
||
|
$cols = $helper->cookHeaders($headers);
|
||
|
$writer->write($helper->getLine($cols, $flavour, false));
|
||
|
$helper->resetOutputHeaders($headers);
|
||
|
}
|
||
|
}
|
||
|
[$skipLine, $outputHeaders, $headers] = $helper->checkHeaders($row);
|
||
|
if ($skipLine) $writer->wnl();
|
||
|
if ($outputHeaders) {
|
||
|
$cols = $helper->cookHeaders($headers);
|
||
|
$writer->write($helper->getLine($cols, $flavour, false));
|
||
|
$helper->resetOutputHeaders($headers);
|
||
|
}
|
||
|
$cols = $helper->cookValues($headers, $row);
|
||
|
$writer->write($helper->getLine($cols, $flavour, false));
|
||
|
}
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
function writeAll(iterable $items): self {
|
||
|
foreach ($items as $value) {
|
||
|
$this->write(A::with($value));
|
||
|
}
|
||
|
$resource = $this->ensureWriter(true)->getResource();
|
||
|
if ($resource !== null) fflush($resource);
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
function close(): void {
|
||
|
$this->teardownWriter(true);
|
||
|
$this->inSession = null;
|
||
|
}
|
||
|
|
||
|
#############################################################################
|
||
|
const _AUTOGEN_CONSTS = [
|
||
|
"" => [self::class, "_AUTOGEN_CONSTS", self::class],
|
||
|
];
|
||
|
const _AUTOGEN_LITERALS = /*autogen*/[
|
||
|
[
|
||
|
\nur\b\params\parametrable_utils::class,
|
||
|
'\\nur\\b\\params\\parametrable_utils::class',
|
||
|
],
|
||
|
[self::class, 'self::class'],
|
||
|
[
|
||
|
self::PARAMETRABLE_PARAMS_SCHEMA,
|
||
|
'self::PARAMETRABLE_PARAMS_SCHEMA',
|
||
|
],
|
||
|
];
|
||
|
const _AUTOGEN_METHODS = /*autogen*/[
|
||
|
[
|
||
|
\nur\b\params\parametrable_utils::class,
|
||
|
'_autogen_methods_getters',
|
||
|
self::PARAMETRABLE_PARAMS_SCHEMA,
|
||
|
self::class,
|
||
|
],
|
||
|
[
|
||
|
\nur\b\params\parametrable_utils::class,
|
||
|
'_autogen_methods_setters',
|
||
|
self::PARAMETRABLE_PARAMS_SCHEMA,
|
||
|
self::class,
|
||
|
],
|
||
|
];
|
||
|
const _AUTO_GETTERS = /*autogen*/[
|
||
|
'getInputEncoding' => 'input_encoding',
|
||
|
'getOutputEncoding' => 'output_encoding',
|
||
|
];
|
||
|
const _AUTO_SETTERS = /*autogen*/[
|
||
|
'setOutput' => 'output',
|
||
|
'setInputEncoding' => 'input_encoding',
|
||
|
'setOutputEncoding' => 'output_encoding',
|
||
|
'setFlavour' => 'flavour',
|
||
|
'setMultiSchema' => 'multi_schema',
|
||
|
'setHeaders' => 'headers',
|
||
|
'setOutputHeaders' => 'output_headers',
|
||
|
];
|
||
|
#--autogen-dynamic--
|
||
|
}
|