XXTEA 加密算法的 JavaScript 和 PHP 實現

XXTEA 加密算法的 JavaScript 和 PHP 實現

微型加密算法(TEA)及其相關變種(XTEA,Block TEA,XXTEA) 都是分組加密算法,它們很容易被描述,實現也很簡單(典型的幾行代碼)。

TEA 算法最初是由劍橋計算機實驗室的 David Wheeler 和 Roger Needham 在 1994 年設計的。該算法使用 128 位的密鑰爲 64 位的信息塊進行加密,它需要進行 64 輪迭代,儘管作者認爲 32 輪已經足夠了。該算法使用了一個神祕常數δ作爲倍數,它來源於黃金比率,以保證每一輪加密都不相同。但δ的精確值似乎並不重要,這裏 TEA 把它定義爲 δ=「(√5 - 1)231」(也就是程序中的 0×9E3779B9)。

之後 TEA 算法被發現存在缺陷,作爲迴應,設計者提出了一個 TEA 的升級版本——XTEA(有時也被稱爲“tean”)。XTEA 跟 TEA 使用了相同的簡單運算,但它採用了截然不同的順序,爲了阻止密鑰表攻擊,四個子密鑰(在加密過程中,原 128 位的密鑰被拆分爲 4 個 32 位的子密鑰)採用了一種不太正規的方式進行混合,但速度更慢了。

在跟描述 XTEA 算法的同一份報告中,還介紹了另外一種被稱爲 Block TEA 算法的變種,它可以對 32 位大小任意倍數的變量塊進行操作。該算法將 XTEA 輪循函數依次應用於塊中的每個字,並且將它附加於它的鄰字。該操作重複多少輪依賴於塊的大小,但至少需要 6 輪。該方法的優勢在於它無需操作模式(CBC,OFB,CFB 等),密鑰可直接用於信息。對於長的信息它可能比 XTEA 更有效率。

在 1998 年,Markku-Juhani Saarinen 給出了一個可有效攻擊 Block TEA 算法的代碼,但之後很快 David J. Wheeler 和 Roger M. Needham 就給出了 Block TEA 算法的修訂版,這個算法被稱爲 XXTEA。XXTEA 使用跟 Block TEA 相似的結構,但在處理塊中每個字時利用了相鄰字。它利用一個更復雜的 MX 函數代替了 XTEA 輪循函數,MX 使用 2 個輸入量。

XXTEA 算法很安全,而且非常快速,非常適合應用於 Web 開發中。但目前似乎很少有人將該算法用於實際開發中。甚至國內尚無介紹該算法的文章(至少在 Google 上搜索不到這方面的中文文章,關於密碼學算法的書中也未見提及)。我在 Google 上搜索到了幾個國外的 XXTEA 算法的實現(見參考文獻),但基本上都是 JavaScript 的,但這些 JavaScript 實現也有一些問題,如果加密字符串長度不是 4 的整數倍,則這些實現的在加密後無法真正還原,還原以後的字符串實際上與原字符串不相等,而是後面多了一些 /0 的字符,或者少了一些 /0 的字符。

原因在於 XXTEA 算法只定義瞭如何對 32 位的信息塊數組(實際上是 32 位無符號整數數組)進行加密,而並沒有定義如何來將字符串編碼爲這種數組。而現有的實現中在將字符串編碼爲整數數組時,都丟失了字符串長度信息,因此還原出現了問題。另外單純的 JavaScript 是沒有意義的,因爲單純的客戶端加密解密是不能有效保證信息的安全性的。因此我寫了這個 JavaScript 和 PHP 實現,這兩種實現在字符串編碼上採用的算法是一致的,因此 JavaScript 加密的內容可以用 PHP 實現的解密算法進行解密,反之亦然。

注意:如果需要在 JavaScript 中加密解密帶有漢字的信息, 在加密時,需要先將帶加密信息用 utf16to8 進行轉換,解密時,需要將解密後的內容再用 utf8to16 還原。如果要在 PHP 和 JavaScript 之間傳遞帶有漢字的加密信息,原信息需要用 UTF-8 字符集。

 

JavaScript 版本的演示程序:http://test.coolcode.cn/xxtea/

JavaScript 實現代碼

