/*
* Copyright (c) Citrix Systems, Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1) Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2) Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
using System;
using System.Collections.Generic;
using System.IO;
using System.Net;
using System.Runtime.Remoting.Messaging;
using System.Threading;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
namespace XenAPI
{
public enum JsonRpcVersion
{
v1,
v2
}
internal abstract class JsonRequest
{
protected JsonRequest(int id, string method, JToken parameters)
{
this.Id = id;
this.Method = method;
this.Parameters = parameters;
}
public static JsonRequest Create(JsonRpcVersion jsonRpcVersion, int id, string method, JToken parameters)
{
switch (jsonRpcVersion)
{
case JsonRpcVersion.v2:
return new JsonRequestV2(id, method, parameters);
default:
return new JsonRequestV1(id, method, parameters);
}
}
///
/// Unique call id. Can be null in JSON_RPC v2.0, but xapi disallows it.
///
[JsonProperty("id", Required = Required.Always)]
public int Id { get; private set; }
///
/// The API function to call.
///
[JsonProperty("method", Required = Required.Always)]
public string Method { get; private set; }
///
/// Any parameters, optional in JSON-RPC v2.0.
///
[JsonProperty("params", Required = Required.Always)]
public JToken Parameters { get; private set; }
public override string ToString()
{
return JsonConvert.SerializeObject(this, Formatting.Indented);
}
}
internal class JsonRequestV1 : JsonRequest
{
public JsonRequestV1(int id, string method, JToken parameters)
: base(id, method, parameters)
{
}
}
internal class JsonRequestV2 : JsonRequest
{
public JsonRequestV2(int id, string method, JToken parameters)
: base(id, method, parameters)
{
}
[JsonProperty("jsonrpc", Required = Required.Always)]
public string JsonRPC
{
get { return "2.0"; }
}
}
internal abstract class JsonResponse
{
[JsonProperty("id", Required = Required.AllowNull)] public int Id;
[JsonProperty("result", Required = Required.Default)] public T Result;
public override string ToString()
{
return JsonConvert.SerializeObject(this, Formatting.Indented);
}
}
internal class JsonResponseV1 : JsonResponse
{
[JsonProperty("error", Required = Required.AllowNull)] public object Error;
}
internal class JsonResponseV2 : JsonResponse
{
[JsonProperty("error", Required = Required.DisallowNull)] public JsonResponseV2Error Error;
[JsonProperty("jsonrpc", Required = Required.Always)] public string JsonRpc;
}
internal class JsonResponseV2Error
{
[JsonProperty("code", Required = Required.Always)] public int Code;
[JsonProperty("message", Required = Required.Always)] public string Message;
[JsonProperty("data", Required = Required.Default)] public JToken Data;
public override string ToString()
{
return JsonConvert.SerializeObject(this, Formatting.Indented);
}
}
public partial class JsonRpcClient
{
private int _globalId;
public JsonRpcClient(string baseUrl)
{
Url = baseUrl;
JsonRpcUrl = new Uri(new Uri(baseUrl), "/jsonrpc").ToString();
JsonRpcVersion = JsonRpcVersion.v1;
}
///
/// Fired when a request has been serialized taking as a parameter the call name in RELEASE and the
/// Json string in DEBUG mode.
/// IMPORTANT: the latter may contain sensitive data, so handle it carefully.
///
public event Action RequestEvent;
public JsonRpcVersion JsonRpcVersion { get; set; }
public string UserAgent { get; set; }
public bool KeepAlive { get; set; }
public IWebProxy WebProxy { get; set; }
public int Timeout { get; set; }
public string ConnectionGroupName { get; set; }
public string Url { get; private set; }
public string JsonRpcUrl { get; private set; }
private void Rpc(string callName, JToken parameters, JsonSerializer serializer)
{
Rpc