157 lines
4.5 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceModel.Channels;
namespace System.ServiceModel.Dispatcher
{
internal class MessageProcessingContext
{
OperationContext operation_context;
RequestContext request_context;
Message incoming_message;
Message reply_message;
InstanceContext instance_context;
Exception processingException;
DispatchOperation operation;
UserEventsHandler user_events_handler;
IChannel reply_or_input;
public MessageProcessingContext (OperationContext opCtx, IChannel replyOrInput)
{
operation_context = opCtx;
request_context = opCtx.RequestContext;
incoming_message = opCtx.IncomingMessage;
user_events_handler = new UserEventsHandler (this);
reply_or_input = replyOrInput;
}
public IChannel Channel {
get { return reply_or_input; }
}
public DispatchOperation Operation
{
get { return operation; }
set { operation = value; }
}
public Exception ProcessingException
{
get { return processingException; }
set { processingException = value; }
}
public Message ReplyMessage
{
get { return reply_message; }
set { reply_message = value; }
}
public InstanceContext InstanceContext
{
get { return instance_context; }
set { instance_context = value; }
}
public Message IncomingMessage
{
get { return incoming_message; }
set { incoming_message = value; }
}
public RequestContext RequestContext
{
get { return request_context; }
set { request_context = value; }
}
public OperationContext OperationContext
{
get { return operation_context; }
set { operation_context = value; }
}
public UserEventsHandler EventsHandler
{
get { return user_events_handler; }
set { user_events_handler = value; }
}
public void Reply (IDuplexChannel channel, bool useTimeout)
{
EventsHandler.BeforeSendReply ();
if (useTimeout && Operation.Parent.ChannelDispatcher != null) // FIXME: this condition is a workaround for NRE, there might be better way to get timeout value.
channel.Send (ReplyMessage, Operation.Parent.ChannelDispatcher.timeouts.SendTimeout);
else
channel.Send (ReplyMessage);
}
public void Reply (bool useTimeout)
{
EventsHandler.BeforeSendReply ();
if (useTimeout && Operation.Parent.ChannelDispatcher != null) // FIXME: this condition is a workaround for NRE, there might be better way to get timeout value.
RequestContext.Reply (ReplyMessage, Operation.Parent.ChannelDispatcher.timeouts.SendTimeout);
else
RequestContext.Reply (ReplyMessage);
}
}
#region user events implementation
internal class UserEventsHandler
{
MessageProcessingContext request_context;
DispatchRuntime dispatch_runtime;
IClientChannel channel;
object [] msg_inspectors_states;
object [] callcontext_initializers_states;
public UserEventsHandler (MessageProcessingContext mrc)
{
request_context = mrc;
dispatch_runtime = mrc.OperationContext.EndpointDispatcher.DispatchRuntime;
msg_inspectors_states = new object [dispatch_runtime.MessageInspectors.Count];
channel = request_context.OperationContext.Channel as IClientChannel;
}
public void AfterReceiveRequest ()
{
Message message = request_context.IncomingMessage;
for (int i = 0; i < dispatch_runtime.MessageInspectors.Count; ++i)
msg_inspectors_states [i] = dispatch_runtime.MessageInspectors [i].AfterReceiveRequest (
ref message, channel, request_context.InstanceContext);
request_context.IncomingMessage = message;
}
public void BeforeSendReply ()
{
Message toBeChanged = request_context.ReplyMessage;
for (int i = 0; i < dispatch_runtime.MessageInspectors.Count; ++i)
dispatch_runtime.MessageInspectors [i].BeforeSendReply (ref toBeChanged, msg_inspectors_states [i]);
request_context.ReplyMessage = toBeChanged;
}
public void BeforeInvoke (DispatchOperation operation)
{
callcontext_initializers_states = new object [operation.CallContextInitializers.Count];
for (int i = 0; i < callcontext_initializers_states.Length; ++i)
callcontext_initializers_states [i] = operation.CallContextInitializers [i].BeforeInvoke (
request_context.InstanceContext, channel, request_context.IncomingMessage);
}
public void AfterInvoke (DispatchOperation operation)
{
for (int i = 0; i < callcontext_initializers_states.Length; ++i)
operation.CallContextInitializers [i].AfterInvoke (callcontext_initializers_states [i]);
}
}
#endregion
}