Encrypting data with ruby decrypting with node

后端 未结 3 890
礼貌的吻别
礼貌的吻别 2021-02-08 19:55

I want to encrypt some data in a ruby app and then decode it in a nodejs app. I have been trying to get this to work and now I am just trying to encrypt the same piece of data i

3条回答
  •  谎友^
    谎友^ (楼主)
    2021-02-08 20:26

    There are several subtle things that make this fail. The most important one - you are not specifying an IV in your code, so a random value will be generated for you. You would notice that you couldn't even decrypt your ciphertext within the same programming language this way.

    So you need to provide an explicit IV to both implementations. But before I show you the code, some advice:

    Key generation:

    Blowfish operates on 64 bit blocks, its key size varies, but OpenSSL (which currently powers both Ruby's and node.js' cipher implementation) uses 128 bit by default, that is 16 bytes.

    So your key violates two principles - the first: it's simply too long. It's the hex representation of a SHA-1 hash, which is 20 bytes * 2 = 40 bytes instead of 16. Most of the time this is fine, because the implementation truncates the values appropriately, but that is something you should not depend on.

    The second mistake, much more severe, is that you use the hex representation instead of the raw bytes: big security issue! Hex characters are not random at all, so in effect you reduce the entropy of your input to half the length (because the underlying bytes were random).

    A secure way to generate random keys is using OpenSSL::Random

    key = OpenSSL::Random.random_bytes(cipher_key_len)
    

    A third mistake is to keep your key hard-coded in the sources. It's a bad idea. The least you should do is to store it elsewhere on the file system, where access is tightly restricted. See also my answer to another question. The key should be stored out-of-band and only loaded dynamically within the application.

    Cipher:

    Blowfish grows old. It's still considered unbroken in the sense that brute-forcing it is the only way to break it. But a search space of 2^64 is not out of reach for resourceful attackers. So you should indeed move on to AES.

    Padding:

    OpenSSL pads using PKCS5Padding (also known as PKCS7Padding) by default. Ruby profits from this and my bet is node.js utilizes this, too - so you should be safe on this.

    Now to the working solution. We need to generate an IV, Blowfish requires it to be 64 bit - 8 bytes. You will need rbytes to get secure random numbers in node. The IV may be hardcoded in your sources (it's public information, no security impact) - but it must be the same on both sides. You should pregenerate a value and use it for both node.js and Ruby.

    /*node.js*/
    
    var rbytes = require('rbytes');
    var iv = rbytes.randomBytes(8);
    
    /*see advice above - this should be out-of-band*/
    var key = rbytes.randomBytes(16);
    var encrypted = "";
    var cipher = crypto.createCipheriv('bf-cbc', key, iv);
    
    encrypted += cipher.update('text');
    encrypted += cipher.final('hex');
    

    Now the Ruby part:

    require 'openssl'
    
    c = OpenSSL::Cipher::Cipher.new("bf-cbc")
    c.encrypt
    # should be out-of-band again
    c.key = OpenSSL::Random.random_bytes(16)
    # may be public but has to be the same for Ruby and node
    iv = OpenSSL::Random.random_bytes(8)
    c.iv = iv 
    e = c.update("text")
    e << c.final
    puts e.unpack('H*')[0]
    

提交回复
热议问题