- Hinzufügen der DependencyInjection-Konfiguration für das Billing-Modul. - Erstellen der Invoice-Entity mit API-Ressourcen und Berechtigungen. - Konfigurieren der Services in services.yaml für das Billing-Modul. - Implementieren von CLI-Commands zur Verwaltung von Modul-Lizenzen und zur Auflistung installierter Module. - Erstellen eines API-Controllers zur Verwaltung von Modulen und Lizenzen. - Hinzufügen eines EventListeners für das Booten von Modulen. - Definieren von Interfaces für Lizenzvalidierung und Modul-Plugins. - Implementieren der ModuleRegistry zur Verwaltung und Booten von Modulen. - Erstellen eines LicenseValidator-Services zur Validierung und Registrierung von Lizenzen.
229 lines
5.7 KiB
PHP
229 lines
5.7 KiB
PHP
<?php
|
|
|
|
namespace MyCRM\BillingModule\Entity;
|
|
|
|
use ApiPlatform\Metadata\ApiResource;
|
|
use ApiPlatform\Metadata\Delete;
|
|
use ApiPlatform\Metadata\Get;
|
|
use ApiPlatform\Metadata\GetCollection;
|
|
use ApiPlatform\Metadata\Post;
|
|
use ApiPlatform\Metadata\Put;
|
|
use App\Entity\ModuleAwareInterface;
|
|
use Doctrine\Common\Collections\ArrayCollection;
|
|
use Doctrine\Common\Collections\Collection;
|
|
use Doctrine\ORM\Mapping as ORM;
|
|
use Symfony\Component\Serializer\Annotation\Groups;
|
|
|
|
/**
|
|
* Rechnung Entity für Billing-Modul
|
|
*
|
|
* Implementiert ModuleAwareInterface für Integration mit Permission-System
|
|
*/
|
|
#[ORM\Entity]
|
|
#[ORM\Table(name: 'billing_invoices')]
|
|
#[ApiResource(
|
|
operations: [
|
|
new GetCollection(
|
|
security: "is_granted('VIEW', 'billing')"
|
|
),
|
|
new Get(
|
|
security: "is_granted('VIEW', object)"
|
|
),
|
|
new Post(
|
|
security: "is_granted('CREATE', 'billing')"
|
|
),
|
|
new Put(
|
|
security: "is_granted('EDIT', object)"
|
|
),
|
|
new Delete(
|
|
security: "is_granted('DELETE', object)"
|
|
),
|
|
],
|
|
normalizationContext: ['groups' => ['invoice:read']],
|
|
denormalizationContext: ['groups' => ['invoice:write']]
|
|
)]
|
|
class Invoice implements ModuleAwareInterface
|
|
{
|
|
#[ORM\Id]
|
|
#[ORM\GeneratedValue]
|
|
#[ORM\Column]
|
|
#[Groups(['invoice:read'])]
|
|
private ?int $id = null;
|
|
|
|
#[ORM\Column(length: 50, unique: true)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private string $invoiceNumber;
|
|
|
|
#[ORM\Column]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private \DateTimeImmutable $invoiceDate;
|
|
|
|
#[ORM\Column(nullable: true)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private ?\DateTimeImmutable $dueDate = null;
|
|
|
|
#[ORM\Column(length: 20)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private string $status = 'draft'; // draft, sent, paid, overdue, cancelled
|
|
|
|
#[ORM\Column(type: 'decimal', precision: 10, scale: 2)]
|
|
#[Groups(['invoice:read'])]
|
|
private string $totalAmount = '0.00';
|
|
|
|
#[ORM\Column(type: 'decimal', precision: 10, scale: 2)]
|
|
#[Groups(['invoice:read'])]
|
|
private string $taxAmount = '0.00';
|
|
|
|
#[ORM\Column(length: 3)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private string $currency = 'EUR';
|
|
|
|
#[ORM\Column(type: 'text', nullable: true)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private ?string $notes = null;
|
|
|
|
/**
|
|
* Verknüpfung zum Contact aus dem Core-System
|
|
* (Über API-Relation, nicht über Foreign Key)
|
|
*/
|
|
#[ORM\Column(nullable: true)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private ?int $contactId = null;
|
|
|
|
/**
|
|
* @var Collection<int, InvoiceItem>
|
|
*/
|
|
#[ORM\OneToMany(
|
|
targetEntity: InvoiceItem::class,
|
|
mappedBy: 'invoice',
|
|
cascade: ['persist', 'remove'],
|
|
orphanRemoval: true
|
|
)]
|
|
#[Groups(['invoice:read', 'invoice:write'])]
|
|
private Collection $items;
|
|
|
|
#[ORM\Column]
|
|
#[Groups(['invoice:read'])]
|
|
private \DateTimeImmutable $createdAt;
|
|
|
|
#[ORM\Column]
|
|
#[Groups(['invoice:read'])]
|
|
private \DateTimeImmutable $updatedAt;
|
|
|
|
public function __construct()
|
|
{
|
|
$this->items = new ArrayCollection();
|
|
$this->invoiceDate = new \DateTimeImmutable();
|
|
$this->createdAt = new \DateTimeImmutable();
|
|
$this->updatedAt = new \DateTimeImmutable();
|
|
}
|
|
|
|
// Getters und Setters...
|
|
|
|
public function getId(): ?int
|
|
{
|
|
return $this->id;
|
|
}
|
|
|
|
public function getInvoiceNumber(): string
|
|
{
|
|
return $this->invoiceNumber;
|
|
}
|
|
|
|
public function setInvoiceNumber(string $invoiceNumber): self
|
|
{
|
|
$this->invoiceNumber = $invoiceNumber;
|
|
return $this;
|
|
}
|
|
|
|
public function getStatus(): string
|
|
{
|
|
return $this->status;
|
|
}
|
|
|
|
public function setStatus(string $status): self
|
|
{
|
|
$this->status = $status;
|
|
$this->updatedAt = new \DateTimeImmutable();
|
|
return $this;
|
|
}
|
|
|
|
public function getTotalAmount(): string
|
|
{
|
|
return $this->totalAmount;
|
|
}
|
|
|
|
public function setTotalAmount(string $totalAmount): self
|
|
{
|
|
$this->totalAmount = $totalAmount;
|
|
return $this;
|
|
}
|
|
|
|
public function getContactId(): ?int
|
|
{
|
|
return $this->contactId;
|
|
}
|
|
|
|
public function setContactId(?int $contactId): self
|
|
{
|
|
$this->contactId = $contactId;
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* @return Collection<int, InvoiceItem>
|
|
*/
|
|
public function getItems(): Collection
|
|
{
|
|
return $this->items;
|
|
}
|
|
|
|
public function addItem(InvoiceItem $item): self
|
|
{
|
|
if (!$this->items->contains($item)) {
|
|
$this->items->add($item);
|
|
$item->setInvoice($this);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function removeItem(InvoiceItem $item): self
|
|
{
|
|
if ($this->items->removeElement($item)) {
|
|
if ($item->getInvoice() === $this) {
|
|
$item->setInvoice(null);
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Berechnet Gesamtsumme aus Items
|
|
*/
|
|
public function calculateTotal(): void
|
|
{
|
|
$total = 0;
|
|
$tax = 0;
|
|
|
|
foreach ($this->items as $item) {
|
|
$itemTotal = $item->getQuantity() * (float) $item->getUnitPrice();
|
|
$total += $itemTotal;
|
|
$tax += $itemTotal * ((float) $item->getTaxRate() / 100);
|
|
}
|
|
|
|
$this->totalAmount = number_format($total + $tax, 2, '.', '');
|
|
$this->taxAmount = number_format($tax, 2, '.', '');
|
|
$this->updatedAt = new \DateTimeImmutable();
|
|
}
|
|
|
|
/**
|
|
* ModuleAwareInterface: Gibt den Modul-Namen zurück
|
|
*/
|
|
public function getModuleName(): string
|
|
{
|
|
return 'billing';
|
|
}
|
|
}
|