/* 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;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Xml;
using XenAdmin.Network;
using XenAPI;
namespace XenAdmin.Actions
{
///
/// Performs a scan of a CSLG server for storage pools given a storage system.
///
public class SrCslgStoragePoolScanAction : SrCslgScanAction
{
private static readonly log4net.ILog Log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
private readonly string _storageSystemId;
private ReadOnlyCollection _cslgStoragePools;
///
/// Initializes a new instance of the class.
///
/// The connection.
/// The hostname.
/// The username.
/// The password secret.
/// The storage system id.
public SrCslgStoragePoolScanAction(IXenConnection connection, string hostname, string username, string passwordSecret, string storageSystemId)
: base(connection, hostname, username, passwordSecret)
{
Util.ThrowIfStringParameterNullOrEmpty(storageSystemId, "storageSystemId");
_storageSystemId = storageSystemId;
}
private readonly string _adapterId;
public SrCslgStoragePoolScanAction(IXenConnection connection, string hostname, string username, string password, string storagesystemid, string adapterid)
: base(connection, hostname, username, password)
{
_storageSystemId = storagesystemid;
_adapterId = adapterid;
}
///
/// Gets the CSLG system storages. Returns null before the action has been run.
///
/// The CSLG system storages.
public ReadOnlyCollection CslgStoragePools
{
get
{
return _cslgStoragePools;
}
}
private List ParseStoragePoolXml(String xml)
{
List output = new List();
XmlDocument doc = new XmlDocument();
doc.LoadXml(xml);
foreach (XmlNode storagePoolInfo in doc.GetElementsByTagName("csl__storagePoolInfo"))
{
string displayName = GetXmlNodeInnerText(storagePoolInfo, "displayName").Trim();
string storagePoolId = GetXmlNodeInnerText(storagePoolInfo, "storagePoolId").Trim();
List raidTypes = new List();
foreach (var raid in GetXmlChildNodeInnerTexts(storagePoolInfo, "provisioningOptions/supportedRaidTypes/raidType"))
{
raidTypes.Add(raid.Trim());
}
List provisioningTypes = new List();
string parentStoragePoolId = GetXmlNodeInnerText(storagePoolInfo, "parentStoragePoolId");
foreach (string innerText in GetXmlChildNodeInnerTexts(storagePoolInfo, "provisioningOptions/supportedProvisioningTypes/provisioningType"))
{
var trimmedInnerText = innerText.Trim();
// CVSM-3277: bridge returns "FULL" for some versions but "THICK" is required.
if (trimmedInnerText == "FULL" || trimmedInnerText == "THICK")
{
provisioningTypes.Add(new CslgParameter("THICK", Messages.NEWSR_CSLG_THICK_PROVISIONING));
}
else if (trimmedInnerText == "THIN")
{
provisioningTypes.Add(new CslgParameter(trimmedInnerText, Messages.NEWSR_CSLG_THIN_PROVISIONING));
}
}
if (raidTypes.Count == 0)
{
raidTypes.Add("RAID_NONE");
}
if (provisioningTypes.Count == 0)
{
provisioningTypes.Add(new CslgParameter(null, Messages.NEWSR_CSLG_DEFAULT_PROVISIONING));
}
uint capacity = 0;
uint usedSpace = 0;
try
{
capacity = UInt32.Parse(GetXmlNodeInnerText(storagePoolInfo, "sizeInMB").Trim());
usedSpace = capacity - UInt32.Parse(GetXmlNodeInnerText(storagePoolInfo, "freeSpaceInMB").Trim());
}
catch { }
StorageLinkPool storageLinkPool = new StorageLinkPool(storagePoolId, displayName, parentStoragePoolId, _storageSystemId, capacity, usedSpace,
(StorageLinkEnums.RaidType)Enum.Parse(typeof(StorageLinkEnums.RaidType), raidTypes[0].ToUpper()),
(StorageLinkEnums.ProvisioningType)Enum.Parse(typeof(StorageLinkEnums.ProvisioningType), provisioningTypes[0].Name.ToUpper()));
output.Add(new CslgStoragePool(displayName, storagePoolId, raidTypes, provisioningTypes, !string.IsNullOrEmpty(parentStoragePoolId), storageLinkPool));
}
return output;
}
///
/// Runs this instance.
///
protected override void Run()
{
Dictionary dconf = GetAuthenticationDeviceConfig();
dconf["storageSystemId"] = _storageSystemId;
if (!string.IsNullOrEmpty(_adapterId))
dconf["adapterid"] = _adapterId;
Log.DebugFormat("Attempting to find pools on {0}.", _storageSystemId);
RunProbe(dconf);
if (!string.IsNullOrEmpty(Result))
_cslgStoragePools = new ReadOnlyCollection(ParseStoragePoolXml(Util.GetContentsOfValueNode(Result)));
}
private static List GetRaidTypes(StorageLinkEnums.RaidType raidType)
{
List output = new List();
foreach (StorageLinkEnums.RaidType r in Enum.GetValues(typeof(StorageLinkEnums.RaidType)))
{
if ((raidType & r) != 0)
{
output.Add(r.ToString());
}
}
if (output.Count == 0)
{
output.Add("RAID_NONE");
}
return output;
}
private static List GetProvisioningTypes(StorageLinkEnums.ProvisioningType provisioningType)
{
List output = new List();
if ((provisioningType & StorageLinkEnums.ProvisioningType.THICK) != 0)
{
output.Add(new CslgParameter("THICK", Messages.NEWSR_CSLG_THICK_PROVISIONING));
}
if ((provisioningType & StorageLinkEnums.ProvisioningType.THIN) != 0)
{
output.Add(new CslgParameter("THIN", Messages.NEWSR_CSLG_THIN_PROVISIONING));
}
return output;
}
}
#region CslgStoragePool class
///
/// Represents of storoage pool on a storage system on a CSLG server.
///
public class CslgStoragePool
{
private readonly string _displayName;
private readonly string _storagePoolId;
private readonly ReadOnlyCollection _raidTypes;
private readonly ReadOnlyCollection _provisioningTypes;
private readonly bool _hasParent;
///
/// Initializes a new instance of the class.
///
/// The display name.
/// The storage pool id.
/// The raid types.
/// The provisioning types.
/// if set to true if this pool has a parent pool.
public CslgStoragePool(string displayName, string storagePoolId, IEnumerable raidTypes, IEnumerable provisioningTypes, bool hasParent, StorageLinkPool storageLinkPool)
{
Util.ThrowIfStringParameterNullOrEmpty(displayName, "displayName");
Util.ThrowIfStringParameterNullOrEmpty(storagePoolId, "storagePoolId");
_displayName = displayName;
_storagePoolId = storagePoolId;
_raidTypes = new ReadOnlyCollection(new List(raidTypes));
_provisioningTypes = new ReadOnlyCollection(new List(provisioningTypes));
_hasParent = hasParent;
StorageLinkPool = storageLinkPool;
}
public StorageLinkPool StorageLinkPool { get; private set; }
///
/// Gets the available raid types for this pool.
///
/// The available raid types for this pool.
public ReadOnlyCollection RaidTypes
{
get
{
return _raidTypes;
}
}
///
/// Gets the available provisioning types for this pool.
///
/// The available provisioning types for this pool.
public ReadOnlyCollection ProvisioningTypes
{
get
{
return _provisioningTypes;
}
}
///
/// Gets the storage pool id.
///
/// The storage pool id.
public string StoragePoolId
{
get
{
return _storagePoolId;
}
}
public override string ToString()
{
return _displayName;
}
///
/// Gets a value indicating whether this pool has a parent pool.
///
///
/// true if this instance has a parent pool; otherwise, false.
///
public bool HasParent
{
get { return _hasParent; }
}
}
#endregion
}