Ссылка на модель MVC через WCF с использованием DataAnnotations - PullRequest
3 голосов
/ 30 октября 2011

Я использую WCF в ASP.Net 4.0 с EntityFramework 4.1 и MVC3 с Razor.

DataAnnotations отлично работают в следующем сценарии:

Наличие класса POCO в другой DLL с EFи затем заполняя модель прямым вызовом из моего уровня представления.Если я попытаюсь отредактировать адрес электронной почты в сетке внешнего интерфейса и преднамеренно вставить поддельное электронное сообщение, для этого свойства появится ошибка DataAnnotation .

Однако, если я использую подход с WCF вызов среднего уровня (вместо прямого вызова), данные возвращаются и обновляются очень хорошо, за исключением того факта, что Аннотации данных не работают в сценарии такого типа .Если я попытаюсь отредактировать адрес электронной почты в сетке внешнего интерфейса и преднамеренно вставить поддельное письмо, DataAnnotation не появится для этого свойства.

Как я могу его получитьработать с использованием WCF?Должен быть какой-то атрибут WCF, который мне не хватает, когда он не распознает аннотации данных.

Ниже приведен соответствующий код:

2-уровневый сценарий:

Модель

 using System;
using System.Collections.Generic;
//using System.ServiceModel;
//using System.Runtime.Serialization;
using System.ComponentModel.DataAnnotations;
using DataAnnotationsExtensions;

namespace YeagerTechModel
{
    public class Customer
    {
        public Customer()
        {
            this.Projects = new HashSet<Project>();
        }


        public short CustomerID { get; set; }


        [Required]
        [StringLength(50)]
        [DataType(DataType.EmailAddress)]
        [Email]
        public string Email { get; set; }


        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string Company { get; set; }


        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string FirstName { get; set; }


        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string LastName { get; set; }


        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string Address1 { get; set; }


        [StringLength(50)]
        [DataType(DataType.Text)]
        public string Address2 { get; set; }


        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string City { get; set; }


        [StringLength(2, MinimumLength = 2, ErrorMessage = "Must have a length of 2.")]
        [DataType(DataType.Text)]
        public string State { get; set; }


        [StringLength(10)]
        [DataType(DataType.Text)]
        [RegularExpression(@"^\d{5}(-\d{4})?$", ErrorMessage = "Invalid Zip")]
        public string Zip { get; set; }


        [StringLength(12)]
        [DataType(DataType.PhoneNumber)]
        [RegularExpression(@"^\s*([\(]?)\[?\s*\d{3}\s*\]?[\)]?\s*[\-]?[\.]?\s*\d{3}\s*[\-]?[\.]?\s*\d{4}$", ErrorMessage = "Invalid Phone")]
        public string HomePhone { get; set; }


        [StringLength(12)]
        [DataType(DataType.PhoneNumber)]
        [RegularExpression(@"^\s*([\(]?)\[?\s*\d{3}\s*\]?[\)]?\s*[\-]?[\.]?\s*\d{3}\s*[\-]?[\.]?\s*\d{4}$", ErrorMessage = "Invalid Phone")]
        public string CellPhone { get; set; }


        [StringLength(100)]
        [DataType(DataType.Url)]
        [Url]
        public string Website { get; set; }


        [StringLength(50)]
        [DataType(DataType.EmailAddress)]
        [Email]
        public string IMAddress { get; set; }


        public System.DateTime CreatedDate { get; set; }


        public Nullable<System.DateTime> UpdatedDate { get; set; }


        public virtual ICollection<Project> Projects { get; set; }
    }
} 

Контроллер:

