You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
105 lines
3.2 KiB
105 lines
3.2 KiB
using System;
|
|
using System.Collections;
|
|
using System.IO;
|
|
|
|
using Org.BouncyCastle.Utilities;
|
|
using Org.BouncyCastle.Utilities.IO;
|
|
|
|
namespace Org.BouncyCastle.Crypto.Tls
|
|
{
|
|
public class ServerNameList
|
|
{
|
|
protected readonly IList mServerNameList;
|
|
|
|
/**
|
|
* @param serverNameList an {@link IList} of {@link ServerName}.
|
|
*/
|
|
public ServerNameList(IList serverNameList)
|
|
{
|
|
if (serverNameList == null)
|
|
throw new ArgumentNullException("serverNameList");
|
|
|
|
this.mServerNameList = serverNameList;
|
|
}
|
|
|
|
/**
|
|
* @return an {@link IList} of {@link ServerName}.
|
|
*/
|
|
public virtual IList ServerNames
|
|
{
|
|
get { return mServerNameList; }
|
|
}
|
|
|
|
/**
|
|
* Encode this {@link ServerNameList} to a {@link Stream}.
|
|
*
|
|
* @param output
|
|
* the {@link Stream} to encode to.
|
|
* @throws IOException
|
|
*/
|
|
public virtual void Encode(Stream output)
|
|
{
|
|
MemoryStream buf = new MemoryStream();
|
|
|
|
byte[] nameTypesSeen = TlsUtilities.EmptyBytes;
|
|
foreach (ServerName entry in ServerNames)
|
|
{
|
|
nameTypesSeen = CheckNameType(nameTypesSeen, entry.NameType);
|
|
if (nameTypesSeen == null)
|
|
throw new TlsFatalAlert(AlertDescription.internal_error);
|
|
|
|
entry.Encode(buf);
|
|
}
|
|
|
|
TlsUtilities.CheckUint16(buf.Length);
|
|
TlsUtilities.WriteUint16((int)buf.Length, output);
|
|
Streams.WriteBufTo(buf, output);
|
|
}
|
|
|
|
/**
|
|
* Parse a {@link ServerNameList} from a {@link Stream}.
|
|
*
|
|
* @param input
|
|
* the {@link Stream} to parse from.
|
|
* @return a {@link ServerNameList} object.
|
|
* @throws IOException
|
|
*/
|
|
public static ServerNameList Parse(Stream input)
|
|
{
|
|
int length = TlsUtilities.ReadUint16(input);
|
|
if (length < 1)
|
|
throw new TlsFatalAlert(AlertDescription.decode_error);
|
|
|
|
byte[] data = TlsUtilities.ReadFully(length, input);
|
|
|
|
MemoryStream buf = new MemoryStream(data, false);
|
|
|
|
byte[] nameTypesSeen = TlsUtilities.EmptyBytes;
|
|
IList server_name_list = Platform.CreateArrayList();
|
|
while (buf.Position < buf.Length)
|
|
{
|
|
ServerName entry = ServerName.Parse(buf);
|
|
|
|
nameTypesSeen = CheckNameType(nameTypesSeen, entry.NameType);
|
|
if (nameTypesSeen == null)
|
|
throw new TlsFatalAlert(AlertDescription.illegal_parameter);
|
|
|
|
server_name_list.Add(entry);
|
|
}
|
|
|
|
return new ServerNameList(server_name_list);
|
|
}
|
|
|
|
private static byte[] CheckNameType(byte[] nameTypesSeen, byte nameType)
|
|
{
|
|
/*
|
|
* RFC 6066 3. The ServerNameList MUST NOT contain more than one name of the same
|
|
* name_type.
|
|
*/
|
|
if (!NameType.IsValid(nameType) || Arrays.Contains(nameTypesSeen, nameType))
|
|
return null;
|
|
|
|
return Arrays.Append(nameTypesSeen, nameType);
|
|
}
|
|
}
|
|
}
|
|
|