xxtea.js
  1. /* XXTEA encryption arithmetic library.
  2. *
  3. * Copyright (C) 2006 Ma Bingyao <[email protected]>
  4. * Version: 1.5
  5. * LastModified: Dec 9, 2006
  6. * This library is free. You can redistribute it and/or modify it.
  7. */
  8. function long2str(v, w) {
  9. var vl = v.length;
  10. var n = (vl - 1) << 2;
  11. if (w) {
  12. var m = v[vl - 1];
  13. if ((m < n - 3) || (m > n)) return null;
  14. n = m;
  15. }
  16. for (var i = 0; i < vl; i++) {
  17. v[i] = String.fromCharCode(v[i] & 0xff,
  18. v[i] >>> 8 & 0xff,
  19. v[i] >>> 16 & 0xff,
  20. v[i] >>> 24 & 0xff);
  21. }
  22. if (w) {
  23. return v.join('').substring(0, n);
  24. }
  25. else {
  26. return v.join('');
  27. }
  28. }
  29. function str2long(s, w) {
  30. var len = s.length;
  31. var v = [];
  32. for (var i = 0; i < len; i += 4) {
  33. v[i >> 2] = s.charCodeAt(i)
  34. | s.charCodeAt(i + 1) << 8
  35. | s.charCodeAt(i + 2) << 16
  36. | s.charCodeAt(i + 3) << 24;
  37. }
  38. if (w) {
  39. v[v.length] = len;
  40. }
  41. return v;
  42. }
  43. function xxtea_encrypt(str, key) {
  44. if (str == "") {
  45. return "";
  46. }
  47. var v = str2long(str, true);
  48. var k = str2long(key, false);
  49. if (k.length < 4) {
  50. k.length = 4;
  51. }
  52. var n = v.length - 1;
  53. var z = v[n], y = v[0], delta = 0x9E3779B9;
  54. var mx, e, p, q = Math.floor(6 + 52 / (n + 1)), sum = 0;
  55. while (0 < q--) {
  56. sum = sum + delta & 0xffffffff;
  57. e = sum >>> 2 & 3;
  58. for (p = 0; p < n; p++) {
  59. y = v[p + 1];
  60. mx = (z >>> 5 ^ y << 2) + (y >>> 3 ^ z << 4) ^ (sum ^ y) + (k[p & 3 ^ e] ^ z);
  61. z = v[p] = v[p] + mx & 0xffffffff;
  62. }
  63. y = v[0];
  64. mx = (z >>> 5 ^ y << 2) + (y >>> 3 ^ z << 4) ^ (sum ^ y) + (k[p & 3 ^ e] ^ z);
  65. z = v[n] = v[n] + mx & 0xffffffff;
  66. }
  67. return long2str(v, false);
  68. }
  69. function xxtea_decrypt(str, key) {
  70. if (str == "") {
  71. return "";
  72. }
  73. var v = str2long(str, false);
  74. var k = str2long(key, false);
  75. if (k.length < 4) {
  76. k.length = 4;
  77. }
  78. var n = v.length - 1;
  79. var z = v[n - 1], y = v[0], delta = 0x9E3779B9;
  80. var mx, e, p, q = Math.floor(6 + 52 / (n + 1)), sum = q * delta & 0xffffffff;
  81. while (sum != 0) {
  82. e = sum >>> 2 & 3;
  83. for (p = n; p > 0; p--) {
  84. z = v[p - 1];
  85. mx = (z >>> 5 ^ y << 2) + (y >>> 3 ^ z << 4) ^ (sum ^ y) + (k[p & 3 ^ e] ^ z);
  86. y = v[p] = v[p] - mx & 0xffffffff;
  87. }
  88. z = v[n];
  89. mx = (z >>> 5 ^ y << 2) + (y >>> 3 ^ z << 4) ^ (sum ^ y) + (k[p & 3 ^ e] ^ z);
  90. y = v[0] = v[0] - mx & 0xffffffff;
  91. sum = sum - delta & 0xffffffff;
  92. }
  93. return long2str(v, true);
  94. }

PHP 實現代碼

