3

ModelState を使用してフォームを検証しようとしていますが、ModelState は常に有効と表示されます。例: 2 人のフォームモデルをそれぞれ同じ SSN で保存しようとすると、ModelState が有効な値を返します。IValidatableObject を使用してフォームモデルを検証しています。私が間違っているかもしれないアイデアはありますか?MVC 3 で .Net 4.0 を使用しています。

public JsonResult LoadOccupantsDetailedInformation()
{
    //Load the personsFormModel with data
    return new JsonpResult(personsFormModel, JsonRequestBehavior.AllowGet);
}

[HttpPost]
public ActionResult SaveOccupantsDetailedInformation(
PersonsFormModel personsFormModel)
    {
//This line is always returning true even if I pass 2 persons with the same ssn
        if (ModelState.IsValid == false)
        {
            var errorList = ModelState.ToDictionary(
                      kvp => kvp.Key,
                      kvp => kvp.Value.Errors.Select(e => e.ErrorMessage).ToArray()
                );
            return Json(new { Errors = errorList });
        }
        //Save the data in personsFormModel to database
        return Json(new JsonResultViewModel { Success = true });
    }


public partial class PersonsFormModel : List<PersonFormModel>, IValidatableObject
{
    public IEnumerable<System.ComponentModel.DataAnnotations.ValidationResult> Validate(
    ValidationContext validationContext)
    {
        var validationResults
            = new List<System.ComponentModel.DataAnnotations.ValidationResult>();
        if (this.SSNs.Count() > this.SSNs.Distinct().Count())
        {
            validationResults.Add(new System.ComponentModel.DataAnnotations.ValidationResult(
            "All the persons in the household should have a unique SSN\\ITIN number",
                new[] { "SSN" }));
        }
        return validationResults;
    }
    private IEnumerable<string> SSNs
    {
        get
        {
            return this.Select(element => element.SSN);
        }
    }
}
    public class PrequalifyOccupantListPersonDetailedFormModel
{
    [Required(ErrorMessage = "SSN is required")]
    public string SSN { get; set; }
}
4

2 に答える 2

0

これが私が抱えていた問題を修正した方法です:

***Controller:***
public JsonResult LoadOccupantsDetailedInformation()
{
    //Load the personsFormModel with data
    return new JsonpResult(personsFormModel, JsonRequestBehavior.AllowGet);
}

[HttpPost]
public ActionResult SaveOccupantsDetailedInformation(
PersonsFormModel personsFormModel)
    {
//This line is always returning true even if I pass 2 persons with the same ssn
        if (ModelState.IsValid == false)
        {
            var errorList = ModelState.ToDictionary(
                      kvp => kvp.Key,
                      kvp => kvp.Value.Errors.Select(e => e.ErrorMessage).ToArray()
                );
            return Json(new { Errors = errorList });
        }
        //Save the data in personsFormModel to database
        return Json(new JsonResultViewModel { Success = true });
    }

***Model:***
public class PersonsFormModel : IValidatableObject
{
    public PersonsFormModel()
    {
        this.Instance = new List<PersonFormModel>();
    }

    public List<PrequalifyOccupantListPersonFormModel> Instance { get; set; }

    public void Add(PersonFormModel personFormModel)
    {
        Instance.Add(personFormModel);
    }

    public IEnumerable<System.ComponentModel.DataAnnotations.ValidationResult> Validate(
    ValidationContext validationContext)
    {
        var validationResults
            = new List<System.ComponentModel.DataAnnotations.ValidationResult>();
        if (this.SSNs.Count() > this.SSNs.Distinct().Count())
        {
            validationResults.Add(new System.ComponentModel.DataAnnotations.ValidationResult(
            "All the persons in the household should have a unique SSN\\ITIN number",
                new[] { "SSN" }));
        }
        return validationResults;
    }
    private IEnumerable<string> SSNs
    {
        get
        {
            return Instance.Select(element => element.SSN);
        }
    }
}

public class PersonFormModel
{
    [Required(ErrorMessage = "SSN is required")]
    public string SSN { get; set; }
}

View: 

// Note, I am passing the "Instance" here. 

 var postData = JSON.stringify({ Instance: list });

            RealPage.DataManager.POST({
                url: "/Approval/SaveOccupantsDetailedInformation"
                , data: postData
                , success: function (data) {
                    if (data && data.Success) {

                    } 
                }
            }); 
于 2012-10-04T23:30:06.163 に答える
0

あなたが検証していることを確認するためのテストを書きましたが、それは良さそうです。

        [Test]
        public void Should_validate_person_form_model()
        {
            var input = new PersonsFormModel();
            input.Add(new PersonFormModel { SSN = "33" });
            input.Add(new PersonFormModel { SSN = "33" });
            _controller.ViewData.ModelState.Clear();

            var results = new List<ValidationResult>();

            bool isValid = Validator.TryValidateObject(input,
                new ValidationContext(input, null, null),
                results,true);

            Assert.IsTrue(!isValid, "Cannot have duplicates.");
        }

したがって、呼び出されていないか、データが有効です。

前者をテストするには、Validate 内にブレーク ポイントを配置し、ヒットしていることを確認できます。

後者をテストするには、投稿された正確なデータと関連する完全なモデルを提供できます。

于 2012-10-03T08:22:22.437 に答える