forked from GNUsocial/gnu-social
		
	
		
			
				
	
	
		
			148 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			148 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * XML Formatted RSA Key Handler
 | |
|  *
 | |
|  * More info:
 | |
|  *
 | |
|  * http://www.w3.org/TR/xmldsig-core/#sec-RSAKeyValue
 | |
|  * http://en.wikipedia.org/wiki/XML_Signature
 | |
|  *
 | |
|  * PHP version 5
 | |
|  *
 | |
|  * @category  Crypt
 | |
|  * @package   RSA
 | |
|  * @author    Jim Wigginton <terrafrost@php.net>
 | |
|  * @copyright 2015 Jim Wigginton
 | |
|  * @license   http://www.opensource.org/licenses/mit-license.html  MIT License
 | |
|  * @link      http://phpseclib.sourceforge.net
 | |
|  */
 | |
| 
 | |
| namespace phpseclib\Crypt\RSA;
 | |
| 
 | |
| use ParagonIE\ConstantTime\Base64;
 | |
| use phpseclib\Math\BigInteger;
 | |
| 
 | |
| /**
 | |
|  * XML Formatted RSA Key Handler
 | |
|  *
 | |
|  * @package RSA
 | |
|  * @author  Jim Wigginton <terrafrost@php.net>
 | |
|  * @access  public
 | |
|  */
 | |
| class XML
 | |
| {
 | |
|     /**
 | |
|      * Break a public or private key down into its constituent components
 | |
|      *
 | |
|      * @access public
 | |
|      * @param string $key
 | |
|      * @param string $password optional
 | |
|      * @return array
 | |
|      */
 | |
|     static function load($key, $password = '')
 | |
|     {
 | |
|         if (!is_string($key)) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         $components = array(
 | |
|             'isPublicKey' => false,
 | |
|             'primes' => array(),
 | |
|             'exponents' => array(),
 | |
|             'coefficients' => array()
 | |
|         );
 | |
| 
 | |
|         $use_errors = libxml_use_internal_errors(true);
 | |
| 
 | |
|         $dom = new \DOMDocument();
 | |
|         if (!$dom->loadXML('<xml>' . $key . '</xml>')) {
 | |
|             return false;
 | |
|         }
 | |
|         $xpath = new \DOMXPath($dom);
 | |
|         $keys = array('modulus', 'exponent', 'p', 'q', 'dp', 'dq', 'inverseq', 'd');
 | |
|         foreach ($keys as $key) {
 | |
|             // $dom->getElementsByTagName($key) is case-sensitive
 | |
|             $temp = $xpath->query("//*[translate(local-name(), 'ABCDEFGHIJKLMNOPQRSTUVWXYZ','abcdefghijklmnopqrstuvwxyz')='$key']");
 | |
|             if (!$temp->length) {
 | |
|                 continue;
 | |
|             }
 | |
|             $value = new BigInteger(Base64::decode($temp->item(0)->nodeValue), 256);
 | |
|             switch ($key) {
 | |
|                 case 'modulus':
 | |
|                     $components['modulus'] = $value;
 | |
|                     break;
 | |
|                 case 'exponent':
 | |
|                     $components['publicExponent'] = $value;
 | |
|                     break;
 | |
|                 case 'p':
 | |
|                     $components['primes'][1] = $value;
 | |
|                     break;
 | |
|                 case 'q':
 | |
|                     $components['primes'][2] = $value;
 | |
|                     break;
 | |
|                 case 'dp':
 | |
|                     $components['exponents'][1] = $value;
 | |
|                     break;
 | |
|                 case 'dq':
 | |
|                     $components['exponents'][2] = $value;
 | |
|                     break;
 | |
|                 case 'inverseq':
 | |
|                     $components['coefficients'][2] = $value;
 | |
|                     break;
 | |
|                 case 'd':
 | |
|                     $components['privateExponent'] = $value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         libxml_use_internal_errors($use_errors);
 | |
| 
 | |
|         return isset($components['modulus']) && isset($components['publicExponent']) ? $components : false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Convert a private key to the appropriate format.
 | |
|      *
 | |
|      * @access public
 | |
|      * @param \phpseclib\Math\BigInteger $n
 | |
|      * @param \phpseclib\Math\BigInteger $e
 | |
|      * @param \phpseclib\Math\BigInteger $d
 | |
|      * @param array $primes
 | |
|      * @param array $exponents
 | |
|      * @param array $coefficients
 | |
|      * @param string $password optional
 | |
|      * @return string
 | |
|      */
 | |
|     static function savePrivateKey(BigInteger $n, BigInteger $e, BigInteger $d, $primes, $exponents, $coefficients, $password = '')
 | |
|     {
 | |
|         if (count($primes) != 2) {
 | |
|             return false;
 | |
|         }
 | |
|         return "<RSAKeyValue>\r\n" .
 | |
|                '  <Modulus>' . Base64::encode($n->toBytes()) . "</Modulus>\r\n" .
 | |
|                '  <Exponent>' . Base64::encode($e->toBytes()) . "</Exponent>\r\n" .
 | |
|                '  <P>' . Base64::encode($primes[1]->toBytes()) . "</P>\r\n" .
 | |
|                '  <Q>' . Base64::encode($primes[2]->toBytes()) . "</Q>\r\n" .
 | |
|                '  <DP>' . Base64::encode($exponents[1]->toBytes()) . "</DP>\r\n" .
 | |
|                '  <DQ>' . Base64::encode($exponents[2]->toBytes()) . "</DQ>\r\n" .
 | |
|                '  <InverseQ>' . Base64::encode($coefficients[2]->toBytes()) . "</InverseQ>\r\n" .
 | |
|                '  <D>' . Base64::encode($d->toBytes()) . "</D>\r\n" .
 | |
|                '</RSAKeyValue>';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Convert a public key to the appropriate format
 | |
|      *
 | |
|      * @access public
 | |
|      * @param \phpseclib\Math\BigInteger $n
 | |
|      * @param \phpseclib\Math\BigInteger $e
 | |
|      * @return string
 | |
|      */
 | |
|     static function savePublicKey(BigInteger $n, BigInteger $e)
 | |
|     {
 | |
|         return "<RSAKeyValue>\r\n" .
 | |
|                '  <Modulus>' . Base64::encode($n->toBytes()) . "</Modulus>\r\n" .
 | |
|                '  <Exponent>' . Base64::encode($e->toBytes()) . "</Exponent>\r\n" .
 | |
|                '</RSAKeyValue>';
 | |
|     }
 | |
| }
 |