Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Change WPS ConnectionContext.States to a Dict<string, BinaryData> #25504

Merged
26 commits merged into from
Dec 6, 2021
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion eng/Packages.Data.props
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,7 @@
<PackageReference Update="Azure.Messaging.EventHubs" Version="5.6.2" />
<PackageReference Update="Azure.Messaging.EventGrid" Version="4.7.0" />
<PackageReference Update="Azure.Messaging.ServiceBus" Version="7.5.0" />
<PackageReference Update="Azure.Messaging.WebPubSub" Version="1.0.0-beta.2" />
<PackageReference Update="Azure.Messaging.WebPubSub" Version="1.0.0" />
<PackageReference Update="Azure.Identity" Version="1.5.0" />
<PackageReference Update="Azure.Security.KeyVault.Secrets" Version="4.2.0" />
<PackageReference Update="Azure.Security.KeyVault.Keys" Version="4.2.0" />
Expand Down
Original file line number Diff line number Diff line change
@@ -1,14 +1,12 @@
# Release History

## 1.1.0-beta.1 (Unreleased)

### Features Added

### Breaking Changes
## 1.1.0 (2021-11-24)

### Bugs Fixed
- Changed the `ConnectionContext`'s `ConnectionStates` to correctly serialize as proper JSON when used with JavaScript.

### Other Changes
### Breaking Changes
- JavaScript developers using `request.connectionContext.states` no longer need to `JSON.parse(...)` its values. The values are already valid JSON.

## 1.0.0 (2021-11-09)

Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

using System;
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
using System.IO;
using System.Text.Json;
using Newtonsoft.Json.Linq;

namespace System
Expand Down Expand Up @@ -37,5 +39,23 @@ public static object Convert(this BinaryData data, Type targetType)

return null;
}

