CommunicationException aumentado ao retornar um EF4 POCO da operação de serviço WCF
-
25-09-2019 - |
Pergunta
O código a seguir levanta um System.Servicemodel.CommunicationException. Está chamando uma operação de serviço WCF chamada Conecte-se que retorna um EF4 poco:
var client = new AuthServiceReference.AuthServiceClient();
try
{
Console.Write("Trying to logon...");
var session = client.Login("user", "password"); // throws CommunicationException
Console.WriteLine("done!");
Console.WriteLine("Session ID: {0}. Expires {1}",
session.Id, session.UtcExpires.ToLocalTime());
}
finally
{
client.Close();
}
Eu tenho depurado e pesquisando horas tentando descobrir Por quê Isso acontece e como corrigi -lo. O que encontrei até agora:
- Este é provavelmente um problema de serialização
- Quando eu removo o DataMemberattribute de Sessão Classe Proprietário Membro, a exceção desaparece, mas isso significa que não será serializada.
Ficaria agradecido se alguém pudesse esclarecer esse assunto.
Abaixo está o código para o contrato de serviço e as classes POCO:
[ServiceContract]
public interface IAuthService
{
[OperationContract]
Session Login(string username, string passwordHash);
[OperationContract]
void Logout(Guid sessionId);
}
[DataContract]
public class Session
{
[DataMember]
public Guid Id { get; set; }
[DataMember]
public DateTime UtcCreated { get; set; }
[DataMember]
public DateTime UtcExpires { get; set; }
[DataMember] // serializes correctly if commented out
public virtual User Owner { get; set; }
public static Session Create(User owner)
{
return new Session
{
Owner = owner,
Id = Guid.NewGuid(),
UtcCreated = DateTime.UtcNow,
UtcExpires = DateTime.UtcNow.AddDays(1)
};
}
}
[DataContract]
public class User
{
[DataMember]
public int Id { get; set; }
[DataMember]
public string Name { get; set; }
[DataMember]
public string PasswordHash { get; set; }
[DataMember]
public string PasswordSalt { get; set; }
[DataMember]
public bool IsContributor { get; set; }
[DataMember]
public bool IsConfirmed { get; set; }
[DataMember]
public bool IsAdmin { get; set; }
[DataMember]
public string Email { get; set; }
[DataMember]
public virtual ICollection<Post> Posts { get; set; }
[DataMember]
public virtual ICollection<Comment> Comments { get; set; }
}
Solução
Acontece que esse é um problema conhecido ao serializar proxies POCO com o WCF. Há uma MSDN WalkHhough Isso explica como contorná -lo usando System.Data.Objects.ProxyDatacontractResolver.
Essencialmente, você cria uma nova classe chamada APLICARDATACONTRACTRESOLVERATTRIBUTIVO e aplique -o aos métodos de serviço que retornam PoCos:
[ServiceContract]
public interface IAuthService
{
[OperationContract]
[ApplyDataContractResolver]
Session Login(string username, string passwordHash);
}
using System;
using System.Data.Objects;
using System.ServiceModel.Channels;
using System.ServiceModel.Description;
using System.ServiceModel.Dispatcher;
namespace WcfExampleBlog.Services
{
public class ApplyDataContractResolverAttribute : Attribute, IOperationBehavior
{
#region IOperationBehavior Members
public void AddBindingParameters(OperationDescription description, BindingParameterCollection parameters)
{
}
public void ApplyClientBehavior(OperationDescription description, ClientOperation proxy)
{
var dataContractSerializerOperationBehavior =
description.Behaviors.Find<DataContractSerializerOperationBehavior>();
dataContractSerializerOperationBehavior.DataContractResolver =
new ProxyDataContractResolver();
}
public void ApplyDispatchBehavior(OperationDescription description, DispatchOperation dispatch)
{
var dataContractSerializerOperationBehavior =
description.Behaviors.Find<DataContractSerializerOperationBehavior>();
dataContractSerializerOperationBehavior.DataContractResolver =
new ProxyDataContractResolver();
}
public void Validate(OperationDescription description)
{
// Do validation.
}
#endregion
}
}
Outras dicas
Presumo que a classe 'Usuário' seja uma classe personalizada? Nesse caso, você precisa adicionar isso logo abaixo do atributo ServiceContract:
[KnownType(typeof(User))]
Você também precisaria configurar seus atributos [Datamember] e [ServiceContract] na classe de usuário.