[GridAction]
        public ActionResult Index()
        {
            ViewData["ErrCode"] = string.Empty;

            if (HttpContext.User.IsInRole("Admin"))
            {
                try
                {
                    //IEnumerable<YeagerTechWcfService.Customer> customerList = db.GetCustomers();

                    //IEnumerable<YeagerTechModel.Customer> customerList = db.GetCustomers();
                    DbContext.Configuration.ProxyCreationEnabled = false;
                    IEnumerable<Customer> customerList = DbContext.Customers.Where(p => p.CustomerID > 0);

                    if (DbContext.Database.Connection != null)
                    {
                        if (DbContext.Database.Connection.State != System.Data.ConnectionState.Closed)
                        {
                            DbContext.Database.Connection.Close();
                            DbContext.Database.Connection.Dispose();
                        }
                    }

                    return View(new GridModel<YeagerTechModel.Customer>
                    {
                        Data = customerList

                    });
                }
                catch (Exception ex)
                {
                    throw ex;
                }
            }
            else
            {
                //HttpCookie cn = Request.Cookies["strCookieName"];
                //if (cn != null)
                //{
                //    YeagerTechWcfService.Customer cust = db.GetCustomerID(Convert.ToInt16(cn.Value),false);

                //    if (cust != null)
                //    {
                //        return View(new GridModel<YeagerTechWcfService.Customer>
                //        {
                //            //Data = cust
                //        });
                //    }
                //    else
                //        return View(new GridModel<YeagerTechWcfService.Customer>());
                //}
                //else
                //{
                //    TempData["ErrCode"] = "CustView";
                return RedirectToAction("Index", "Home");
                //}
            }
        }

Вид:

@model Telerik.Web.Mvc.GridModel<YeagerTechModel.Customer>
@{
    ViewBag.Title = "Customer Index";
}
<h2>
    Customer Index</h2>
@(  Html.Telerik().Grid<YeagerTechModel.Customer>(Model.Data)
      .Name("Customers")
            .DataKeys(dataKeys => dataKeys.Add(o => o.CustomerID)
                                            .RouteKey("CustomerID"))
                .ToolBar(commands => commands.Insert().ButtonType(GridButtonType.Text).ImageHtmlAttributes(new { style = "margin-left:0" }))
      .Columns(columns =>
            {
                columns.Bound(o => o.CustomerID).Hidden(true);
                columns.Command(commands =>
                {
                    commands.Edit().ButtonType(GridButtonType.Text);
                }).Width(200).Title("Command");
                columns.Bound(o => o.Email).Width(200).Filterable(false);
                columns.Bound(o => o.Company).Width(200).Filterable(false);
                columns.Bound(o => o.FirstName).Width(100).Title("FName").Filterable(false);
                columns.Bound(o => o.LastName).Width(100).Title("LName").Filterable(false);
                columns.Bound(o => o.Address1).Width(200).Title("Addr1").Filterable(false).Sortable(false);
                columns.Bound(o => o.Address2).Width(100).Title("Addr2").Filterable(false).Sortable(false);
                columns.Bound(o => o.City).Width(100);
                columns.Bound(o => o.State).Width(40).Title("ST");
                columns.Bound(o => o.Zip).Width(60);
                columns.Bound(o => o.HomePhone).Width(120).Filterable(false).Sortable(false);
                columns.Bound(o => o.CellPhone).Width(120).Filterable(false).Sortable(false);
                columns.Bound(o => o.Website).Width(100).Filterable(false).Sortable(false);
                columns.Bound(o => o.IMAddress).Width(100).Filterable(false).Sortable(false);
                columns.Bound(o => o.CreatedDate).Format("{0:MM/dd/yyyy}").ReadOnly(true).Width(120).Filterable(false).Sortable(false);
                columns.Bound(o => o.UpdatedDate).Format("{0:MM/dd/yyyy}").ReadOnly(true).Width(120).Filterable(false).Sortable(false);
            }).DataBinding(dataBinding =>
                dataBinding.Ajax()
                        .Insert("_InsertAjaxEditing", "Customer")
                        .Update("_SaveAjaxEditing", "Customer"))
    .Editable(editing => editing.Mode(GridEditMode.InLine))
    .Pageable()
    .Sortable()
    .Filterable()
    .Scrollable()
 )

Сценарий NTier Модель:

using System;
using System.Collections.Generic;
using System.ServiceModel;
using System.Runtime.Serialization;
using System.ComponentModel.DataAnnotations;
using DataAnnotationsExtensions;

namespace YeagerTechModel
{
    [Serializable]
    [DataContract(IsReference = true)]
    public class Customer
    {
        public Customer()
        {
            this.Projects = new HashSet<Project>();
        }

        [DataMember]
        public short CustomerID { get; set; }

        [DataMember]
        [Required]
        [StringLength(50)]
        [DataType(DataType.EmailAddress)]
        [Email]
        public string Email { get; set; }

