その記事で提供されているソリューションでは、検証ロジックとサービス ロジックが混在しています。これらは 2 つの懸念事項であり、分離する必要があります。アプリケーションが成長すると、検証ロジックが複雑になり、サービス レイヤー全体で重複することがすぐにわかります。したがって、私は別のアプローチを提案したいと思います。
まず第一に、検証エラーが発生したときにサービス層に例外をスローさせる方がはるかに優れています。これにより、エラーのチェックがより明確になり、忘れにくくなります。これにより、エラーの処理方法はプレゼンテーション層に委ねられます。次のリストは、ProductController
このアプローチを使用する を示しています。
public class ProductController : Controller
{
private readonly IProductService service;
public ProductController(IProductService service) => this.service = service;
public ActionResult Create(
[Bind(Exclude = "Id")] Product productToCreate)
{
try
{
this.service.CreateProduct(productToCreate);
}
catch (ValidationException ex)
{
this.ModelState.AddModelErrors(ex);
return View();
}
return RedirectToAction("Index");
}
}
public static class MvcValidationExtension
{
public static void AddModelErrors(
this ModelStateDictionary state, ValidationException exception)
{
foreach (var error in exception.Errors)
{
state.AddModelError(error.Key, error.Message);
}
}
}
クラス自体に検証を含めるべきではProductService
ありませんが、それを検証に特化したクラスに委譲する必要がありIValidationProvider
ます。
public interface IValidationProvider
{
void Validate(object entity);
void ValidateAll(IEnumerable entities);
}
public class ProductService : IProductService
{
private readonly IValidationProvider validationProvider;
private readonly IProductRespository repository;
public ProductService(
IProductRespository repository,
IValidationProvider validationProvider)
{
this.repository = repository;
this.validationProvider = validationProvider;
}
// Does not return an error code anymore. Just throws an exception
public void CreateProduct(Product productToCreate)
{
// Do validation here or perhaps even in the repository...
this.validationProvider.Validate(productToCreate);
// This call should also throw on failure.
this.repository.CreateProduct(productToCreate);
}
}
ただし、 thisIValidationProvider
はそれ自体を検証するのではなく、特定のタイプの検証に特化した検証クラスに検証を委譲する必要があります。オブジェクト (またはオブジェクトのセット) が有効でない場合、検証プロバイダーは をスローする必要ValidationException
があります。これは、コール スタックの上位でキャッチできます。プロバイダーの実装は次のようになります。
sealed class ValidationProvider : IValidationProvider
{
private readonly Func<Type, IValidator> validatorFactory;
public ValidationProvider(Func<Type, IValidator> validatorFactory)
{
this.validatorFactory = validatorFactory;
}
public void Validate(object entity)
{
IValidator validator = this.validatorFactory(entity.GetType());
var results = validator.Validate(entity).ToArray();
if (results.Length > 0)
throw new ValidationException(results);
}
public void ValidateAll(IEnumerable entities)
{
var results = (
from entity in entities.Cast<object>()
let validator = this.validatorFactory(entity.GetType())
from result in validator.Validate(entity)
select result)
.ToArray();
if (results.Length > 0)
throw new ValidationException(results);
}
}
実際の検証を行うインスタンスに依存ValidationProvider
します。IValidator
プロバイダー自体は、これらのインスタンスを作成する方法を知りませんが、そのために注入されたFunc<Type, IValidator>
デリゲートを使用します。このメソッドには、コンテナー固有のコードが含まれます。たとえば、Ninject の場合は次のようになります。
var provider = new ValidationProvider(type =>
{
var valType = typeof(Validator<>).MakeGenericType(type);
return (IValidator)kernel.Get(valType);
});
このスニペットはValidator<T>
クラスを示しています。このクラスはすぐに示します。まず、ValidationProvider
は次のクラスに依存します。
public interface IValidator
{
IEnumerable<ValidationResult> Validate(object entity);
}
public class ValidationResult
{
public ValidationResult(string key, string message)
{
this.Key = key;
this.Message = message;
}
public string Key { get; }
public string Message { get; }
}
public class ValidationException : Exception
{
public ValidationException(ValidationResult[] r) : base(r[0].Message)
{
this.Errors = new ReadOnlyCollection<ValidationResult>(r);
}
public ReadOnlyCollection<ValidationResult> Errors { get; }
}
上記のコードはすべて、検証を行うために必要な配管です。検証するエンティティごとに検証クラスを定義できるようになりました。ただし、DI コンテナーを少しでも支援するには、バリデーターのジェネリック基本クラスを定義する必要があります。これにより、検証タイプを登録できます。
public abstract class Validator<T> : IValidator
{
IEnumerable<ValidationResult> IValidator.Validate(object entity)
{
if (entity == null) throw new ArgumentNullException("entity");
return this.Validate((T)entity);
}
protected abstract IEnumerable<ValidationResult> Validate(T entity);
}
ご覧のとおり、この抽象クラスは を継承していIValidator
ます。ProductValidator
から派生するクラスを定義できるようになりましたValidator<Product>
。
public sealed class ProductValidator : Validator<Product>
{
protected override IEnumerable<ValidationResult> Validate(
Product entity)
{
if (entity.Name.Trim().Length == 0)
yield return new ValidationResult(
nameof(Product.Name), "Name is required.");
if (entity.Description.Trim().Length == 0)
yield return new ValidationResult(
nameof(Product.Description), "Description is required.");
if (entity.UnitsInStock < 0)
yield return new ValidationResult(
nameof(Product.UnitsInStock),
"Units in stock cnnot be less than zero.");
}
}
ご覧のとおり、このProductValidator
クラスは C#yield return
ステートメントを使用しているため、検証エラーがよりスムーズに返されます。
これをすべて機能させるために最後に行うべきことは、Ninject 構成をセットアップすることです。
kernel.Bind<IProductService>().To<ProductService>();
kernel.Bind<IProductRepository>().To<L2SProductRepository>();
Func<Type, IValidator> validatorFactory = type =>
{
var valType = typeof(Validator<>).MakeGenericType(type);
return (IValidator)kernel.Get(valType);
};
kernel.Bind<IValidationProvider>()
.ToConstant(new ValidationProvider(validatorFactory));
kernel.Bind<Validator<Product>>().To<ProductValidator>();
本当に終わりですか?場合によります。上記の構成の欠点は、ドメイン内のエンティティごとにValidator<T>
実装が必要になることです。おそらくほとんどの実装が空になる場合でも。
この問題は、次の 2 つの方法で解決できます。
- 自動登録を使用して、特定のアセンブリから動的にすべての実装を自動的にロードできます。
- 登録が存在しない場合は、デフォルトの実装に戻すことができます。
このようなデフォルトの実装は次のようになります。
sealed class NullValidator<T> : Validator<T>
{
protected override IEnumerable<ValidationResult> Validate(T entity)
{
return Enumerable.Empty<ValidationResult>();
}
}
NullValidator<T>
これは次のように構成できます。
kernel.Bind(typeof(Validator<>)).To(typeof(NullValidator<>));
これを行った後、Ninject はNullValidator<Customer>
aValidator<Customer>
が要求され、特定の実装が登録されていない場合に a を返します。
現在欠けている最後のものは、自動登録です。これにより、実装ごとに登録を追加する必要がValidator<T>
なくなり、Ninject がアセンブリを動的に検索できるようになります。この例は見つかりませんでしたが、Ninject でこれができると思います。
更新:これらのタイプを自動登録する方法については、Kayess の回答を参照してください。
最後に 1 つ注意してください。これを行うには、かなりの量の配管が必要です。そのため、プロジェクトがかなり小さい (そしてそのままである) 場合、このアプローチではオーバーヘッドが大きくなりすぎる可能性があります。しかし、プロジェクトが大きくなると、このように柔軟な設計ができると非常に嬉しくなります。検証を変更したい場合に何をしなければならないかを考えてください (たとえば、Validation Application Block または DataAnnotations)。あなたがしなければならない唯一のことは、の実装を書くことですNullValidator<T>
(その場合は名前を変更しDefaultValidator<T>
ます。それ以外にも、他の検証テクノロジーでは実装が難しい追加の検証用にカスタム検証クラスを使用することも可能です.
IProductService
やなどの抽象化の使用はICustomerService
SOLID の原則に違反するため、このパターンからユース ケースを抽象化するパターンに移行することでメリットが得られる可能性があることに注意してください。
更新:こちらの q/a もご覧ください。同じ記事に関するフォローアップの質問について説明します。