public sealed class SystemStringFormat()

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);
}