/* 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.Linq; using XenAPI; using System.IO; using System.Xml; using XenAdmin.Core; using System.Diagnostics; using System.Net; using System.Text; namespace XenAdmin.Actions { public class DownloadUpdatesXmlAction : AsyncAction { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); private const string XenCenterVersionsNode = "xencenterversions"; private const string XenServerVersionsNode = "serverversions"; private const string PatchesNode = "patches"; private const string ConflictingPatchesNode = "conflictingpatches"; private const string RequiredPatchesNode = "requiredpatches"; private const string ConflictingPatchNode = "conflictingpatch"; private const string RequiredPatchNode = "requiredpatch"; public List XenCenterVersions { get; private set; } public List XenServerVersions { get; private set; } public List XenServerPatches { get; private set; } public List XenServerVersionsForAutoCheck { get { if(_checkForServerVersion) { return XenServerVersions; } return new List(); } } private readonly bool _checkForXenCenter; private readonly bool _checkForServerVersion; private readonly bool _checkForPatches; private readonly string _checkForUpdatesUrl; private readonly string _userAgent; private readonly string _userAgentId; public DownloadUpdatesXmlAction(bool checkForXenCenter, bool checkForServerVersion, bool checkForPatches, string userAgent, string userAgentId, string checkForUpdatesUrl = null) : base(null, "_get_updates", "_get_updates", true) { Debug.Assert(checkForUpdatesUrl != null, "Parameter checkForUpdatesUrl should not be null. This class does not default its value anymore."); Debug.Assert(!string.IsNullOrWhiteSpace(userAgent) && !string.IsNullOrWhiteSpace(userAgentId)); XenServerPatches = new List(); XenServerVersions = new List(); XenCenterVersions = new List(); _checkForXenCenter = checkForXenCenter; _checkForServerVersion = checkForServerVersion; _checkForPatches = checkForPatches; _checkForUpdatesUrl = checkForUpdatesUrl; _userAgent = userAgent; _userAgentId = userAgentId; } protected override void Run() { this.Description = Messages.AVAILABLE_UPDATES_SEARCHING; XmlDocument xdoc = FetchCheckForUpdatesXml(_checkForUpdatesUrl); GetXenCenterVersions(xdoc); GetXenServerPatches(xdoc); GetXenServerVersions(xdoc); } private void GetXenCenterVersions(XmlDocument xdoc) { if (!_checkForXenCenter) return; foreach (XmlNode versions in xdoc.GetElementsByTagName(XenCenterVersionsNode)) { foreach (XmlNode version in versions.ChildNodes) { string version_lang = ""; string name = ""; bool latest = false; bool latest_cr = false; string url = ""; string timestamp = ""; foreach (XmlAttribute attrib in version.Attributes) { if (attrib.Name == "value") version_lang = attrib.Value; else if (attrib.Name == "name") name = attrib.Value; else if (attrib.Name == "latest") latest = attrib.Value.ToUpperInvariant() == bool.TrueString.ToUpperInvariant(); else if (attrib.Name == "latestcr") latest_cr = attrib.Value.ToUpperInvariant() == bool.TrueString.ToUpperInvariant(); else if (attrib.Name == "url") url = attrib.Value; else if (attrib.Name == "timestamp") timestamp = attrib.Value; } XenCenterVersions.Add(new XenCenterVersion(version_lang, name, latest, latest_cr, url, timestamp)); } } } private void GetXenServerPatches(XmlDocument xdoc) { if (!_checkForPatches) return; foreach (XmlNode versions in xdoc.GetElementsByTagName(PatchesNode)) { foreach (XmlNode version in versions.ChildNodes) { string uuid = ""; string name = ""; string description = ""; string guidance = ""; string guidance_mandatory = ""; string patchVersion = ""; string url = ""; string patchUrl = ""; string timestamp = ""; string priority = ""; string installationSize = ""; foreach (XmlAttribute attrib in version.Attributes) { if (attrib.Name == "uuid") uuid = attrib.Value; else if (attrib.Name == "name-label") name = attrib.Value; else if (attrib.Name == "name-description") description = attrib.Value; else if (attrib.Name == "after-apply-guidance") guidance = attrib.Value; else if (attrib.Name == "guidance-mandatory") guidance_mandatory = attrib.Value; else if (attrib.Name == "version") patchVersion = attrib.Value; else if (attrib.Name == "url") url = attrib.Value; else if (attrib.Name == "patch-url") patchUrl = attrib.Value; else if (attrib.Name == "timestamp") timestamp = attrib.Value; else if (attrib.Name == "priority") priority = attrib.Value; else if (attrib.Name == "installation-size") installationSize = attrib.Value; } var conflictingPatches = GetPatchDependencies(version, ConflictingPatchesNode, ConflictingPatchNode); var requiredPatches = GetPatchDependencies(version, RequiredPatchesNode, RequiredPatchNode); XenServerPatches.Add(new XenServerPatch(uuid, name, description, guidance, guidance_mandatory, patchVersion, url, patchUrl, timestamp, priority, installationSize, conflictingPatches, requiredPatches)); } } } // dependencies patches are listed in the xml as below: // // // // // // // // private static List GetPatchDependencies(XmlNode patchsNode, string dependenciesNodeName, string dependencyNodeName) { var dependenciesNode = patchsNode.ChildNodes.Cast().FirstOrDefault(childNode => childNode.Name == dependenciesNodeName); if (dependenciesNode == null) return null; var dependencies = new List(); dependencies.AddRange(from XmlNode node in dependenciesNode.ChildNodes where node.Attributes != null from XmlAttribute attrib in node.Attributes where node.Name == dependencyNodeName && node.Attributes != null && attrib.Name == "uuid" select attrib.Value); return dependencies; } private void GetXenServerVersions(XmlDocument xdoc) { if (!_checkForServerVersion && !_checkForPatches) return; foreach (XmlNode versions in xdoc.GetElementsByTagName(XenServerVersionsNode)) { foreach (XmlNode version in versions.ChildNodes) { string version_oem = ""; string name = ""; bool is_latest = false; bool is_latest_cr = false; string url = ""; string timestamp = ""; string buildNumber = ""; string patchUuid = ""; bool presentAsUpdate = false; string minXcVersion = ""; foreach (XmlAttribute attrib in version.Attributes) { if (attrib.Name == "value") version_oem = attrib.Value; else if (attrib.Name == "name") name = attrib.Value; else if (attrib.Name == "latest") is_latest = attrib.Value.ToUpperInvariant() == bool.TrueString.ToUpperInvariant(); else if (attrib.Name == "latestcr") is_latest_cr = attrib.Value.ToUpperInvariant() == bool.TrueString.ToUpperInvariant(); else if (attrib.Name == "url") url = attrib.Value; else if (attrib.Name == "timestamp") timestamp = attrib.Value; else if (attrib.Name == "build-number") buildNumber = attrib.Value; else if (attrib.Name == "patch-uuid") patchUuid = attrib.Value; else if (attrib.Name == "present-as-update") presentAsUpdate = attrib.Value.ToUpperInvariant() == bool.TrueString.ToUpperInvariant(); else if (attrib.Name == "minimum-xc-version") minXcVersion = attrib.Value; } List patches = new List(); List minimalPatches = null; //keep it null to indicate that there is no a minimalpatches tag foreach (XmlNode childnode in version.ChildNodes) { if (childnode.Name == "minimalpatches") { minimalPatches = new List(); foreach (XmlNode minimalpatch in childnode.ChildNodes) { if (minimalpatch.Name != "patch") continue; XenServerPatch mp = XenServerPatches.Find(p => string.Equals(p.Uuid, minimalpatch.Attributes["uuid"].Value, StringComparison.OrdinalIgnoreCase)); if (mp == null) continue; minimalPatches.Add(mp); } } if (childnode.Name == "patch") { XenServerPatch patch = XenServerPatches.Find(item => string.Equals(item.Uuid, childnode.Attributes["uuid"].Value, StringComparison.OrdinalIgnoreCase)); if (patch == null) continue; patches.Add(patch); } } XenServerVersions.Add(new XenServerVersion(version_oem, name, is_latest, is_latest_cr, url, patches, minimalPatches, timestamp, buildNumber, patchUuid, presentAsUpdate, minXcVersion)); } } } protected virtual XmlDocument FetchCheckForUpdatesXml(string location) { var xdoc = new XmlDocument(); var uri = new Uri(location); var proxy = XenAdminConfigManager.Provider.GetProxyFromSettings(Connection, false); if (uri.IsFile) { xdoc.Load(location); } else { using (var webClient = new WebClient()) { webClient.Proxy = proxy; webClient.Headers.Add("User-Agent", _userAgent); webClient.Headers.Add("X-User-Agent-Id", _userAgentId); using (var stream = new MemoryStream(webClient.DownloadData(uri))) { xdoc.Load(stream); } } } return xdoc; } } }