CommunicationException elevó al devolver un POCO EF4 de operación de servicio WCF
-
25-09-2019 - |
Pregunta
El siguiente código plantea un System.ServiceModel.CommunicationException . Está llamando a una operación de servicio WCF llamado Iniciar sesión que devuelve un POCO EF4:
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();
}
he estado depuración y de buscar por horas tratando de averiguar ¿Por qué esto ocurre y cómo solucionarlo. Lo que he encontrado hasta ahora:
- Este es probablemente un problema de serialización
- Cuando quito el DataMemberAttribute del Sesión clase de propietario miembro, desaparece la excepción, pero esto significa que no va a ser serializado.
Yo estaría agradecido si alguien puede arrojar algo de luz sobre esta cuestión.
A continuación se muestra el código para el contrato de servicios y clases 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; }
}
Solución
Resulta que este es un problema conocido cuando se serializa proxies POCO con WCF. Hay una MSDN walkthough que explica cómo trabajar alrededor de ella utilizando System.Data.Objects.ProxyDataContractResolver .
En esencia, se crea una nueva clase llamada ApplyDataContractResolverAttribute y aplicarlo a los métodos de servicio que regresan 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
}
}
Otros consejos
Estoy suponiendo que la clase 'Usuario' es una clase personalizada? Si es así, es necesario agregar esta justo debajo del atributo ServiceContract:
[KnownType(typeof(User))]
También tendría que configurar su [DataMember] y [ServiceContract] atributos de la clase de usuario también.