[SEC] Add `keying` module
The keying modules tries to solve two problems, the lack of key
separation and the lack of AEAD being used for encryption. The currently
used `secrets` doesn't provide this and is hard to adjust to provide
this functionality.
For encryption, the additional data is now a parameter that can be used,
as the underlying primitive is an AEAD constructions. This allows for
context binding to happen and can be seen as defense-in-depth; it
ensures that if a value X is encrypted for context Y (e.g. ID=3,
Column="private_key") it will only decrypt if that context Y is also
given in the Decrypt function. This makes confused deputy attack harder
to exploit.[^1]
For key separation, HKDF is used to derives subkeys from some IKM, which
is the value of the `[service].SECRET_KEY` config setting. The context
for subkeys are hardcoded, any variable should be shuffled into the the
additional data parameter when encrypting.
[^1]: This is still possible, because the used AEAD construction is not
key-comitting. For Forgejo's current use-case this risk is negligible,
because the subkeys aren't known to a malicious user (which is required
for such attack), unless they also have access to the IKM (at which
point you can assume the whole system is compromised). See
https://scottarc.blog/2022/10/17/lucid-multi-key-deputies-require-commitment/
2024-08-20 23:13:04 +02:00
|
|
|
// Copyright 2024 The Forgejo Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
|
|
|
package keying_test
|
|
|
|
|
|
|
|
import (
|
2024-08-04 14:46:05 -04:00
|
|
|
"math"
|
[SEC] Add `keying` module
The keying modules tries to solve two problems, the lack of key
separation and the lack of AEAD being used for encryption. The currently
used `secrets` doesn't provide this and is hard to adjust to provide
this functionality.
For encryption, the additional data is now a parameter that can be used,
as the underlying primitive is an AEAD constructions. This allows for
context binding to happen and can be seen as defense-in-depth; it
ensures that if a value X is encrypted for context Y (e.g. ID=3,
Column="private_key") it will only decrypt if that context Y is also
given in the Decrypt function. This makes confused deputy attack harder
to exploit.[^1]
For key separation, HKDF is used to derives subkeys from some IKM, which
is the value of the `[service].SECRET_KEY` config setting. The context
for subkeys are hardcoded, any variable should be shuffled into the the
additional data parameter when encrypting.
[^1]: This is still possible, because the used AEAD construction is not
key-comitting. For Forgejo's current use-case this risk is negligible,
because the subkeys aren't known to a malicious user (which is required
for such attack), unless they also have access to the IKM (at which
point you can assume the whole system is compromised). See
https://scottarc.blog/2022/10/17/lucid-multi-key-deputies-require-commitment/
2024-08-20 23:13:04 +02:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"code.gitea.io/gitea/modules/keying"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"golang.org/x/crypto/chacha20poly1305"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestKeying(t *testing.T) {
|
|
|
|
t.Run("Not initalized", func(t *testing.T) {
|
|
|
|
assert.Panics(t, func() {
|
|
|
|
keying.DeriveKey(keying.Context("TESTING"))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Initialization", func(t *testing.T) {
|
|
|
|
keying.Init([]byte{0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Context seperation", func(t *testing.T) {
|
|
|
|
key1 := keying.DeriveKey(keying.Context("TESTING"))
|
|
|
|
key2 := keying.DeriveKey(keying.Context("TESTING2"))
|
|
|
|
|
|
|
|
ciphertext := key1.Encrypt([]byte("This is for context TESTING"), nil)
|
|
|
|
|
|
|
|
plaintext, err := key2.Decrypt(ciphertext, nil)
|
|
|
|
require.Error(t, err)
|
|
|
|
assert.Empty(t, plaintext)
|
|
|
|
|
|
|
|
plaintext, err = key1.Decrypt(ciphertext, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.EqualValues(t, "This is for context TESTING", plaintext)
|
|
|
|
})
|
|
|
|
|
|
|
|
context := keying.Context("TESTING PURPOSES")
|
|
|
|
plainText := []byte("Forgejo is run by [Redacted]")
|
|
|
|
var cipherText []byte
|
|
|
|
t.Run("Encrypt", func(t *testing.T) {
|
|
|
|
key := keying.DeriveKey(context)
|
|
|
|
|
|
|
|
cipherText = key.Encrypt(plainText, []byte{0x05, 0x06})
|
|
|
|
cipherText2 := key.Encrypt(plainText, []byte{0x05, 0x06})
|
|
|
|
|
|
|
|
// Ensure ciphertexts don't have an determistic output.
|
|
|
|
assert.NotEqualValues(t, cipherText, cipherText2)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Decrypt", func(t *testing.T) {
|
|
|
|
key := keying.DeriveKey(context)
|
|
|
|
|
|
|
|
t.Run("Succesful", func(t *testing.T) {
|
|
|
|
convertedPlainText, err := key.Decrypt(cipherText, []byte{0x05, 0x06})
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.EqualValues(t, plainText, convertedPlainText)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Not enougn additional data", func(t *testing.T) {
|
|
|
|
plainText, err := key.Decrypt(cipherText, []byte{0x05})
|
|
|
|
require.Error(t, err)
|
|
|
|
assert.Empty(t, plainText)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Too much additional data", func(t *testing.T) {
|
|
|
|
plainText, err := key.Decrypt(cipherText, []byte{0x05, 0x06, 0x07})
|
|
|
|
require.Error(t, err)
|
|
|
|
assert.Empty(t, plainText)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Incorrect nonce", func(t *testing.T) {
|
|
|
|
// Flip the first byte of the nonce.
|
|
|
|
cipherText[0] = ^cipherText[0]
|
|
|
|
|
|
|
|
plainText, err := key.Decrypt(cipherText, []byte{0x05, 0x06})
|
|
|
|
require.Error(t, err)
|
|
|
|
assert.Empty(t, plainText)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Incorrect ciphertext", func(t *testing.T) {
|
|
|
|
assert.Panics(t, func() {
|
|
|
|
key.Decrypt(nil, nil)
|
|
|
|
})
|
|
|
|
|
|
|
|
assert.Panics(t, func() {
|
|
|
|
cipherText := make([]byte, chacha20poly1305.NonceSizeX)
|
|
|
|
key.Decrypt(cipherText, nil)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
2024-08-04 14:46:05 -04:00
|
|
|
|
|
|
|
func TestKeyingColumnAndID(t *testing.T) {
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x3a, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, keying.ColumnAndID("table", math.MinInt64))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x3a, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, keying.ColumnAndID("table", -1))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x3a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, keying.ColumnAndID("table", 0))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x3a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}, keying.ColumnAndID("table", 1))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x3a, 0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, keying.ColumnAndID("table", math.MaxInt64))
|
|
|
|
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x32, 0x3a, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, keying.ColumnAndID("table2", math.MinInt64))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x32, 0x3a, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, keying.ColumnAndID("table2", -1))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x32, 0x3a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, keying.ColumnAndID("table2", 0))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x32, 0x3a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}, keying.ColumnAndID("table2", 1))
|
|
|
|
assert.EqualValues(t, []byte{0x74, 0x61, 0x62, 0x6c, 0x65, 0x32, 0x3a, 0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, keying.ColumnAndID("table2", math.MaxInt64))
|
|
|
|
}
|