191 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| namespace nur\io\fsv;
 | |
| 
 | |
| use IteratorAggregate;
 | |
| use nur\A;
 | |
| use nur\b\ICloseable;
 | |
| use nur\b\io\EOFException;
 | |
| use nur\b\io\IReader;
 | |
| use nur\b\io\Tfilter;
 | |
| use nur\b\params\Parametrable;
 | |
| use nur\b\params\Tparametrable;
 | |
| use nur\b\ValueException;
 | |
| use nur\reader;
 | |
| use nur\str;
 | |
| 
 | |
| /**
 | |
|  * Class FsvReader: produit un flux de données à partir d'une source au format
 | |
|  * FSV.
 | |
|  *
 | |
|  * --autogen-properties-and-methods--
 | |
|  * @method int getSkipLines()
 | |
|  * @method setInput($value)
 | |
|  * @method array setSchema(array $value)
 | |
|  * @method string|null setInputEncoding(?string $value)
 | |
|  * @method string|null setOutputEncoding(?string $value)
 | |
|  * @method string|null setMapEmpty(?string $value)
 | |
|  * @method bool setOutputSeq(bool $value)
 | |
|  * @method int setSkipLines(int $value)
 | |
|  */
 | |
| class FsvReader extends Parametrable implements IteratorAggregate, ICloseable {
 | |
|   use Tparametrable, Tfilter;
 | |
| 
 | |
|   function __construct($input=null, ?array $params=null) {
 | |
|     A::set_nn($params, "input", $input);
 | |
|     $this->schema = new FsvSchema();
 | |
|     parent::__construct($params);
 | |
|   }
 | |
| 
 | |
|   const PARAMETRABLE_PARAMS_SCHEMA = [
 | |
|     "input" => [null, null, "fichier en entrée"],
 | |
|     "schema" => ["array", null, "schéma des données"],
 | |
|     "input_encoding" => ["?string", null, "encoding en entrée"],
 | |
|     "output_encoding" => ["?string", null, "encoding en sortie"],
 | |
|     "map_empty" => ["?string", null, "valeur adoptée pour les chaines vides"],
 | |
|     "output_seq" => ["bool", null, "faut-il générer en-têtes et données en tableaux séquentiels?"],
 | |
|     "skip_lines" => ["int", 0, "nombre de lignes à sauter en entrée"],
 | |
|   ];
 | |
| 
 | |
|   protected $ppInput;
 | |
| 
 | |
|   function pp_setInput($input): void {
 | |
|     if ($input instanceof IReader) $this->reader = $input;
 | |
|     else $this->ppInput = $input;
 | |
|   }
 | |
| 
 | |
|   /** @var FsvSchema */
 | |
|   protected $schema;
 | |
| 
 | |
|   function pp_setSchema($schema): void {
 | |
|     if ($schema instanceof FsvSchema) $this->schema = $schema;
 | |
|     elseif (is_array($schema)) $this->schema->setSchema($schema);
 | |
|     elseif ($schema !== null) throw ValueException::invalid_value($schema, "schema");
 | |
|   }
 | |
|   function pp_setInputEncoding($inputEncoding): void {
 | |
|     $this->schema->setInputEncoding($inputEncoding);
 | |
|   }
 | |
|   function pp_setOutputEncoding($outputEncoding): void {
 | |
|     $this->schema->setDataEncoding($outputEncoding);
 | |
|   }
 | |
|   function pp_setMapEmpty($mapEmpty): void {
 | |
|     $this->schema->setMapEmpty($mapEmpty);
 | |
|   }
 | |
|   function pp_setOutputSeq(bool $outputSeq): void {
 | |
|     $this->schema->setOutputSeq($outputSeq);
 | |
|   }
 | |
| 
 | |
|   /** @var int */
 | |
|   protected $ppSkipLines;
 | |
| 
 | |
|   /** @var IReader */
 | |
|   protected $reader;
 | |
| 
 | |
|   protected function ensureOpen(): IReader {
 | |
|     if ($this->reader === null) {
 | |
|       $this->reader = reader::with($this->ppInput);
 | |
|       $this->_rwAppendFilters($this->reader);
 | |
|     }
 | |
|     return $this->reader;
 | |
|   }
 | |
| 
 | |
|   protected function _parseLine(): bool {
 | |
|     if ($this->ppSkipLines > 0) {
 | |
|       $this->ppSkipLines--;
 | |
|       return false;
 | |
|     }
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   protected function _outputKeys(bool $reset=true): bool {
 | |
|     return $this->schema->_outputKeys($reset);
 | |
|   }
 | |
| 
 | |
|   protected function _getKeys(): array {
 | |
|     return $this->schema->_getKeys();
 | |
|   }
 | |
| 
 | |
|   protected function _parseColumns(string $line): array {
 | |
|     return $this->schema->parseRow($line);
 | |
|   }
 | |
| 
 | |
|   function getIterator() {
 | |
|     $reader = $this->ensureOpen();
 | |
|     try {
 | |
|       $resource = $reader->getResource();
 | |
|       if ($resource !== null) {
 | |
|         while (!$this->_parseLine()) {
 | |
|           if (fgets($resource) === false) break;
 | |
|         }
 | |
|         if ($this->_outputKeys()) yield $this->_getKeys();
 | |
|         while (($line = fgets($resource)) !== false) {
 | |
|           $line = str::strip_nl($line);
 | |
|           yield $this->_parseColumns($line);
 | |
|         }
 | |
|       } else {
 | |
|         while (true) {
 | |
|           try {
 | |
|             $line = $reader->readLine();
 | |
|           } catch (EOFException $e) {
 | |
|             break;
 | |
|           }
 | |
|           if (!$this->_parseLine()) continue;
 | |
|           if ($this->_outputKeys()) yield $this->_getKeys();
 | |
|           yield $this->_parseColumns($line);
 | |
|         }
 | |
|       }
 | |
|     } finally {
 | |
|       $this->close();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   function close(): void {
 | |
|     if ($this->reader !== null) {
 | |
|       $this->reader->close();
 | |
|       $this->reader = 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*/[
 | |
|     'getSkipLines' => 'skip_lines',
 | |
|   ];
 | |
|   const _AUTO_SETTERS = /*autogen*/[
 | |
|     'setInput' => 'input',
 | |
|     'setSchema' => 'schema',
 | |
|     'setInputEncoding' => 'input_encoding',
 | |
|     'setOutputEncoding' => 'output_encoding',
 | |
|     'setMapEmpty' => 'map_empty',
 | |
|     'setOutputSeq' => 'output_seq',
 | |
|     'setSkipLines' => 'skip_lines',
 | |
|   ];
 | |
|   #--autogen-dynamic--
 | |
| }
 |