Files
serenity/Tests/LibSSH/TestSSHEncodeDecode.cpp
Lucas Chollet c8e5a4581c LibSSH: Make TypedBlob constructible from raw bytes or a string
This will later be used for reading public keys from the network or from
a local file.
2026-04-08 13:43:19 +02:00

132 lines
4.8 KiB
C++

/*
* Copyright (c) 2026, Lucas Chollet <lucas.chollet@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibSSH/DataTypes.h>
#include <LibTest/TestCase.h>
TEST_CASE(string)
{
auto test_cases = to_array({
"Serenity"sv,
"zlib"sv,
"Including\0null"sv,
"Including Space"sv,
"Including Unicode ˢˢʰ"sv,
"testing"sv,
});
for (auto test : test_cases) {
AllocatingMemoryStream encoded_stream {};
TRY_OR_FAIL(SSH::encode_string(encoded_stream, test));
auto encoded = TRY_OR_FAIL(encoded_stream.read_until_eof());
FixedMemoryStream decoded_stream { encoded.bytes() };
auto decoded = TRY_OR_FAIL(SSH::decode_string(decoded_stream));
EXPECT_EQ(test.bytes(), decoded);
}
}
// Examples are from:
// https://datatracker.ietf.org/doc/html/rfc4251#section-5
TEST_CASE(string_spec)
{
// "For example: the US-ASCII string "testing" is represented as 00 00 00 07 t e s t i n g"
auto test_string = "testing"sv;
auto encoded_string = to_array<u8>({ 0, 0, 0, 7, 't', 'e', 's', 't', 'i', 'n', 'g' });
AllocatingMemoryStream encoded_stream {};
TRY_OR_FAIL(SSH::encode_string(encoded_stream, test_string));
auto encoded = TRY_OR_FAIL(encoded_stream.read_until_eof());
EXPECT_EQ(encoded.span(), encoded_string);
}
TEST_CASE(name_list)
{
// value representation (hex)
// ----- --------------------
// (), the empty name-list 00 00 00 00
// ("zlib") 00 00 00 04 7a 6c 69 62
// ("zlib,none") 00 00 00 09 7a 6c 69 62 2c 6e 6f 6e 65
auto test = [](Span<StringView> list, ReadonlyBytes expected) -> ErrorOr<void> {
AllocatingMemoryStream stream;
TRY(SSH::encode_name_list(stream, list));
auto encoded = TRY(stream.read_until_eof());
EXPECT_EQ(encoded, expected);
return {};
};
TRY_OR_FAIL(test({}, to_array<u8>({ 0, 0, 0, 0 })));
TRY_OR_FAIL(test(to_array<StringView>({ "zlib"sv }), to_array<u8>({ 0, 0, 0, 4, 0x7a, 0x6c, 0x69, 0x62 })));
TRY_OR_FAIL(test(to_array<StringView>({ "zlib"sv, "none"sv }), to_array<u8>({ 0, 0, 0, 9, 0x7a, 0x6c, 0x69, 0x62, 0x2c, 0x6e, 0x6f, 0x6e, 0x65 })));
}
TEST_CASE(mpint)
{
// value (hex) representation (hex)
// ----------- --------------------
// 0 00 00 00 00
// 9a378f9b2e332a7 00 00 00 08 09 a3 78 f9 b2 e3 32 a7
// 80 00 00 00 02 00 80
// -1234 00 00 00 02 ed cc
// -deadbeef 00 00 00 05 ff 21 52 41 11
auto test = [](ReadonlyBytes value, ReadonlyBytes expected) -> ErrorOr<void> {
auto encoded = TRY(SSH::as_mpint(value));
EXPECT_EQ(encoded.bytes(), expected);
return {};
};
TRY_OR_FAIL(test(to_array<u8>({ 0x09, 0xa3, 0x78, 0xf9, 0xb2, 0xe3, 0x32, 0xa7 }), to_array<u8>({ 0x00, 0x00, 0x00, 0x08, 0x09, 0xa3, 0x78, 0xf9, 0xb2, 0xe3, 0x32, 0xa7 })));
TRY_OR_FAIL(test(to_array<u8>({ 0x80 }), to_array<u8>({ 0x00, 0x00, 0x00, 0x02, 0x00, 0x80 })));
// FIXME: Whenever we add support for them add the corresponding tests:
// - for input with leading zeros
// - for negative input numbers
}
TEST_CASE(typed_blob)
{
ByteBuffer a;
a.resize(8);
a.bytes().fill(0x42);
auto input = SSH::TypedBlob { SSH::TypedBlob::Type::SSH_ED25519, a };
AllocatingMemoryStream stream;
TRY_OR_FAIL(input.encode(stream));
auto encoded = TRY_OR_FAIL(stream.read_until_eof());
auto expected = to_array<u8>({
0x00, 0x00, 0x00, 27, // Total size (string size + blob size + 8)
0x00, 0x00, 0x00, 11, // string size
's', 's', 'h', '-', 'e', 'd', '2', '5', '5', '1', '9', // key type
0x00, 0x00, 0x00, 8, // blob size
0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, // blob
});
EXPECT_EQ(encoded.bytes(), expected.span());
FixedMemoryStream encoded_stream { encoded.bytes() };
auto roundtrip_blob = TRY_OR_FAIL(SSH::TypedBlob::decode(encoded_stream));
EXPECT_EQ(roundtrip_blob.type, input.type);
EXPECT_EQ(roundtrip_blob.key, input.key);
}
TEST_CASE(typed_blob_from_string)
{
static constexpr auto public_key = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAICOZUtxv/6qw3GjU4OwgEeSrpKC/81n8yYLwAf/T26Ni lucas"sv;
auto typed_blob = TRY_OR_FAIL(SSH::TypedBlob::read_from_string(public_key));
EXPECT_EQ(typed_blob.type, SSH::TypedBlob::Type::SSH_ED25519);
EXPECT_EQ(typed_blob.key, "\x23\x99\x52\xdc\x6f\xff\xaa\xb0\xdc\x68\xd4\xe0\xec"
"\x20\x11\xe4\xab\xa4\xa0\xbf\xf3\x59\xfc\xc9\x82\xf0"
"\x01\xff\xd3\xdb\xa3\x62"sv.bytes());
}