Part-DB.Part-DB-server/src/Serializer/PartNormalizer.php

197 lines
6.9 KiB
PHP
Raw Normal View History

<?php
declare(strict_types=1);
/*
* This file is part of Part-DB (https://github.com/Part-DB/Part-DB-symfony).
*
* Copyright (C) 2019 - 2023 Jan Böhmer (https://github.com/jbtronics)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
namespace App\Serializer;
use App\Entity\Parts\Part;
use App\Entity\Parts\PartLot;
use App\Entity\Parts\StorageLocation;
use App\Entity\Parts\Supplier;
use App\Entity\PriceInformations\Orderdetail;
use App\Entity\PriceInformations\Pricedetail;
use Brick\Math\BigDecimal;
2023-06-11 18:12:22 +02:00
use Symfony\Component\DependencyInjection\Attribute\Autowire;
use Symfony\Component\Serializer\Normalizer\CacheableSupportsMethodInterface;
use Symfony\Component\Serializer\Normalizer\DenormalizerInterface;
2023-06-11 18:12:22 +02:00
use Symfony\Component\Serializer\Normalizer\NormalizerAwareInterface;
use Symfony\Component\Serializer\Normalizer\NormalizerAwareTrait;
use Symfony\Component\Serializer\Normalizer\NormalizerInterface;
use Symfony\Component\Serializer\Normalizer\ObjectNormalizer;
2023-06-11 15:02:59 +02:00
/**
* @see \App\Tests\Serializer\PartNormalizerTest
*/
2023-06-11 18:12:22 +02:00
class PartNormalizer implements NormalizerInterface, DenormalizerInterface
{
2023-06-11 18:12:22 +02:00
private const DENORMALIZE_KEY_MAPPING = [
'notes' => 'comment',
'quantity' => 'instock',
'amount' => 'instock',
2023-03-13 00:52:22 +01:00
'mpn' => 'manufacturer_product_number',
'spn' => 'supplier_part_number',
2023-03-14 00:02:40 +01:00
'supplier_product_number' => 'supplier_part_number',
'storage_location' => 'storelocation',
];
2023-06-11 18:12:22 +02:00
public function __construct(
private readonly StructuralElementFromNameDenormalizer $locationDenormalizer,
#[Autowire(service: ObjectNormalizer::class)]
2023-06-11 23:16:07 +02:00
private readonly NormalizerInterface $normalizer,
#[Autowire(service: ObjectNormalizer::class)]
private readonly DenormalizerInterface $denormalizer,
)
{
}
2023-06-11 17:38:08 +02:00
public function supportsNormalization($data, string $format = null, array $context = []): bool
{
return $data instanceof Part;
}
/**
* @return (float|mixed)[]|\ArrayObject|null|scalar
*
* @psalm-return \ArrayObject|array{total_instock: float|mixed,...}|null|scalar
*/
public function normalize($object, string $format = null, array $context = [])
{
if (!$object instanceof Part) {
throw new \InvalidArgumentException('This normalizer only supports Part objects!');
}
$data = $this->normalizer->normalize($object, $format, $context);
//Remove type field for CSV export
if ($format === 'csv') {
unset($data['type']);
}
$data['total_instock'] = $object->getAmountSum();
return $data;
}
2023-06-11 17:38:08 +02:00
public function supportsDenormalization($data, string $type, string $format = null, array $context = []): bool
{
return is_array($data) && is_a($type, Part::class, true);
}
private function normalizeKeys(array &$data): array
{
//Rename keys based on the mapping, while leaving the data untouched
foreach ($data as $key => $value) {
if (isset(self::DENORMALIZE_KEY_MAPPING[$key])) {
$data[self::DENORMALIZE_KEY_MAPPING[$key]] = $value;
unset($data[$key]);
}
}
return $data;
}
2023-04-15 21:18:11 +02:00
public function denormalize($data, string $type, string $format = null, array $context = []): ?Part
{
$this->normalizeKeys($data);
//Empty IPN should be null, or we get a constraint error
2023-03-14 00:17:13 +01:00
if (isset($data['ipn']) && $data['ipn'] === '') {
$data['ipn'] = null;
}
//Fill empty needs_review and needs_review_comment fields with false
if (empty($data['needs_review'])) {
$data['needs_review'] = false;
}
if (empty($data['favorite'])) {
$data['favorite'] = false;
}
if (empty($data['minamount'])) {
$data['minamount'] = 0.0;
}
2023-06-11 23:16:07 +02:00
$object = $this->denormalizer->denormalize($data, $type, $format, $context);
if (!$object instanceof Part) {
throw new \InvalidArgumentException('This normalizer only supports Part objects!');
}
if ((isset($data['instock']) && trim((string) $data['instock']) !== "") || (isset($data['storelocation']) && trim((string) $data['storelocation']) !== "")) {
$partLot = new PartLot();
if (isset($data['instock']) && $data['instock'] !== "") {
//Replace comma with dot
$instock = (float) str_replace(',', '.', (string) $data['instock']);
$partLot->setAmount($instock);
} else {
$partLot->setInstockUnknown(true);
}
if (isset($data['storelocation']) && $data['storelocation'] !== "") {
$location = $this->locationDenormalizer->denormalize($data['storelocation'], StorageLocation::class, $format, $context);
$partLot->setStorageLocation($location);
}
$object->addPartLot($partLot);
}
if (isset($data['supplier']) && $data['supplier'] !== "") {
$supplier = $this->locationDenormalizer->denormalize($data['supplier'], Supplier::class, $format, $context);
if ($supplier) {
$orderdetail = new Orderdetail();
$orderdetail->setSupplier($supplier);
if (isset($data['supplier_part_number']) && $data['supplier_part_number'] !== "") {
$orderdetail->setSupplierpartnr($data['supplier_part_number']);
}
$object->addOrderdetail($orderdetail);
if (isset($data['price']) && $data['price'] !== "") {
$pricedetail = new Pricedetail();
$pricedetail->setMinDiscountQuantity(1);
$pricedetail->setPriceRelatedQuantity(1);
$price = BigDecimal::of(str_replace(',', '.', (string) $data['price']));
$pricedetail->setPrice($price);
$orderdetail->addPricedetail($pricedetail);
}
}
}
return $object;
}
/**
* @return bool[]
*/
public function getSupportedTypes(?string $format): array
{
//Must be false, because we rely on is_array($data) in supportsDenormalization()
2023-06-11 18:12:22 +02:00
return [
Part::class => false,
];
}
}