Skip to content

phpcfdi/image-captcha-resolver

Repository files navigation

phpcfdi/image-captcha-resolver

Source Code Packagist PHP Version Support Discord Latest Version Software License Build Status Reliability Maintainability Code Coverage Violations Total Downloads

Conectores para resolver captchas de imágenes

🇺🇸 The documentation of this project is in spanish as this is the natural language for the intended audience.

Acerca de phpcfdi/image-captcha-resolver

Esta librería contiene conectores con algunos servicios populares o de prueba para resolver captchas.

Es utilizado en algunos proyectos de PhpCfdi.

Instalación

Usa composer

composer require phpcfdi/image-captcha-resolver

Es posible que para su correcta implementación requiera también instalar algunos paquetes adicionales. Vea https://docs.php-http.org/en/latest/clients.html

# uso de guzzle, con el adaptador del cliente y su contructor de request y response
composer require guzzlehttp/guzzle php-http/guzzle7-adapter guzzlehttp/psr7

# uso de symfony http client con nyholm/psr7 como constructor de request y response
composer require symfony/http-client nyholm/psr7

# uso de cliente basado en curl con laminas/laminas-diactoros como constructor de request y response
composer require php-http/curl-client laminas/laminas-diactoros

Sin embargo, si lo que está desarrollando es una librería debería usar como dependencia de cliente el paquete php-http/mock-client y cualquier fábrica de mensajes (en composer.json:require-dev).

Uso básico

Llamar a resolver un captcha

Para este ejemplo se asume que ya existe un resolvedor de captchas en $resolver y que la imagen del captcha se encuentra como imagen embedida y su contenido en $theImgElementSrcAtributte.

<?php declare(strict_types=1);

use PhpCfdi\ImageCaptchaResolver\CaptchaImage;
use PhpCfdi\ImageCaptchaResolver\CaptchaResolverInterface;
use PhpCfdi\ImageCaptchaResolver\UnableToResolveCaptchaException;

/**
 * @var string $theImgElementSrcAtributte
 * @var CaptchaResolverInterface $resolver
 */

$image = CaptchaImage::newFromInlineHtml($theImgElementSrcAtributte);

try {
    $answer = $resolver->resolve($image);
} catch (UnableToResolveCaptchaException $exception) {
    echo 'No se pudo resolver el captcha', PHP_EOL;
    return;
}

echo "Respuesta del captcha: {$answer->getValue()}", PHP_EOL;

Creación de un resolvedor de captchas basado en Anti-Captcha

Para crear el resolvedor se puede hacer de dos formas: de manera específica y por descubrimiento.

Servicio: https://anti-captcha.com

<?php declare(strict_types=1);

use PhpCfdi\ImageCaptchaResolver\Resolvers\AntiCaptchaResolver;
use PhpCfdi\ImageCaptchaResolver\Timer\Timer;

/**
 * @var string $clientKey La clave de cliente para conectarse a anti-captcha 
 */

// Se puede crear el resolvedor usando los parámetros predefinidos
$resolverDefault = AntiCaptchaResolver::create($clientKey);

// O especificando parámetros funcionales
$resolverWithSettings = AntiCaptchaResolver::create(
    $clientKey,
    5, // segundos antes de intentar leer la respuesta
    60, // segundos antes de considerar que el captcha no tiene solución
    500 // milisegundos antes de reintentar obtener la respuesta
);

// O directamente creando los objetos
$resolverConstructed = new AntiCaptchaResolver(
    new AntiCaptchaResolver\AntiCaptchaConnector($clientKey),
    new Timer(4, 60, 500)
);

Creación de un resolvedor de captchas basado en Local Captcha Resolver

Para crear el resolvedor se puede hacer de dos formas: de manera específica y por descubrimiento.

Servicio: https://github.com/eclipxe13/captcha-local-resolver

<?php declare(strict_types=1);

use PhpCfdi\ImageCaptchaResolver\Resolvers\CaptchaLocalResolver;
use PhpCfdi\ImageCaptchaResolver\Timer\Timer;

/**
 * @var string $baseUrl La dirección donde está corriendo el servicio, como http://localhost:9095 
 */

// Se puede crear el resolvedor usando los parámetros predefinidos
$resolverDefault = CaptchaLocalResolver::create($baseUrl);

// O especificando parámetros funcionales
$resolverWithSettings = CaptchaLocalResolver::create(
    $baseUrl,
    5, // segundos antes de intentar leer la respuesta
    60, // segundos antes de considerar que el captcha no tiene solución
    500 // milisegundos antes de reintentar obtener la respuesta
);

// O directamente creando los objetos
$resolverConstructed = new CaptchaLocalResolver(
    new CaptchaLocalResolver\CaptchaLocalResolverConnector($baseUrl),
    new Timer(4, 60, 500)
);

Creación de un resolvedor de captchas basado en línea de comandos

La implementación dependerá siempre de la herramienta que se esté utilizando, es probable que fabrique su propio punto de entrada a la herramienta para que devuelva el exit code correcto y la respuesta.

Esta herramienta podría ser útil en caso de que el captcha se pueda resolver utilizando alguna herramienta como tesseract.

El siguiente ejemplo supone que tiene la imagen del captcha a resolver en $image y que existe un commando llamado my-captcha-breaker que se le entrega una imagen y devuelve en el último renglón de la salida la respuesta del captcha.

<?php declare(strict_types=1);

use PhpCfdi\ImageCaptchaResolver\CaptchaImageInterface;
use PhpCfdi\ImageCaptchaResolver\Resolvers\CommandLineResolver;
use PhpCfdi\ImageCaptchaResolver\UnableToResolveCaptchaException;

