GQ_Communicate/GQ_VR/Assets/Best HTTP/Source/SecureProtocol/asn1/Asn1Sequence.cs

285 lines
9.0 KiB
C#

#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
#pragma warning disable
using System;
using System.Collections.Generic;
using System.IO;
using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities;
using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Collections;
namespace BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1
{
public abstract class Asn1Sequence
: Asn1Object, IEnumerable<Asn1Encodable>
{
internal class Meta : Asn1UniversalType
{
internal static readonly Asn1UniversalType Instance = new Meta();
private Meta() : base(typeof(Asn1Sequence), Asn1Tags.Sequence) {}
internal override Asn1Object FromImplicitConstructed(Asn1Sequence sequence)
{
return sequence;
}
}
/**
* return an Asn1Sequence from the given object.
*
* @param obj the object we want converted.
* @exception ArgumentException if the object cannot be converted.
*/
public static Asn1Sequence GetInstance(object obj)
{
if (obj == null)
return null;
if (obj is Asn1Sequence asn1Sequence)
return asn1Sequence;
if (obj is IAsn1Convertible asn1Convertible)
{
Asn1Object asn1Object = asn1Convertible.ToAsn1Object();
if (asn1Object is Asn1Sequence converted)
return converted;
}
else if (obj is byte[] bytes)
{
try
{
return (Asn1Sequence)Meta.Instance.FromByteArray(bytes);
}
catch (IOException e)
{
throw new ArgumentException("failed to construct sequence from byte[]: " + e.Message);
}
}
throw new ArgumentException("illegal object in GetInstance: " + Org.BouncyCastle.Utilities.Platform.GetTypeName(obj), "obj");
}
/**
* Return an ASN1 sequence from a tagged object. There is a special
* case here, if an object appears to have been explicitly tagged on
* reading but we were expecting it to be implicitly tagged in the
* normal course of events it indicates that we lost the surrounding
* sequence - so we need to add it back (this will happen if the tagged
* object is a sequence that contains other sequences). If you are
* dealing with implicitly tagged sequences you really <b>should</b>
* be using this method.
*
* @param taggedObject the tagged object.
* @param declaredExplicit true if the object is meant to be explicitly tagged, false otherwise.
* @exception ArgumentException if the tagged object cannot be converted.
*/
public static Asn1Sequence GetInstance(Asn1TaggedObject taggedObject, bool declaredExplicit)
{
return (Asn1Sequence)Meta.Instance.GetContextInstance(taggedObject, declaredExplicit);
}
// NOTE: Only non-readonly to support LazyDLSequence
internal Asn1Encodable[] elements;
protected internal Asn1Sequence()
{
this.elements = Asn1EncodableVector.EmptyElements;
}
protected internal Asn1Sequence(Asn1Encodable element)
{
if (null == element)
throw new ArgumentNullException(nameof(element));
this.elements = new Asn1Encodable[]{ element };
}
protected internal Asn1Sequence(Asn1Encodable element1, Asn1Encodable element2)
{
if (null == element1)
throw new ArgumentNullException(nameof(element1));
if (null == element2)
throw new ArgumentNullException(nameof(element2));
this.elements = new Asn1Encodable[]{ element1, element2 };
}
protected internal Asn1Sequence(params Asn1Encodable[] elements)
{
if (Arrays.IsNullOrContainsNull(elements))
throw new NullReferenceException("'elements' cannot be null, or contain null");
this.elements = Asn1EncodableVector.CloneElements(elements);
}
internal Asn1Sequence(Asn1Encodable[] elements, bool clone)
{
this.elements = clone ? Asn1EncodableVector.CloneElements(elements) : elements;
}
protected internal Asn1Sequence(Asn1EncodableVector elementVector)
{
if (null == elementVector)
throw new ArgumentNullException("elementVector");
this.elements = elementVector.TakeElements();
}
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
public virtual IEnumerator<Asn1Encodable> GetEnumerator()
{
IEnumerable<Asn1Encodable> e = elements;
return e.GetEnumerator();
}
private class Asn1SequenceParserImpl
: Asn1SequenceParser
{
private readonly Asn1Sequence outer;
private readonly int max;
private int index;
public Asn1SequenceParserImpl(
Asn1Sequence outer)
{
this.outer = outer;
// NOTE: Call Count here to 'force' a LazyDerSequence
this.max = outer.Count;
}
public IAsn1Convertible ReadObject()
{
if (index == max)
return null;
Asn1Encodable obj = outer[index++];
if (obj is Asn1Sequence)
return ((Asn1Sequence)obj).Parser;
if (obj is Asn1Set)
return ((Asn1Set)obj).Parser;
// NB: Asn1OctetString implements Asn1OctetStringParser directly
// if (obj is Asn1OctetString)
// return ((Asn1OctetString)obj).Parser;
return obj;
}
public Asn1Object ToAsn1Object()
{
return outer;
}
}
public virtual Asn1SequenceParser Parser
{
get { return new Asn1SequenceParserImpl(this); }
}
/**
* return the object at the sequence position indicated by index.
*
* @param index the sequence number (starting at zero) of the object
* @return the object at the sequence position indicated by index.
*/
public virtual Asn1Encodable this[int index]
{
get { return elements[index]; }
}
public virtual int Count
{
get { return elements.Length; }
}
public virtual T[] MapElements<T>(Func<Asn1Encodable, T> func)
{
// NOTE: Call Count here to 'force' a LazyDerSequence
int count = Count;
T[] result = new T[count];
for (int i = 0; i < count; ++i)
{
result[i] = func(elements[i]);
}
return result;
}
public virtual Asn1Encodable[] ToArray()
{
return Asn1EncodableVector.CloneElements(elements);
}
protected override int Asn1GetHashCode()
{
// NOTE: Call Count here to 'force' a LazyDerSequence
int i = Count;
int hc = i + 1;
while (--i >= 0)
{
hc *= 257;
hc ^= elements[i].ToAsn1Object().CallAsn1GetHashCode();
}
return hc;
}
protected override bool Asn1Equals(Asn1Object asn1Object)
{
Asn1Sequence that = asn1Object as Asn1Sequence;
if (null == that)
return false;
// NOTE: Call Count here (on both) to 'force' a LazyDerSequence
int count = this.Count;
if (that.Count != count)
return false;
for (int i = 0; i < count; ++i)
{
Asn1Object o1 = this.elements[i].ToAsn1Object();
Asn1Object o2 = that.elements[i].ToAsn1Object();
if (!o1.Equals(o2))
return false;
}
return true;
}
public override string ToString()
{
return CollectionUtilities.ToString(elements);
}
// TODO[asn1] Preferably return an Asn1BitString[] (doesn't exist yet)
internal DerBitString[] GetConstructedBitStrings()
{
return MapElements(DerBitString.GetInstance);
}
internal Asn1OctetString[] GetConstructedOctetStrings()
{
return MapElements(Asn1OctetString.GetInstance);
}
// TODO[asn1] Preferably return an Asn1BitString (doesn't exist yet)
internal abstract DerBitString ToAsn1BitString();
// TODO[asn1] Preferably return an Asn1External (doesn't exist yet)
internal abstract DerExternal ToAsn1External();
internal abstract Asn1OctetString ToAsn1OctetString();
internal abstract Asn1Set ToAsn1Set();
}
}
#pragma warning restore
#endif