// // System.Runtime.Remoting.Channels.BinaryClientFormatterSink.cs // // Author: Rodrigo Moya (rodrigo@ximian.com) // Dietmar Maurer (dietmar@ximian.com) // Lluis Sanchez Gual (lluis@ideary.com) // // 2002 (C) Copyright, Ximian, Inc. // using System.Collections; using System.IO; using System.Runtime.Remoting.Messaging; using System.Runtime.Serialization; using System.Runtime.Serialization.Formatters.Binary; namespace System.Runtime.Remoting.Channels { public class BinaryClientFormatterSink : IClientFormatterSink, IMessageSink, IClientChannelSink, IChannelSinkBase { static BinaryFormatter _serializationFormatter; static BinaryFormatter _deserializationFormatter; IClientChannelSink nextInChain; static BinaryClientFormatterSink () { RemotingSurrogateSelector surrogateSelector = new RemotingSurrogateSelector (); StreamingContext context = new StreamingContext(StreamingContextStates.Remoting, null); _serializationFormatter = new BinaryFormatter (surrogateSelector, context); _deserializationFormatter = new BinaryFormatter (null, context); } public BinaryClientFormatterSink (IClientChannelSink nextSink) { nextInChain = nextSink; } public IClientChannelSink NextChannelSink { get { return nextInChain; } } public IMessageSink NextSink { get { // This is the last sink in the IMessageSink sink chain return null; } } public IDictionary Properties { get { return null; } } public void AsyncProcessRequest (IClientChannelSinkStack sinkStack, IMessage msg, ITransportHeaders headers, Stream stream) { // never called because the formatter sink is // always the first in the chain throw new NotSupportedException("BinaryClientFormatterSink must be the first sink in the IClientChannelSink chain"); } public void AsyncProcessResponse (IClientResponseChannelSinkStack sinkStack, object state, ITransportHeaders headers, Stream stream) { IMessage replyMessage = (IMessage)_deserializationFormatter.DeserializeMethodResponse (stream, null, (IMethodCallMessage)state); sinkStack.DispatchReplyMessage (replyMessage); } public Stream GetRequestStream (IMessage msg, ITransportHeaders headers) { // never called throw new NotSupportedException (); } public void ProcessMessage (IMessage msg, ITransportHeaders requestHeaders, Stream requestStream, out ITransportHeaders responseHeaders, out Stream responseStream) { // never called because the formatter sink is // always the first in the chain throw new NotSupportedException (); } public IMessageCtrl AsyncProcessMessage (IMessage msg, IMessageSink replySink) { ITransportHeaders transportHeaders = new TransportHeaders(); transportHeaders[CommonTransportKeys.RequestUri] = ((IMethodCallMessage)msg).Uri; Stream stream = nextInChain.GetRequestStream(msg, transportHeaders); if (stream == null) stream = new MemoryStream (); _serializationFormatter.Serialize (stream, msg, null); if (stream is MemoryStream) stream.Position = 0; ClientChannelSinkStack stack = new ClientChannelSinkStack(replySink); stack.Push (this, msg); nextInChain.AsyncProcessRequest (stack, msg, transportHeaders, stream); // FIXME: No idea about how to implement IMessageCtrl return null; } public IMessage SyncProcessMessage (IMessage msg) { try { ITransportHeaders call_headers = new TransportHeaders(); call_headers[CommonTransportKeys.RequestUri] = ((IMethodCallMessage)msg).Uri; Stream call_stream = nextInChain.GetRequestStream(msg, call_headers); if (call_stream == null) call_stream = new MemoryStream (); // Serialize msg to the stream _serializationFormatter.Serialize (call_stream, msg, null); if (call_stream is MemoryStream) call_stream.Position = 0; Stream response_stream; ITransportHeaders response_headers; nextInChain.ProcessMessage (msg, call_headers, call_stream, out response_headers, out response_stream); // Deserialize response_stream return (IMessage) _deserializationFormatter.DeserializeMethodResponse (response_stream, null, (IMethodCallMessage)msg); } catch (Exception e) { return new ReturnMessage (e, (IMethodCallMessage)msg); } } } }