blob: afb8b720e2bf0072bce7e60525a8ce0c3a42b617 [file] [log] [blame] [edit]
// <copyright file="Response.cs" company="Selenium Committers">
// Licensed to the Software Freedom Conservancy (SFC) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The SFC 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.
// </copyright>
using OpenQA.Selenium.Internal;
using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Globalization;
using System.Text.Json;
using System.Text.Json.Serialization;
#nullable enable
namespace OpenQA.Selenium
{
/// <summary>
/// Handles reponses from the browser
/// </summary>
public class Response
{
private static readonly JsonSerializerOptions s_jsonSerializerOptions = new()
{
TypeInfoResolver = ResponseJsonSerializerContext.Default,
Converters = { new ResponseValueJsonConverter() } // we still need it to make `Object` as `Dictionary`
};
/// <summary>
/// Initializes a new instance of the <see cref="Response"/> class
/// </summary>
[Obsolete("Set all values using the Response(string, object, WebDriverResult) constructor instead. This constructor will be removed in Selenium 4.30")]
public Response()
{
}
/// <summary>
/// Initializes a new instance of the <see cref="Response"/> class
/// </summary>
/// <param name="sessionId">Session ID in use</param>
[Obsolete("Set all values using the Response(string, object, WebDriverResult) constructor instead. This constructor will be removed in Selenium 4.30")]
public Response(SessionId? sessionId)
{
this.SessionId = sessionId?.ToString();
}
/// <summary>
/// Initializes a new instance of the <see cref="Response"/> class
/// </summary>
/// <param name="sessionId">The Session ID in use, if any.</param>
/// <param name="value">The JSON payload of the response.</param>
/// <param name="status">The WebDriver result status of the response.</param>
public Response(string? sessionId, object? value, WebDriverResult status)
{
#pragma warning disable CS0618 // Type or member is obsolete
this.SessionId = sessionId;
this.Value = value;
this.Status = status;
#pragma warning restore CS0618 // Type or member is obsolete
}
/// <summary>
/// Returns a new <see cref="Response"/> from a JSON-encoded string.
/// </summary>
/// <param name="value">The JSON string to deserialize into a <see cref="Response"/>.</param>
/// <returns>A <see cref="Response"/> object described by the JSON string.</returns>
/// <exception cref="ArgumentNullException">If <paramref name="value"/> is <see langword="null"/>.</exception>
/// <exception cref="JsonException">If <paramref name="value"/> is not a valid JSON object.</exception>
public static Response FromJson(string value)
{
Dictionary<string, object?> rawResponse = JsonSerializer.Deserialize<Dictionary<string, object?>>(value, s_jsonSerializerOptions)
?? throw new WebDriverException("JSON success response returned \"null\" value");
object? contents;
string? sessionId = null;
if (rawResponse.TryGetValue("sessionId", out object? s) && s is not null)
{
sessionId = s.ToString();
}
if (rawResponse.TryGetValue("value", out object? valueObj))
{
contents = valueObj;
}
else
{
// If the returned object does *not* have a "value" property
// the response value should be the entirety of the response.
// TODO: Remove this if statement altogether; there should
// never be a spec-compliant response that does not contain a
// value property.
// Special-case for the new session command, where the "capabilities"
// property of the response is the actual value we're interested in.
if (rawResponse.TryGetValue("capabilities", out object? capabilities))
{
contents = capabilities;
}
else
{
contents = rawResponse;
}
}
if (contents is Dictionary<string, object?> valueDictionary)
{
// Special case code for the new session command. If the response contains
// sessionId and capabilities properties, fix up the session ID and value members.
if (valueDictionary.TryGetValue("sessionId", out object? session))
{
sessionId = session.ToString();
if (valueDictionary.TryGetValue("capabilities", out object? capabilities))
{
contents = capabilities;
}
else
{
contents = valueDictionary["value"];
}
}
}
return new Response(sessionId, contents, WebDriverResult.Success);
}
/// <summary>
/// Gets or sets the value from JSON.
/// </summary>
public object? Value
{
get;
[Obsolete("The Response type will be immutable and this setter will be removed in Selenium 4.30")]
set;
}
/// <summary>
/// Gets or sets the session ID.
/// </summary>
public string? SessionId
{
get;
[Obsolete("The Response type will be immutable and this setter will be removed in Selenium 4.30")]
set;
}
/// <summary>
/// Gets or sets the status value of the response.
/// </summary>
public WebDriverResult Status
{
get;
[Obsolete("The Response type will be immutable and this setter will be removed in Selenium 4.30")]
set;
}
/// <summary>
/// Returns a new <see cref="Response"/> from a JSON-encoded string.
/// </summary>
/// <param name="value">The JSON string to deserialize into a <see cref="Response"/>.</param>
/// <returns>A <see cref="Response"/> object described by the JSON string.</returns>
/// <exception cref="ArgumentNullException">If <paramref name="value"/> is <see langword="null"/>.</exception>
/// <exception cref="JsonException">If <paramref name="value"/> is not a valid JSON object.</exception>
/// <exception cref="WebDriverException">If the JSON dictionary is not in the expected state, per spec.</exception>
public static Response FromErrorJson(string value)
{
Dictionary<string, object?> deserializedResponse = JsonSerializer.Deserialize<Dictionary<string, object?>>(value, s_jsonSerializerOptions)
?? throw new WebDriverException("JSON error response returned \"null\" value");
if (!deserializedResponse.TryGetValue("value", out object? valueObject))
{
throw new WebDriverException($"The 'value' property was not found in the response:{Environment.NewLine}{value}");
}
if (valueObject is not Dictionary<string, object?> valueDictionary)
{
throw new WebDriverException($"The 'value' property is not a dictionary of <string, object>{Environment.NewLine}{value}");
}
if (!valueDictionary.TryGetValue("error", out object? errorObject))
{
throw new WebDriverException($"The 'value > error' property was not found in the response:{Environment.NewLine}{value}");
}
if (errorObject is not string errorString)
{
throw new WebDriverException($"The 'value > error' property is not a string{Environment.NewLine}{value}");
}
WebDriverResult status = WebDriverError.ResultFromError(errorString);
return new Response(sessionId: null, valueDictionary, status);
}
/// <summary>
/// Returns this object as a JSON-encoded string.
/// </summary>
/// <returns>A JSON-encoded string representing this <see cref="Response"/> object.</returns>
public string ToJson()
{
return JsonSerializer.Serialize(this);
}
/// <summary>
/// Throws if <see cref="Value"/> is <see langword="null"/>.
/// </summary>
/// <exception cref="WebDriverException">If <see cref="Value"/> is <see langword="null"/>.</exception>
[MemberNotNull(nameof(Value))]
internal Response EnsureValueIsNotNull()
{
if (Value is null)
{
throw new WebDriverException("Response from remote end doesn't have $.Value property");
}
return this;
}
/// <summary>
/// Returns the object as a string.
/// </summary>
/// <returns>A string with the Session ID, status value, and the value from JSON.</returns>
public override string ToString()
{
return string.Format(CultureInfo.InvariantCulture, "({0} {1}: {2})", this.SessionId, this.Status, this.Value);
}
}
[JsonSerializable(typeof(Dictionary<string, object>))]
internal sealed partial class ResponseJsonSerializerContext : JsonSerializerContext;
}