2017-09-13 18:14:07 +02:00
|
|
|
/*
|
|
|
|
* 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.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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Unique call id. Can be null in JSON_RPC v2.0, but xapi disallows it.
|
|
|
|
/// </summary>
|
|
|
|
[JsonProperty("id", Required = Required.Always)]
|
|
|
|
public int Id { get; private set; }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// The API function to call.
|
|
|
|
/// </summary>
|
|
|
|
[JsonProperty("method", Required = Required.Always)]
|
|
|
|
public string Method { get; private set; }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Any parameters, optional in JSON-RPC v2.0.
|
|
|
|
/// </summary>
|
|
|
|
[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<T>
|
|
|
|
{
|
2018-03-20 14:57:04 +01:00
|
|
|
[JsonProperty("id", Required = Required.AllowNull)] public int Id = 0;
|
2017-09-13 18:14:07 +02:00
|
|
|
|
2017-11-23 16:01:05 +01:00
|
|
|
[JsonProperty("result", Required = Required.Default)] public T Result = default(T);
|
2017-09-13 18:14:07 +02:00
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
{
|
|
|
|
return JsonConvert.SerializeObject(this, Formatting.Indented);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class JsonResponseV1<T> : JsonResponse<T>
|
|
|
|
{
|
2018-02-27 17:35:27 +01:00
|
|
|
[JsonProperty("error", Required = Required.AllowNull)] public JToken Error = null;
|
2017-09-13 18:14:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
internal class JsonResponseV2<T> : JsonResponse<T>
|
|
|
|
{
|
2017-11-23 16:01:05 +01:00
|
|
|
[JsonProperty("error", Required = Required.DisallowNull)] public JsonResponseV2Error Error = null;
|
2017-09-13 18:14:07 +02:00
|
|
|
|
2017-11-23 16:01:05 +01:00
|
|
|
[JsonProperty("jsonrpc", Required = Required.Always)] public string JsonRpc = null;
|
2017-09-13 18:14:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
internal class JsonResponseV2Error
|
|
|
|
{
|
2017-11-23 16:01:05 +01:00
|
|
|
[JsonProperty("code", Required = Required.Always)] public int Code = 0;
|
2017-09-13 18:14:07 +02:00
|
|
|
|
2017-11-23 16:01:05 +01:00
|
|
|
[JsonProperty("message", Required = Required.Always)] public string Message = null;
|
2017-09-13 18:14:07 +02:00
|
|
|
|
2017-11-23 16:01:05 +01:00
|
|
|
[JsonProperty("data", Required = Required.Default)] public JToken Data = null;
|
2017-09-13 18:14:07 +02:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// 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.
|
|
|
|
/// </summary>
|
|
|
|
public event Action<string> 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<object>(callName, parameters, serializer);
|
|
|
|
}
|
|
|
|
|
|
|
|
private T Rpc<T>(string callName, JToken parameters, JsonSerializer serializer)
|
|
|
|
{
|
|
|
|
// Note that the following method handles an overflow condition by wrapping;
|
|
|
|
// if the _globalId reaches Int32.MaxValue, _globalId + 1 starts over from
|
|
|
|
// Int32.MinValue and no exception is thrown.
|
|
|
|
var id = Interlocked.Increment(ref _globalId);
|
|
|
|
|
|
|
|
JsonRequest request = JsonRequest.Create(JsonRpcVersion, id, callName, parameters);
|
|
|
|
var webRequest = (HttpWebRequest)WebRequest.Create(JsonRpcUrl);
|
|
|
|
webRequest.Method = "POST";
|
|
|
|
webRequest.ContentType = "application/json";
|
|
|
|
webRequest.Accept = "application/json";
|
|
|
|
if (Timeout > 0)
|
|
|
|
webRequest.Timeout = Timeout;
|
|
|
|
if (WebProxy != null)
|
|
|
|
webRequest.Proxy = WebProxy;
|
|
|
|
if (KeepAlive)
|
|
|
|
webRequest.KeepAlive = KeepAlive;
|
|
|
|
if (!string.IsNullOrEmpty(UserAgent))
|
|
|
|
webRequest.UserAgent = UserAgent;
|
|
|
|
if (!string.IsNullOrEmpty(ConnectionGroupName))
|
|
|
|
webRequest.ConnectionGroupName = ConnectionGroupName;
|
|
|
|
|
|
|
|
// for performance reasons it's preferable to deserialize directly
|
|
|
|
// from the Stream rather than allocating strings inbetween
|
|
|
|
// therefore the latter will be done only in DEBUG mode
|
2018-03-20 14:57:04 +01:00
|
|
|
|
2017-11-30 16:34:36 +01:00
|
|
|
#if DEBUG
|
2018-03-20 14:57:04 +01:00
|
|
|
var settings = CreateSettings(serializer.Converters);
|
2017-11-30 16:34:36 +01:00
|
|
|
#endif
|
2017-09-13 18:14:07 +02:00
|
|
|
|
|
|
|
using (var str = webRequest.GetRequestStream())
|
|
|
|
using (var sw = new StreamWriter(str))
|
|
|
|
{
|
|
|
|
#if DEBUG
|
|
|
|
string jsonReq = JsonConvert.SerializeObject(request, settings);
|
|
|
|
if (RequestEvent != null)
|
|
|
|
RequestEvent(jsonReq);
|
|
|
|
sw.Write(jsonReq);
|
|
|
|
#else
|
|
|
|
if (RequestEvent != null)
|
|
|
|
RequestEvent(callName);
|
|
|
|
serializer.Serialize(sw, request);
|
|
|
|
#endif
|
|
|
|
sw.Flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
using (var webResponse = (HttpWebResponse)webRequest.GetResponse())
|
|
|
|
{
|
|
|
|
if (webResponse.StatusCode != HttpStatusCode.OK)
|
|
|
|
throw new WebException(webResponse.StatusCode.ToString());
|
|
|
|
|
|
|
|
using (var str = webResponse.GetResponseStream())
|
|
|
|
{
|
|
|
|
if (str == null)
|
|
|
|
throw new WebException();
|
|
|
|
|
|
|
|
using (var responseReader = new StreamReader(str))
|
|
|
|
{
|
|
|
|
switch (JsonRpcVersion)
|
|
|
|
{
|
|
|
|
case JsonRpcVersion.v2:
|
|
|
|
#if DEBUG
|
|
|
|
string json2 = responseReader.ReadToEnd();
|
|
|
|
var res2 = JsonConvert.DeserializeObject<JsonResponseV2<T>>(json2, settings);
|
|
|
|
#else
|
|
|
|
var res2 = (JsonResponseV2<T>)serializer.Deserialize(responseReader, typeof(JsonResponseV2<T>));
|
|
|
|
#endif
|
|
|
|
if (res2.Error != null)
|
|
|
|
{
|
|
|
|
var descr = new List<string> {res2.Error.Message};
|
|
|
|
descr.AddRange(res2.Error.Data.ToObject<string[]>());
|
|
|
|
throw new Failure(descr);
|
|
|
|
}
|
|
|
|
return res2.Result;
|
|
|
|
default:
|
|
|
|
#if DEBUG
|
|
|
|
string json1 = responseReader.ReadToEnd();
|
|
|
|
var res1 = JsonConvert.DeserializeObject<JsonResponseV1<T>>(json1, settings);
|
|
|
|
#else
|
|
|
|
var res1 = (JsonResponseV1<T>)serializer.Deserialize(responseReader, typeof(JsonResponseV1<T>));
|
|
|
|
#endif
|
|
|
|
if (res1.Error != null)
|
2018-02-27 17:35:27 +01:00
|
|
|
{
|
|
|
|
var errorArray = res1.Error.ToObject<string[]>();
|
|
|
|
if (errorArray != null)
|
|
|
|
throw new Failure(errorArray);
|
|
|
|
}
|
2017-09-13 18:14:07 +02:00
|
|
|
return res1.Result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-03-20 14:57:04 +01:00
|
|
|
|
|
|
|
private JsonSerializerSettings CreateSettings(IList<JsonConverter> converters)
|
|
|
|
{
|
|
|
|
return new JsonSerializerSettings
|
|
|
|
{
|
|
|
|
#if DEBUG
|
|
|
|
Formatting = Formatting.Indented,
|
|
|
|
#endif
|
|
|
|
Converters = converters,
|
|
|
|
DateParseHandling = DateParseHandling.None
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
private JsonSerializer CreateSerializer(IList<JsonConverter> converters)
|
|
|
|
{
|
|
|
|
var settings = CreateSettings(converters);
|
|
|
|
return JsonSerializer.Create(settings);
|
|
|
|
}
|
2017-09-13 18:14:07 +02:00
|
|
|
}
|
|
|
|
}
|