Stack & Setup
- Dockerisierte Dev-Umgebung (PHP 8.4-FPM, Nginx, MariaDB 11.4)
- Symfony 7.4 + API Platform 4.3, Doctrine ORM, LexikJWT, Messenger
- Vue 3 + TS (Vite), Vue Router, Pinia, Axios
Kern-Domäne & Auth
- Entitäten: User, PlatformPlan, Reseller, Company, Domain, Location,
Employee, ContactLink (UUIDv7)
- JWT-Login (/api/login), Rollen-Hierarchie, /api/me
- Mandantentrennung via API-Platform-Query-Extension (Lesen) +
TenantStampProcessor (Schreiben)
Öffentliche Profile (SSR)
- Profil-Landingpage, vCard-Download, QR-Code im Marken-Look
- Stabiler NFC/QR-Kurz-Link /t/{code} -> Redirect aufs aktuelle Profil
- Firmenspezifisches Branding (Farben/Logo) auf der Profilseite
Verwaltungsoberfläche (SPA)
- Brand-Look (dunkle Sidebar), rollenbasierte Navigation
- Dashboard, Reseller (+Provisioning), Firmen, Mitarbeiter, Standorte,
Domains, Design/Branding mit Live-Vorschau
Konzept & Doku: docs/KONZEPT.md (inkl. Wallet/Sync §12), README.md
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
135 lines
2.9 KiB
PHP
135 lines
2.9 KiB
PHP
<?php
|
|
|
|
namespace App\Entity;
|
|
|
|
use ApiPlatform\Metadata\ApiResource;
|
|
use App\Repository\DomainRepository;
|
|
use Doctrine\ORM\Mapping as ORM;
|
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
|
use Symfony\Component\Uid\Uuid;
|
|
|
|
/**
|
|
* Sub- oder Custom-Domain eines Firmenkunden (siehe KONZEPT §11).
|
|
*/
|
|
#[ORM\Entity(repositoryClass: DomainRepository::class)]
|
|
#[ApiResource(security: "is_granted('ROLE_COMPANY_ADMIN')")]
|
|
class Domain implements ResellerOwnedInterface
|
|
{
|
|
public const TYPE_SUBDOMAIN = 'subdomain';
|
|
public const TYPE_CUSTOM = 'custom';
|
|
|
|
public const STATUS_PENDING = 'pending';
|
|
public const STATUS_VERIFIED = 'verified';
|
|
public const STATUS_FAILED = 'failed';
|
|
|
|
#[ORM\Id]
|
|
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
|
private Uuid $id;
|
|
|
|
#[ORM\Column(length: 255, unique: true)]
|
|
private string $hostname;
|
|
|
|
#[ORM\Column(length: 20)]
|
|
private string $type = self::TYPE_SUBDOMAIN;
|
|
|
|
#[ORM\Column(length: 20)]
|
|
private string $status = self::STATUS_PENDING;
|
|
|
|
#[ORM\Column(length: 20)]
|
|
private string $tlsStatus = 'none';
|
|
|
|
#[ORM\Column(type: 'datetime_immutable', nullable: true)]
|
|
private ?\DateTimeImmutable $verificationCheckedAt = null;
|
|
|
|
#[ORM\ManyToOne(targetEntity: Company::class, inversedBy: 'domains')]
|
|
#[ORM\JoinColumn(nullable: false)]
|
|
private Company $company;
|
|
|
|
public function __construct()
|
|
{
|
|
$this->id = Uuid::v7();
|
|
}
|
|
|
|
public function getId(): Uuid
|
|
{
|
|
return $this->id;
|
|
}
|
|
|
|
public function getHostname(): string
|
|
{
|
|
return $this->hostname;
|
|
}
|
|
|
|
public function setHostname(string $hostname): self
|
|
{
|
|
$this->hostname = $hostname;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function getType(): string
|
|
{
|
|
return $this->type;
|
|
}
|
|
|
|
public function setType(string $type): self
|
|
{
|
|
$this->type = $type;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function getStatus(): string
|
|
{
|
|
return $this->status;
|
|
}
|
|
|
|
public function setStatus(string $status): self
|
|
{
|
|
$this->status = $status;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function getTlsStatus(): string
|
|
{
|
|
return $this->tlsStatus;
|
|
}
|
|
|
|
public function setTlsStatus(string $tlsStatus): self
|
|
{
|
|
$this->tlsStatus = $tlsStatus;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function getVerificationCheckedAt(): ?\DateTimeImmutable
|
|
{
|
|
return $this->verificationCheckedAt;
|
|
}
|
|
|
|
public function setVerificationCheckedAt(?\DateTimeImmutable $verificationCheckedAt): self
|
|
{
|
|
$this->verificationCheckedAt = $verificationCheckedAt;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function getCompany(): Company
|
|
{
|
|
return $this->company;
|
|
}
|
|
|
|
public function setCompany(Company $company): self
|
|
{
|
|
$this->company = $company;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function getReseller(): ?Reseller
|
|
{
|
|
return $this->company->getReseller();
|
|
}
|
|
}
|