        [DataMember]
        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string Company { get; set; }

        [DataMember]
        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string FirstName { get; set; }

        [DataMember]
        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string LastName { get; set; }

        [DataMember]
        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string Address1 { get; set; }

        [DataMember]
        [StringLength(50)]
        [DataType(DataType.Text)]
        public string Address2 { get; set; }

        [DataMember]
        [StringLength(50, MinimumLength = 3, ErrorMessage = "Must have a minimum length of 3.")]
        [DataType(DataType.Text)]
        public string City { get; set; }

        [DataMember]
        [StringLength(2, MinimumLength = 2, ErrorMessage = "Must have a length of 2.")]
        [DataType(DataType.Text)]
        public string State { get; set; }

        [DataMember]
        [StringLength(10)]
        [DataType(DataType.Text)]
        [RegularExpression(@"^\d{5}(-\d{4})?$", ErrorMessage = "Invalid Zip")]
        public string Zip { get; set; }

        [DataMember]
        [StringLength(12)]
        [DataType(DataType.PhoneNumber)]
        [RegularExpression(@"^\s*([\(]?)\[?\s*\d{3}\s*\]?[\)]?\s*[\-]?[\.]?\s*\d{3}\s*[\-]?[\.]?\s*\d{4}$", ErrorMessage = "Invalid Phone")]
        public string HomePhone { get; set; }

        [DataMember]
        [StringLength(12)]
        [DataType(DataType.PhoneNumber)]
        [RegularExpression(@"^\s*([\(]?)\[?\s*\d{3}\s*\]?[\)]?\s*[\-]?[\.]?\s*\d{3}\s*[\-]?[\.]?\s*\d{4}$", ErrorMessage = "Invalid Phone")]
        public string CellPhone { get; set; }

        [DataMember]
        [StringLength(100)]
        [DataType(DataType.Url)]
        [Url]
        public string Website { get; set; }

        [DataMember]
        [StringLength(50)]
        [DataType(DataType.EmailAddress)]
        [Email]
        public string IMAddress { get; set; }

        [DataMember]
        public System.DateTime CreatedDate { get; set; }

        [DataMember]
        public Nullable<System.DateTime> UpdatedDate { get; set; }

        [DataMember]
        public virtual ICollection<Project> Projects { get; set; }
    }
}

Вызов WCF:

public IEnumerable<Customer> GetCustomers()
        {
            YeagerTechEntities DbContext = new YeagerTechEntities();

            DbContext.Configuration.ProxyCreationEnabled = false;

            IQueryable<Customer> customer = DbContext.Customers.Where(p => p.CustomerID > 0);

            CloseConnection(DbContext);

            return customer;
        }

Контроллер:

[GridAction]
        public ActionResult Index()
        {
            ViewData["ErrCode"] = string.Empty;

            if (HttpContext.User.IsInRole("Admin"))
            {
                try
                {
                    IEnumerable<YeagerTechWcfService.Customer> customerList = db.GetCustomers();

                    return View(new GridModel<YeagerTechWcfService.Customer>
                    {
                        Data = customerList
                    });
                }
                catch (Exception ex)
                {
                    throw ex;
                }
            }
            else
            {
                return RedirectToAction("Index", "Home");
            }
        }

Вид:

@model Telerik.Web.Mvc.GridModel<YeagerTech.YeagerTechWcfService.Customer>
@{
    ViewBag.Title = "Customer Index";
}
<h2>
    Customer Index</h2>
