Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Mod custom assertion examples #16

Merged
merged 2 commits into from
Nov 8, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 23 additions & 28 deletions Assets/APIExamples/Tests/Runtime/NUnit/CustomComparerExample.cs
Original file line number Diff line number Diff line change
@@ -1,67 +1,62 @@
// Copyright (c) 2021 Koji Hasegawa.
// Copyright (c) 2021-2023 Koji Hasegawa.
// This software is released under the MIT License.

using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using NUnit.Framework;
using NUnit.Framework.Constraints;
using UnityEngine;

// ReSharper disable AccessToStaticMemberViaDerivedType

namespace APIExamples.NUnit
{
/// <summary>
/// <see cref="EqualConstraint"/> + Using修飾子と、カスタム<see cref="IComparer{T}"/>の使用例
/// Using修飾子と、カスタム<see cref="IComparer{T}"/>の使用例
/// </summary>
public class CustomComparerExample
{
[Test]
public void EqualConstraint_Using修飾子で比較()
public void EqualConstraint_Using修飾子でカスタムComparerを指定して比較()
{
var actual = new CompositeKeySUT("Foo", "Bar", 1);
var actual = new GameObject("test object");
var expected = new GameObject("test object");

Assert.That(actual, Is.EqualTo(new CompositeKeySUT("Foo", "Bar", 10)).Using(new CompositeKeySUTComparer()));
Assert.That(actual, Is.EqualTo(expected).Using(new GameObjectNameComparer()));
// 失敗時メッセージ例:
// Expected: <APIExamples.NUnit.CompositeKeySUT>
// But was: <APIExamples.NUnit.CompositeKeySUT>
// Expected: <test object (UnityEngine.GameObject)>
// But was: <test (UnityEngine.GameObject)>
}

[Test]
public void EqualConstraint_コレクションの要素をUsing修飾子で比較()
public void CollectionContainsConstraint_コレクションの要素をUsing修飾子でカスタムComparerを指定して比較()
{
var actual = new[] { new CompositeKeySUT("Foo", "Bar", 1), new CompositeKeySUT("Bar", "Baz", 2) };
var actual = new[] { new GameObject("test1"), new GameObject("test2"), new GameObject("test3"), };
var expected = new GameObject("test3");

Assert.That(actual,
Is.EqualTo(new[] { new CompositeKeySUT("Foo", "Bar", 10), new CompositeKeySUT("Bar", "Baz", 20) })
.Using(new CompositeKeySUTComparer()));
Assert.That(actual, Does.Contain(expected).Using(new GameObjectNameComparer()));
// 失敗時メッセージ例:
// Expected and actual are both <APIExamples.NUnit.CompositeKeySUT[2]>
// Values differ at index [1]
// Expected: <APIExamples.NUnit.CompositeKeySUT>
// But was: <APIExamples.NUnit.CompositeKeySUT>
// Expected: collection containing <test4 (UnityEngine.GameObject)>
// But was: < <test1 (UnityEngine.GameObject)>, <test2 (UnityEngine.GameObject)>, <test3 (UnityEngine.GameObject)> >
}
}

/// <summary>
/// カスタム<see cref="IComparer{T}"/>の実装例
/// カスタム<see cref="IComparer{T}"/>の実装例.
/// このコードは、Test Helperパッケージ(com.nowsprinting.test-helper)内のコードに日本語コメントをつけたものです。
/// </summary>
public class CompositeKeySUTComparer : IComparer<CompositeKeySUT>
public class GameObjectNameComparer : IComparer<GameObject>
{
/// <summary>
/// Key1 + Key2 で比較
/// <c>GameObject</c> 同士を、参照でなく <c>name</c> プロパティで比較するカスタムComparer
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <returns></returns>
public int Compare(CompositeKeySUT x, CompositeKeySUT y)
[SuppressMessage("ReSharper", "PossibleNullReferenceException")]
public int Compare(GameObject x, GameObject y)
{
var key1Comparison = string.Compare(x.Key1, y.Key1, StringComparison.Ordinal);
if (key1Comparison != 0)
{
return key1Comparison;
}

return string.Compare(x.Key2, y.Key2, StringComparison.Ordinal);
return string.Compare(x.name, y.name, StringComparison.Ordinal);
}
}
}
116 changes: 78 additions & 38 deletions Assets/APIExamples/Tests/Runtime/NUnit/CustomConstraintExample.cs
Original file line number Diff line number Diff line change
@@ -1,98 +1,138 @@
// Copyright (c) 2021 Koji Hasegawa.
// Copyright (c) 2021-2023 Koji Hasegawa.
// This software is released under the MIT License.

