From 678363ce8d1e6a5828dacb3166df7fde390cad73 Mon Sep 17 00:00:00 2001 From: Merlin Date: Sat, 16 Mar 2024 14:59:38 +0100 Subject: [PATCH] Add CodeExporter, fixes #30 --- README.md | 22 ++++++++++++++ src/Code.php | 29 +++++++++++++++++++ src/CodeInterface.php | 13 +++++++++ src/Internal/GenericExporter.php | 9 ++++++ src/Internal/ObjectExporter/CodeExporter.php | 30 ++++++++++++++++++++ src/VarExporter.php | 5 ++++ 6 files changed, 108 insertions(+) create mode 100644 src/Code.php create mode 100644 src/CodeInterface.php create mode 100644 src/Internal/ObjectExporter/CodeExporter.php diff --git a/README.md b/README.md index ea5b30a..1681c54 100644 --- a/README.md +++ b/README.md @@ -505,6 +505,28 @@ try { } ``` +## Literal code expressions + +```php +use Brick\VarExporter\VarExporter; +use Brick\Code; + +echo VarExporter::export([ + 'literal_expression' => Code::create("$app_root", ". '/../private'"); +], VarExporter::ALLOW_CODE); +``` + +This code will output: + +```php +[ + 'literal_expression' => $app_root + . '/../private', +] +``` + +Also, you can implement CodeInterface in your custom code. + ## Limitations - Exporting internal classes other than `stdClass` and `Closure`, and classes implementing `__set_state()` (most notably DateTime classes) is currently not supported. `VarExporter` will throw an `ExportException` if it finds one. diff --git a/src/Code.php b/src/Code.php new file mode 100644 index 0000000..dcb7847 --- /dev/null +++ b/src/Code.php @@ -0,0 +1,29 @@ +lines = $lines; + } + + public static function create(string ...$lines): self { + return new self($lines); + } + + public function toLines(): array + { + return $this->lines; + } + +} diff --git a/src/CodeInterface.php b/src/CodeInterface.php new file mode 100644 index 0000000..84219e4 --- /dev/null +++ b/src/CodeInterface.php @@ -0,0 +1,13 @@ +objectExporters[] = new ObjectExporter\CodeExporter($this); $this->objectExporters[] = new ObjectExporter\StdClassExporter($this); if (! ($options & VarExporter::NO_CLOSURES)) { @@ -113,6 +121,7 @@ public function __construct(int $options, int $indentLevel = 0) $this->inlineScalarList = (bool) ($options & VarExporter::INLINE_SCALAR_LIST); $this->closureSnapshotUses = (bool) ($options & VarExporter::CLOSURE_SNAPSHOT_USES); $this->trailingCommaInArray = (bool) ($options & VarExporter::TRAILING_COMMA_IN_ARRAY); + $this->allowCode = (bool) ($options & VarExporter::ALLOW_CODE); $this->indentLevel = $indentLevel; } diff --git a/src/Internal/ObjectExporter/CodeExporter.php b/src/Internal/ObjectExporter/CodeExporter.php new file mode 100644 index 0000000..74c1ebe --- /dev/null +++ b/src/Internal/ObjectExporter/CodeExporter.php @@ -0,0 +1,30 @@ +exporter->allowCode + && $reflectionObject->implementsInterface(CodeInterface::class); + } + + public function export(object $object, \ReflectionObject $reflectionObject, array $path, array $parentIds): array + { + assert($object instanceof CodeInterface); + return $object->toLines(); + } + +} diff --git a/src/VarExporter.php b/src/VarExporter.php index 138c431..08ad03e 100644 --- a/src/VarExporter.php +++ b/src/VarExporter.php @@ -74,6 +74,11 @@ final class VarExporter */ public const INLINE_ARRAY = 1 << 11; + /** + * Any value implementing CodeInterface will print a literal expression. + */ + public const ALLOW_CODE = 1 << 12; + /** * @param mixed $var The variable to export. * @param int $options A bitmask of options. Possible values are `VarExporter::*` constants.