ethsam / easy-hcaptcha
A modern, zero-config Symfony bundle for hCaptcha: server-side verification, Twig widget, Form Type and Validation Constraint.
v2.0.0
2026-04-24 20:20 UTC
Requires
- php: >=8.1
- symfony/config: ^6.4 || ^7.0
- symfony/dependency-injection: ^6.4 || ^7.0
- symfony/form: ^6.4 || ^7.0
- symfony/http-client: ^6.4 || ^7.0
- symfony/http-foundation: ^6.4 || ^7.0
- symfony/http-kernel: ^6.4 || ^7.0
- symfony/twig-bundle: ^6.4 || ^7.0
- symfony/validator: ^6.4 || ^7.0
- twig/twig: ^3.0
Requires (Dev)
- phpunit/phpunit: ^10.5 || ^11.0
- symfony/framework-bundle: ^6.4 || ^7.0
- symfony/phpunit-bridge: ^6.4 || ^7.0
This package is not auto-updated.
Last update: 2026-05-08 20:53:06 UTC
README
A modern, zero-config Symfony bundle for hCaptcha: server-side verification, Twig widget, Form Type and Validation Constraint — ready in under a minute.
Features
- Zero-config install via Symfony Flex
- Strict-typed
HcaptchaVerifierreturning a richHcaptchaResponsevalue object (success,errorCodes,hostname,scorefor Enterprise, …) - Symfony Form Type (
HcaptchaType) with auto-attached validator - Symfony Validation Constraint (
#[Hcaptcha]) for any DTO/property - Twig functions:
hcaptcha_script(),hcaptcha_widget(),hcaptcha_invisible_widget() - HTTPS-only verify URL, attribute name allow-list, JS callback identifier check
- Test-mode toggle (
enabled: false) — no HTTP call, always succeeds - PSR-3 logging on transport failures
- 100% PHP 8.1+ / Symfony 6.4 LTS · 7.x compatible
Requirements
- PHP 8.1 or higher
- Symfony 6.4 LTS or 7.x
Quick install
composer require ethsam/easy-hcaptcha
Then in .env:
HCAPTCHA_SITE_KEY=your_public_site_key HCAPTCHA_SECRET_KEY=your_secret_key
And in config/packages/easy_hcaptcha.yaml:
easy_hcaptcha: site_key: '%env(HCAPTCHA_SITE_KEY)%' secret_key: '%env(HCAPTCHA_SECRET_KEY)%'
30-second usage
{# templates/contact.html.twig #} <head>{{ hcaptcha_script() }}</head> <form method="post"> {{ hcaptcha_widget() }} <button>Send</button> </form>
// In your controller — pick your favourite style: // 1) Form Type (recommended — auto-validated) $form = $this->createFormBuilder() ->add('email', EmailType::class) ->add('captcha', HcaptchaType::class) ->getForm(); // 2) Validation Constraint on a DTO class ContactDto { #[\Ethsam\EasyHcaptcha\Validator\Constraint\Hcaptcha] public ?string $captchaToken = null; } // 3) Manual call (full control) public function submit(Request $r, HcaptchaVerifierInterface $verifier): Response { $result = $verifier->verify($r->request->get('h-captcha-response'), $r->getClientIp()); if (!$result->isSuccess()) { return $this->redirectToRoute('contact'); } // ... }
Documentation
Full documentation in three languages:
Testing your app without hitting the API
In config/packages/test/easy_hcaptcha.yaml:
easy_hcaptcha: enabled: false # verifier returns success without any network call
Contributing
Pull requests welcome. Run the test suite locally:
composer install vendor/bin/phpunit
Changelog
See CHANGELOG.md for a detailed history of releases.
License
Released under the MIT License — see the file for details.
Originally authored by Samuel ETHEVE.