in src/log4net/Util/SystemStringFormat.cs [32:182]
public sealed class SystemStringFormat(IFormatProvider? provider, string format, params object?[]? args)
{
/// <summary>
/// Format
/// </summary>
public string Format { get; set; } = format;
/// <summary>
/// Args
/// </summary>
[System.Diagnostics.CodeAnalysis.SuppressMessage("Performance", "CA1819:Properties should not return arrays")]
public object?[]? Args { get; set; } = args;
/// <summary>
/// Format the string and arguments
/// </summary>
/// <returns>the formatted string</returns>
public override string? ToString() => StringFormat(provider, Format, Args);
/// <summary>
/// Replaces the format item in a specified <see cref="string"/> with the text equivalent
/// of the value of a corresponding <see cref="object"/> instance in a specified array.
/// A specified parameter supplies culture-specific formatting information.
/// </summary>
/// <param name="provider">An <see cref="System.IFormatProvider"/> that supplies culture-specific formatting information.</param>
/// <param name="format">A <see cref="string"/> containing zero or more format items.</param>
/// <param name="args">An <see cref="object"/> array containing zero or more objects to format.</param>
/// <returns>
/// A copy of format in which the format items have been replaced by the <see cref="string"/>
/// equivalent of the corresponding instances of <see cref="object"/> in args.
/// </returns>
/// <remarks>
/// <para>
/// This method does not throw exceptions. If an exception thrown while formatting the result the
/// exception and arguments are returned in the result string.
/// </para>
/// </remarks>
private static string? StringFormat(IFormatProvider? provider, string? format, params object?[]? args)
{
try
{
// The format is missing, log null value
if (format is null)
{
return null;
}
// The args are missing - should not happen unless we are called explicitly with a null array
if (args is null)
{
return format;
}
// Try to format the string
return string.Format(provider, format, args);
}
catch (Exception e) when (!e.IsFatal())
{
LogLog.Warn(_declaringType, $"Exception while rendering format [{format}]", e);
return StringFormatError(e, format, args);
}
}
/// <summary>
/// Process an error during StringFormat
/// </summary>
private static string StringFormatError(Exception formatException, string? format, object?[]? args)
{
try
{
var buf = new StringBuilder("<log4net.Error>", 100);
buf.Append("Exception during StringFormat: ").Append(formatException.Message);
buf.Append(" <format>").Append(format).Append("</format>");
buf.Append("<args>");
RenderArray(args, buf);
buf.Append("</args>");
buf.Append("</log4net.Error>");
return buf.ToString();
}
catch (Exception e) when (!e.IsFatal())
{
LogLog.Error(_declaringType, "INTERNAL ERROR during StringFormat error handling", e);
return "<log4net.Error>Exception during StringFormat. See Internal Log.</log4net.Error>";
}
}
/// <summary>
/// Dump the contents of an array into a string builder
/// </summary>
private static void RenderArray(Array? array, StringBuilder buffer)
{
if (array is null)
{
buffer.Append(SystemInfo.NullText);
}
else
{
if (array.Rank != 1)
{
buffer.Append(array);
}
else
{
buffer.Append('{');
int len = array.Length;
if (len > 0)
{
RenderObject(array.GetValue(0), buffer);
for (int i = 1; i < len; i++)
{
buffer.Append(", ");
RenderObject(array.GetValue(i), buffer);
}
}
buffer.Append('}');
}
}
}
/// <summary>
/// Dump an object to a string
/// </summary>
private static void RenderObject(object? obj, StringBuilder buffer)
{
if (obj is null)
{
buffer.Append(SystemInfo.NullText);
}
else
{
try
{
buffer.Append(obj);
}
catch (Exception e) when (!e.IsFatal())
{
buffer.Append("<Exception: ").Append(e.Message).Append('>');
}
}
}
/// <summary>
/// The fully qualified type of the SystemStringFormat class.
/// </summary>
/// <remarks>
/// Used by the internal logger to record the Type of the
/// log message.
/// </remarks>
private static readonly Type _declaringType = typeof(SystemStringFormat);
}