using NUnit.Framework;
using NUnit.Framework.Constraints;
using UnityEngine;

// ReSharper disable AccessToStaticMemberViaDerivedType

namespace APIExamples.NUnit
{
/// <summary>
/// カスタム<see cref="Constraint"/>の使用例
/// カスタム制約の使用例
/// </summary>
[TestFixture]
public class CustomConstraintExample
{
private static GameObject CreateDestroyedObject()
{
var gameObject = new GameObject("Foo");
GameObject.DestroyImmediate(gameObject);
return gameObject;
}

[Test]
public void CustomConstraint_Constraintの実装だけで可能な書きかた()
{
var actual = CreateDestroyedObject();

Assert.That(actual, new DestroyedConstraint());
// 失敗時メッセージ例:
// Expected: destroyed GameObject
// But was: <Foo (UnityEngine.GameObject)>
}

[Test]
public void CustomEqConstraint_Constraintの実装だけで可能な書きかた()
public void CustomConstraint_Extensionsの実装も行なうと可能な書きかた()
{
var actual = "Foo bar";
var actual = CreateDestroyedObject();

Assert.That(actual, new CustomEqConstraint("Foo bar"));
Assert.That(actual, Is.Not.Null.And.Destroyed());
// 失敗時メッセージ例:
// Expected: "Foo bar"(カスタム制約)
// But was: "Foo bar baz"
// Expected: not null and destroyed GameObject
// But was: <Foo (UnityEngine.GameObject)>
}

[Test]
public void CustomEqConstraint_Extensionsの実装も行なうと可能な書きかた()
public void CustomConstraint_Isの実装も行なうと可能な書きかた()
{
var actual = "Foo bar";
var actual = CreateDestroyedObject();

Assert.That(actual, Is.Not.Null.And.CustomEq("Foo bar"));
Assert.That(actual, Is.Destroyed);
// 失敗時メッセージ例:
// Expected: not null and "Foo bar"(カスタム制約)
// But was: "Foo bar baz"
// Expected: destroyed GameObject
// But was: <Foo (UnityEngine.GameObject)>
}

[Test]
public void CustomEqConstraint_Isの実装も行なうと可能な書きかた()
public void CustomConstraint_Isの実装も行なうと可能な書きかた_Not()
{
var actual = "Foo bar";
var actual = new GameObject("Bar");

Assert.That(actual, Is.CustomEq("Foo bar"));
Assert.That(actual, Is.Not.Destroyed());
// 失敗時メッセージ例:
// Expected: "Foo bar"(カスタム制約)
// But was: "Foo bar baz"
// Expected: not destroyed GameObject
// But was: <null>
}
}

/// <summary>
/// カスタム<see cref="Constraint"/>の実装例
/// カスタム制約の実装例.
/// このコードは、Test Helperパッケージ(com.nowsprinting.test-helper)内のコードに日本語コメントをつけたものです。
/// </summary>
/// <remarks>
/// 内容は文字列が一致するか判定しているだけ
/// </remarks>
public class CustomEqConstraint : Constraint
public class DestroyedConstraint : Constraint
{
public CustomEqConstraint(params object[] args) : base(args) { }
public DestroyedConstraint(params object[] args) : base(args)
{
base.Description = "destroyed GameObject"; // 失敗時メッセージに出力される
}

/// <summary>
/// Assert.Thatから渡されるactualオブジェクトを検証する、カスタム制約の本体
/// </summary>
/// <param name="actual">検証対象オブジェクト</param>
/// <returns>制約の成否</returns>
public override ConstraintResult ApplyTo(object actual)
{
return new ConstraintResult(this, actual, actual.ToString() == Arguments[0].ToString());
}
if (actual is GameObject actualGameObject)
{
return new ConstraintResult(this, actual, (bool)actualGameObject == false);
// GameObjectであれば、破棄されているかを判定して結果として返す
}

public override string Description { get { return $"\"{Arguments[0]}\"(カスタム制約)"; } }
return new ConstraintResult(this, actual, false);
// GameObjectでなければ常にfalse
}
}

