使用Interceptor进行WCF参数验证

mga*_*mer 9 validation parameters service wcf

我有一个WCF服务,其操作都需要MyServiceRequest参数(或派生类型)并返回MyServiceResponse(或dervived类型),即:

    [OperationContract]
    MySeviceResponse FindAppointments(FindAppointmentRequest request);

    [OperationContract]
    MyServiceResponse MakeAnAppointment(MakeAnAppointmentRequest request);

    [OperationContract]
    MyServiceResponse RegisterResource(RegisterResourceRequest request);
Run Code Online (Sandbox Code Playgroud)

FindAppointmentRequest,MakeAnAppointmentRequest和RegisterResourceRequest扩展MyServiceRequest,其中包含UserName和UserPassword属性.

如果Request中有错误的UserName/UserPassword对,则此方法都不会正确执行.

我想创建一个拦截器,它不仅检查给定的UserName/UserPassword对是否正确(使用IParameterInspector非常简单),而且还返回扩展MyServiceResponse的ErrorRespone类型的客户端对象.

IParameterInspector可以阻止服务执行请求的方法并返回ErrorResponse strightaway吗?

Dar*_*rov 22

IParameterInspector可以通过抛出异常来阻止执行操作.这是一个例子.

定义自定义故障合同:

public class DataAccessFaultContract
{
    public string ErrorMessage { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

然后检查员:

public class InspectorAttribute : Attribute, IParameterInspector, IOperationBehavior
{
    public void AfterCall(string operationName, object[] outputs, object returnValue, object correlationState)
    {

    }

    public object BeforeCall(string operationName, object[] inputs)
    {
        MyServiceRequest request = null;
        if (inputs != null && inputs.Length > 0)
        {
            request = inputs[0] as MyServiceRequest;
        }

        if (request != null && request.Username != "user" && request.Password != "secret")
        {
            var fc = new DataAccessFaultContract{ ErrorMessage = "Invalid user" };
            throw new FaultException<DataAccessFaultContract>(fc);
        }
        return null;
    }

    public void AddBindingParameters(OperationDescription operationDescription, BindingParameterCollection bindingParameters)
    {
    }

    public void ApplyClientBehavior(OperationDescription operationDescription, ClientOperation clientOperation)
    {
    }

    public void ApplyDispatchBehavior(OperationDescription operationDescription, DispatchOperation dispatchOperation)
    {
        dispatchOperation.ParameterInspectors.Add(this);
    }

    public void Validate(OperationDescription operationDescription)
    {
    }
}
Run Code Online (Sandbox Code Playgroud)

最后用适当的属性装饰你的操作:

[ServiceContract]
public interface IMyServiceContract
{
    [Inspector]
    [FaultContract(typeof(DataAccessFaultContract))]
    [OperationContract]
    MySeviceResponse FindAppointments(FindAppointmentRequest request);

    ...
}
Run Code Online (Sandbox Code Playgroud)

调用服务时,客户端可以检查FaultException:

try
{
    var response = client.FindAppointments(request);
}
catch (FaultException<DataAccessFaultContract> ex)
{
    string errorMessage = ex.Detail.ErrorMessage;
    // ...
}
Run Code Online (Sandbox Code Playgroud)