Мне нужно подражать тому, что MySQL делает при шифровании и дешифровании строк, используя встроенные функции AES_ENCRYPT () и AES_DECRYPT ().
Я прочитал пару постов в блоге, и, очевидно, MySQL использует 128-битное шифрование AES для этих функций. Кроме того, поскольку для этого шифрования требуется 16-битный ключ, MySQL дополняет строку символами x0 (\ 0s) до размера 16-бит.
Алгоритм на C из исходного кода MySQL замечен здесь .
Теперь мне нужно воспроизвести то, что MySQL делает в приложении на Rails, но все, что я пробовал, не работает.
Вот способ воспроизвести поведение, которое я получаю:
1) Создать новое приложение Rails
rails encryption-test
cd encryption-test
2) Создать новые строительные леса
script/generate scaffold user name:string password:binary
3) Отредактируйте ваш config / database.yml и добавьте тестовую базу данных MySQL
development:
adapter: mysql
host: localhost
database: test
user: <<user>>
password: <<password>>
4) Запустить миграцию
rake db:migrate
5) Войдите в консоль, создайте пользователя и обновите его пароль из запроса MySQL
script/console
Loading development environment (Rails 2.2.2)
>> User.create(:name => "John Doe")
>> key = "82pjd12398JKBSDIGUSisahdoahOUASDHsdapdjqwjeASIduAsdh078asdASD087asdADSsdjhA7809asdajhADSs"
>> ActiveRecord::Base.connection.execute("UPDATE users SET password = AES_ENCRYPT('password', '#{key}') WHERE name='John Doe'")
Вот где я застрял. Если я пытаюсь расшифровать его, используя MySQL, он работает:
>> loaded_user = User.find_by_sql("SELECT AES_DECRYPT(password, '#{key}') AS password FROM users WHERE id=1").first
>> loaded_user['password']
=> "password"
Однако, если я попытаюсь использовать библиотеку OpenSSL, я не смогу заставить ее работать:
cipher = OpenSSL::Cipher::Cipher.new("AES-128-ECB")
cipher.padding = 0
cipher.key = key
cipher.decrypt
user = User.find(1)
cipher.update(user.password) << cipher.final #=> "########gf####\027\227"
Я попытался заполнить ключ:
desired_length = 16 * ((key.length / 16) + 1)
padded_key = key + "\0" * (desired_length - key.length)
cipher = OpenSSL::Cipher::Cipher.new("AES-128-ECB")
cipher.key = key
cipher.decrypt
user = User.find(1)
cipher.update(user.password) << cipher.final #=> ""|\e\261\205:\032s\273\242\030\261\272P##"
Но это действительно не работает.
Кто-нибудь знает, как можно имитировать поведение функций MySQL AES_ENCRYPT () и AES_DECRYPT () в Ruby?
Спасибо!