Files

182 lines
8.2 KiB
C#
Raw Permalink Normal View History

2025-09-24 09:00:48 -04:00
using System;
using System.Numerics;
using System.Text;
2025-09-26 13:06:18 -04:00
using SabreTools.Data.Models.ASN1;
2026-03-18 16:18:10 -04:00
using SabreTools.ObjectIdentifier;
2025-09-24 09:00:48 -04:00
2025-09-26 13:15:55 -04:00
namespace SabreTools.Data.Extensions
2025-09-24 09:00:48 -04:00
{
public static class TypeLengthValueExtensions
2025-09-24 09:00:48 -04:00
{
/// <summary>
/// Format a TypeLengthValue as a string
/// </summary>
/// <param name="paddingLevel">Padding level of the item when formatting</param>
/// <returns>String representing the TypeLengthValue, if possible</returns>
public static string Format(this TypeLengthValue tlv, int paddingLevel = 0)
2025-09-24 09:00:48 -04:00
{
// Create the left-padding string
string padding = new(' ', paddingLevel);
// Create the string builder
var formatBuilder = new StringBuilder();
// Append the type
formatBuilder.Append($"{padding}Type: {tlv.Type}");
2025-09-24 09:22:23 -04:00
if (tlv.Type == ASN1Type.V_ASN1_EOC)
2025-09-24 09:00:48 -04:00
return formatBuilder.ToString();
// Append the length
formatBuilder.Append($", Length: {tlv.Length}");
if (tlv.Length == 0)
return formatBuilder.ToString();
// If we have a constructed type
#if NET20 || NET35
2025-09-24 09:22:23 -04:00
if ((tlv.Type & ASN1Type.V_ASN1_CONSTRUCTED) != 0)
2025-09-24 09:00:48 -04:00
#else
2025-09-24 09:22:23 -04:00
if (tlv.Type.HasFlag(ASN1Type.V_ASN1_CONSTRUCTED))
2025-09-24 09:00:48 -04:00
#endif
{
if (tlv.Value is not TypeLengthValue[] valueAsObjectArray)
2025-09-24 09:00:48 -04:00
{
formatBuilder.Append(", Value: [INVALID DATA TYPE]");
return formatBuilder.ToString();
}
formatBuilder.Append(", Value:\n");
for (int i = 0; i < valueAsObjectArray.Length; i++)
{
var child = valueAsObjectArray[i];
string childString = child.Format(paddingLevel + 1);
formatBuilder.Append($"{childString}\n");
}
return formatBuilder.ToString().TrimEnd('\n');
}
// Get the value as a byte array
if (tlv.Value is not byte[] valueAsByteArray)
{
formatBuilder.Append(", Value: [INVALID DATA TYPE]");
return formatBuilder.ToString();
}
else if (valueAsByteArray.Length == 0)
{
formatBuilder.Append(", Value: [NO DATA]");
return formatBuilder.ToString();
}
// If we have a primitive type
switch (tlv.Type)
{
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-boolean"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_BOOLEAN:
2025-09-24 09:00:48 -04:00
if (tlv.Length > 1)
formatBuilder.Append($" [Expected length of 1]");
else if (valueAsByteArray.Length > 1)
formatBuilder.Append($" [Expected value length of 1]");
bool booleanValue = valueAsByteArray[0] != 0x00;
formatBuilder.Append($", Value: {booleanValue}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-integer"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_INTEGER:
2025-09-24 09:00:48 -04:00
Array.Reverse(valueAsByteArray);
var integerValue = new BigInteger(valueAsByteArray);
formatBuilder.Append($", Value: {integerValue}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-bit-string"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_BIT_STRING:
2025-09-24 09:00:48 -04:00
// TODO: Read into a BitArray and print that out instead?
int unusedBits = valueAsByteArray[0];
if (unusedBits == 0)
formatBuilder.Append($", Value with {unusedBits} unused bits");
else
formatBuilder.Append($", Value with {unusedBits} unused bits: {BitConverter.ToString(valueAsByteArray, 1).Replace('-', ' ')}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-octet-string"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_OCTET_STRING:
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {BitConverter.ToString(valueAsByteArray).Replace('-', ' ')}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-object-identifier"/>
/// <see cref="http://snmpsharpnet.com/index.php/2009/03/02/ber-encoding-and-decoding-oid-values/"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_OBJECT:
2025-09-24 09:00:48 -04:00
// Derive array of values
ulong[] objectNodes = Parser.ParseDERIntoArray(valueAsByteArray, tlv.Length);
2025-09-24 09:00:48 -04:00
// Append the dot and modified OID-IRI notations
string? dotNotationString = Parser.ParseOIDToDotNotation(objectNodes);
string? oidIriString = Parser.ParseOIDToOIDIRINotation(objectNodes);
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {dotNotationString} ({oidIriString})");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-utf8string"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_UTF8STRING:
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {Encoding.UTF8.GetString(valueAsByteArray)}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-printablestring"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_PRINTABLESTRING:
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {Encoding.ASCII.GetString(valueAsByteArray)}");
break;
//case ASN1Type.V_ASN1_T61STRING:
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_TELETEXSTRING:
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {Encoding.ASCII.GetString(valueAsByteArray)}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-ia5string"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_IA5STRING:
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {Encoding.ASCII.GetString(valueAsByteArray)}");
break;
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_UTCTIME:
2025-09-24 09:00:48 -04:00
string utctimeString = Encoding.ASCII.GetString(valueAsByteArray);
if (DateTime.TryParse(utctimeString, out DateTime utctimeDateTime))
formatBuilder.Append($", Value: {utctimeDateTime:yyyy-MM-dd HH:mm:ss}");
else
formatBuilder.Append($", Value: {utctimeString}");
break;
/// <see href="https://learn.microsoft.com/en-us/windows/win32/seccertenroll/about-bmpstring"/>
2025-09-24 09:22:23 -04:00
case ASN1Type.V_ASN1_BMPSTRING:
2025-09-24 09:00:48 -04:00
formatBuilder.Append($", Value: {Encoding.Unicode.GetString(valueAsByteArray)}");
break;
2026-01-25 16:15:05 -05:00
// Unimplemented
case ASN1Type.V_ASN1_EOC:
case ASN1Type.V_ASN1_NULL:
case ASN1Type.V_ASN1_OBJECT_DESCRIPTOR:
case ASN1Type.V_ASN1_EXTERNAL:
case ASN1Type.V_ASN1_REAL:
case ASN1Type.V_ASN1_ENUMERATED:
case ASN1Type.V_ASN1_SEQUENCE:
case ASN1Type.V_ASN1_SET:
case ASN1Type.V_ASN1_NUMERICSTRING:
case ASN1Type.V_ASN1_VIDEOTEXSTRING:
case ASN1Type.V_ASN1_GENERALIZEDTIME:
case ASN1Type.V_ASN1_GRAPHICSTRING:
case ASN1Type.V_ASN1_ISO64STRING:
case ASN1Type.V_ASN1_GENERALSTRING:
case ASN1Type.V_ASN1_UNIVERSALSTRING:
case ASN1Type.V_ASN1_PRIMITIVE_TAG:
case ASN1Type.V_ASN1_CONSTRUCTED:
case ASN1Type.V_ASN1_APPLICATION:
case ASN1Type.V_ASN1_CONTEXT_SPECIFIC:
case ASN1Type.V_ASN1_PRIVATE:
2025-09-24 09:00:48 -04:00
default:
formatBuilder.Append($", Value: {BitConverter.ToString(valueAsByteArray).Replace('-', ' ')}");
break;
}
// Return the formatted string
return formatBuilder.ToString();
}
}
}