Switch from CBC to GCM for Geo logout tokens

The AES-256-GCM cipher is the current best practice for symmetric
encryption. It also allows us to protect with authentication the
content of the return_to and validate everything on the Geo primary
node, otherwise, we can open a security hole.
parent 096ddf92
......@@ -4,37 +4,42 @@ module Gitlab
module Geo
module Oauth
class LogoutState
include ::Gitlab::Utils::StrongMemoize
def self.from_state(state)
salt, encrypted, return_to = state.to_s.split(':', 3)
self.new(salt: salt, token: encrypted, return_to: return_to)
salt, tag, encrypted, return_to = state.to_s.split(':', 4)
self.new(salt: salt, tag: tag, token: encrypted, return_to: return_to)
end
def initialize(token:, salt: nil, return_to: nil)
def initialize(token:, salt: nil, tag: nil, return_to: nil)
@token = token
@salt = salt
@salt = decode_base64(salt)
@tag = decode_base64(tag)
@return_to_location = Gitlab::ReturnToLocation.new(return_to)
end
def decode
return unless salt && token
return unless salt && tag && token
return unless tag.bytesize == 16
encrypted = decode_base64(token)
return unless encrypted
decoded = Base64.urlsafe_decode64(token)
decrypt = cipher(salt, :decrypt)
decrypt.update(decoded) + decrypt.final
rescue OpenSSL::OpenSSLError
decrypt.update(encrypted) + decrypt.final
rescue ArgumentError, OpenSSL::OpenSSLError
nil
end
def encode
return unless token
iv = salt || SecureRandom.hex(8)
encrypt = cipher(iv, :encrypt)
encrypted = encrypt.update(token) + encrypt.final
encoded = Base64.urlsafe_encode64(encrypted)
salt_base64 = encode_base64(salt)
auth_tag_base64 = encode_base64(encrypt.auth_tag)
encrypted_base64 = encode_base64(encrypted)
"#{iv}:#{encoded}:#{return_to}"
rescue OpenSSL::OpenSSLError
"#{salt_base64}:#{auth_tag_base64}:#{encrypted_base64}:#{return_to}"
rescue ArgumentError, OpenSSL::OpenSSLError
nil
end
......@@ -44,15 +49,41 @@ module Gitlab
private
attr_reader :token, :salt, :return_to_location
attr_reader :token, :salt, :tag, :return_to_location
def encrypt
strong_memoize(:encrypt) do
with_cipher { |cipher| cipher.encrypt }
end
end
def decrypt
strong_memoize(:decrypt) do
with_cipher(tag) { |cipher| cipher.decrypt }
end
end
def with_cipher(auth_tag = nil)
cipher = OpenSSL::Cipher::AES256.new(:GCM)
def cipher(salt, operation)
cipher = OpenSSL::Cipher::AES.new(128, :CBC)
cipher.__send__(operation) # rubocop:disable GitlabSecurity/PublicSend
cipher.iv = salt
cipher.key = Settings.attr_encrypted_db_key_base.first(16)
yield cipher
cipher.key = Settings.attr_encrypted_db_key_base_truncated
cipher.iv = @salt ||= cipher.random_iv
cipher.auth_tag = auth_tag if auth_tag
cipher.auth_data = return_to.to_s
cipher
end
def encode_base64(value)
Base64.urlsafe_encode64(value)
end
def decode_base64(value)
Base64.urlsafe_decode64(value)
rescue ArgumentError, NoMethodError
nil
end
end
end
end
......
......@@ -2,18 +2,21 @@
module Gitlab
class ReturnToLocation
include ::Gitlab::Utils::StrongMemoize
def initialize(location)
@location = location
end
def full_path
uri = parse_uri
if uri
path = remove_domain_from_uri(uri)
path = add_fragment_back_to_path(uri, path)
path
strong_memoize(:full_path) do
uri = parse_uri
if uri
path = remove_domain_from_uri(uri)
path = add_fragment_back_to_path(uri, path)
path
end
end
end
......
......@@ -3,14 +3,15 @@
require 'spec_helper'
describe Gitlab::Geo::Oauth::LogoutState do
let(:salt) { '100d8cbd1750a2bb' }
let(:salt) { 'MTAwZDhjYmQxNzUw' }
let(:tag) { 'Y0D_b1xDW3uO-qN86c83HQ==' }
let(:return_to) { 'http://fake-secondary.com:3000/project/test' }
let(:access_token) { '48622af3df5b5b3e09b9754f2a3e5f3f10a94b4147d155b1029d827c112524d1' }
let(:encrypted_token) { 'fDyMq6IrHGhToG5NHiXnQ4O8AsHmSDqDTqbLP64MK0L9j0rkPEnrNDBSoWU-QS2l7sIt_Q4UMItxFhFH6xMh68uspgydVysRG9fmr_PXIU4=' }
let(:encrypted_token) { 't5fPL8_1KcFC5L945n9fcMRr7N-1J60LrOREQ9BAdur_K97tU1IpmWrN5-9P9aqpFvdL3SxzvP_z6CfO92BPsA==' }
before do
allow(Settings).to receive(:attr_encrypted_db_key_base)
.and_return('4587f5984bf8f807ee320ed7b783e0c56b644a18fdcf5bc79bb2b5b38edbbb1a7037e8d79cbc880cc593880cd3ce87906ebb38466428dfd0dc70a626bb28b7ba')
allow(Settings).to receive(:attr_encrypted_db_key_base_truncated)
.and_return('4587f5984bf8f807ee320ed7b783e0c5')
end
describe '#encode' do
......@@ -21,7 +22,7 @@ describe Gitlab::Geo::Oauth::LogoutState do
end
it 'returns nil when encryption fails' do
allow_any_instance_of(OpenSSL::Cipher::AES)
allow_any_instance_of(OpenSSL::Cipher::AES256)
.to receive(:final) { raise OpenSSL::OpenSSLError }
subject = described_class.new(token: access_token, return_to: return_to)
......@@ -29,10 +30,10 @@ describe Gitlab::Geo::Oauth::LogoutState do
expect(subject.encode).to be_nil
end
it 'returns a string with salt, encrypted access token, and return_to full path colon separated' do
it 'returns a string with salt, tag, encrypted access token, and return_to full path colon separated' do
subject = described_class.new(salt: salt, token: access_token, return_to: return_to)
expect(subject.encode).to eq("#{salt}:#{encrypted_token}:/project/test")
expect(subject.encode).to eq("#{salt}:#{tag}:#{encrypted_token}:/project/test")
end
it 'includes a empty value for return_to into state when return_to is nil' do
......@@ -40,7 +41,7 @@ describe Gitlab::Geo::Oauth::LogoutState do
state = subject.encode
expect(state.split(':', 3)[2]).to eq ''
expect(state.split(':', 4)[3]).to eq ''
end
end
......@@ -51,23 +52,59 @@ describe Gitlab::Geo::Oauth::LogoutState do
expect(subject.decode).to be_nil
end
it 'returns nil when salt has invalid base64' do
subject = described_class.new(salt: 'invalid', tag: tag, token: encrypted_token, return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when tag is nil' do
subject = described_class.new(salt: salt, tag: nil, token: encrypted_token, return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when encrypted token has invalid base64' do
subject = described_class.new(salt: salt, tag: tag, token: 'invalid', return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when encrypted token is nil' do
subject = described_class.new(salt: salt, token: nil, return_to: return_to)
subject = described_class.new(salt: salt, tag: tag, token: nil, return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when decryption fails' do
allow_any_instance_of(OpenSSL::Cipher::AES)
allow_any_instance_of(OpenSSL::Cipher::AES256)
.to receive(:final) { raise OpenSSL::OpenSSLError }
subject = described_class.new(salt: salt, token: encrypted_token, return_to: return_to)
subject = described_class.new(salt: salt, tag: tag, token: encrypted_token, return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when tag has an invalid byte size' do
subject = described_class.new(salt: salt, tag: 'aW52YWxpZA==', token: encrypted_token, return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when tag has been modified' do
subject = described_class.new(salt: salt, tag: 'MGY4MzY5YmU0OTk0', token: encrypted_token, return_to: return_to)
expect(subject.decode).to be_nil
end
it 'returns nil when return_to has been modified' do
subject = described_class.new(salt: salt, tag: tag, token: encrypted_token, return_to: '/foo/bar')
expect(subject.decode).to be_nil
end
it 'returns access_token when token is recoverable' do
subject = described_class.new(salt: salt, token: encrypted_token, return_to: return_to)
subject = described_class.new(salt: salt, tag: tag, token: encrypted_token, return_to: return_to)
expect(subject.decode).to eq(access_token)
end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment