. */ declare(strict_types=1); namespace FireflyIII\Services\Password; use Exception; use GuzzleHttp\Client; use GuzzleHttp\Exception\GuzzleException; use Log; use RuntimeException; /** * Class PwndVerifierV2. * @codeCoverageIgnore */ class PwndVerifierV2 implements Verifier { /** * Constructor. */ public function __construct() { if ('testing' === config('app.env')) { Log::warning(sprintf('%s should not be instantiated in the TEST environment!', get_class($this))); } } /** * Verify the given password against (some) service. * * @param string $password * * @return bool */ public function validPassword(string $password): bool { $hash = sha1($password); $prefix = substr($hash, 0, 5); $rest = substr($hash, 5); $uri = sprintf('https://api.pwnedpasswords.com/range/%s', $prefix); $opt = [ 'headers' => ['User-Agent' => 'Firefly III v' . config('firefly.version')], 'timeout' => 5]; Log::debug(sprintf('hash prefix is %s', $prefix)); Log::debug(sprintf('rest is %s', $rest)); try { $client = new Client(); $res = $client->request('GET', $uri, $opt); } catch (GuzzleException|Exception $e) { Log::error(sprintf('Could not verify password security: %s', $e->getMessage())); return true; } Log::debug(sprintf('Status code returned is %d', $res->getStatusCode())); if (404 === $res->getStatusCode()) { return true; } try { $strpos = stripos($res->getBody()->getContents(), $rest); } catch (RuntimeException $e) { Log::error(sprintf('Could not get body from Pwnd result: %s', $e->getMessage())); $strpos = false; } if (false === $strpos) { Log::debug(sprintf('%s was not found in result body. Return true.', $rest)); return true; } Log::debug(sprintf('Could not find %s, return FALSE.', $rest)); return false; } }