2015-07-10 14:36:26 +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:
|
|
|
|
|
*
|
|
|
|
|
* * Redistributions of source code must retain the above
|
|
|
|
|
* copyright notice, this list of conditions and the
|
|
|
|
|
* following disclaimer.
|
|
|
|
|
* * 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;
|
2015-07-01 13:06:19 +02:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Net;
|
2015-07-24 10:01:04 +02:00
|
|
|
|
using System.Runtime.Serialization;
|
2015-07-01 13:06:19 +02:00
|
|
|
|
using XenAPI;
|
|
|
|
|
using System.Web.Script.Serialization;
|
|
|
|
|
|
|
|
|
|
namespace XenAdmin.Actions
|
|
|
|
|
{
|
2015-07-29 09:44:41 +02:00
|
|
|
|
public class HealthCheckAuthenticationAction : AsyncAction
|
2015-07-01 13:06:19 +02:00
|
|
|
|
{
|
|
|
|
|
private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
|
|
|
|
|
|
|
|
|
|
private readonly string username;
|
|
|
|
|
private readonly string password;
|
|
|
|
|
|
|
|
|
|
private readonly long tokenExpiration;
|
|
|
|
|
|
|
|
|
|
private string uploadToken;
|
2015-09-03 12:49:53 +02:00
|
|
|
|
private string diagnosticToken;
|
2015-07-01 13:06:19 +02:00
|
|
|
|
|
|
|
|
|
private const string identityTokenUrl = "/auth/api/create_identity/";
|
|
|
|
|
private const string uploadGrantTokenUrl = "/feeds/api/create_grant/";
|
|
|
|
|
private const string uploadTokenUrl = "/feeds/api/create_upload/";
|
2015-08-27 01:30:02 +02:00
|
|
|
|
private const string diagnosticTokenUrl = "/diag_sdk/token_grant/";
|
2015-07-01 13:06:19 +02:00
|
|
|
|
|
2015-08-20 17:59:58 +02:00
|
|
|
|
private readonly string identityTokenDomainName = "https://cis.citrix.com";
|
|
|
|
|
private readonly string uploadGrantTokenDomainName = "https://rttf.citrix.com";
|
|
|
|
|
private readonly string uploadTokenDomainName = "https://rttf.citrix.com";
|
2016-03-16 11:00:23 +01:00
|
|
|
|
private readonly string diagnosticTokenDomainName = " https://cis.citrix.com";
|
2015-07-01 13:06:19 +02:00
|
|
|
|
|
2015-08-20 17:59:58 +02:00
|
|
|
|
private readonly string productKey = "1a2d94a4263cd016dd7a7d510bde87f058a0b75d";
|
2015-07-01 13:06:19 +02:00
|
|
|
|
|
2015-09-03 12:49:53 +02:00
|
|
|
|
public HealthCheckAuthenticationAction(string username, string password, long tokenExpiration, bool suppressHistory)
|
|
|
|
|
: base(null, Messages.ACTION_HEALTHCHECK_AUTHENTICATION, Messages.ACTION_HEALTHCHECK_AUTHENTICATION_PROGRESS, suppressHistory)
|
2015-07-01 13:06:19 +02:00
|
|
|
|
{
|
|
|
|
|
this.username = username;
|
|
|
|
|
this.password = password;
|
|
|
|
|
this.tokenExpiration = tokenExpiration;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-03 12:49:53 +02:00
|
|
|
|
public HealthCheckAuthenticationAction(string username, string password,
|
2015-08-27 01:30:02 +02:00
|
|
|
|
string identityTokenDomainName, string uploadGrantTokenDomainName, string uploadTokenDomainName, string diagnosticTokenDomainName,
|
2015-09-03 12:49:53 +02:00
|
|
|
|
string productKey, long tokenExpiration, bool suppressHistory)
|
|
|
|
|
: this(username, password, tokenExpiration, suppressHistory)
|
2015-07-01 13:06:19 +02:00
|
|
|
|
{
|
|
|
|
|
if (!string.IsNullOrEmpty(identityTokenDomainName))
|
|
|
|
|
this.identityTokenDomainName = identityTokenDomainName;
|
2015-08-27 01:30:02 +02:00
|
|
|
|
if (!string.IsNullOrEmpty(uploadGrantTokenDomainName))
|
2015-07-01 13:06:19 +02:00
|
|
|
|
this.uploadGrantTokenDomainName = uploadGrantTokenDomainName;
|
2015-08-27 01:30:02 +02:00
|
|
|
|
if (!string.IsNullOrEmpty(uploadTokenDomainName))
|
2015-07-01 13:06:19 +02:00
|
|
|
|
this.uploadTokenDomainName = uploadTokenDomainName;
|
2015-08-27 01:30:02 +02:00
|
|
|
|
if (!string.IsNullOrEmpty(diagnosticTokenDomainName))
|
|
|
|
|
this.diagnosticTokenDomainName = diagnosticTokenDomainName;
|
2015-07-15 13:21:07 +02:00
|
|
|
|
if (!string.IsNullOrEmpty(productKey))
|
|
|
|
|
this.productKey = productKey;
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void Run()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
string identityToken = GetIdentityToken();
|
|
|
|
|
string uploadGrantToken = GetUploadGrantToken(identityToken);
|
|
|
|
|
uploadToken = GetUploadToken(uploadGrantToken);
|
2015-09-03 12:49:53 +02:00
|
|
|
|
diagnosticToken = GetDiagnosticToken(identityToken);
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
2015-07-24 10:01:04 +02:00
|
|
|
|
catch (HealthCheckAuthenticationException)
|
|
|
|
|
{
|
|
|
|
|
throw;
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2015-07-24 10:01:04 +02:00
|
|
|
|
throw new HealthCheckAuthenticationException(e);
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
2015-07-01 04:35:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-07-01 13:06:19 +02:00
|
|
|
|
public string UploadToken
|
2015-07-01 04:35:59 +02:00
|
|
|
|
{
|
2015-07-01 13:06:19 +02:00
|
|
|
|
get { return uploadToken; }
|
2015-07-01 04:35:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-09-03 12:49:53 +02:00
|
|
|
|
public string DiagnosticToken
|
2015-07-06 18:02:55 +02:00
|
|
|
|
{
|
2015-09-03 12:49:53 +02:00
|
|
|
|
get { return diagnosticToken; }
|
2015-07-06 18:02:55 +02:00
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
|
|
|
|
|
private string GetIdentityToken()
|
|
|
|
|
{
|
|
|
|
|
var json = new JavaScriptSerializer().Serialize(new
|
|
|
|
|
{
|
|
|
|
|
username,
|
|
|
|
|
password
|
|
|
|
|
});
|
|
|
|
|
var urlString = string.Format("{0}{1}", identityTokenDomainName, identityTokenUrl);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
return GetToken(urlString, json, null);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
}
|
|
|
|
|
catch (WebException e)
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
log.InfoFormat("WebException while getting identity token from {0}. Exception Message: {1} ", identityTokenDomainName, e.Message);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
if (e.Status == WebExceptionStatus.ProtocolError && ((HttpWebResponse) e.Response).StatusCode == HttpStatusCode.Forbidden)
|
|
|
|
|
throw new HealthCheckAuthenticationException(Messages.HEALTH_CHECK_AUTHENTICATION_INVALID_CREDENTIALS, e);
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
log.InfoFormat("Exception while getting identity token from {0}. Exception Message: {1} ", identityTokenDomainName, e.Message);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetUploadGrantToken(string identityToken)
|
|
|
|
|
{
|
|
|
|
|
var json = new JavaScriptSerializer().Serialize(new
|
|
|
|
|
{
|
2015-08-11 10:07:59 +02:00
|
|
|
|
identity_token = identityToken
|
2015-07-01 13:06:19 +02:00
|
|
|
|
});
|
2015-08-11 10:07:59 +02:00
|
|
|
|
if (tokenExpiration != 0)
|
|
|
|
|
{
|
|
|
|
|
json = new JavaScriptSerializer().Serialize(new
|
|
|
|
|
{
|
|
|
|
|
identity_token = identityToken,
|
|
|
|
|
expiration = tokenExpiration
|
|
|
|
|
});
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
var urlString = string.Format("{0}{1}", uploadGrantTokenDomainName, uploadGrantTokenUrl);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
return GetToken(urlString, json, null);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
log.InfoFormat("Exception while getting upload grant token from {0}. Exception Message: {1} ", uploadGrantTokenDomainName, e.Message);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetUploadToken(string grantToken)
|
|
|
|
|
{
|
|
|
|
|
var json = new JavaScriptSerializer().Serialize(new
|
|
|
|
|
{
|
|
|
|
|
grant_token = grantToken,
|
|
|
|
|
product_key = productKey,
|
|
|
|
|
expiration = tokenExpiration
|
|
|
|
|
});
|
|
|
|
|
var urlString = string.Format("{0}{1}", uploadTokenDomainName, uploadTokenUrl);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
return GetToken(urlString, json, null);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
log.InfoFormat("Exception while getting upload token from {0}. Exception Message: {1} ", uploadTokenDomainName, e.Message);
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetDiagnosticToken(string identityToken)
|
|
|
|
|
{
|
|
|
|
|
var json = new JavaScriptSerializer().Serialize(new
|
|
|
|
|
{
|
|
|
|
|
agent = "XenServer",
|
2016-03-16 11:00:23 +01:00
|
|
|
|
max_age = tokenExpiration
|
2015-08-27 01:30:02 +02:00
|
|
|
|
});
|
|
|
|
|
var urlString = string.Format("{0}{1}", diagnosticTokenDomainName, diagnosticTokenUrl);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
return GetToken(urlString, json, "BT " + identityToken);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2015-08-27 01:30:02 +02:00
|
|
|
|
log.InfoFormat("Exception while getting diagnostic token from {0}. Exception Message: {1} ", diagnosticTokenDomainName, e.Message);
|
2015-07-24 10:01:04 +02:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 01:30:02 +02:00
|
|
|
|
private string GetToken(string urlString, string jsonParameters, string authorizationHeader)
|
2015-07-01 13:06:19 +02:00
|
|
|
|
{
|
|
|
|
|
var httpWebRequest = (HttpWebRequest) WebRequest.Create(urlString);
|
2015-08-27 01:30:02 +02:00
|
|
|
|
if (authorizationHeader != null)
|
|
|
|
|
{
|
|
|
|
|
httpWebRequest.Headers.Add("Authorization", authorizationHeader);
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
httpWebRequest.ContentType = "application/json";
|
|
|
|
|
httpWebRequest.Method = "POST";
|
|
|
|
|
|
|
|
|
|
using (var streamWriter = new StreamWriter(httpWebRequest.GetRequestStream()))
|
|
|
|
|
{
|
|
|
|
|
streamWriter.Write(jsonParameters);
|
|
|
|
|
}
|
|
|
|
|
string result;
|
|
|
|
|
var httpResponse = (HttpWebResponse) httpWebRequest.GetResponse();
|
|
|
|
|
using (var streamReader = new StreamReader(httpResponse.GetResponseStream()))
|
2015-07-01 04:35:59 +02:00
|
|
|
|
{
|
2015-07-01 13:06:19 +02:00
|
|
|
|
result = streamReader.ReadToEnd();
|
|
|
|
|
}
|
|
|
|
|
TemplateResponse response = new JavaScriptSerializer().Deserialize<TemplateResponse>(result);
|
|
|
|
|
return response.Token;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class TemplateResponse
|
|
|
|
|
{
|
|
|
|
|
public string Token { get; set; }
|
|
|
|
|
}
|
2015-07-24 10:01:04 +02:00
|
|
|
|
|
|
|
|
|
[Serializable]
|
|
|
|
|
public class HealthCheckAuthenticationException : Exception
|
|
|
|
|
{
|
|
|
|
|
public HealthCheckAuthenticationException() : base() { }
|
|
|
|
|
|
|
|
|
|
public HealthCheckAuthenticationException(string message) : base(message) { }
|
|
|
|
|
|
|
|
|
|
public HealthCheckAuthenticationException(string message, Exception exception) : base(message, exception) { }
|
|
|
|
|
|
|
|
|
|
public HealthCheckAuthenticationException(Exception exception) : base(Messages.HEALTH_CHECK_AUTHENTICATION_FAILED, exception) { }
|
|
|
|
|
|
|
|
|
|
public HealthCheckAuthenticationException(SerializationInfo serialinfo, StreamingContext context) : base(serialinfo, context) { }
|
|
|
|
|
}
|
2015-07-01 13:06:19 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|