/**
 * @var CaptchaImageInterface $image 
 */

$resolver = CommandLineResolver::create(explode(' ', 'my-captcha-breaker --in {file} --stdout'));

try {
    $answer = $resolver->resolve($image);
} catch (UnableToResolveCaptchaException $exception) {
    echo 'No se pudo resolver el captcha: ', $exception->getMessage(), PHP_EOL;
    return;
}

echo $answer, PHP_EOL;

Resolvedores

Multiresolvedor

El resolvedor MultiResolver es en sí mismo un resolvedor que intenta resolver el captcha usando un conjunto predefinido de resolvedores. Podría ser útil para intentar resolver utilizando diferentes estrategias o reintentando con un mismo resolvedor el número de veces en las que esté incluído.

Resolvedores para pruebas

El resolvedor CaptchaLocalResolver usa servicio de resolución de captchas local y es comúnmente utilizado para pruebas.

También podría usar el resolvedor ConsoleResolver en donde se almacena en un archivo predefinido la imagen del captcha a resolver y se espera que escriba la ventana en la misma terminal donde está ejecutando el proceso. Solo es útil si puede escribir la respuesta. Si no se recibe la respuesta en un tiempo predeterminado el resolvedor fallará lo tomará como una respuesta vacía.

Si está haciendo pruebas unitarias, la mejor alternativa es usar el resolvedor MockResolver, que se construye con respuestas prestablecidas CaptchaAnswerInterface o excepciones UnableToResolveCaptchaException y falla con una excepción OutOfRangeException si se le pide una respuesta y ya no tiene más.

Nuevos resolvedores

Si ya tienes contratado un servicio de solución de captchas o deseas implementar uno, por ejemplo, basado en tesseract, debes implementar la interfaz ResolverInterface en tu proyecto.

Si el resolvedor puede beneficiar a toda la comunidad entonces haz tu solicitud de que se incluya en esta librería y con gusto lo evaluaremos, tomando en cuenta las dependencias y las pruebas.

Si el servicio requiere de registrar una nueva cuenta, por favor ponte en contacto con nosotros y podemos crear el resolvedor. Considera que será necesario que patrocines la suscripción para poder hacer pruebas de funcionamiento punto a punto.

Especificación de clases e interfaces

CaptchaImage

Contiene la imagen en base64 de la imagen, se puede construir desde un archivo, desde datos binarios, desde datos binarios codificados como base64 o desde un texto de imagen html embedido.

  • static CaptchaImage::newFromFile(string $filename): self
  • static CaptchaImage::newFromBinary(string $contents): self
  • static CaptchaImage::newFromBase64(string $contents): self
  • static CaptchaImage::newFromInlineHtml(string $contents): self

También contiene métodos para expresar la imagen u obtener el tipo MIME:

  • CaptchaImage::asBinary(): string
  • CaptchaImage::asBase64(): string
  • CaptchaImage::asBinary(): string
  • CaptchaImage::asInlineHtml(): string
  • CaptchaImage::getMimeType(): string

También se puede expresar como JSON o como string pues implementa Stringable y JsonSerializable y si lo desea, puede establecer su propia implementación usando CaptchaImageInterface.

CaptchaResolverInterface

Contiene un único método de resolución en donde toma un CaptchaImageInterface y entrega un CaptchaAnswer: CaptchaResolverInterface::resolve(CaptchaImageInterface $image): CaptchaAnswerInterface.

Hay diferentes implementaciones y se pueden agregar más a esta librería o en una librería independiente.

CaptchaAnswer

Contiene la respuesta del captcha como un valor de texto, no puede ser una cadena vacía.

La respuesta se obtiene con el método CaptchaAnswer::getValue(): string.

Y se puede comparar contra cualquier otro valor usando CaptchaAnswer::equalsTo($value): bool.

También se puede expresar como JSON o como string pues implementa Stringable y JsonSerializable y si lo desea, puede establecer su propia implementación usando CaptchaAnswerInterface.

HttpClientInterface y HttpClient

Este método es un adaptador para facilitar las comunicaciones de HTTP con el exterior.

La implementación actual contiene muy pocos métodos y serán agregados nuevos conforme se necesite.

Utiliza los estándares PSR-18 HTTP Client, y PSR-17 HTTP Factories que usan el PSR-7 HTTP message interfaces. Requieren de un cliente http que implemente el PSR-18 y de una librería que implemente PSR-17 para construir los mensajes de tipo Request o Response.

Para que sea fácil poder crear el objeto, se usa el paquete de HTTPlug Discovery php-http/discovery que permite encontrar implementaciones instaladas y utilizarlas.

El método HttpClient::discovery() es el que se utiliza de forma predeterminada para construir el objeto, aunque también se puede crear utilizando el constructor y entregando los objetos necesarios.

Soporte

Puedes obtener soporte abriendo un ticket en Github.

Adicionalmente, esta librería pertenece a la comunidad PhpCfdi, así que puedes usar los mismos canales de comunicación para obtener ayuda de algún miembro de la comunidad.

Compatibilidad

Esta librería se mantendrá compatible con al menos la versión con soporte activo de PHP más reciente.

También utilizamos Versionado Semántico 2.0.0 por lo que puedes usar esta librería sin temor a romper tu aplicación.

Contribuciones

Las contribuciones con bienvenidas. Por favor lee CONTRIBUTING para más detalles y recuerda revisar el archivo de tareas pendientes TODO y el archivo CHANGELOG.

Copyright and License

The phpcfdi/image-captcha-resolver library is copyright © PhpCfdi and licensed for use under the MIT License (MIT). Please see LICENSE for more information.