我是否正确编写了我的第一个MSpec规范?

Sek*_*hat 41 c# bdd mspec

我正在编写我的第一个MSpec规范,我想要一些指导.我将规范保留在"待处理"状态,但上下文已填写完整.有什么改进吗?

作为参考,这是故事和第一个场景:

Story: "Blog admin logs in to the system"

As a blog writer
I want to be able to log in to my blog
So that I can write posts and administer my blog

Scenario: "Logs in from the login page"

Given the user enters in correct credentials for a user in the system
When the user clicks the "Login" button
Then log the user in and redirect to the admin panel with a message 
stating that he logged in correctly
Run Code Online (Sandbox Code Playgroud)

并且MSpec代码(某些部分被剪切),请注意It由于与以下内容冲突,我必须为MSpec 委托设置别名Moq.It:

using MoqIt = Moq.It;
using ThenIt = Machine.Specifications.It;

[Subject("User tries logging in")]
public class When_user_enters_valid_credentials : With_user_existing_in_membership
{
    protected static ActionResult result;

    Because of = () =>
    {
        result = loginController.Login(validUsername, validPassword);
    };

    ThenIt should_log_the_user_in;
    ThenIt should_redirect_the_user_to_the_admin_panel;
    ThenIt should_show_message_confirming_successful_login;
}

public abstract class With_user_existing_in_membership
{
    protected static Mock<ISiteMembership> membershipMock;
    protected static string validUsername;
    protected static string validPassword;
    protected static LoginController loginController;

    Establish context =()=>
    {
        membershipMock = new Mock<ISiteMembership>();
        validUsername = "ValidUsername";
        validPassword = "ValidPassword";
        //make sure it's treated as valid usernames and password
        membershipMock
            .Setup<bool>(m => m.Validate(
                MoqIt.Is<string>(s => s == validUsername), 
                MoqIt.Is<string>(s => s == validPassword)))
            .Returns(true);
        loginController = new LoginController(membershipMock.Object);
    };
}
Run Code Online (Sandbox Code Playgroud)

Ale*_*roß 56

上下文看起来不错.我喜欢你解决It与别名冲突的方式.我认为可以改进Moq别名.考虑类似句子的东西.例如,Param.Is<T>Value.Is<T>.

一些注释,使用代码片段,然后在底部重写整个规范.

情景是你的 Subject

主题可以是故事中的场景.另外,它会使用您的测试运行报告进行渲染(在HTML报告中尤为出色).

[Subject("Login Page")]
Run Code Online (Sandbox Code Playgroud)

不要在"With"命名基类上浪费时间

MSpec的创造者,亚伦詹森,已经恢复从完全利用"同向"的语法.上下文类名称不会显示在任何报告中,因此请避免花时间创建有意义的名称.

public abstract class MembershipContext
Run Code Online (Sandbox Code Playgroud)

Given是你的spec类名

在故事中的Given之后命名具体的spec类.特别是因为基类名称没有在任何地方报告,您可能会在报告中丢失一半的上下文!您还应该避免在上下文类名称中输入正在测试的系统的名称.这使您的上下文更友好地重构被测系统.

public class When_an_existing_user_enters_valid_credentials
Run Code Online (Sandbox Code Playgroud)

基本规范类应仅包含常规初始化

并且通常是不必要的.它们导致安排和行动阶段的分离.使用基类进行公共字段初始化,例如设置模拟依赖项.但是,您不应该在基类中模拟行为.并且您不应将特定于上下文的信息放在基类中.在您的示例中,用户名/密码.这样,您可以使用无效凭据创建第二个上下文.

Establish context = () =>
{
    membership = new Mock<ISiteMembership>();
    loginController = new LoginController(membership.Object);
};
Run Code Online (Sandbox Code Playgroud)

具体spec类中的字段应该是私有的

它减少了测试中语言的"仪式".您应该将它们放在所有MSpec特定代表的下方,因为规范的那些部分告诉了大部分故事.

static ActionResult result;
Run Code Online (Sandbox Code Playgroud)

规格检修

这里的规范是建立全局上下文MembershipContext并在特定于规范的上下文中继承它的一个很好的例子(因此,附加Establish).

[Subject("Login Page")]
public class When_an_existing_user_enters_valid_credentials : MembershipContext 
{
    Establish context = () =>
    {
        membership
            .Setup<bool>(m => m.Validate(
                Param.Is<string>(s => s == username), 
                Param.Is<string>(s => s == password)))
            .Returns(true);
    };

    Because of = () => result = loginController.Login(username, password);

    It should_log_the_user_in;
    It should_redirect_the_user_to_the_admin_panel;
    It should_show_message_confirming_successful_login;

    static ActionResult result;
    const string username = "username";
    const string password = "password";
}

public abstract class MembershipContext 
{
    Establish context = () =>
    {
        membership = new Mock<ISiteMembership>();
        loginController = new LoginController(membership.Object);
    };

    protected static Mock<ISiteMembership> membership;
    protected static LoginController loginController;
}
Run Code Online (Sandbox Code Playgroud)

  • 这是正确的,但MSpec不会并行运行规范.ReSharper确实支持并行运行,但这是按照程序集实现的,因此在单个"AppDomain"内不会发生共享. (4认同)