2016-03-28 16:42:34 +02:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Cryptography key holder.
|
|
|
|
*
|
|
|
|
* @author Julián Gutiérrez <juliangut@gmail.com>
|
|
|
|
* @copyright Copyright (c) Alex Bilbie
|
|
|
|
* @license http://mit-license.org/
|
|
|
|
*
|
|
|
|
* @link https://github.com/thephpleague/oauth2-server
|
|
|
|
*/
|
2016-07-09 01:00:44 +02:00
|
|
|
|
2016-03-28 16:42:34 +02:00
|
|
|
namespace League\OAuth2\Server;
|
|
|
|
|
2018-11-13 01:00:23 +01:00
|
|
|
use LogicException;
|
|
|
|
use RuntimeException;
|
|
|
|
|
2016-03-28 16:42:34 +02:00
|
|
|
class CryptKey
|
|
|
|
{
|
2016-07-19 15:01:31 +02:00
|
|
|
const RSA_KEY_PATTERN =
|
2018-01-04 12:17:31 +01:00
|
|
|
'/^(-----BEGIN (RSA )?(PUBLIC|PRIVATE) KEY-----)\R.*(-----END (RSA )?(PUBLIC|PRIVATE) KEY-----)\R?$/s';
|
2016-07-19 15:01:31 +02:00
|
|
|
|
2016-03-28 16:42:34 +02:00
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $keyPath;
|
|
|
|
|
|
|
|
/**
|
2016-07-09 01:00:44 +02:00
|
|
|
* @var null|string
|
2016-03-28 16:42:34 +02:00
|
|
|
*/
|
|
|
|
protected $passPhrase;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $keyPath
|
|
|
|
* @param null|string $passPhrase
|
2017-08-01 14:59:21 +02:00
|
|
|
* @param bool $keyPermissionsCheck
|
2016-03-28 16:42:34 +02:00
|
|
|
*/
|
2017-08-01 14:59:21 +02:00
|
|
|
public function __construct($keyPath, $passPhrase = null, $keyPermissionsCheck = true)
|
2016-03-28 16:42:34 +02:00
|
|
|
{
|
2016-07-19 15:01:31 +02:00
|
|
|
if (preg_match(self::RSA_KEY_PATTERN, $keyPath)) {
|
|
|
|
$keyPath = $this->saveKeyToFile($keyPath);
|
|
|
|
}
|
|
|
|
|
2016-03-28 16:42:34 +02:00
|
|
|
if (strpos($keyPath, 'file://') !== 0) {
|
|
|
|
$keyPath = 'file://' . $keyPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!file_exists($keyPath) || !is_readable($keyPath)) {
|
2018-11-13 01:00:23 +01:00
|
|
|
throw new LogicException(sprintf('Key path "%s" does not exist or is not readable', $keyPath));
|
2016-03-28 16:42:34 +02:00
|
|
|
}
|
|
|
|
|
2017-08-01 14:59:21 +02:00
|
|
|
if ($keyPermissionsCheck === true) {
|
|
|
|
// Verify the permissions of the key
|
|
|
|
$keyPathPerms = decoct(fileperms($keyPath) & 0777);
|
2018-06-21 17:02:01 +02:00
|
|
|
if (in_array($keyPathPerms, ['400', '440', '600', '640', '660'], true) === false) {
|
2017-08-01 14:59:21 +02:00
|
|
|
trigger_error(sprintf(
|
2018-02-11 22:12:55 +00:00
|
|
|
'Key file "%s" permissions are not correct, recommend changing to 600 or 660 instead of %s',
|
2017-08-01 14:59:21 +02:00
|
|
|
$keyPath,
|
|
|
|
$keyPathPerms
|
|
|
|
), E_USER_NOTICE);
|
2017-06-16 16:58:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-28 16:42:34 +02:00
|
|
|
$this->keyPath = $keyPath;
|
|
|
|
$this->passPhrase = $passPhrase;
|
|
|
|
}
|
|
|
|
|
2016-07-19 15:01:31 +02:00
|
|
|
/**
|
|
|
|
* @param string $key
|
|
|
|
*
|
2018-11-13 01:00:23 +01:00
|
|
|
* @throws RuntimeException
|
2016-07-19 17:15:36 +02:00
|
|
|
*
|
2016-07-19 15:01:31 +02:00
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function saveKeyToFile($key)
|
|
|
|
{
|
2017-06-16 16:59:29 +01:00
|
|
|
$tmpDir = sys_get_temp_dir();
|
|
|
|
$keyPath = $tmpDir . '/' . sha1($key) . '.key';
|
2016-07-19 15:01:31 +02:00
|
|
|
|
2018-01-29 11:05:10 +01:00
|
|
|
if (file_exists($keyPath)) {
|
|
|
|
return 'file://' . $keyPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!touch($keyPath)) {
|
2016-07-19 17:15:36 +02:00
|
|
|
// @codeCoverageIgnoreStart
|
2018-11-13 01:00:23 +01:00
|
|
|
throw new RuntimeException(sprintf('"%s" key file could not be created', $keyPath));
|
2016-07-19 17:15:36 +02:00
|
|
|
// @codeCoverageIgnoreEnd
|
2016-07-19 15:01:31 +02:00
|
|
|
}
|
|
|
|
|
2017-06-16 16:59:29 +01:00
|
|
|
if (file_put_contents($keyPath, $key) === false) {
|
|
|
|
// @codeCoverageIgnoreStart
|
2018-11-13 01:00:23 +01:00
|
|
|
throw new RuntimeException(sprintf('Unable to write key file to temporary directory "%s"', $tmpDir));
|
2017-06-16 16:59:29 +01:00
|
|
|
// @codeCoverageIgnoreEnd
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chmod($keyPath, 0600) === false) {
|
|
|
|
// @codeCoverageIgnoreStart
|
2018-11-13 01:00:23 +01:00
|
|
|
throw new RuntimeException(sprintf('The key file "%s" file mode could not be changed with chmod to 600', $keyPath));
|
2017-06-16 16:59:29 +01:00
|
|
|
// @codeCoverageIgnoreEnd
|
|
|
|
}
|
2016-07-19 15:01:31 +02:00
|
|
|
|
|
|
|
return 'file://' . $keyPath;
|
|
|
|
}
|
|
|
|
|
2016-03-28 16:42:34 +02:00
|
|
|
/**
|
|
|
|
* Retrieve key path.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getKeyPath()
|
|
|
|
{
|
|
|
|
return $this->keyPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve key pass phrase.
|
|
|
|
*
|
|
|
|
* @return null|string
|
|
|
|
*/
|
|
|
|
public function getPassPhrase()
|
|
|
|
{
|
|
|
|
return $this->passPhrase;
|
|
|
|
}
|
|
|
|
}
|