mirror of
https://github.com/vichan-devel/vichan.git
synced 2024-12-02 19:17:21 +01:00
Merge pull request #832 from Zankaria/captcha-breakup
Breakup the captcha file
This commit is contained in:
commit
fed2170923
@ -39,7 +39,6 @@
|
|||||||
"inc/functions/net.php",
|
"inc/functions/net.php",
|
||||||
"inc/functions/num.php",
|
"inc/functions/num.php",
|
||||||
"inc/functions/theme.php",
|
"inc/functions/theme.php",
|
||||||
"inc/service/captcha-queries.php",
|
|
||||||
"inc/context.php"
|
"inc/context.php"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
47
inc/Service/HCaptchaQuery.php
Normal file
47
inc/Service/HCaptchaQuery.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
namespace Vichan\Service;
|
||||||
|
|
||||||
|
use Vichan\Data\Driver\HttpDriver;
|
||||||
|
|
||||||
|
defined('TINYBOARD') or exit;
|
||||||
|
|
||||||
|
|
||||||
|
class HCaptchaQuery implements RemoteCaptchaQuery {
|
||||||
|
private HttpDriver $http;
|
||||||
|
private string $secret;
|
||||||
|
private string $sitekey;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new HCaptchaQuery using the hCaptcha service.
|
||||||
|
*
|
||||||
|
* @param HttpDriver $http The http client.
|
||||||
|
* @param string $secret Server side secret.
|
||||||
|
* @return HCaptchaQuery A new hCaptcha query instance.
|
||||||
|
*/
|
||||||
|
public function __construct(HttpDriver $http, string $secret, string $sitekey) {
|
||||||
|
$this->http = $http;
|
||||||
|
$this->secret = $secret;
|
||||||
|
$this->sitekey = $sitekey;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function responseField(): string {
|
||||||
|
return 'h-captcha-response';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function verify(string $response, ?string $remote_ip): bool {
|
||||||
|
$data = [
|
||||||
|
'secret' => $this->secret,
|
||||||
|
'response' => $response,
|
||||||
|
'sitekey' => $this->sitekey
|
||||||
|
];
|
||||||
|
|
||||||
|
if ($remote_ip !== null) {
|
||||||
|
$data['remoteip'] = $remote_ip;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ret = $this->http->requestGet('https://hcaptcha.com/siteverify', $data);
|
||||||
|
$resp = \json_decode($ret, true, 16, \JSON_THROW_ON_ERROR);
|
||||||
|
|
||||||
|
return isset($resp['success']) && $resp['success'];
|
||||||
|
}
|
||||||
|
}
|
44
inc/Service/ReCaptchaQuery.php
Normal file
44
inc/Service/ReCaptchaQuery.php
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
<?php
|
||||||
|
namespace Vichan\Service;
|
||||||
|
|
||||||
|
use Vichan\Data\Driver\HttpDriver;
|
||||||
|
|
||||||
|
defined('TINYBOARD') or exit;
|
||||||
|
|
||||||
|
|
||||||
|
class ReCaptchaQuery implements RemoteCaptchaQuery {
|
||||||
|
private HttpDriver $http;
|
||||||
|
private string $secret;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new ReCaptchaQuery using the google recaptcha service.
|
||||||
|
*
|
||||||
|
* @param HttpDriver $http The http client.
|
||||||
|
* @param string $secret Server side secret.
|
||||||
|
* @return ReCaptchaQuery A new ReCaptchaQuery query instance.
|
||||||
|
*/
|
||||||
|
public function __construct(HttpDriver $http, string $secret) {
|
||||||
|
$this->http = $http;
|
||||||
|
$this->secret = $secret;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function responseField(): string {
|
||||||
|
return 'g-recaptcha-response';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function verify(string $response, ?string $remote_ip): bool {
|
||||||
|
$data = [
|
||||||
|
'secret' => $this->secret,
|
||||||
|
'response' => $response
|
||||||
|
];
|
||||||
|
|
||||||
|
if ($remote_ip !== null) {
|
||||||
|
$data['remoteip'] = $remote_ip;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ret = $this->http->requestGet('https://www.google.com/recaptcha/api/siteverify', $data);
|
||||||
|
$resp = \json_decode($ret, true, 16, \JSON_THROW_ON_ERROR);
|
||||||
|
|
||||||
|
return isset($resp['success']) && $resp['success'];
|
||||||
|
}
|
||||||
|
}
|
24
inc/Service/RemoteCaptchaQuery.php
Normal file
24
inc/Service/RemoteCaptchaQuery.php
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
namespace Vichan\Service;
|
||||||
|
|
||||||
|
defined('TINYBOARD') or exit;
|
||||||
|
|
||||||
|
|
||||||
|
interface RemoteCaptchaQuery {
|
||||||
|
/**
|
||||||
|
* Name of the response field in the form data expected by the implementation.
|
||||||
|
*
|
||||||
|
* @return string The name of the field.
|
||||||
|
*/
|
||||||
|
public function responseField(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the user at the remote ip passed the captcha.
|
||||||
|
*
|
||||||
|
* @param string $response User provided response.
|
||||||
|
* @param ?string $remote_ip User ip. Leave to null to only check the response value.
|
||||||
|
* @return bool Returns true if the user passed the captcha.
|
||||||
|
* @throws RuntimeException|JsonException Throws on IO errors or if it fails to decode the answer.
|
||||||
|
*/
|
||||||
|
public function verify(string $response, ?string $remote_ip): bool;
|
||||||
|
}
|
47
inc/Service/SecureimageCaptchaQuery.php
Normal file
47
inc/Service/SecureimageCaptchaQuery.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
namespace Vichan\Service;
|
||||||
|
|
||||||
|
use Vichan\Data\Driver\HttpDriver;
|
||||||
|
|
||||||
|
defined('TINYBOARD') or exit;
|
||||||
|
|
||||||
|
|
||||||
|
class SecureImageCaptchaQuery {
|
||||||
|
private HttpDriver $http;
|
||||||
|
private string $domain;
|
||||||
|
private string $provider_check;
|
||||||
|
private string $extra;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param HttpDriver $http The http client.
|
||||||
|
* @param string $domain The server's domain.
|
||||||
|
* @param string $provider_check Path to the endpoint.
|
||||||
|
* @param string $extra Extra http parameters.
|
||||||
|
*/
|
||||||
|
function __construct(HttpDriver $http, string $domain, string $provider_check, string $extra) {
|
||||||
|
$this->http = $http;
|
||||||
|
$this->domain = $domain;
|
||||||
|
$this->provider_check = $provider_check;
|
||||||
|
$this->extra = $extra;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the user at the remote ip passed the native vichan captcha.
|
||||||
|
*
|
||||||
|
* @param string $user_text Remote user's text input.
|
||||||
|
* @param string $user_cookie Remote user cookie.
|
||||||
|
* @return bool Returns true if the user passed the check.
|
||||||
|
* @throws RuntimeException Throws on IO errors.
|
||||||
|
*/
|
||||||
|
public function verify(string $user_text, string $user_cookie): bool {
|
||||||
|
$data = [
|
||||||
|
'mode' => 'check',
|
||||||
|
'text' => $user_text,
|
||||||
|
'extra' => $this->extra,
|
||||||
|
'cookie' => $user_cookie
|
||||||
|
];
|
||||||
|
|
||||||
|
$ret = $this->http->requestGet($this->domain . '/' . $this->provider_check, $data);
|
||||||
|
return $ret === '1';
|
||||||
|
}
|
||||||
|
}
|
@ -3,7 +3,7 @@ namespace Vichan;
|
|||||||
|
|
||||||
use Vichan\Data\Driver\{CacheDriver, HttpDriver, ErrorLogLogDriver, FileLogDriver, LogDriver, StderrLogDriver, SyslogLogDriver};
|
use Vichan\Data\Driver\{CacheDriver, HttpDriver, ErrorLogLogDriver, FileLogDriver, LogDriver, StderrLogDriver, SyslogLogDriver};
|
||||||
use Vichan\Service\HCaptchaQuery;
|
use Vichan\Service\HCaptchaQuery;
|
||||||
use Vichan\Service\NativeCaptchaQuery;
|
use Vichan\Service\SecureImageCaptchaQuery;
|
||||||
use Vichan\Service\ReCaptchaQuery;
|
use Vichan\Service\ReCaptchaQuery;
|
||||||
use Vichan\Service\RemoteCaptchaQuery;
|
use Vichan\Service\RemoteCaptchaQuery;
|
||||||
|
|
||||||
@ -71,12 +71,12 @@ function build_context(array $config): Context {
|
|||||||
throw new \RuntimeException('No remote captcha service available');
|
throw new \RuntimeException('No remote captcha service available');
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
NativeCaptchaQuery::class => function($c) {
|
SecureImageCaptchaQuery::class => function($c) {
|
||||||
$config = $c->get('config');
|
$config = $c->get('config');
|
||||||
if ($config['captcha']['provider'] !== 'native') {
|
if ($config['captcha']['provider'] !== 'native') {
|
||||||
throw new \RuntimeException('No native captcha service available');
|
throw new \RuntimeException('No native captcha service available');
|
||||||
}
|
}
|
||||||
return new NativeCaptchaQuery(
|
return new SecureImageCaptchaQuery(
|
||||||
$c->get(HttpDriver::class),
|
$c->get(HttpDriver::class),
|
||||||
$config['domain'],
|
$config['domain'],
|
||||||
$config['captcha']['native']['provider_check'],
|
$config['captcha']['native']['provider_check'],
|
||||||
|
@ -1,143 +0,0 @@
|
|||||||
<?php // Verify captchas server side.
|
|
||||||
namespace Vichan\Service;
|
|
||||||
|
|
||||||
use Vichan\Data\Driver\HttpDriver;
|
|
||||||
|
|
||||||
defined('TINYBOARD') or exit;
|
|
||||||
|
|
||||||
|
|
||||||
class ReCaptchaQuery implements RemoteCaptchaQuery {
|
|
||||||
private HttpDriver $http;
|
|
||||||
private string $secret;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a new ReCaptchaQuery using the google recaptcha service.
|
|
||||||
*
|
|
||||||
* @param HttpDriver $http The http client.
|
|
||||||
* @param string $secret Server side secret.
|
|
||||||
* @return ReCaptchaQuery A new ReCaptchaQuery query instance.
|
|
||||||
*/
|
|
||||||
public function __construct(HttpDriver $http, string $secret) {
|
|
||||||
$this->http = $http;
|
|
||||||
$this->secret = $secret;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function responseField(): string {
|
|
||||||
return 'g-recaptcha-response';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function verify(string $response, ?string $remote_ip): bool {
|
|
||||||
$data = [
|
|
||||||
'secret' => $this->secret,
|
|
||||||
'response' => $response
|
|
||||||
];
|
|
||||||
|
|
||||||
if ($remote_ip !== null) {
|
|
||||||
$data['remoteip'] = $remote_ip;
|
|
||||||
}
|
|
||||||
|
|
||||||
$ret = $this->http->requestGet('https://www.google.com/recaptcha/api/siteverify', $data);
|
|
||||||
$resp = json_decode($ret, true, 16, JSON_THROW_ON_ERROR);
|
|
||||||
|
|
||||||
return isset($resp['success']) && $resp['success'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class HCaptchaQuery implements RemoteCaptchaQuery {
|
|
||||||
private HttpDriver $http;
|
|
||||||
private string $secret;
|
|
||||||
private string $sitekey;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a new HCaptchaQuery using the hCaptcha service.
|
|
||||||
*
|
|
||||||
* @param HttpDriver $http The http client.
|
|
||||||
* @param string $secret Server side secret.
|
|
||||||
* @return HCaptchaQuery A new hCaptcha query instance.
|
|
||||||
*/
|
|
||||||
public function __construct(HttpDriver $http, string $secret, string $sitekey) {
|
|
||||||
$this->http = $http;
|
|
||||||
$this->secret = $secret;
|
|
||||||
$this->sitekey = $sitekey;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function responseField(): string {
|
|
||||||
return 'h-captcha-response';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function verify(string $response, ?string $remote_ip): bool {
|
|
||||||
$data = [
|
|
||||||
'secret' => $this->secret,
|
|
||||||
'response' => $response,
|
|
||||||
'sitekey' => $this->sitekey
|
|
||||||
];
|
|
||||||
|
|
||||||
if ($remote_ip !== null) {
|
|
||||||
$data['remoteip'] = $remote_ip;
|
|
||||||
}
|
|
||||||
|
|
||||||
$ret = $this->http->requestGet('https://hcaptcha.com/siteverify', $data);
|
|
||||||
$resp = json_decode($ret, true, 16, JSON_THROW_ON_ERROR);
|
|
||||||
|
|
||||||
return isset($resp['success']) && $resp['success'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
interface RemoteCaptchaQuery {
|
|
||||||
/**
|
|
||||||
* Name of the response field in the form data expected by the implementation.
|
|
||||||
*
|
|
||||||
* @return string The name of the field.
|
|
||||||
*/
|
|
||||||
public function responseField(): string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the user at the remote ip passed the captcha.
|
|
||||||
*
|
|
||||||
* @param string $response User provided response.
|
|
||||||
* @param ?string $remote_ip User ip. Leave to null to only check the response value.
|
|
||||||
* @return bool Returns true if the user passed the captcha.
|
|
||||||
* @throws RuntimeException|JsonException Throws on IO errors or if it fails to decode the answer.
|
|
||||||
*/
|
|
||||||
public function verify(string $response, ?string $remote_ip): bool;
|
|
||||||
}
|
|
||||||
|
|
||||||
class NativeCaptchaQuery {
|
|
||||||
private HttpDriver $http;
|
|
||||||
private string $domain;
|
|
||||||
private string $provider_check;
|
|
||||||
private string $extra;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param HttpDriver $http The http client.
|
|
||||||
* @param string $domain The server's domain.
|
|
||||||
* @param string $provider_check Path to the endpoint.
|
|
||||||
* @param string $extra Extra http parameters.
|
|
||||||
*/
|
|
||||||
function __construct(HttpDriver $http, string $domain, string $provider_check, string $extra) {
|
|
||||||
$this->http = $http;
|
|
||||||
$this->domain = $domain;
|
|
||||||
$this->provider_check = $provider_check;
|
|
||||||
$this->extra = $extra;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the user at the remote ip passed the native vichan captcha.
|
|
||||||
*
|
|
||||||
* @param string $user_text Remote user's text input.
|
|
||||||
* @param string $user_cookie Remote user cookie.
|
|
||||||
* @return bool Returns true if the user passed the check.
|
|
||||||
* @throws RuntimeException Throws on IO errors.
|
|
||||||
*/
|
|
||||||
public function verify(string $user_text, string $user_cookie): bool {
|
|
||||||
$data = [
|
|
||||||
'mode' => 'check',
|
|
||||||
'text' => $user_text,
|
|
||||||
'extra' => $this->extra,
|
|
||||||
'cookie' => $user_cookie
|
|
||||||
];
|
|
||||||
|
|
||||||
$ret = $this->http->requestGet($this->domain . '/' . $this->provider_check, $data);
|
|
||||||
return $ret === '1';
|
|
||||||
}
|
|
||||||
}
|
|
6
post.php
6
post.php
@ -7,7 +7,7 @@ require_once 'inc/bootstrap.php';
|
|||||||
|
|
||||||
use Vichan\{Context, WebDependencyFactory};
|
use Vichan\{Context, WebDependencyFactory};
|
||||||
use Vichan\Data\Driver\{LogDriver, HttpDriver};
|
use Vichan\Data\Driver\{LogDriver, HttpDriver};
|
||||||
use Vichan\Service\{RemoteCaptchaQuery, NativeCaptchaQuery};
|
use Vichan\Service\{RemoteCaptchaQuery, SecureImageCaptchaQuery};
|
||||||
use Vichan\Functions\Format;
|
use Vichan\Functions\Format;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -516,7 +516,7 @@ if (isset($_POST['delete'])) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$query = new NativeCaptchaQuery(
|
$query = new SecureImageCaptchaQuery(
|
||||||
$context->get(HttpDriver::class),
|
$context->get(HttpDriver::class),
|
||||||
$config['domain'],
|
$config['domain'],
|
||||||
$config['captcha']['provider_check'],
|
$config['captcha']['provider_check'],
|
||||||
@ -636,7 +636,7 @@ if (isset($_POST['delete'])) {
|
|||||||
// With our custom captcha provider
|
// With our custom captcha provider
|
||||||
if (($provider === 'native' && !$new_thread_capt)
|
if (($provider === 'native' && !$new_thread_capt)
|
||||||
|| ($provider === 'native' && $new_thread_capt && $post['op'])) {
|
|| ($provider === 'native' && $new_thread_capt && $post['op'])) {
|
||||||
$query = $context->get(NativeCaptchaQuery::class);
|
$query = $context->get(SecureImageCaptchaQuery::class);
|
||||||
$success = $query->verify($_POST['captcha_text'], $_POST['captcha_cookie']);
|
$success = $query->verify($_POST['captcha_text'], $_POST['captcha_cookie']);
|
||||||
|
|
||||||
if (!$success) {
|
if (!$success) {
|
||||||
|
Loading…
Reference in New Issue
Block a user