/* 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;
using System.Collections.Generic;
using System.Xml;
namespace XenAdmin
{
public enum RoundingBehaviour
{
Up, Down, Nearest, None
}
///
/// Miscellaneous utility functions
///
public static class Util
{
public const long BINARY_KILO = 1024;
public const long BINARY_MEGA = BINARY_KILO * BINARY_KILO;
public const long BINARY_GIGA = BINARY_KILO * BINARY_MEGA;
public const long BINARY_TERA = BINARY_KILO * BINARY_GIGA;
public const long DEC_KILO = 1000;
public const long DEC_MEGA = DEC_KILO * DEC_KILO;
public const long DEC_GIGA = DEC_KILO * DEC_MEGA;
public const long DEC_TERA = DEC_KILO * DEC_GIGA;
///
/// The default iSCSI filer port.
///
public const UInt16 DEFAULT_ISCSI_PORT = 3260;
public static string MemorySizeStringSuitableUnits(double bytes, bool showPoint0Decimal)
{
if (bytes >= 1 * BINARY_GIGA)
{
string format = Messages.VAL_GB_ONE_DECIMAL;
int dp = 1;
double valGB = bytes / BINARY_GIGA;
if (valGB > 100)
{
dp = 0;
format = Messages.VAL_GB;
}
if(!showPoint0Decimal)
{
format = Messages.VAL_GB;
}
return string.Format(format, Math.Round(valGB, dp, MidpointRounding.AwayFromZero));
}
else if (bytes >= 1 * BINARY_MEGA)
{
return string.Format(Messages.VAL_MB, Math.Round(bytes / BINARY_MEGA));
}
else if (bytes >= 1 * BINARY_KILO)
{
return string.Format(Messages.VAL_KB, Math.Round(bytes / BINARY_KILO));
}
if(bytes == 0)
{
return bytes.ToString();
}
else
{
return string.Format(Messages.VAL_B, bytes);
}
}
///
/// Returns the string in suitable units and when the number of bytes is 0 the units are the ones given in formatStringWhenZero.
/// E.g. : if bytes = 0, showPoint0Decimal = false, formatStringWhenZero = Messages.VAL_MB, then the function will return "0 MB".
///
///
///
///
///
public static string MemorySizeStringSuitableUnits(double bytes, bool showPoint0Decimal, string formatStringWhenZero)
{
if(bytes == 0)
{
return string.Format(formatStringWhenZero, bytes);
}
else
{
return MemorySizeStringSuitableUnits(bytes, showPoint0Decimal);
}
}
public static string DataRateString(double bytesPerSec)
{
string unit;
string value = ByteSizeString(bytesPerSec, 1, true, out unit);
return string.Format(Messages.VAL_FORMAT, value, unit);
}
public static string DataRateValue(double bytesPerSec, out string unit)
{
return ByteSizeString(bytesPerSec, 1, true, out unit);
}
public static string DiskSizeString(ulong bytes)
{
string unit;
string value = ByteSizeString(bytes, 1, false, out unit);
return string.Format(Messages.VAL_FORMAT, value, unit);
}
public static string DiskSizeString(long bytes)
{
return DiskSizeString(bytes, 1);
}
public static string DiskSizeString(long bytes, int dp)
{
ulong abs = (ulong)Math.Abs(bytes);
string unit;
string value = ByteSizeString(abs, dp, false, out unit);
return string.Format(Messages.VAL_FORMAT, value, unit);
}
public static string DiskSizeStringWithoutUnits(long bytes)
{
string unit;
return ByteSizeString(bytes, 1, false, out unit);
}
public static string MemorySizeStringVariousUnits(double bytes)
{
string unit;
string value = ByteSizeString(bytes, 0, false, out unit);
return string.Format(Messages.VAL_FORMAT, value, unit);
}
public static string MemorySizeValueVariousUnits(double bytes, out string unit)
{
return ByteSizeString(bytes, 0, false, out unit);
}
private static string ByteSizeString(double bytes, int decPlaces, bool isRate, out string unit)
{
if (bytes >= BINARY_GIGA)
{
unit = isRate ? Messages.VAL_GIGRATE : Messages.VAL_GIGB;
return Math.Round(bytes / BINARY_GIGA, decPlaces).ToString();
}
if (bytes >= BINARY_MEGA)
{
unit = isRate ? Messages.VAL_MEGRATE : Messages.VAL_MEGB;
return Math.Round(bytes / BINARY_MEGA, decPlaces).ToString();
}
if (bytes >= BINARY_KILO)
{
unit = isRate ? Messages.VAL_KILRATE : Messages.VAL_KILB;
return Math.Round(bytes / BINARY_KILO, decPlaces).ToString();
}
unit = isRate ? Messages.VAL_RATE : Messages.VAL_BYTE;
return bytes.ToString();
}
///
/// nothing actually is measured in nanoseconds, we actually output microseconds
///
public static string NanoSecondsString(double t)
{
string unit;
string value = NanoSecondsValue(t, out unit);
return string.Format(Messages.VAL_FORMAT_SECONDS, value, unit);
}
public static string NanoSecondsValue(double t, out string unit)
{
if (t >= DEC_GIGA)
{
unit = Messages.VAL_SEC;
return (t / DEC_GIGA).ToString("0");
}
if (t >= DEC_MEGA)
{
unit = Messages.VAL_MILSEC;
return (t / DEC_MEGA).ToString("0");
}
if (t >= DEC_KILO)
{
unit = Messages.VAL_MICSEC;
return (t / DEC_KILO).ToString("0");
}
unit = Messages.VAL_NANOSEC;
return t.ToString("0");
}
public static string MilliWattString(double t)
{
string unit;
string value = MilliWattValue(t, out unit);
return string.Format(Messages.VAL_FORMAT, value, unit);
}
public static string MilliWattValue(double t, out string unit)
{
if (t >= DEC_GIGA)
{
unit = Messages.VAL_MWATT;
return (t / DEC_GIGA).ToString("0");
}
if (t >= DEC_MEGA)
{
unit = Messages.VAL_KILOWATT;
return (t / DEC_MEGA).ToString("0");
}
if (t >= DEC_KILO)
{
unit = Messages.VAL_WATT;
return (t / DEC_KILO).ToString("0");
}
unit = Messages.VAL_MILWATT;
return t.ToString("0");
}
public static double ToGB(double bytes, int dp, RoundingBehaviour rounding)
{
double value = (double)bytes / BINARY_GIGA;
int decimalsAdjustment = (int)Math.Pow(10, dp);
switch (rounding)
{
case RoundingBehaviour.None:
return value;
case RoundingBehaviour.Down:
return (Math.Floor(value * decimalsAdjustment) / decimalsAdjustment);
case RoundingBehaviour.Up:
return (Math.Ceiling(value * decimalsAdjustment) / decimalsAdjustment);
default: // case RoundingBehaviour.Nearest:
return (Math.Round(value, 1, MidpointRounding.AwayFromZero));
}
}
public static double ToMB(double bytes, RoundingBehaviour rounding)
{
switch (rounding)
{
case RoundingBehaviour.None:
return bytes / BINARY_MEGA;
case RoundingBehaviour.Down:
return Math.Floor(bytes / BINARY_MEGA);
case RoundingBehaviour.Up:
return Math.Ceiling(bytes / BINARY_MEGA);
default: // case RoundingBehaviour.Nearest:
return Math.Round(bytes / BINARY_MEGA, MidpointRounding.AwayFromZero);
}
}
public static double CorrectRoundingErrors(double amount)
{
// Special case to cope with choosing an amount that's a multiple of 0.1G but not 0.5G --
// sending it to the server as the nearest byte and getting it back later --
// and finding it's fractionally changed, messing up our spinner permitted ranges.
double amountRounded = ToGB(amount, 1, RoundingBehaviour.Nearest) * BINARY_GIGA;
double roundingDiff = amountRounded - amount;
if (roundingDiff > -1.0 && roundingDiff < 1.0) // within 1 byte: although I think it will always be positive in the case we want to correct
return amountRounded;
else
return amount;
}
public static double ToUnixTime(DateTime time)
{
TimeSpan diff = time - new DateTime(1970, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc);
return diff.TotalSeconds;
}
public static DateTime FromUnixTime(double time)
{
DateTime bootTime = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);
return bootTime.AddSeconds(time);
}
public static string TimeString(long t)
{
if (t >= 120)
return string.Format(Messages.TIME_MINUTES, t / 60);
if (t > 0)
return string.Format(Messages.TIME_SECONDS, t);
return Messages.TIME_NEGLIGIBLE;
}
public static string TimeRangeString(long t1, long t2)
{
return t1 > 60 && t2 > 60 ?
string.Format(Messages.TIME_RANGE_MINUTES, t1 / 60, t2 / 60) :
string.Format(Messages.TIME_RANGE_SECONDS, t1, t2);
}
internal static string LThanTime(long max)
{
return string.Format(Messages.LESS_THAN, TimeString(max));
}
internal static string GThanSize(long min)
{
return string.Format(Messages.GREATER_THAN, DiskSizeString(min));
}
internal static string GThanTime(long min)
{
return string.Format(Messages.GREATER_THAN, TimeString(min));
}
internal static string LThanSize(long max)
{
return string.Format(Messages.LESS_THAN, DiskSizeString(max));
}
public static string CountsPerSecondString(double p)
{
return string.Format(Messages.VAL_FORMAT, p, Messages.COUNTS_PER_SEC_UNIT);
}
public static string MegaHertzString(double t)
{
string unit;
string value = MegaHertzValue(t, out unit);
return string.Format(Messages.VAL_FORMAT, value, unit);
}
public static string MegaHertzValue(double t, out string unit)
{
return MegaHertzValue(t, 4, out unit);
}
///
/// Converts the input value from MHz to GHz if needed, rounding it to the specified decimal places
///
private static string MegaHertzValue(double t, int decPlaces, out string unit)
{
if (t >= DEC_KILO)
{
unit = Messages.VAL_GIGHZ;
return Math.Round(t / DEC_KILO, decPlaces).ToString();
}
unit = Messages.VAL_MEGHZ;
return Math.Round(t, decPlaces).ToString();
}
public static string PercentageString(double fraction)
{
return string.Format("{0}%", (fraction * 100d).ToString("0.0"));
}
public static void ThrowIfParameterNull(object obj, string name)
{
if (name == null)
{
throw new ArgumentNullException("name");
}
if (name.Length == 0)
{
ThrowBecauseZeroLength("name");
}
if (obj == null)
{
throw new ArgumentNullException(name);
}
}
public static void ThrowIfStringParameterNullOrEmpty(string value, string name)
{
ThrowIfParameterNull(value, name);
if (value.Length == 0)
{
ThrowBecauseZeroLength(name);
}
}
public static void ThrowIfEnumerableParameterNullOrEmpty(IEnumerable value, string name)
{
ThrowIfParameterNull(value, name);
#pragma warning disable 0168
foreach (object _ in value)
{
return;
}
#pragma warning restore 0168
ThrowBecauseZeroLength(name);
}
private static void ThrowBecauseZeroLength(string name)
{
throw new ArgumentException(string.Format("{0} cannot have 0 length.", name), name);
}
///
/// Loads the specified non-generic IEnumerable into a generic List<T>.
///
/// The type to convert each element to
/// The input non-generic IEnumerable.
/// Generic List<T>
public static List PopulateList(IEnumerable input)
{
ThrowIfParameterNull(input, "input");
List output = new List();
foreach (T t in input)
{
output.Add(t);
}
return output;
}
///
/// Gets a List that represents the specified IEnumerable. If the input isn't a List then one gets created by passing the input into
/// List's constructor. If the input is already a List then it is returned directly.
///
///
/// The input.
/// A list for the specified enumerable
public static List GetList(IEnumerable input)
{
if (input == null)
{
return null;
}
var list = input as List;
if (list != null)
{
return list;
}
return new List(input);
}
///
/// Matches 1-65535 inclusive.
///
/// The string to be parsed
/// True if the specified string contains only a valid port, otherwise false.
public static bool IsValidPort(string s)
{
int port;
if (!int.TryParse(s, out port))
return false;
return 0 < port && port <= 65535;
}
public static string GetXmlNodeInnerText(XmlNode node, string xPath)
{
ThrowIfParameterNull(node, "node");
ThrowIfStringParameterNullOrEmpty(xPath, "xPath");
XmlNodeList nodes = node.SelectNodes(xPath);
if (nodes == null || nodes.Count == 0)
{
throw new InvalidOperationException("Node not found: " + xPath);
}
return nodes[0].InnerText;
}
///
/// Get the first node with name 'value' and returns its innerText. Used for gettings results of CGSL async actions.
///
/// The XML.
/// The contents of the first node with name 'value'.
public static string GetContentsOfValueNode(string xml)
{
ThrowIfStringParameterNullOrEmpty(xml, "xml");
XmlDocument doc = new XmlDocument();
doc.LoadXml(xml);
// If we've got this from an async task result, then it will be wrapped
// in a element.
foreach (XmlNode node in doc.GetElementsByTagName("value"))
{
return node.InnerText;
}
return null;
}
}
}