public static BinaryData FromObjectAsJsonExtended<T>(T item, JsonSerializerOptions? options = null)
{
try
{
return BinaryData.FromObjectAsJson(item, options);
}
catch (NotSupportedException e)
{
return item is IJEnumerable<JToken> ?
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
BinaryData.FromString(item.ToString()) :
throw e;
}
catch (Exception)
{
throw;
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

using System;
using System.Collections.Generic;
using Microsoft.Azure.WebPubSub.Common;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;

using SystemJson = System.Text.Json;

namespace Microsoft.Azure.WebJobs.Extensions.WebPubSub
{
/// <summary>
/// JsonConverter works for Functions JavaScript language object converters.
/// </summary>
internal class ConnectionStatesNewtonsoftConverter : JsonConverter<IReadOnlyDictionary<string, BinaryData>>
{
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
public override IReadOnlyDictionary<string, BinaryData> ReadJson(JsonReader reader, Type objectType, IReadOnlyDictionary<string, BinaryData> existingValue, bool hasExistingValue, JsonSerializer serializer)
tg-msft marked this conversation as resolved.
Show resolved Hide resolved
{
var rawJson = JObject.Load(reader).ToString();

return LoadWithSystemJson(rawJson);
}

public override void WriteJson(JsonWriter writer, IReadOnlyDictionary<string, BinaryData> value, JsonSerializer serializer)
{
writer.WriteStartObject();
if (value != null)
{
foreach (KeyValuePair<string, BinaryData> pair in value)
{
writer.WritePropertyName(pair.Key);
writer.WriteRawValue(pair.Value.ToString());
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
}
}
writer.WriteEndObject();
}

private static IReadOnlyDictionary<string, BinaryData> LoadWithSystemJson(string rawJson)
{
var dic = new Dictionary<string, BinaryData>();
var element = SystemJson.JsonDocument.Parse(rawJson).RootElement;
foreach (var elementInfo in element.EnumerateObject())
{
dic.Add(elementInfo.Name, BinaryData.FromString(elementInfo.Value.GetRawText()));
}
return dic;
}
}
}

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -21,12 +21,10 @@ public override WebPubSubContext ReadJson(JsonReader reader, Type objectType, We
public override void WriteJson(JsonWriter writer, WebPubSubContext value, JsonSerializer serializer)
{
serializer.Converters.Add(new HttpResponseMessageJsonConverter());
// Request is using System.Json, use string as bridge to convert.
var request = ConvertString(value.Request);
var jobj = new JObject();
if (value.Request != null)
{
jobj.Add(new JProperty(WebPubSubContext.RequestPropertyName, JObject.Parse(request)));
jobj.Add(new JProperty(WebPubSubContext.RequestPropertyName, JObject.FromObject(value.Request, serializer)));
}
if (value.Response != null)
{
Expand All @@ -37,16 +35,5 @@ public override void WriteJson(JsonWriter writer, WebPubSubContext value, JsonSe
jobj.Add(WebPubSubContext.IsPreflightPropertyName, value.IsPreflight);
jobj.WriteTo(writer);
}

private static string ConvertString(WebPubSubEventRequest request) =>
request switch
{
ConnectedEventRequest connected => SystemJson.JsonSerializer.Serialize<ConnectedEventRequest>(connected),
ConnectEventRequest connect => SystemJson.JsonSerializer.Serialize<ConnectEventRequest>(connect),
UserEventRequest userEvent => SystemJson.JsonSerializer.Serialize<UserEventRequest>(userEvent),
DisconnectedEventRequest disconnected => SystemJson.JsonSerializer.Serialize<DisconnectedEventRequest>(disconnected),
PreflightRequest validation => SystemJson.JsonSerializer.Serialize<PreflightRequest>(validation),
_ => null
};
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -160,7 +160,7 @@ internal static void RegisterJsonConverter()
{
new StringEnumConverter(),
new BinaryDataJsonConverter(),
new JsonElementJsonConverter(),
new ConnectionStatesNewtonsoftConverter(),
},
ContractResolver = new CamelCasePropertyNamesContractResolver()
};
Expand Down
Original file line number Diff line number Diff line change
@@ -1,17 +1,21 @@
<Project Sdk="Microsoft.NET.Sdk">
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<TargetFrameworks>$(RequiredTargetFrameworks)</TargetFrameworks>
<PackageId>Microsoft.Azure.WebJobs.Extensions.WebPubSub</PackageId>
<PackageTags>Azure, WebPubSub</PackageTags>
<Description>Azure Functions extension for the WebPubSub service</Description>
<Version>1.1.0-beta.1</Version>
<Version>1.1.0</Version>
<!--The ApiCompatVersion is managed automatically and should not generally be modified manually.-->
<ApiCompatVersion>1.0.0</ApiCompatVersion>
<NoWarn>$(NoWarn);AZC0001;CS8632;CA1056;CA2227</NoWarn>
<IsExtensionClientLibrary>true</IsExtensionClientLibrary>
</PropertyGroup>

<ItemGroup>
<Compile Include="..\..\Microsoft.Azure.WebPubSub.Common\src\Shared\ConnectionStatesConverter.cs" Link="Bindings\ConnectionStatesConverter.cs" />
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
</ItemGroup>

<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.Http" />
<PackageReference Include="Microsoft.Azure.WebJobs" />
Expand All @@ -20,6 +24,11 @@

<ItemGroup>
<ProjectReference Include="..\..\Microsoft.Azure.WebPubSub.Common\src\Microsoft.Azure.WebPubSub.Common.csproj" />
<ProjectReference Include="..\..\Azure.Messaging.WebPubSub\src\Azure.Messaging.WebPubSub.csproj" />

<!--
TODO: Changing to a PackageReference since we only want to depend on the already GA'ed version of WebPubSub. Change back after release.
<ProjectReference Include="..\..\Azure.Messaging.WebPubSub\src\Azure.Messaging.WebPubSub.csproj" />
-->
<PackageReference Include="Azure.Messaging.WebPubSub" />
</ItemGroup>
</Project>
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@

using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.IO;
using System.Linq;
using System.Net.Http.Headers;
Expand All @@ -21,6 +22,7 @@ namespace Microsoft.Azure.WebJobs.Extensions.WebPubSub
/// </summary>
internal static class WebPubSubRequestExtensions
{
private static JsonSerializerOptions _innerSerializer => CreateSystemTestJsonSerializer();
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
/// <summary>
/// Parse request to system/user type ServiceRequest.
/// </summary>
Expand Down Expand Up @@ -151,10 +153,10 @@ internal static Dictionary<string, object> DecodeConnectionStates(this string co
if (!string.IsNullOrEmpty(connectionStates))
{
var states = new Dictionary<string, object>();
var rawData = JsonSerializer.Deserialize<Dictionary<string, JsonElement>>(Convert.FromBase64String(connectionStates));
foreach (var state in rawData)
var rawValue = JsonSerializer.Deserialize<IReadOnlyDictionary<string, BinaryData>>(Convert.FromBase64String(connectionStates), _innerSerializer);
foreach (var item in rawValue)
{
states.Add(state.Key, state.Value);
states.Add(item.Key, item.Value);
}
return states;
}
Expand Down Expand Up @@ -194,7 +196,8 @@ internal static Dictionary<string,object> UpdateStates(this WebPubSubConnectionC

internal static string EncodeConnectionStates(this Dictionary<string, object> value)
{
return Convert.ToBase64String(Encoding.UTF8.GetBytes(JsonSerializer.Serialize(value)));
IReadOnlyDictionary<string, BinaryData> readOnlyDic = value.ToDictionary(x => x.Key, y => y.Value is BinaryData data ? data : BinaryDataExtensions.FromObjectAsJsonExtended(y.Value));
return Convert.ToBase64String(Encoding.UTF8.GetBytes(JsonSerializer.Serialize(readOnlyDic, _innerSerializer)));
}

private static bool TryParseCloudEvents(this HttpRequest request, out WebPubSubConnectionContext connectionContext)
Expand Down Expand Up @@ -298,5 +301,12 @@ private static WebPubSubDataType GetDataType(this string mediaType) =>
Constants.ContentTypes.JsonContentType => WebPubSubDataType.Json,
_ => throw new ArgumentException($"Invalid content type: {mediaType}")
};

private static JsonSerializerOptions CreateSystemTestJsonSerializer()
{
var options = new JsonSerializerOptions();
options.Converters.Add(new ConnectionStatesConverter());
return options;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,10 @@
using System.Net.Http.Headers;
using System.Reflection;
using System.Text.Json;
using System.Text.Json.Serialization;

using Microsoft.Azure.WebPubSub.Common;

using NewtonsoftJson = Newtonsoft.Json;

namespace Microsoft.Azure.WebJobs.Extensions.WebPubSub
{
internal static class Utilities
Expand Down Expand Up @@ -120,7 +120,7 @@ public static HttpResponseMessage BuildValidResponse(
{
if (needConvert)
{
var states = GetStatesFromJson(converted, originStr);
var states = GetStatesFromJson(converted);
var mergedStates = context.UpdateStates(states);
return BuildConnectEventResponse(originStr, mergedStates);
}
Expand All @@ -134,7 +134,7 @@ public static HttpResponseMessage BuildValidResponse(
{
if (needConvert)
{
var states = GetStatesFromJson(converted, originStr);
var states = GetStatesFromJson(converted);
var mergedStates = context.UpdateStates(states);
return BuildUserEventResponse(JsonSerializer.Deserialize<UserEventResponse>(originStr), mergedStates);
}
Expand Down Expand Up @@ -218,23 +218,23 @@ public static bool IsValidationRequest(this HttpRequestMessage req, out List<str
return false;
}

private static Dictionary<string, object> GetStatesFromJson(JsonDocument converted, string originStr)
private static Dictionary<string, object> GetStatesFromJson(JsonDocument converted)
{
var states = new Dictionary<string, object>();
if (converted.RootElement.TryGetProperty("states", out var val))
{
if (val.ValueKind == JsonValueKind.Object)
{
return JsonSerializer.Deserialize<StatesEntity>(originStr).States;
var rawJson = val.GetRawText();
var strongType = NewtonsoftJson.JsonConvert.DeserializeObject<IReadOnlyDictionary<string, BinaryData>>(rawJson);
JialinXin marked this conversation as resolved.
Show resolved Hide resolved
foreach (var item in strongType)
{
states.Add(item.Key, item.Value);
}
}
}
// We don't support clear states for JS
return new Dictionary<string, object>();
}

private sealed class StatesEntity
{
[JsonPropertyName("states")]
public Dictionary<string, object> States { get; set; }
return states;
}
}
}
Loading