openpgp-php/lib/openpgp_crypt_aes_tripledes.php

105 lines
3.7 KiB
PHP
Raw Normal View History

2013-01-21 23:18:13 +00:00
<?php
require_once dirname(__FILE__).'/openpgp.php';
require_once 'Crypt/AES.php';
2013-01-26 16:08:18 +00:00
require_once 'Crypt/TripleDES.php';
2013-01-21 23:18:13 +00:00
2013-01-26 16:08:18 +00:00
class OpenPGP_Crypt_AES_TripleDES {
2013-01-21 23:18:13 +00:00
public static function decryptSymmetric($pass, $m) {
2013-01-26 19:00:00 +00:00
$epacket = self::getEncryptedData($m);
2013-01-21 23:18:13 +00:00
foreach($m as $p) {
if($p instanceof OpenPGP_SymmetricSessionKeyPacket) {
2013-01-26 16:26:55 +00:00
if(strlen($p->encrypted_data) > 0) {
2013-01-26 19:00:00 +00:00
list($cipher, $key_bytes, $key_block_bytes) = self::getCipher($p->symmetric_algorithm);
if(!$cipher) continue;
$cipher->setKey($p->s2k->make_key($pass, $key_bytes));
2013-01-26 16:26:55 +00:00
$padAmount = $key_block_bytes - (strlen($p->encrypted_data) % $key_block_bytes);
$data = substr($cipher->decrypt($p->encrypted_data . str_repeat("\0", $padAmount)), 0, strlen($p->encrypted_data));
2013-01-26 19:00:00 +00:00
$decrypted = self::decryptPacket($epacket, ord($data{0}), substr($data, 1));
} else {
list($cipher, $key_bytes, $key_block_bytes) = self::getCipher($p->symmetric_algorithm);
$decrypted = self::decryptPacket($epacket, $p->symmetric_algorithm, $p->s2k->make_key($pass, $key_bytes));
2013-01-21 23:18:13 +00:00
}
2013-01-26 19:00:00 +00:00
if($decrypted) return $decrypted;
}
}
2013-01-21 23:18:13 +00:00
2013-01-26 19:00:00 +00:00
return NULL; /* If we get here, we failed */
}
2013-01-21 23:18:13 +00:00
2013-01-26 19:00:00 +00:00
public static function decryptPacket($epacket, $symmetric_algorithm, $key) {
list($cipher, $key_bytes, $key_block_bytes) = self::getCipher($symmetric_algorithm);
if(!$cipher) return NULL;
$cipher->setKey($key);
2013-01-21 23:18:13 +00:00
2013-01-26 19:00:00 +00:00
$padAmount = $key_block_bytes - (strlen($epacket->data) % $key_block_bytes);
2013-01-26 16:17:11 +00:00
2013-01-26 19:00:00 +00:00
if($epacket instanceof OpenPGP_IntegrityProtectedDataPacket) {
$data = substr($cipher->decrypt($epacket->data . str_repeat("\0", $padAmount)), 0, strlen($epacket->data));
$prefix = substr($data, 0, $key_block_bytes + 2);
$mdc = substr(substr($data, -22, 22), 2);
$data = substr($data, $key_block_bytes + 2, -22);
2013-01-26 16:17:11 +00:00
2013-01-26 19:00:00 +00:00
$mkMDC = hash("sha1", $prefix . $data . "\xD3\x14", true);
if($mkMDC !== $mdc) return false;
try {
$msg = OpenPGP_Message::parse($data);
} catch (Exception $ex) { $msg = NULL; }
if($msg) return $msg; /* Otherwise keep trying */
} else {
// No MDC mean decrypt with resync
$iv = substr($epacket->data, 2, $key_block_bytes);
$edata = substr($epacket->data, $key_block_bytes + 2);
$cipher->setIV($iv);
$data = substr($cipher->decrypt($edata . str_repeat("\0", $padAmount)), 0, strlen($edata));
try {
$msg = OpenPGP_Message::parse($data);
} catch (Exception $ex) { $msg = NULL; }
if($msg) return $msg; /* Otherwise keep trying */
2013-01-21 23:18:13 +00:00
}
2013-01-26 16:10:45 +00:00
2013-01-26 19:00:00 +00:00
return NULL; /* Failed */
2013-01-21 23:18:13 +00:00
}
2013-01-26 16:26:55 +00:00
public static function getCipher($algo) {
switch($algo) {
case 2:
$cipher = new Crypt_TripleDES(CRYPT_DES_MODE_CFB);
$key_bytes = 24;
$key_block_bytes = 8;
break;
case 7:
$cipher = new Crypt_AES(CRYPT_AES_MODE_CFB);
$cipher->setKeyLength(128);
break;
case 8:
$cipher = new Crypt_AES(CRYPT_AES_MODE_CFB);
$cipher->setKeyLength(192);
break;
case 9:
$cipher = new Crypt_AES(CRYPT_AES_MODE_CFB);
$cipher->setKeyLength(256);
break;
default:
$cipher = NULL;
}
if(!$cipher) return array(NULL, NULL, NULL); // Unsupported cipher
if(!isset($key_bytes)) $key_bytes = $cipher->key_size;
if(!isset($key_block_bytes)) $key_block_bytes = $cipher->block_size;
return array($cipher, $key_bytes, $key_block_bytes);
}
2013-01-21 23:18:13 +00:00
public static function getEncryptedData($m) {
foreach($m as $p) {
if($p instanceof OpenPGP_EncryptedDataPacket) return $p;
}
throw new Exception("Can only decrypt EncryptedDataPacket");
}
}