/* 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 NUnit.Framework; using Moq; using XenAdmin.Core; using XenAPI; using System.Linq; using XenAdmin.Network; namespace XenAdminTests.UnitTests { [TestFixture, Category(TestCategories.Unit)] public class AutomatedUpdatesTests { private static Mock GetAMockHost(string productVersion, string buildNumber, List applied_patches = null) { var id = System.Guid.NewGuid().ToString(); MockObjectManager mom = new MockObjectManager(); mom.CreateNewConnection(id); Mock master = mom.NewXenObject(id); Mock pool = mom.NewXenObject(id); XenRef masterRef = new XenRef("master-ref"); pool.Setup(p => p.master).Returns(masterRef); mom.MockCacheFor(id).Setup(c => c.Resolve(It.IsAny>())).Returns(pool.Object); mom.MockConnectionFor(id).Setup(c => c.Resolve(masterRef)).Returns(master.Object); mom.MockConnectionFor(id).Setup(c => c.Cache.Hosts).Returns(new Host[] { master.Object }); mom.MockConnectionFor(id).Setup(c => c.Cache.Pools).Returns(new Pool[] { pool.Object }); master.Setup(h => h.software_version).Returns(new Dictionary()); master.Setup(h => h.ProductVersion()).Returns(productVersion); master.Setup(h => h.AppliedPatches()).Returns(applied_patches ?? new List()); master.Setup(h => h.BuildNumberRaw()).Returns(buildNumber); master.Setup(h => h.uuid).Returns(id); return master; } private static List GetAVersionWithGivenNumberOfPatches(int numberOfPatches) { var serverVersions = new List(); var version = new XenServerVersion("7.0.0", "XenServer Test 7", true, false, "", new List(), new List(), DateTime.MinValue.ToString(), "buildNo", "", false, ""); for (int ii = 0; ii < numberOfPatches; ii++) { var patch = new XenServerPatch("patch_uuid_" + ii, "patch name " + ii, "patch description" + ii, "", "", "1.0", "", "", "1970-01-01T00:00:00Z", "", "1000", "1000"); version.Patches.Add(patch); version.MinimalPatches.Add(patch); } serverVersions.Add(version); return serverVersions; } private static void RemoveANumberOfPatchesPseudoRandomly(List patches, int numberOfPatchesToRemove) { int expectedCount = patches.Count - numberOfPatchesToRemove; var rnd = new Random(777); //pseudo random to be able to repeat this test... while (patches.Count > expectedCount) { patches.RemoveAt(rnd.Next(patches.Count)); } } [Test] public void GetUpgradeSequenceForNullConnection() { Assert.AreEqual(Updates.GetMinimalPatches((IXenConnection) null), null); } /// /// Version exists /// No patches in updates /// Nothing installed on host /// Result: update sequence has the host, but empty sequence /// [Test] public void NoPatchesForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(0); var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber); SetXenServerVersionsInUpdates(serverVersions); var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(0, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(0, upgradeSequence.Count); } /// /// Version exists /// 1 patch in updates. 1 in minimal list. /// 1 has to be installed on host /// Result: update sequence has the host, with the one single patch in it /// [Test] public void OnePatchToBeInstalledForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(1); var patch = serverVersions.First().Patches[0]; var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber); SetXenServerVersionsInUpdates(serverVersions); var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(1, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(1, upgradeSequence.Count); Assert.NotNull(upgradeSequence[0]); Assert.AreEqual(upgradeSequence[0], patch); } /// /// Version exists /// 1 patch in updates. 1 in minimal list. But it is applied already on master. /// 0 has to be installed on host /// [Test] public void AlreadyInstalledOneFromMinimalForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(1); var patch = serverVersions.First().Patches[0]; Pool_patch pool_patch = new Pool_patch(); pool_patch.uuid = patch.Uuid; var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber, new List {pool_patch}); SetXenServerVersionsInUpdates(serverVersions); var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(1, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(0, upgradeSequence.Count); } /// /// Version exists /// 100 patch in updates. 100 in minimal list. /// 100 has to be installed on host /// Result: update sequence has all the 100 patches in it /// [Test] public void HundredPatchToBeInstalledForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(100); var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber); SetXenServerVersionsInUpdates(serverVersions); var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(100, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(100, upgradeSequence.Count); for (int i = 100; i < 0; --i) Assert.AreEqual(upgradeSequence[i], serverVersions[0].Patches[i]); } /// /// Version exists /// 100 patch in updates. 51 in minimal list. /// 51 has to be installed on host /// Result: update sequence has all the 51 patches in it /// [Test] public void FiftyOnePatchToBeInstalledForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(100); var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber); RemoveANumberOfPatchesPseudoRandomly(serverVersions.First().MinimalPatches, 49); SetXenServerVersionsInUpdates(serverVersions); // Assert var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(51, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(51, upgradeSequence.Count); foreach (var patch in upgradeSequence) Assert.IsTrue(serverVersions.First().MinimalPatches.Contains(patch)); Assert.False(upgradeSequence.Exists(seqpatch => !serverVersions.First().MinimalPatches.Contains(seqpatch))); } /// /// Version exists /// 100 patch in updates. 51 in minimal list. /// 41 has to be installed on host (only 41 of 51, because 10 is already installed from the Minimal list (as well as 5 other)) /// Result: update sequence has all the 41 patches in it and all are from Minimal List and also not installed on the host /// [Test] public void FourtyOnePatchToBeInstalledForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(100); RemoveANumberOfPatchesPseudoRandomly(serverVersions.First().MinimalPatches, 49); var pool_patches = new List(); for (int ii = 0; ii < 10; ii++) { pool_patches.Add(new Pool_patch() { uuid = serverVersions.First().MinimalPatches[ii].Uuid }); } var notMinimalPatchesInVersion = serverVersions.First().Patches.Where(p => !serverVersions.First().MinimalPatches.Exists(mp => mp.Uuid == p.Uuid)).ToList(); for (int ii = 0; ii < 5; ii++) { pool_patches.Add(new Pool_patch() { uuid = notMinimalPatchesInVersion[ii].Uuid }); } var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber, pool_patches); SetXenServerVersionsInUpdates(serverVersions); // Assert var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(51, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(41, upgradeSequence.Count); foreach (var patch in upgradeSequence) Assert.IsTrue(serverVersions.First().MinimalPatches.Contains(patch) && !pool_patches.Exists(p => p.uuid == patch.Uuid)); Assert.False(upgradeSequence.Exists(seqpatch => !serverVersions.First().MinimalPatches.Contains(seqpatch))); Assert.True(upgradeSequence.Exists(seqpatch => !pool_patches.Exists(pp => pp.uuid == seqpatch.Uuid))); } /// /// Version exists /// 2 patch in updates. 1 in minimal list. /// 1 has to be installed on host /// Result: update sequence has the host, with the one single patch in it /// [Test] public void OnePatchOutOfTwoToBeInstalledForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(2); var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber); RemoveANumberOfPatchesPseudoRandomly(serverVersions.First().MinimalPatches, 1); SetXenServerVersionsInUpdates(serverVersions); // Assert var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(1, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(1, upgradeSequence.Count); Assert.AreEqual(upgradeSequence[0], serverVersions.First().MinimalPatches[0]); } /// /// Version exists /// 2 patch in updates. 2 in minimal list. /// 2 has to be installed on host /// Result: update sequence has the host, with the one single patch in it /// [Test] public void TwoPatchToBeInstalleOutOfTwodForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(2); var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber); SetXenServerVersionsInUpdates(serverVersions); // Assert var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.NotNull(minimalPatches); Assert.AreEqual(2, minimalPatches.Count); var upgradeSequence = Updates.GetPatchSequenceForHost(master.Object, minimalPatches); Assert.NotNull(upgradeSequence); Assert.AreEqual(2, upgradeSequence.Count); for (int i = 0; i < upgradeSequence.Count; i++) Assert.AreEqual(upgradeSequence[i], serverVersions.First().MinimalPatches[i]); } /// /// Version does not exist /// Result: update sequence is null /// [Test] public void NoInfoForCurrentVersion() { // Arrange var serverVersions = GetAVersionWithGivenNumberOfPatches(200); var master = GetAMockHost(serverVersions.First().Oem, serverVersions.First().BuildNumber + "to_make_it_not_match"); SetXenServerVersionsInUpdates(serverVersions); // assert var minimalPatches = Updates.GetMinimalPatches(master.Object.Connection); Assert.Null(minimalPatches); } /// /// Updates is static class, can't be mocked without refactoring - using this method instead /// /// private void SetXenServerVersionsInUpdates(List serverVersions) { Updates.XenServerVersions = serverVersions; } } }