Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/fluffynuts/nexpect
An assertions framework for .NET with a BDD-like feel, inspired by Chai and Jasmine, designed to be user-extensible
https://github.com/fluffynuts/nexpect
Last synced: 5 days ago
JSON representation
An assertions framework for .NET with a BDD-like feel, inspired by Chai and Jasmine, designed to be user-extensible
- Host: GitHub
- URL: https://github.com/fluffynuts/nexpect
- Owner: fluffynuts
- License: bsd-3-clause
- Created: 2017-07-09T19:11:43.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2024-04-29T06:16:08.000Z (9 months ago)
- Last Synced: 2024-05-02T00:05:04.821Z (9 months ago)
- Language: C#
- Size: 2.74 MB
- Stars: 21
- Watchers: 6
- Forks: 8
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# NExpect
An assertions framework for .NET with a BDD-like feel, inspired by Chai and Jasmine, designed to be user-extensible![Build and Test](https://github.com/fluffynuts/NExpect/workflows/Build%20and%20Test/badge.svg)
![Nuget current version badge](https://img.shields.io/nuget/v/NExpect)
## Goals
- Expect(NExpect).To.Be.Readable();
- Because code is for co-workers, not compilers. And your tests are part of your documentation.
- Expect(NExpect).To.Be.Expressive();
- Because the intent of a test should be easy to understand. The reader can delve into the details when she cares to.
- Expect(NExpect).To.Be.Extensible();
- Because I can't predict every use-case. I believe that your assertions framework should enable expressive, readable tests through extension.## Tutorial / blog posts:
[https://fluffynuts.github.io/NExpect](https://fluffynuts.github.io/NExpect)
[dev.to](https://dev.to/fluffynuts/introducing-nexpect-555c)## Usage
1. Download from [nuget.org](https://nuget.org): `install-package nexpect`
2. Import Expectations statically:
```csharp
using static NExpect.Expectations;
```
3. `Expect` inside your tests, with fluent syntax:
```csharp
// simple equality checks
Expect(1).To.Equal(1);
Expect(true).To.Not.Be.False(); // alt. grammar
Expect(null).To.Be.Null();
// - with negation, order doesn't matter
Expect("moo").Not.To.Equal("cow");
Expect("moo").To.Not.Equal("cow");
Expect(true).Not.To.Be.False();
Expect(false).To.Not.Be.True();// exceptions
Expect(() => { }).Not.To.Throw();
Expect(() =>
{
throw new ArgumentException("moo", "moo cow");
}).To.Throw()
.With.Message.Containing("moo")
.And.("cow");// smarter string tests, with fluency
Expect(someString).To.Contain("moo").And("cow");
Expect("moo, said the cow")
.To.Start.With("moo")
.And.Contain("said")
.Then("the")
.And.End.With("cow");// collection tests
Expect(someCollection).To.Contain.Exactly(2)
.Matched.By(item => item.IsWhatWeWant());
Expect(someCollection).To.Contain.Only(1)
.Deep.Equal.To(new { id = 42, name = "Douglas" });
Expect(someFlags).To.Contain.At.Least(3)
.Equal.To(true);
Expect(new[] { 1, 2, 3 })
.To.Be.Ordered.Ascending();
Expect(new[] { "c", "b", "a" })
.To.Be.Ordered.Descending();// type testing
Expect(someObject).To.Be
.An.Instance.Of();// deep and intersection equality testing
var person = new {
id = 1,
name = "bob"
};
Expect(person)
.To.Deep.Equal(new { id = 1, name = "bob" });
Expect(person)
.To.Intersection.Equal(new { name = "bob" });
```## Extending
Mostly, you can extend by adding extension methods for ICanAddMatcher where T is the
type you want. You can also extend at any point in the grammar -- some of the "better"
points are ITo, IBe, IHave, IA, IAn. You will need another namespace import:
```csharp
using NExpect.MatcherLogic
```
And your extension methods can be like:```csharp
public static class MyMatchers
{
public static void Five(this IBe continuation)
{
continuation.AddMatcher(actual =>
{
var passed = actual == 5;
var message = passed
? $"Expected {actual} not to be 5"
: $"Expected {actual} to be 5";
return new MatcherResult(passed, message);
});
}
}
``````csharp
// somewhere else...
[Test]
public void FifteenDividedByThree_ShouldEqual_Five()
{
var result = 15 / 3;
Expect(result).To.Be.Five();
}
// Yes, yes, simple example is simple.
```If you've ever written a Jasmine matcher, this should feel familiar.
If you have a bunch of existing expectations that you'd like to wrap
up into a nicely-named matcher, `.Compose` has you covered:```csharp
// before
var cow = animalFactory.MakeCow();
var beetle = animalFactory.MakeBeetle();// animal factory should make a Jersey cow
Expect(cow.Classification).To.Equal("Mammal");
Expect(cow.Legs).To.Equal(4);
Expect(cow.HasTail).To.Be.True();
Expect(cow.HasHorns).To.Be.True();
Expect(cow.HasSpots).To.Be.True();// Animal factory should make a rhinoceros beetle
Expect(beetle.Classification).To.Equal("Insect");
Expect(beetle.Legs).To.Equal(6);
Expect(beetle.HasTail).To.Be.False();
Expect(beetle.HasHorns).To.Be.True();
Expect(beetle.HasSpots).To.Be.False();
``````csharp
// after
var cow = animalFactory.MakeJerseyCow();
var beetle = animalFactory.MakeRhinocerosBeetle();Expect(cow).To.Be.A.JerseyCow();
Expect(beetle).To.Be.A.RhinocerosBeetle();// elsewhere:
public static class AnimalMatchers
{
// the IMore interface allows fluent chaining of expectations
// eg:
// Expect(cow).To.Be.A.JerseyCow()
// .And
// .Not.To.Be.A.FrieslandCow();
public static IMore JerseyCow(this IA a)
{
return a.Compose(actual =>
{
Expect(cow.Classification).To.Equal("Mammal");
Expect(cow.Legs).To.Equal(4);
Expect(cow.HasTail).To.Be.True();
Expect(cow.HasHorns).To.Be.True();
Expect(cow.HasSpots).To.Be.True();
});
}
public static IMore RhinocerosBeetle(this IA a)
{
return a.Compose(actual =>
{
Expect(beetle.Classification).To.Equal("Insect");
Expect(beetle.Legs).To.Equal(6);
Expect(beetle.HasTail).To.Be.False();
Expect(beetle.HasHorns).To.Be.True();
Expect(beetle.HasSpots).To.Be.False();
});
}
}
```When one of the inner expectations fails, NExpect attempts to construct
a nice failure message. As with all expectations, you can always make
failures easier to understand with a custom message string or generator:```csharp
using NExpect.Implementations;
using NExpect.MatcherLogic;
using NExpect;
using static NExpect.Expectations;public static class AnimalMatchers
{
public static IMore JerseyCow(this IA a)
{
return a.Compose(actual =>
{
// the Stringify extension method, available on all types,
// comes from NExpect.Implementation.MessageHelpers and
// produces a string representation of the object it's
// operating on which is similar to JSON, so it's easier
// to read what the object was
var customMessage = $"Expected {actual.Stringify()} to be a cow";
Expect(cow.Classification).To.Equal("Mammal", customMessage);
Expect(cow.Legs).To.Equal(4, customMessage);
Expect(cow.HasTail).To.Be.True(customMessage);
Expect(cow.HasHorns).To.Be.True(customMessage);
Expect(cow.HasSpots).To.Be.True(customMessage);
});
}
public static IMore RhinocerosBeetle(this IA a)
{
return a.Compose(actual =>
{
// we can use a generator func to delay generation of the message
// which is especially helpful if message generation is expensive
// and we'd only like to spend that cpu time on a failure
Func customMessageGenerator = () => $"Expected {actual.Stringify()} to be a cow";
Expect(beetle.Classification).To.Equal("Insect", customMessageGenerator);
Expect(beetle.Legs).To.Equal(6, customMessageGenerator);
Expect(beetle.HasTail).To.Be.False(customMessageGenerator);
Expect(beetle.HasHorns).To.Be.True(customMessageGenerator);
Expect(beetle.HasSpots).To.Be.False(customMessageGenerator);
});
}
}
```