Convert Moderation to a service+validator
continuous-integration/drone/push Build is failing Details

pull/14/head
Clément FRÉVILLE 11 months ago
parent 32e0774302
commit b0f09a8217

@ -62,6 +62,10 @@ steps:
CODEFIRST_CLIENTDRONE_ENV_SERVER_NAME: http://codefirst.iut.uca.fr
CODEFIRST_CLIENTDRONE_ENV_CORS_ALLOW_ORIGIN: https://codefirst.iut.uca.fr
CODEFIRST_CLIENTDRONE_ENV_TRUSTED_PROXIES: REMOTE_ADDR
CODEFIRST_CLIENTDRONE_ENV_API_USER_SIGHT_ENGINE:
from_secret: API_USER_SIGHT_ENGINE
CODEFIRST_CLIENTDRONE_ENV_API_KEY_SIGHT_ENGINE:
from_secret: API_KEY_SIGHT_ENGINE
depends_on:
- docker-image
when:

@ -47,8 +47,7 @@
"symfonycasts/verify-email-bundle": "^1.17",
"twig/extra-bundle": "^2.12|^3.0",
"twig/twig": "^2.12|^3.0",
"vich/uploader-bundle": "^2.3",
"ext-curl": "*"
"vich/uploader-bundle": "^2.3"
},
"config": {
"allow-plugins": {

@ -6,7 +6,6 @@
parameters:
trusted_proxies: '%env(TRUSTED_PROXIES)%'
env(TRUSTED_PROXIES): 127.0.0.1
app.API_KEY_SIGHT_ENGINE: '%env(API_KEY_SIGHT_ENGINE)%'
services:
# default configuration for services in *this* file
@ -27,5 +26,11 @@ services:
bind:
$processor: '@api_platform.doctrine.orm.state.persist_processor'
App\Service\ImageSafetyServiceInterface: '@App\Service\DummyImageSafetyService'
# add more service definitions when explicit configuration is needed
# please note that last definitions always *replace* previous ones
when@prod:
services:
App\Service\ImageSafetyServiceInterface: '@App\Service\SightEngineImageSafetyService'

@ -8,11 +8,8 @@ use App\Entity\User;
use App\Form\CommentType;
use App\Form\PostType;
use App\Repository\PostRepository;
use App\Security\Moderation\Moderation;
use CURLFile;
use Doctrine\ORM\EntityManagerInterface;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\Routing\Attribute\Route;
@ -48,19 +45,15 @@ class PostController extends AbstractController
#[Route('/post/new', name: 'app_post_new', methods: ['GET', 'POST'])]
#[IsGranted('ROLE_USER')]
public function new(Request $request, EntityManagerInterface $entityManager, ParameterBagInterface $env): Response
public function new(Request $request, EntityManagerInterface $entityManager): Response
{
$post = new Post();
$form = $this->createForm(PostType::class, $post);
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid()) {
$api_key = $this->getParameter('app.API_KEY_SIGHT_ENGINE');
$moderation = new Moderation($api_key);
if($moderation->valide($post->getImageFile()->getRealPath())){
$entityManager->persist($post);
$entityManager->flush();
}
$entityManager->persist($post);
$entityManager->flush();
return $this->redirectToRoute('app_posts', [], Response::HTTP_SEE_OTHER);
}

@ -8,6 +8,7 @@ use ApiPlatform\Metadata\ApiProperty;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GetCollection;
use App\Repository\PostRepository;
use App\Validator\ImageSafety;
use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\Common\Collections\Collection;
use Doctrine\DBAL\Types\Types;
@ -64,6 +65,7 @@ class Post
#[Vich\UploadableField(mapping: 'posts', fileNameProperty: 'image')]
#[Assert\Image]
#[ImageSafety]
private ?File $imageFile = null;
#[ORM\Column(type: Types::TEXT)]
@ -184,6 +186,9 @@ class Post
public function setImageFile(?File $imageFile): static
{
$this->imageFile = $imageFile;
if ($imageFile !== null) {
$this->updatedAt = new \DateTimeImmutable();
}
return $this;
}

@ -21,7 +21,9 @@ class PostType extends AbstractType
->add('latitude')
->add('longitude')
->add('altitude')
->add('imageFile', FileType::class)
->add('imageFile', FileType::class, [
'required' => false,
])
->add('commentary')
->add('species', EntityType::class, [
'class' => Species::class,

@ -1,39 +0,0 @@
<?php
namespace App\Security\Moderation;
use CURLFile;
class Moderation
{
private String $api_key;
public function __construct(String $api_key)
{
$this->api_key = $api_key;
}
public function valide($file_image)
{
$params_api = array(
'media' => new CurlFile($file_image),
'models' => 'nudity-2.1',
'api_user' => '26959338',
'api_secret' => $this->api_key,
);
$ch = curl_init('https://api.sightengine.com/1.0/check.json');
curl_setopt($ch, CURLOPT_POST, true);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_POSTFIELDS, $params_api);
$response = curl_exec($ch);
curl_close($ch);
$output = json_decode($response, true);
$score_nudity = $output["nudity"];
return $score_nudity["sexual_activity"] < 0.8 &&
$score_nudity["sexual_display"] < 0.8 &&
$score_nudity["erotica"] < 0.8;
}
}

@ -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,39 @@
<?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');
$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,27 @@
<?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)
{
}
public function validate($value, Constraint $constraint): void
{
/* @var ImageSafety $constraint */
if (null === $value || '' === $value) {
return;
}
if (!$this->imageSafetyService->isValid($value)) {
$this->context->buildViolation($constraint->message)->addViolation();
}
}
}
Loading…
Cancel
Save