Skip to content

Commit fe49de6

Browse files
committed
crypto/internal/cryptotest: add ChaCha20-Poly1305 Wycheproof coverage
Tests the vendored x/crypto chacha20poly1305 package using Wycheproof test vectors. Specifically, chacha20_poly1305_test.json and xchacha20_poly1305_test.json. Test vectors with invalid nonce sizes are skipped since the Go implementation panics rather than returning an error. Change-Id: I2e01cfa6d9e3836e5c4ef001adde98170379696d Reviewed-on: https://go-review.googlesource.com/c/go/+/758864 LUCI-TryBot-Result: [email protected] <[email protected]> Reviewed-by: David Chase <[email protected]> Reviewed-by: Roland Shoemaker <[email protected]>
1 parent 5ac12e6 commit fe49de6

1 file changed

Lines changed: 81 additions & 0 deletions

File tree

Lines changed: 81 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
// Copyright 2026 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package cryptotest_test
6+
7+
import (
8+
"bytes"
9+
"crypto/cipher"
10+
"crypto/internal/cryptotest/wycheproof"
11+
"testing"
12+
13+
"golang.org/x/crypto/chacha20poly1305"
14+
)
15+
16+
func TestChaCha20Poly1305Wycheproof(t *testing.T) {
17+
for _, tc := range []struct {
18+
file string
19+
newCipher func([]byte) (cipher.AEAD, error)
20+
}{
21+
{"chacha20_poly1305_test.json", chacha20poly1305.New},
22+
{"xchacha20_poly1305_test.json", chacha20poly1305.NewX},
23+
} {
24+
var testdata wycheproof.AeadTestSchemaV1Json
25+
wycheproof.LoadVectorFile(t, tc.file, &testdata)
26+
27+
for _, tg := range testdata.TestGroups {
28+
for _, tv := range tg.Tests {
29+
t.Run(wycheproof.TestName(tc.file, tv), func(t *testing.T) {
30+
t.Parallel()
31+
32+
aead, err := tc.newCipher(wycheproof.MustDecodeHex(tv.Key))
33+
if err != nil {
34+
t.Fatalf("failed to construct cipher: %s", err)
35+
}
36+
37+
iv := wycheproof.MustDecodeHex(tv.Iv)
38+
tag := wycheproof.MustDecodeHex(tv.Tag)
39+
ct := wycheproof.MustDecodeHex(tv.Ct)
40+
msg := wycheproof.MustDecodeHex(tv.Msg)
41+
aad := wycheproof.MustDecodeHex(tv.Aad)
42+
43+
// The Go implementation panics on invalid nonce sizes rather
44+
// than returning an error. Verify this behavior.
45+
if len(iv) != aead.NonceSize() {
46+
ctWithTag := append(ct, tag...)
47+
wycheproof.MustPanic(t, "Seal", func() { aead.Seal(nil, iv, msg, aad) })
48+
wycheproof.MustPanic(t, "Open", func() { aead.Open(nil, iv, ctWithTag, aad) })
49+
return
50+
}
51+
52+
genCT := aead.Seal(nil, iv, msg, aad)
53+
genMsg, err := aead.Open(nil, iv, genCT, aad)
54+
if err != nil {
55+
t.Errorf("failed to decrypt generated ciphertext: %s", err)
56+
}
57+
if !bytes.Equal(genMsg, msg) {
58+
t.Errorf("unexpected roundtripped plaintext: got %x, want %x", genMsg, msg)
59+
}
60+
61+
ctWithTag := append(ct, tag...)
62+
msg2, err := aead.Open(nil, iv, ctWithTag, aad)
63+
wantPass := wycheproof.ShouldPass(t, tv.Result, tv.Flags, nil)
64+
if !wantPass && err == nil {
65+
t.Error("decryption succeeded when it should've failed")
66+
} else if wantPass {
67+
if err != nil {
68+
t.Fatalf("decryption failed: %s", err)
69+
}
70+
if !bytes.Equal(genCT, ctWithTag) {
71+
t.Errorf("generated ciphertext doesn't match expected: got %x, want %x", genCT, ctWithTag)
72+
}
73+
if !bytes.Equal(msg, msg2) {
74+
t.Errorf("decrypted ciphertext doesn't match expected: got %x, want %x", msg2, msg)
75+
}
76+
}
77+
})
78+
}
79+
}
80+
}
81+
}

0 commit comments

Comments
 (0)