xxtea.php
  1. <?php
  2. /* XXTEA encryption arithmetic library.
  3. *
  4. * Copyright (C) 2006 Ma Bingyao <[email protected]>
  5. * Version: 1.5
  6. * LastModified: Dec 5, 2006
  7. * This library is free. You can redistribute it and/or modify it.
  8. */
  9. function long2str($v, $w) {
  10. $len = count($v);
  11. $n = ($len - 1) << 2;
  12. if ($w) {
  13. $m = $v[$len - 1];
  14. if (($m < $n - 3) || ($m > $n)) return false;
  15. $n = $m;
  16. }
  17. $s = array();
  18. for ($i = 0; $i < $len; $i++) {
  19. $s[$i] = pack("V", $v[$i]);
  20. }
  21. if ($w) {
  22. return substr(join('', $s), 0, $n);
  23. }
  24. else {
  25. return join('', $s);
  26. }
  27. }
  28. function str2long($s, $w) {
  29. $v = unpack("V*", $s. str_repeat("/0", (4 - strlen($s) % 4) & 3));
  30. $v = array_values($v);
  31. if ($w) {
  32. $v[count($v)] = strlen($s);
  33. }
  34. return $v;
  35. }
  36. function int32($n) {
  37. while ($n >= 2147483648) $n -= 4294967296;
  38. while ($n <= -2147483649) $n += 4294967296;
  39. return (int)$n;
  40. }
  41. function xxtea_encrypt($str, $key) {
  42. if ($str == "") {
  43. return "";
  44. }
  45. $v = str2long($str, true);
  46. $k = str2long($key, false);
  47. if (count($k) < 4) {
  48. for ($i = count($k); $i < 4; $i++) {
  49. $k[$i] = 0;
  50. }
  51. }
  52. $n = count($v) - 1;
  53. $z = $v[$n];
  54. $y = $v[0];
  55. $delta = 0x9E3779B9;
  56. $q = floor(6 + 52 / ($n + 1));
  57. $sum = 0;
  58. while (0 < $q--) {
  59. $sum = int32($sum + $delta);
  60. $e = $sum >> 2 & 3;
  61. for ($p = 0; $p < $n; $p++) {
  62. $y = $v[$p + 1];
  63. $mx = int32((($z >> 5 & 0x07ffffff) ^ $y << 2) + (($y >> 3 & 0x1fffffff) ^ $z << 4)) ^ int32(($sum ^ $y) + ($k[$p & 3 ^ $e] ^ $z));
  64. $z = $v[$p] = int32($v[$p] + $mx);
  65. }
  66. $y = $v[0];
  67. $mx = int32((($z >> 5 & 0x07ffffff) ^ $y << 2) + (($y >> 3 & 0x1fffffff) ^ $z << 4)) ^ int32(($sum ^ $y) + ($k[$p & 3 ^ $e] ^ $z));
  68. $z = $v[$n] = int32($v[$n] + $mx);
  69. }
  70. return long2str($v, false);
  71. }
  72. function xxtea_decrypt($str, $key) {
  73. if ($str == "") {
  74. return "";
  75. }
  76. $v = str2long($str, false);
  77. $k = str2long($key, false);
  78. if (count($k) < 4) {
  79. for ($i = count($k); $i < 4; $i++) {
  80. $k[$i] = 0;
  81. }
  82. }
  83. $n = count($v) - 1;
  84. $z = $v[$n];
  85. $y = $v[0];
  86. $delta = 0x9E3779B9;
  87. $q = floor(6 + 52 / ($n + 1));
  88. $sum = int32($q * $delta);
  89. while ($sum != 0) {
  90. $e = $sum >> 2 & 3;
  91. for ($p = $n; $p > 0; $p--) {
  92. $z = $v[$p - 1];
  93. $mx = int32((($z >> 5 & 0x07ffffff) ^ $y << 2) + (($y >> 3 & 0x1fffffff) ^ $z << 4)) ^ int32(($sum ^ $y) + ($k[$p & 3 ^ $e] ^ $z));
  94. $y = $v[$p] = int32($v[$p] - $mx);
  95. }
  96. $z = $v[$n];
  97. $mx = int32((($z >> 5 & 0x07ffffff) ^ $y << 2) + (($y >> 3 & 0x1fffffff) ^ $z << 4)) ^ int32(($sum ^ $y) + ($k[$p & 3 ^ $e] ^ $z));
  98. $y = $v[0] = int32($v[0] - $mx);
  99. $sum = int32($sum - $delta);
  100. }
  101. return long2str($v, true);
  102. }
  103. ?>
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章