/// <summary>
/// カスタム<see cref="Constraint"/>向けの<see cref="ConstraintExpression"/>拡張メソッド
/// カスタム制約のための<see cref="ConstraintExpression"/>拡張クラス.
/// このコードは、Test Helperパッケージ(com.nowsprinting.test-helper)内のコードに日本語コメントをつけたものです。
/// </summary>
public static class CustomConstraintExtensions
public static class ConstraintExtensions
{
public static CustomEqConstraint CustomEq(this ConstraintExpression expression, object expected)
/// <summary>
/// カスタム制約のための<see cref="ConstraintExpression"/>拡張メソッド.
/// 式にDestroyedが指定されたとき、Destroyed制約のインスタンスをexpressionに追加して返します。
/// </summary>
/// <param name="expression"></param>
/// <returns>constraint to destroyed GameObject</returns>
public static DestroyedConstraint Destroyed(this ConstraintExpression expression)
// Note: 比較対象がある制約の場合、第2引数で object expected を受け取ります
{
var constraint = new CustomEqConstraint(expected);
var constraint = new DestroyedConstraint();
expression.Append(constraint);
return constraint;
}
}

/// <summary>
/// カスタム<see cref="Constraint"/>向けの<see cref="Is"/>クラス
/// カスタム制約のための<see cref="Is"/>クラス.
/// このコードは、Test Helperパッケージ(com.nowsprinting.test-helper)内のコードに日本語コメントをつけたものです。
/// </summary>
/// <remarks>
/// Unity Test Framework には NUnit.Framework.Is を継承した UnityEngine.TestTools.Constraints.Is がすでにあるため、
/// これを継承して作ります
/// UnityEngine.TestTools.Constraints.Is が必要なければ、直接 global::NUnit.Framework.Is を継承しても構いません。
/// Unity Test Framework には NUnit.Framework.Is を継承した UnityEngine.TestTools.Constraints.Is がすでにあるため、これを継承して作ります。
/// Test Helperパッケージも使用するのであれば、TestHelper.Constraints.Is を継承します
/// UnityEngine.TestTools.Constraints.Is も TestHelper.Constraints.Is も必要なければ、直接 global::NUnit.Framework.Is を継承しても構いません。
/// </remarks>
// ReSharper disable once ClassNeverInstantiated.Global
public class Is : UnityEngine.TestTools.Constraints.Is
{
public static CustomEqConstraint CustomEq(object expected)
{
return new CustomEqConstraint(expected);
}
/// <summary>
/// Create constraint to destroyed GameObject.
/// </summary>
public static DestroyedConstraint Destroyed => new DestroyedConstraint();
// Note: 比較対象がある制約の場合、引数で object expected を受け取ります
}
}
6 changes: 6 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,12 @@ Assets
    └── 第5章 非同期処理のテスト, 第9章 Unity Test Framework Tips
```

> **Note**
> 『Unity Test Framework完全攻略ガイド』6.4 カスタムアサーションの例を、v2.1.0(2023-11-11版)で書き換えました。
> 変更差分はコミット &lt;[dc1b643](https://github.com/nowsprinting/UnityTestExamples/commit/dc1b643cd7e1275388881933b5edfcabde0413ba)&gt; を参照してください。



### BasicExample

第2章 Unity Test Frameworkの基本
Expand Down