/* 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 XenAPI; using System.Reflection; using XenAdmin.Model; namespace XenAdminTests.XenModelTests { [TestFixture, Category(TestCategories.Unit)] public class XenObjectEqualsTests { private readonly Random _random = new Random(); /// /// Gets all Types that derive from IXenObject except Folder and DockerContainer /// private static IEnumerable AllXenObjectTypesExceptFolder { get { var theTypes = typeof(IXenObject).Assembly.GetTypes(); foreach (Type t in theTypes) { if (!t.IsAbstract && typeof(IXenObject).IsAssignableFrom(t) && t.GetConstructor(new Type[0]) != null && !typeof(Folder).IsAssignableFrom(t) && !typeof(DockerContainer).IsAssignableFrom(t)) { yield return t; } } } } [Test] [Description("Tests object.Equals uses opaque_ref for all types of IXenObject except folders.")] public void TestObjectEquals([ValueSource(typeof(XenObjectEqualsTests), nameof(AllXenObjectTypesExceptFolder))] Type xenObjectType) { IXenObject a = (IXenObject)Activator.CreateInstance(xenObjectType); SetRandomStuffToFields(a); a.opaque_ref = "hello"; IXenObject b = (IXenObject)Activator.CreateInstance(xenObjectType); SetRandomStuffToFields(b); b.opaque_ref = "hello"; IXenObject c = (IXenObject)Activator.CreateInstance(xenObjectType); SetRandomStuffToFields(c); c.opaque_ref = "goodbye"; Assert.IsTrue(a.Equals(b), xenObjectType.Name + " Equals failed"); Assert.IsFalse(a.Equals(c), xenObjectType.Name + " Equals failed"); } [Test] [Description("Tests that folder uses _name_label for object.Equals instead of opaque_ref.")] public void TestFolderObjectEquals() { Folder a = new Folder(null, "hello") { opaque_ref = "a" }; Folder b = new Folder(null, "hello") { opaque_ref = "b" }; Folder c = new Folder(null, "goodbye") { opaque_ref = "c" }; Assert.AreEqual(a, b, "Folder Equals failed"); Assert.AreNotEqual(a, c, "Folder Equals failed"); } [Test] [Description("Tests that IEquatable has the same result as object.Equals for all types that derive from IXenObject except Folder.")] public void TestIEquatableUsage([ValueSource(typeof(XenObjectEqualsTests), nameof(AllXenObjectTypesExceptFolder))] Type xenObjectType) { IXenObject a = (IXenObject)Activator.CreateInstance(xenObjectType); SetRandomStuffToFields(a); a.opaque_ref = "hello"; IXenObject b = (IXenObject)Activator.CreateInstance(xenObjectType); SetRandomStuffToFields(b); b.opaque_ref = "hello"; IXenObject c = (IXenObject)Activator.CreateInstance(xenObjectType); SetRandomStuffToFields(c); c.opaque_ref = "goodbye"; Assert.IsTrue(IEquatableEquals(a, b), xenObjectType.Name + " Equals failed"); Assert.IsFalse(IEquatableEquals(a, c), xenObjectType.Name + " Equals failed"); } /// /// Iterate through all implemented IEquatable<T> interfaces and call Equals on each, if any of them /// return false, then this method returns false, otherwise it returns true. /// /// Both o and oo should be of the same type. /// private bool IEquatableEquals(IXenObject o, IXenObject oo) { Assert.AreEqual(o.GetType(), oo.GetType()); foreach (Type iface in o.GetType().GetInterfaces()) { if (iface.Name.StartsWith("IEquatable")) { MethodInfo mi = iface.GetMethod("Equals"); if (!(bool)mi.Invoke(o, new object[] { oo })) { return false; } } } return true; } [Test] public void TestFolderIEquatableUsage() { var a = new Folder(null, "hello") { opaque_ref = "a" }; var b = new Folder(null, "hello") { opaque_ref = "b" }; var c = new Folder(null, "goodbye") { opaque_ref = "c" }; IEquatable aa = a; IEquatable bb = b; IEquatable cc = c; Assert.IsTrue(aa.Equals(bb), "Folder Equals failed"); Assert.IsFalse(aa.Equals(cc), "Folder Equals failed"); } /// /// Sets the random stuff to the string, int, long, double fields of the specified object. /// private void SetRandomStuffToFields(IXenObject xenObject) { foreach (FieldInfo fi in xenObject.GetType().GetFields(BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic)) { if (!fi.IsInitOnly && !fi.IsLiteral) //check field isn't const or readonly { if (fi.FieldType == typeof(string)) { fi.SetValue(xenObject, Guid.NewGuid().ToString()); } else if (fi.FieldType == typeof(int) || fi.FieldType == typeof(long) || fi.FieldType == typeof(double)) { fi.SetValue(xenObject, _random.Next()); } } } } [Test] public void TestCanDeriveHostAndOverrideEquals() { var h1 = new DerivedFromHost { opaque_ref = "hello" }; var h2 = new DerivedFromHost { opaque_ref = "hello" }; Assert.IsFalse(h1.Equals(h2)); object hh1 = h1; object hh2 = h2; Assert.IsFalse(hh1.Equals(hh2)); IEquatable hhh1 = h1; IEquatable hhh2 = h2; Assert.IsFalse(hhh1.Equals(hhh2)); } private class DerivedFromHost : Host { private readonly object _o = new object(); public override bool Equals(object other) { return ReferenceEquals(this, other); } public override int GetHashCode() { return _o.GetHashCode(); } } } }