<?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--
}