blob: c9f2b881f6fada916e767861141b3e114fe76aec [file] [log] [blame]
#region License
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
#endregion
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
using System.Text.Json;
using System.Threading;
using System.Threading.Tasks;
using Gremlin.Net.Driver;
using Gremlin.Net.Driver.Messages;
namespace Gremlin.Net.Structure.IO.GraphSON
{
/// <summary>
/// Serializes data to and from Gremlin Server in GraphSON format.
/// </summary>
public abstract class GraphSONMessageSerializer : IMessageSerializer
{
private static readonly JsonSerializerOptions JsonDeserializingOptions = new JsonSerializerOptions
{PropertyNamingPolicy = JsonNamingPolicy.CamelCase};
private readonly string _mimeType;
private readonly GraphSONReader _graphSONReader;
private readonly GraphSONWriter _graphSONWriter;
/// <summary>
/// Initializes a new instance of the <see cref="GraphSONMessageSerializer" /> class.
/// </summary>
/// <param name="mimeType">The MIME type supported by this serializer.</param>
/// <param name="graphSONReader">The <see cref="GraphSONReader"/> used to deserialize from GraphSON.</param>
/// <param name="graphSonWriter">The <see cref="GraphSONWriter"/> used to serialize to GraphSON.</param>
protected GraphSONMessageSerializer(string mimeType, GraphSONReader graphSONReader,
GraphSONWriter graphSonWriter)
{
_mimeType = mimeType;
_graphSONReader = graphSONReader;
_graphSONWriter = graphSonWriter;
}
/// <inheritdoc />
public virtual Task<byte[]> SerializeMessageAsync(RequestMessage requestMessage,
CancellationToken cancellationToken = default)
{
if (cancellationToken.CanBeCanceled) cancellationToken.ThrowIfCancellationRequested();
var graphSONMessage = _graphSONWriter.WriteObject(requestMessage);
return Task.FromResult(Encoding.UTF8.GetBytes(MessageWithHeader(graphSONMessage)));
}
private string MessageWithHeader(string messageContent)
{
return $"{(char) _mimeType.Length}{_mimeType}{messageContent}";
}
/// <inheritdoc />
public virtual Task<ResponseMessage<List<object>>?> DeserializeMessageAsync(byte[] message,
CancellationToken cancellationToken = default)
{
if (message == null) throw new ArgumentNullException(nameof(message));
if (cancellationToken.CanBeCanceled) cancellationToken.ThrowIfCancellationRequested();
if (message.Length == 0) return Task.FromResult<ResponseMessage<List<object>>?>(null);
var reader = new Utf8JsonReader(message);
var responseMessage =
JsonSerializer.Deserialize<ResponseMessage<JsonElement>>(ref reader, JsonDeserializingOptions);
if (responseMessage == null) return Task.FromResult<ResponseMessage<List<object>>?>(null);
var data = _graphSONReader.ToObject(responseMessage.Result.Data);
return Task.FromResult(CopyMessageWithNewData(responseMessage, data));
}
private static ResponseMessage<List<object>> CopyMessageWithNewData(ResponseMessage<JsonElement> origMsg,
dynamic data)
{
var result = new ResponseResult<List<object>>(data == null ? null : new List<object>(data),
origMsg.Result.Meta);
return new ResponseMessage<List<object>>(origMsg.RequestId, origMsg.Status, result);
}
}
}