diff --git a/src/StreamJsonRpc/JsonRpcEventSource.cs b/src/StreamJsonRpc/JsonRpcEventSource.cs index 7aff5585..0925407e 100644 --- a/src/StreamJsonRpc/JsonRpcEventSource.cs +++ b/src/StreamJsonRpc/JsonRpcEventSource.cs @@ -291,12 +291,22 @@ internal static string GetArgumentsString(JsonRpcRequest request) stringBuilder.Append('['); for (int i = 0; i < request.ArgumentCount; ++i) { - if (request.TryGetArgumentByNameOrIndex(null, i, null, out object? value)) + bool formatted = false; + try { - Format(stringBuilder, value); - stringBuilder.Append(", "); + if (request.TryGetArgumentByNameOrIndex(null, i, null, out object? value)) + { + Format(stringBuilder, value); + stringBuilder.Append(", "); + formatted = true; + } } - else + catch + { + // Swallow exceptions when deserializing args for ETW logging. It's never worth causing a functional failure. + } + + if (!formatted) { stringBuilder.Append(""); } @@ -317,14 +327,24 @@ internal static string GetArgumentsString(JsonRpcRequest request) stringBuilder.Append('{'); foreach (string key in request.ArgumentNames) { - if (request.TryGetArgumentByNameOrIndex(key, -1, null, out object? value)) + bool formatted = false; + try { - stringBuilder.Append(key); - stringBuilder.Append(": "); - Format(stringBuilder, value); - stringBuilder.Append(", "); + if (request.TryGetArgumentByNameOrIndex(key, -1, null, out object? value)) + { + stringBuilder.Append(key); + stringBuilder.Append(": "); + Format(stringBuilder, value); + stringBuilder.Append(", "); + formatted = true; + } } - else + catch + { + // Swallow exceptions when deserializing args for ETW logging. It's never worth causing a functional failure. + } + + if (!formatted) { stringBuilder.Append(""); } diff --git a/test/StreamJsonRpc.Tests/JsonRpcMessagePackLengthTests.cs b/test/StreamJsonRpc.Tests/JsonRpcMessagePackLengthTests.cs index 5f9eb84f..43b63064 100644 --- a/test/StreamJsonRpc.Tests/JsonRpcMessagePackLengthTests.cs +++ b/test/StreamJsonRpc.Tests/JsonRpcMessagePackLengthTests.cs @@ -29,6 +29,8 @@ internal interface IMessagePackServer Task ProgressUnionType(IProgress progress, CancellationToken cancellationToken); IAsyncEnumerable GetAsyncEnumerableOfUnionType(CancellationToken cancellationToken); + + Task IsExtensionArgNonNull(CustomExtensionType extensionValue); } [Fact] @@ -363,6 +365,25 @@ public async Task UnionType_AsAsyncEnumerableTypeArgument() Assert.IsType(actualItem); } + /// + /// Verifies that an argument that cannot be deserialized by the msgpack primitive formatter will not cause a failure. + /// + /// + /// This is a regression test for a bug where + /// verbose ETW tracing would fail to deserialize arguments with the primitive formatter that deserialize just fine for the actual method dispatch. + /// This test is effective because custom msgpack extensions cause the to throw an exception when deserializing. + /// + [Theory, PairwiseData] + public async Task VerboseLoggingDoesNotFailWhenArgsDoNotDeserializePrimitively(bool namedArguments) + { + var server = new MessagePackServer(); + this.serverRpc.AllowModificationWhileListening = true; + this.serverRpc.AddLocalRpcTarget(server); + var clientProxy = this.clientRpc.Attach(new JsonRpcProxyOptions { ServerRequiresNamedArguments = namedArguments }); + + Assert.True(await clientProxy.IsExtensionArgNonNull(new CustomExtensionType())); + } + protected override void InitializeFormattersAndHandlers(bool controlledFlushingClient) { this.serverMessageFormatter = new MessagePackFormatter(); @@ -370,7 +391,7 @@ protected override void InitializeFormattersAndHandlers(bool controlledFlushingC var options = MessagePackFormatter.DefaultUserDataSerializationOptions .WithResolver(CompositeResolver.Create( - new IMessagePackFormatter[] { new UnserializableTypeFormatter(), new TypeThrowsWhenDeserializedFormatter() }, + new IMessagePackFormatter[] { new UnserializableTypeFormatter(), new TypeThrowsWhenDeserializedFormatter(), new CustomExtensionFormatter() }, new IFormatterResolver[] { StandardResolverAllowPrivate.Instance })); ((MessagePackFormatter)this.serverMessageFormatter).SetMessagePackSerializerOptions(options); ((MessagePackFormatter)this.clientMessageFormatter).SetMessagePackSerializerOptions(options); @@ -392,6 +413,40 @@ public class UnionDerivedClass : UnionBaseClass { } + internal class CustomExtensionType + { + } + + private class CustomExtensionFormatter : IMessagePackFormatter + { + public CustomExtensionType? Deserialize(ref MessagePackReader reader, MessagePackSerializerOptions options) + { + if (reader.TryReadNil()) + { + return null; + } + + if (reader.ReadExtensionFormat() is { Header: { TypeCode: 1, Length: 0 } }) + { + return new(); + } + + throw new Exception("Unexpected extension header."); + } + + public void Serialize(ref MessagePackWriter writer, CustomExtensionType? value, MessagePackSerializerOptions options) + { + if (value is null) + { + writer.WriteNil(); + } + else + { + writer.WriteExtensionFormat(new ExtensionResult(1, default(Memory))); + } + } + } + private class UnserializableTypeFormatter : IMessagePackFormatter { public CustomSerializedType Deserialize(ref MessagePackReader reader, MessagePackSerializerOptions options) @@ -448,6 +503,8 @@ public async IAsyncEnumerable GetAsyncEnumerableOfUnionType([Enu await Task.Yield(); yield return new UnionDerivedClass(); } + + public Task IsExtensionArgNonNull(CustomExtensionType extensionValue) => Task.FromResult(extensionValue is not null); } private class DelayedFlushingHandler : LengthHeaderMessageHandler, IControlledFlushHandler