如何在 MSpec 中每次测试运行之前运行代码

本文关键字:测试运行 运行 代码 MSpec | 更新日期: 2023-09-27 18:31:29

我正在尝试在测试前运行一些初始化代码。我已经尝试了其他问题中的建议,但它似乎不起作用。我的域模型通过以下类引发事件:

public static class DomainEvents
{
    private static readonly object @lock = new object();
    private static Action<IDomainEvent> raiseEvent;
    public static void Raise<TEvent>(TEvent @event) where TEvent : class, IDomainEvent 
    {
         // omitted for brevity
    }
    public static void RegisterEventPublisher(Action<IDomainEvent> eventPublisher)
    {
        lock (@lock)
        {
            raiseEvent = eventPublisher;
        }
    }
}

出于测试目的,我想在静态列表中捕获这些事件。最好的方法是什么?

更新

问题是由测试的运行顺序引起的(正如亚历山大在下面指出的那样,不能保证)。在我的一个规范中,我注册了一个模拟事件发布者。规范通常会以不同的顺序运行的事实意味着 a) 一开始我不知道我有问题("问题"规范总是最后运行)和 b) 一旦我开始遇到问题,失败的测试数量通常会在运行之间变化(使其更加混乱)。

经验教训 - 在每个上下文运行后清理任何静态资源。您可以通过实现 ICleanupAfterEveryContextInAssembly .

如何在 MSpec 中每次测试运行之前运行代码

也许我误解了这个问题,但基本模式是:

public class WhenSomeDomainEventIsRaised
{
    private IList<IDomainEvent> EventsRaised = new List<IDomainEvent>();
    Establish context = () => 
    {
        // subscribe to events; when raised, add to EventsRaised list
    }
}

如果要对所有测试或测试子集执行此操作:

public abstract class DomainSpecification
{
    protected IList<IDomainEvent> EventsRaised = new List<IDomainEvent>();
    Establish context = () => 
    {
        // subscribe to events; when raised, add to EventsRaised list
    }
}

你可以让所有需要此行为的规范从此类继承,MSpec 将负责沿继承层次结构运行所有Establish块。

这对

我有用:

using System;
using System.Collections.Generic;
using Machine.Specifications;
namespace AssemblyContextSpecs
{
  public static class DomainEvents
  {
    static readonly object @lock = new object();
    static Action<IDomainEvent> raiseEvent;
    public static void Raise<TEvent>(TEvent @event) where TEvent : class, IDomainEvent
    {
      raiseEvent(@event);
    }
    public static void RegisterEventPublisher(Action<IDomainEvent> eventPublisher)
    {
      lock (@lock)
      {
        raiseEvent = eventPublisher;
      }
    }
  }
  public interface IDomainEvent
  {
  }
  class FooEvent : IDomainEvent
  {
  }
  public class DomainEventsContext : IAssemblyContext
  {
    internal static IList<IDomainEvent> Events = new List<IDomainEvent>();
    public void OnAssemblyStart()
    {
      DomainEvents.RegisterEventPublisher(x => Events.Add(x));
    }
    public void OnAssemblyComplete()
    {
    }
  }
  public class When_a_domain_event_is_raised
  {
    Because of = () => DomainEvents.Raise(new FooEvent());
    It should_capture_the_event =
      () => DomainEventsContext.Events.ShouldContain(x => x.GetType() == typeof(FooEvent));
  }
}

RegisterEventPublisher不应该RegisterEventSubscriber吗?