MARC4.js 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. (function () {
  2. var C = typeof window === 'undefined' ? require('./Crypto').Crypto : window.Crypto;
  3. // Shortcuts
  4. var util = C.util;
  5. var charenc = C.charenc;
  6. var UTF8 = charenc.UTF8;
  7. var Binary = charenc.Binary;
  8. var MARC4 = (C.MARC4 = {
  9. /**
  10. * Public API
  11. */
  12. encrypt: function (message, password) {
  13. var // Convert to bytes
  14. m = UTF8.stringToBytes(message);
  15. var // Generate random IV
  16. iv = util.randomBytes(16);
  17. var // Generate key
  18. k =
  19. password.constructor == String
  20. ? // Derive key from passphrase
  21. C.PBKDF2(password, iv, 32, {
  22. asBytes: true
  23. })
  24. : // else, assume byte array representing cryptographic key
  25. password; // Encrypt
  26. MARC4._marc4(m, k, 1536);
  27. // Return ciphertext
  28. return util.bytesToBase64(iv.concat(m));
  29. },
  30. decrypt: function (ciphertext, password) {
  31. var // Convert to bytes
  32. c = util.base64ToBytes(ciphertext);
  33. var // Separate IV and message
  34. iv = c.splice(0, 16);
  35. var // Generate key
  36. k =
  37. password.constructor == String
  38. ? // Derive key from passphrase
  39. C.PBKDF2(password, iv, 32, {
  40. asBytes: true
  41. })
  42. : // else, assume byte array representing cryptographic key
  43. password; // Decrypt
  44. MARC4._marc4(c, k, 1536);
  45. // Return plaintext
  46. return UTF8.bytesToString(c);
  47. },
  48. /**
  49. * Internal methods
  50. */
  51. // The core
  52. _marc4: function (m, k, drop) {
  53. // State variables
  54. var i;
  55. var j;
  56. var s;
  57. var temp; // Key setup
  58. for (i = 0, s = []; i < 256; i++) {
  59. s[i] = i;
  60. }
  61. for (i = 0, j = 0; i < 256; i++) {
  62. j = (j + s[i] + k[i % k.length]) % 256;
  63. // Swap
  64. temp = s[i];
  65. s[i] = s[j];
  66. s[j] = temp;
  67. }
  68. // Clear counters
  69. i = j = 0;
  70. // Encryption
  71. for (var k = -drop; k < m.length; k++) {
  72. i = (i + 1) % 256;
  73. j = (j + s[i]) % 256;
  74. // Swap
  75. temp = s[i];
  76. s[i] = s[j];
  77. s[j] = temp;
  78. // Stop here if we're still dropping keystream
  79. if (k < 0) {
  80. continue;
  81. }
  82. // Encrypt
  83. m[k] ^= s[(s[i] + s[j]) % 256];
  84. }
  85. }
  86. });
  87. })();