@(  Html.Telerik().Grid<YeagerTech.YeagerTechWcfService.Customer>(Model.Data)
      .Name("Customers")
            .DataKeys(dataKeys => dataKeys.Add(o => o.CustomerID)
                                            .RouteKey("CustomerID"))
                .ToolBar(commands => commands.Insert().ButtonType(GridButtonType.Text).ImageHtmlAttributes(new { style = "margin-left:0" }))
      .Columns(columns =>
            {
                columns.Bound(o => o.CustomerID).Hidden(true);
                columns.Command(commands =>
                {
                    commands.Edit().ButtonType(GridButtonType.Text);
                }).Width(200).Title("Command");
                columns.Bound(o => o.Email).Width(200).Filterable(false);
                columns.Bound(o => o.Company).Width(200).Filterable(false);
                columns.Bound(o => o.FirstName).Width(100).Title("FName").Filterable(false);
                columns.Bound(o => o.LastName).Width(100).Title("LName").Filterable(false);
                columns.Bound(o => o.Address1).Width(200).Title("Addr1").Filterable(false).Sortable(false);
                columns.Bound(o => o.Address2).Width(100).Title("Addr2").Filterable(false).Sortable(false);
                columns.Bound(o => o.City).Width(100);
                columns.Bound(o => o.State).Width(40).Title("ST");
                columns.Bound(o => o.Zip).Width(60);
                columns.Bound(o => o.HomePhone).Width(120).Filterable(false).Sortable(false);
                columns.Bound(o => o.CellPhone).Width(120).Filterable(false).Sortable(false);
                columns.Bound(o => o.Website).Width(100).Filterable(false).Sortable(false);
                columns.Bound(o => o.IMAddress).Width(100).Filterable(false).Sortable(false);
                columns.Bound(o => o.CreatedDate).Format("{0:MM/dd/yyyy}").ReadOnly(true).Width(120).Filterable(false).Sortable(false);
                columns.Bound(o => o.UpdatedDate).Format("{0:MM/dd/yyyy}").ReadOnly(true).Width(120).Filterable(false).Sortable(false);
            }).DataBinding(dataBinding =>
                dataBinding.Ajax()
                        .Insert("_InsertAjaxEditing", "Customer")
                        .Update("_SaveAjaxEditing", "Customer"))
    .Editable(editing => editing.Mode(GridEditMode.InLine))
    .Pageable()
    .Sortable()
    .Filterable()
    .Scrollable()
 )

1 Ответ

5 голосов
/ 30 октября 2011

Нет, вы не упускаете ничего, кроме теории о технологиях, которые вы используете.WCF - это технология для предоставления услуг, и она делает это совместимым образом.Контракт с данными, предоставляемый сервисом, просто создан для данных.Не имеет значения, сколько необычных атрибутов вы используете в контракте или сколько пользовательских логик вы помещаете в get и set методы свойства.На стороне клиента вы всегда видите это:

[DataContract(IsReference = true)]
public partial class Customer
{
    [DataMember]
    public short CustomerID { get; set; }

    [DataMember]
    public string Email { get; set; }

    [DataMember]
    public string Company { get; set; }

    [DataMember]
    public string FirstName { get; set; }

    [DataMember]
    public string LastName { get; set; }

    [DataMember]
    public string Address1 { get; set; }

    [DataMember]
    public string Address2 { get; set; }

    [DataMember]
    public string City { get; set; }

    [DataMember]
    public string State { get; set; }

    [DataMember]
    public string Zip { get; set; }

    [DataMember]
    public string HomePhone { get; set; }

    [DataMember]
    public string CellPhone { get; set; }

    [DataMember]
    public string Website { get; set; }

    [DataMember]
    public string IMAddress { get; set; }

    [DataMember]
    public System.DateTime CreatedDate { get; set; }

    [DataMember]
    public Nullable<System.DateTime> UpdatedDate { get; set; }

    [DataMember]
    public Project[] Projects { get; set; }
}

Причина этого заключается в том, что, как только вы открываете сервис, он выставляет все свои контракты взаимодействующим образом - контракты на обслуживание и операции описываются WSDL, а контракты на данныеописанный XSD.XSD может описывать только структуру данных, но не логику.Сама проверка может быть ограниченным образом описана в XSD, но генератор .NET XSD этого не делает.Как только вы добавите ссылку на службу в службу WCF, прокси-генератор примет WSDL и XSD в качестве источника и снова создаст ваши классы без всех этих атрибутов.

Если вы хотите иметь проверку на стороне клиента, вы должны в первую очередь реализовать эту проверку на стороне клиента - это можно сделать с помощью классов собеседников для частичных классов, используемых прокси-сервером WCF.Если вы не хотите использовать этот способ, вы должны совместно использовать сборку со своими сущностями между вашим клиентом WCF и службой WCF и повторно использовать эти типы при добавлении ссылки на службу.Это создаст тесную связь между вашим сервисом и приложением ASP.NET MVC.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...