如何构建Fluent嵌套Guard API

Gra*_*ham 11 c# lambda fluent

我正在构建一个简单的Guard API来防止传递给函数的非法参数等.

我有以下代码:

public static class Guard
{
    public static GuardArgument<T> Ensure<T>(T value, string argumentName)
    {
        return new GuardArgument<T>(value, argumentName);
    }
}

public class GuardArgument<T>
{
    public GuardArgument(T value, string argumentName)
    {
        Value = value;
        Name = Name;
    }

    public T Value { get; private set; }
    public string Name { get; private set; }
}

// Example extension for validity checks
public static GuardArgument<T> IsNotNull<T>(this GuardArgument<T> guardArgument, string errorMessage)
{
    if (guardArgument.Value == null)
    {
        throw new ArgumentNullException(guardArgument.Name, errorMessage);
    }

    return guardArgument;
}     
Run Code Online (Sandbox Code Playgroud)

目前代码可以以类似的方式使用(注意这只是一个愚蠢的例子):

void DummyMethod(int? someObject) {

    Guard.Ensure(someObject, "someObject")
       .IsNotNull()
       .IsGreaterThan(0)
       .IsLessThan(10);
}
Run Code Online (Sandbox Code Playgroud)

一切正常.我现在想要做的是扩展API以在支票中包含子属性,方法如下:

Guard.Ensure(someObject, "someObject")
    .IsNotNull()
    .Property(
        (x => x.ChildProp1, "childProp1")
           .IsNotNull()
           .IsGreaterThan(10)
     )
     .Property(
        (x => x.ChildProp2, "childProp2")
           .IsNotNull()
           .IsLessThan(10)
     );
Run Code Online (Sandbox Code Playgroud)

显然,新.Property方法需要返回父GuardArgument级才能链接.此外,子属性需要能够使用现有的检查方法(IsNotNull()等)来避免代码重复.

我无法弄清楚如何构造lambda/Property函数参数或.Property方法应该位于何处- 即它应该是GuardArgument或其他地方的属性,或者即使API有更好的结构.

Sea*_*aal 7

以下函数允许使用与您想要的语法类似的语法.

public static GuardArgument<T> Property<T, TProp>(this GuardArgument<T> guardArgument, Func<T, TProp> getProperty, string propertyName, Action<GuardArgument<TProp>> validate)
{
    GuardArgument<TProp> propertyGuardArgument = new GuardArgument<TProp>(getProperty(guardArgument.Value), propertyName);

    validate(propertyGuardArgument);

    return guardArgument;
}
Run Code Online (Sandbox Code Playgroud)

该函数为selected属性创建一个新的GuardArgument,然后将其传递给Action参数以允许您根据需要进行验证.

这也允许无限链接属性,虽然我不确定它是否特别易读.

用法:

Guard.Ensure(someObject, "someObject")
     .IsNotNull()
     .Property(x => x.ChildProp1, "childProp1", childProp1 =>
         childProp1.IsNotNull()
                   .IsLessThan(10)
                   .Property(y => y.InnerChildProperty, "innerChildProperty", innerChildProperty =>
                       innerChildProperty.IsNotNull()
                    )
     )
     .Property(x => x.ChildProp2, "childProp2", childProp2 =>
         childProp2.IsNotNull()
                   .IsGreaterThan(10)
     );
Run Code Online (Sandbox Code Playgroud)