1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181
|
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel.Channels;
using System.ServiceModel;
using System.Reflection;
using System.Threading;
namespace System.ServiceModel.Dispatcher
{
internal class OperationInvokerHandler : BaseRequestProcessorHandler
{
IDuplexChannel duplex;
public OperationInvokerHandler (IChannel channel)
{
duplex = channel as IDuplexChannel;
}
protected override bool ProcessRequest (MessageProcessingContext mrc)
{
RequestContext rc = mrc.RequestContext;
DispatchRuntime dispatchRuntime = mrc.OperationContext.EndpointDispatcher.DispatchRuntime;
DispatchOperation operation = GetOperation (mrc.IncomingMessage, dispatchRuntime);
mrc.Operation = operation;
try {
DoProcessRequest (mrc);
if (!mrc.Operation.IsOneWay)
Reply (mrc, true);
} catch (TargetInvocationException ex) {
mrc.ReplyMessage = BuildExceptionMessage (mrc, ex.InnerException,
dispatchRuntime.ChannelDispatcher.IncludeExceptionDetailInFaults);
if (!mrc.Operation.IsOneWay)
Reply (mrc, true);
ProcessCustomErrorHandlers (mrc, ex);
}
return false;
}
void DoProcessRequest (MessageProcessingContext mrc)
{
DispatchOperation operation = mrc.Operation;
Message req = mrc.IncomingMessage;
object instance = mrc.InstanceContext.GetServiceInstance(req);
object [] parameters, outParams;
BuildInvokeParams (mrc, out parameters);
if (operation.Invoker.IsSynchronous) {
object result = operation.Invoker.Invoke (instance, parameters, out outParams);
HandleInvokeResult (mrc, outParams, result);
} else {
var ar = operation.Invoker.InvokeBegin (instance, parameters, null, null);
object result = operation.Invoker.InvokeEnd (instance, out outParams, ar);
HandleInvokeResult (mrc, outParams, result);
}
}
void Reply (MessageProcessingContext mrc, bool useTimeout)
{
if (duplex != null)
mrc.Reply (duplex, useTimeout);
else
mrc.Reply (useTimeout);
}
DispatchOperation GetOperation (Message input, DispatchRuntime dispatchRuntime)
{
if (dispatchRuntime.OperationSelector != null) {
string name = dispatchRuntime.OperationSelector.SelectOperation (ref input);
foreach (DispatchOperation d in dispatchRuntime.Operations)
if (d.Name == name)
return d;
} else {
string action = input.Headers.Action;
foreach (DispatchOperation d in dispatchRuntime.Operations)
if (d.Action == action)
return d;
}
return dispatchRuntime.UnhandledDispatchOperation;
}
void HandleInvokeResult (MessageProcessingContext mrc, object [] outputs, object result)
{
DispatchOperation operation = mrc.Operation;
mrc.EventsHandler.AfterInvoke (operation);
if (operation.IsOneWay)
return;
Message res = null;
if (operation.SerializeReply)
res = operation.Formatter.SerializeReply (
mrc.OperationContext.EndpointDispatcher.ChannelDispatcher.MessageVersion, outputs, result);
else
res = (Message) result;
res.Headers.CopyHeadersFrom (mrc.OperationContext.OutgoingMessageHeaders);
res.Properties.CopyProperties (mrc.OperationContext.OutgoingMessageProperties);
mrc.ReplyMessage = res;
}
Message CreateActionNotSupported (Message req)
{
FaultCode fc = new FaultCode (
req.Version.Addressing.ActionNotSupported,
req.Version.Addressing.Namespace);
// FIXME: set correct namespace URI
return Message.CreateMessage (req.Version, fc,
String.Format ("action '{0}' is not supported in this service contract.", req.Headers.Action), String.Empty);
}
void BuildInvokeParams (MessageProcessingContext mrc, out object [] parameters)
{
DispatchOperation operation = mrc.Operation;
EnsureValid (operation);
if (operation.DeserializeRequest) {
parameters = operation.Invoker.AllocateInputs ();
operation.Formatter.DeserializeRequest (mrc.IncomingMessage, parameters);
} else
parameters = new object [] { mrc.IncomingMessage };
mrc.EventsHandler.BeforeInvoke (operation);
}
void ProcessCustomErrorHandlers (MessageProcessingContext mrc, Exception ex)
{
var dr = mrc.OperationContext.EndpointDispatcher.DispatchRuntime;
bool shutdown = false;
foreach (var eh in dr.ChannelDispatcher.ErrorHandlers)
shutdown |= eh.HandleError (ex);
if (shutdown)
ProcessSessionErrorShutdown (mrc);
}
void ProcessSessionErrorShutdown (MessageProcessingContext mrc)
{
var dr = mrc.OperationContext.EndpointDispatcher.DispatchRuntime;
var session = mrc.OperationContext.Channel.InputSession;
var dcc = mrc.OperationContext.Channel as IDuplexContextChannel;
if (session == null || dcc == null)
return;
foreach (var h in dr.InputSessionShutdownHandlers)
h.ChannelFaulted (dcc);
}
Message BuildExceptionMessage (MessageProcessingContext mrc, Exception ex, bool includeDetailsInFault)
{
var dr = mrc.OperationContext.EndpointDispatcher.DispatchRuntime;
var cd = dr.ChannelDispatcher;
Message msg = null;
foreach (var eh in cd.ErrorHandlers)
eh.ProvideFault (ex, cd.MessageVersion, ref msg);
if (msg != null)
return msg;
var req = mrc.IncomingMessage;
// FIXME: set correct name
FaultCode fc = new FaultCode (
"InternalServiceFault",
req.Version.Addressing.Namespace);
if (includeDetailsInFault) {
return Message.CreateMessage (req.Version, fc, ex.Message, new ExceptionDetail (ex), req.Headers.Action);
}
string faultString =
@"The server was unable to process the request due to an internal error. The server may be able to return exception details (it depends on the server settings).";
return Message.CreateMessage (req.Version, fc, faultString, req.Headers.Action);
}
void EnsureValid (DispatchOperation operation)
{
if (operation.Invoker == null)
throw new InvalidOperationException (String.Format ("DispatchOperation '{0}' for contract '{1}' requires Invoker.", operation.Name, operation.Parent.EndpointDispatcher.ContractName));
if ((operation.DeserializeRequest || operation.SerializeReply) && operation.Formatter == null)
throw new InvalidOperationException ("The DispatchOperation '" + operation.Name + "' requires Formatter, since DeserializeRequest and SerializeReply are not both false.");
}
}
}
|