Co-authored-by: bastien <bastien.ollier1@gmail.com> Co-authored-by: clfreville2 <clement.freville2@etu.uca.fr> Reviewed-on: #14 Reviewed-by: Clément FRÉVILLE <clement.freville2@etu.uca.fr>pull/18/head
parent
6a6a135891
commit
ccb2b541ea
@ -0,0 +1,13 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
use Symfony\Component\HttpFoundation\File\File;
|
||||||
|
|
||||||
|
class DummyImageSafetyService implements ImageSafetyServiceInterface
|
||||||
|
{
|
||||||
|
public function isValid(File $file): bool
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,10 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
use Symfony\Component\HttpFoundation\File\File;
|
||||||
|
|
||||||
|
interface ImageSafetyServiceInterface
|
||||||
|
{
|
||||||
|
public function isValid(File $file): bool;
|
||||||
|
}
|
@ -0,0 +1,42 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||||
|
use Symfony\Component\HttpFoundation\File\File;
|
||||||
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
|
||||||
|
readonly class SightEngineImageSafetyService implements ImageSafetyServiceInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private HttpClientInterface $client,
|
||||||
|
#[Autowire(env: 'API_USER_SIGHT_ENGINE')] private string $apiUser,
|
||||||
|
#[Autowire(env: 'API_KEY_SIGHT_ENGINE')] private string $apiKey,
|
||||||
|
)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isValid(File $file): bool
|
||||||
|
{
|
||||||
|
$handle = fopen($file->getRealPath(), 'r');
|
||||||
|
if ($handle === false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$response = $this->client->request('POST', 'https://api.sightengine.com/1.0/check.json', [
|
||||||
|
'body' => [
|
||||||
|
'media' => $handle,
|
||||||
|
'models' => 'nudity-2.1',
|
||||||
|
'api_user' => $this->apiUser,
|
||||||
|
'api_secret' => $this->apiKey,
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
fclose($handle);
|
||||||
|
|
||||||
|
$output = $response->toArray();
|
||||||
|
$scoreNudity = $output['nudity'];
|
||||||
|
|
||||||
|
return $scoreNudity['sexual_activity'] < 0.8 &&
|
||||||
|
$scoreNudity['sexual_display'] < 0.8 &&
|
||||||
|
$scoreNudity['erotica'] < 0.8;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,15 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Validator;
|
||||||
|
|
||||||
|
use Symfony\Component\Validator\Constraint;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @Annotation
|
||||||
|
* @Target({"PROPERTY", "METHOD", "ANNOTATION"})
|
||||||
|
*/
|
||||||
|
#[\Attribute(\Attribute::TARGET_PROPERTY | \Attribute::TARGET_METHOD | \Attribute::IS_REPEATABLE)]
|
||||||
|
class ImageSafety extends Constraint
|
||||||
|
{
|
||||||
|
public string $message = 'The uploaded image is not safe.';
|
||||||
|
}
|
@ -0,0 +1,30 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Validator;
|
||||||
|
|
||||||
|
use App\Service\ImageSafetyServiceInterface;
|
||||||
|
use Symfony\Component\Validator\Constraint;
|
||||||
|
use Symfony\Component\Validator\ConstraintValidator;
|
||||||
|
|
||||||
|
class ImageSafetyValidator extends ConstraintValidator
|
||||||
|
{
|
||||||
|
public function __construct(private readonly ImageSafetyServiceInterface $imageSafetyService)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param mixed $value
|
||||||
|
* @param ImageSafety $constraint
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function validate(mixed $value, Constraint $constraint): void
|
||||||
|
{
|
||||||
|
if (null === $value || '' === $value) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$this->imageSafetyService->isValid($value)) {
|
||||||
|
$this->context->buildViolation($constraint->message)->addViolation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue