diff --git a/composer.json b/composer.json index 1bba1f4..9a8f189 100644 --- a/composer.json +++ b/composer.json @@ -32,6 +32,7 @@ "require": { "php": "^7.4 || ^8.0", "ext-libxml": "*", + "ext-dom": "*", "symfony/yaml": "^5.0", "symfony/routing": "^5.0", "symfony/console": "^5.0", diff --git a/phpstan-baseline-7.4.neon b/phpstan-baseline-7.4.neon new file mode 100644 index 0000000..1178cb3 --- /dev/null +++ b/phpstan-baseline-7.4.neon @@ -0,0 +1,7 @@ +parameters: + ignoreErrors: + - + message: '#^Parameter \#2 \$callback of function array_filter expects \(callable\(mixed\)\: bool\)\|null, Closure\(mixed\)\: \(array\|null\) given\.$#' + identifier: argument.type + count: 1 + path: src/bundle/DependencyInjection/IbexaAutomatedTranslationExtension.php \ No newline at end of file diff --git a/phpstan-baseline-gte-8.0.neon b/phpstan-baseline-gte-8.0.neon new file mode 100644 index 0000000..46bfb61 --- /dev/null +++ b/phpstan-baseline-gte-8.0.neon @@ -0,0 +1,7 @@ +parameters: + ignoreErrors: + - + message: '#^Parameter \#2 \$callback of function array_filter expects \(callable\(mixed\)\: bool\)\|null, Closure\(mixed\)\: array given\.$#' + identifier: argument.type + count: 1 + path: src/bundle/DependencyInjection/IbexaAutomatedTranslationExtension.php diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index d3ecc67..6179edf 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -1,11 +1,5 @@ parameters: ignoreErrors: - - - message: '#^Parameter \#2 \$callback of function array_filter expects \(callable\(mixed\)\: bool\)\|null, Closure\(mixed\)\: \(array\|null\) given\.$#' - identifier: argument.type - count: 1 - path: src/bundle/DependencyInjection/IbexaAutomatedTranslationExtension.php - - message: '#^Access to protected property Ibexa\\AdminUi\\Form\\Data\\ContentTranslationData\:\:\$content\.$#' identifier: property.protected @@ -30,6 +24,18 @@ parameters: count: 1 path: src/bundle/Form/TranslationAddDataTransformer.php + - + message: '#^Call to an undefined method DOMNode\:\:getAttribute\(\)\.$#' + identifier: method.notFound + count: 1 + path: src/lib/Encoder.php + + - + message: '#^Call to an undefined method DOMNode\:\:getAttribute\(\)\.$#' + identifier: method.notFound + count: 1 + path: src/lib/Encoder/Field/PageBuilderFieldEncoder.php + - message: '#^Cannot call method setValue\(\) on Ibexa\\Contracts\\FieldTypePage\\FieldType\\LandingPage\\Model\\Attribute\|null\.$#' identifier: method.nonObject diff --git a/phpstan-baseline.neon.php b/phpstan-baseline.neon.php new file mode 100644 index 0000000..7d0bd65 --- /dev/null +++ b/phpstan-baseline.neon.php @@ -0,0 +1,19 @@ +encode($results, XmlEncoder::FORMAT); + + $dom = new DOMDocument(); + $dom->loadXML($payload); + $xpath = new DOMXPath($dom); + $textNodes = $xpath->query('//text()'); + if ($textNodes !== false) { + foreach ($textNodes as $textNode) { + if ($textNode instanceof DOMCdataSection) { + $parent = $textNode->parentNode; + if ($parent === null) { + continue; + } + + $type = $parent->getAttribute('type'); + + if ($type !== RichTextValue::class) { + $newText = $dom->createTextNode($textNode->data); + $parent->replaceChild($newText, $textNode); + } + } + } + $payload = $dom->saveXML(); + + if (!$payload) { + throw new RuntimeException( + sprintf('Saving XML failed after removing CDATA, error: %s', preg_last_error_msg()) + ); + } + } + // here Encoder has decorated with CDATA, we don't want the CDATA return str_replace( [''], diff --git a/src/lib/Encoder/Field/PageBuilderFieldEncoder.php b/src/lib/Encoder/Field/PageBuilderFieldEncoder.php index e9588b7..22bb9d3 100644 --- a/src/lib/Encoder/Field/PageBuilderFieldEncoder.php +++ b/src/lib/Encoder/Field/PageBuilderFieldEncoder.php @@ -8,6 +8,9 @@ namespace Ibexa\AutomatedTranslation\Encoder\Field; +use DOMCdataSection; +use DOMDocument; +use DOMXPath; use Ibexa\AutomatedTranslation\Encoder\BlockAttribute\BlockAttributeEncoderManager; use Ibexa\AutomatedTranslation\Exception\EmptyTranslatedAttributeException; use Ibexa\Contracts\AutomatedTranslation\Encoder\Field\FieldEncoderInterface; @@ -15,7 +18,9 @@ use Ibexa\Core\FieldType\Value as APIValue; use Ibexa\FieldTypePage\FieldType\LandingPage\Value; use Ibexa\FieldTypePage\FieldType\Page\Block\Definition\BlockDefinitionFactoryInterface; +use Ibexa\FieldTypeRichText\FieldType\RichText\Value as RichTextValue; use InvalidArgumentException; +use RuntimeException; use Symfony\Component\Serializer\Encoder\XmlEncoder; final class PageBuilderFieldEncoder implements FieldEncoderInterface @@ -84,6 +89,35 @@ public function encode(Field $field): string XmlEncoder::ROOT_NODE_NAME => 'blocks', ]); + $dom = new DOMDocument(); + $dom->loadXML($payload); + $xpath = new DOMXPath($dom); + $textNodes = $xpath->query('//text()'); + if ($textNodes !== false) { + foreach ($textNodes as $textNode) { + if ($textNode instanceof DOMCdataSection) { + $parent = $textNode->parentNode; + if ($parent === null) { + continue; + } + + $type = $parent->getAttribute('type'); + + if ($type !== RichTextValue::class) { + $newText = $dom->createTextNode($textNode->data); + $parent->replaceChild($newText, $textNode); + } + } + } + $payload = $dom->saveXML(); + + if (!$payload) { + throw new RuntimeException( + sprintf('Saving XML failed after removing CDATA, error: %s', preg_last_error_msg()) + ); + } + } + $payload = str_replace('' . "\n", '', $payload); $payload = str_replace(