net.luminis.tls.extension.SupportedGroupsExtension Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of agent15 Show documentation
Show all versions of agent15 Show documentation
A (partial) TLS 1.3 implementation in Java, suitable and intended for use in a QUIC implementation.
/*
* Copyright © 2019, 2020, 2021, 2022, 2023 Peter Doornbosch
*
* This file is part of Agent15, an implementation of TLS 1.3 in Java.
*
* Agent15 is free software: you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by the
* Free Software Foundation, either version 3 of the License, or (at your option)
* any later version.
*
* Agent15 is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for
* more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see .
*/
package net.luminis.tls.extension;
import net.luminis.tls.alert.DecodeErrorException;
import net.luminis.tls.TlsConstants;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
/**
* The TLS supported groups extension.
* See https://tools.ietf.org/html/rfc8446#section-4.2.7
*/
public class SupportedGroupsExtension extends Extension {
private final List namedGroups = new ArrayList<>();
public SupportedGroupsExtension(TlsConstants.NamedGroup namedGroup) {
namedGroups.add(namedGroup);
}
public SupportedGroupsExtension(ByteBuffer buffer) throws DecodeErrorException {
int extensionDataLength = parseExtensionHeader(buffer, TlsConstants.ExtensionType.supported_groups, 2 + 2);
int namedGroupsLength = buffer.getShort();
if (extensionDataLength != 2 + namedGroupsLength) {
throw new DecodeErrorException("inconsistent length");
}
if (namedGroupsLength % 2 != 0) {
throw new DecodeErrorException("invalid group length");
}
for (int i = 0; i < namedGroupsLength; i += 2) {
int namedGroupBytes = buffer.getShort() % 0xffff;
TlsConstants.NamedGroup namedGroup = Arrays.stream(TlsConstants.NamedGroup.values())
.filter(item -> item.value == namedGroupBytes)
.findFirst()
.orElseThrow(() -> new DecodeErrorException("invalid group value"));
namedGroups.add(namedGroup);
}
}
@Override
public byte[] getBytes() {
int extensionLength = 2 + namedGroups.size() * 2;
ByteBuffer buffer = ByteBuffer.allocate(4 + extensionLength);
buffer.putShort(TlsConstants.ExtensionType.supported_groups.value);
buffer.putShort((short) extensionLength); // Extension data length (in bytes)
buffer.putShort((short) (namedGroups.size() * 2));
for (TlsConstants.NamedGroup namedGroup: namedGroups) {
buffer.putShort(namedGroup.value);
}
return buffer.array();
}
@Override
public String toString() {
return "SupportedGroupsExtension" + namedGroups;
}
public List getNamedGroups() {
return namedGroups;
}
}