You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			389 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			389 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | // | ||
|  | // System.Runtime.Remoting.Channels.Ipc.Win32.IpcServerChannel.cs | ||
|  | // | ||
|  | // Author: Robert Jordan (robertj@gmx.net) | ||
|  | // | ||
|  | // Copyright (C) 2005 Novell, Inc (http://www.novell.com) | ||
|  | // | ||
|  | // | ||
|  | // Permission is hereby granted, free of charge, to any person obtaining | ||
|  | // a copy of this software and associated documentation files (the | ||
|  | // "Software"), to deal in the Software without restriction, including | ||
|  | // without limitation the rights to use, copy, modify, merge, publish, | ||
|  | // distribute, sublicense, and/or sell copies of the Software, and to | ||
|  | // permit persons to whom the Software is furnished to do so, subject to | ||
|  | // the following conditions: | ||
|  | //  | ||
|  | // The above copyright notice and this permission notice shall be | ||
|  | // included in all copies or substantial portions of the Software. | ||
|  | //  | ||
|  | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||
|  | // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||
|  | // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||
|  | // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE | ||
|  | // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION | ||
|  | // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION | ||
|  | // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||
|  | // | ||
|  | 
 | ||
|  | 
 | ||
|  | using System; | ||
|  | using System.Collections; | ||
|  | using System.IO; | ||
|  | using System.Runtime.Remoting; | ||
|  | using System.Runtime.Remoting.Channels; | ||
|  | using System.Runtime.Remoting.Messaging; | ||
|  | using System.Threading; | ||
|  | 
 | ||
|  | namespace System.Runtime.Remoting.Channels.Ipc.Win32 | ||
|  | { | ||
|  |     internal class IpcServerChannel : IChannelReceiver, IChannel | ||
|  |     { | ||
|  |         readonly string portName; | ||
|  |         readonly string channelName = IpcChannelHelper.Scheme; | ||
|  |         readonly int channelPriority = 1; | ||
|  |         readonly IServerChannelSinkProvider serverProvider; | ||
|  |         readonly IpcServerChannelSink sink; | ||
|  |         readonly ChannelDataStore dataStore; | ||
|  |         Thread worker; | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Builds the default channel properties | ||
|  |         /// </summary> | ||
|  |         /// <param name="portName">The pipe name.</param> | ||
|  |         /// <returns></returns> | ||
|  |         internal static IDictionary BuildDefaultProperties(string portName) | ||
|  |         { | ||
|  |             Hashtable h = new Hashtable(); | ||
|  |             h.Add("portName", portName); | ||
|  |             return h; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Builds the default channel properties | ||
|  |         /// </summary> | ||
|  |         /// <param name="portName">The pipe name.</param> | ||
|  |         /// <returns></returns> | ||
|  |         internal static IDictionary BuildDefaultProperties(string name, string portName) | ||
|  |         { | ||
|  |             Hashtable h = new Hashtable(); | ||
|  |             h.Add("name", name); | ||
|  |             h.Add("portName", portName); | ||
|  |             return h; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Creates a server channel | ||
|  |         /// </summary> | ||
|  |         /// <param name="portName">The port name.</param> | ||
|  |         public IpcServerChannel(string portName) | ||
|  |             : this(BuildDefaultProperties(portName), null) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Creates a server channel | ||
|  |         /// </summary> | ||
|  |         /// <param name="mame">The channel name.</param> | ||
|  |         /// <param name="portName">The port name.</param> | ||
|  |         public IpcServerChannel(string name, string portName) | ||
|  |             : this(BuildDefaultProperties(name, portName), null) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Creates a server channel | ||
|  |         /// </summary> | ||
|  |         /// <param name="mame">The channel name.</param> | ||
|  |         /// <param name="portName">The port name.</param> | ||
|  |         /// <param name="provider">The sink provider.</param> | ||
|  |         public IpcServerChannel(string name, string portName, | ||
|  |                                 IServerChannelSinkProvider provider) | ||
|  |             : this(BuildDefaultProperties(name, portName), provider) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Creates a server channel. | ||
|  |         /// </summary> | ||
|  |         /// <param name="properties">The channel properties.</param> | ||
|  |         /// <param name="provider">The sink provider.</param> | ||
|  |         public IpcServerChannel(IDictionary properties, IServerChannelSinkProvider provider) | ||
|  |         { | ||
|  |             bool impersonate = false; | ||
|  | 
 | ||
|  |             if (properties != null)  | ||
|  |             { | ||
|  |                 foreach (DictionaryEntry e in properties)  | ||
|  |                 { | ||
|  |                     switch ((string)e.Key)  | ||
|  |                     { | ||
|  |                         case "name": | ||
|  |                             channelName = (string)e.Value; | ||
|  |                             break; | ||
|  | 
 | ||
|  |                         case "priority": | ||
|  |                             channelPriority = Convert.ToInt32(e.Value); | ||
|  |                             break; | ||
|  | 
 | ||
|  |                         case "portName": | ||
|  |                             portName = (string)e.Value; | ||
|  |                             if (!IpcChannelHelper.IsValidPipeName(portName)) | ||
|  |                                 throw new ArgumentException("Invalid pipe name.", "portName"); | ||
|  |                             break; | ||
|  | 
 | ||
|  |                         case "impersonate": | ||
|  |                             impersonate = Boolean.Parse((string)e.Value); | ||
|  |                             break; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             if (portName == null)  | ||
|  |             { | ||
|  |                 portName = Guid.NewGuid().ToString("N"); | ||
|  |             } | ||
|  | 
 | ||
|  |             serverProvider = provider; | ||
|  | 
 | ||
|  |             if (serverProvider == null)  | ||
|  |             { | ||
|  |                 serverProvider = new BinaryServerFormatterSinkProvider(); | ||
|  |             } | ||
|  | 
 | ||
|  |             dataStore = new ChannelDataStore( | ||
|  |                 new string[] {IpcChannelHelper.SchemeStart + portName} | ||
|  |                 ); | ||
|  |             PopulateChannelData(dataStore, serverProvider); | ||
|  | 
 | ||
|  |             sink = new IpcServerChannelSink( | ||
|  |                 ChannelServices.CreateServerChannelSinkChain(serverProvider, this), | ||
|  |                 portName, | ||
|  |                 impersonate | ||
|  |                 ); | ||
|  | 
 | ||
|  |             StartListening(null); | ||
|  |         } | ||
|  | 
 | ||
|  |         void PopulateChannelData( ChannelDataStore channelData, | ||
|  |             IServerChannelSinkProvider provider) | ||
|  |         { | ||
|  |             while (provider != null) | ||
|  |             { | ||
|  |                 provider.GetChannelData(channelData); | ||
|  |                 provider = provider.Next; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         #region IChannelReceiver Members | ||
|  | 
 | ||
|  |         public void StartListening(object data) | ||
|  |         { | ||
|  |             if (worker == null)  | ||
|  |             { | ||
|  |                 worker = new Thread(new ThreadStart(sink.Listen)); | ||
|  |                 worker.IsBackground = true; | ||
|  |                 worker.Start(); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public object ChannelData | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return dataStore; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public void StopListening(object data) | ||
|  |         { | ||
|  |             if (worker != null)  | ||
|  |             { | ||
|  |                 worker.Abort(); | ||
|  |                 worker = null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public string[] GetUrlsForUri(string objectURI) | ||
|  |         { | ||
|  |             if (!objectURI.StartsWith("/")) objectURI = "/" + objectURI; | ||
|  |             string[] urls = new string[1]; | ||
|  |             urls[0] = IpcChannelHelper.SchemeStart + portName + objectURI; | ||
|  |             return urls; | ||
|  |         } | ||
|  | 
 | ||
|  |         #endregion | ||
|  | 
 | ||
|  |         #region IChannel Members | ||
|  | 
 | ||
|  |         public string ChannelName | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return channelName; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public int ChannelPriority | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return channelPriority; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public string Parse(string url, out string objectURI) | ||
|  |         { | ||
|  |             return IpcChannelHelper.Parse(url, out objectURI); | ||
|  |         } | ||
|  | 
 | ||
|  |         #endregion | ||
|  |     } | ||
|  | 
 | ||
|  |     internal class IpcServerChannelSink : IServerChannelSink | ||
|  |     { | ||
|  |         IServerChannelSink nextSink; | ||
|  |         string portName; | ||
|  |         bool impersonate; | ||
|  | 
 | ||
|  |         public IpcServerChannelSink(IServerChannelSink nextSink, string portName, bool impersonate)  | ||
|  |         { | ||
|  |             this.nextSink = nextSink; | ||
|  |             this.portName = portName; | ||
|  |             this.impersonate = impersonate; | ||
|  |         } | ||
|  | 
 | ||
|  |         #region IServerChannelSink Members | ||
|  | 
 | ||
|  |         public Stream GetResponseStream(IServerResponseChannelSinkStack sinkStack, object state, IMessage msg, ITransportHeaders headers) | ||
|  |         { | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         public ServerProcessing ProcessMessage(IServerChannelSinkStack sinkStack, IMessage requestMsg, ITransportHeaders requestHeaders, Stream requestStream, out IMessage responseMsg, out ITransportHeaders responseHeaders, out Stream responseStream) | ||
|  |         { | ||
|  |             throw new NotSupportedException(); | ||
|  |         } | ||
|  | 
 | ||
|  |         public void AsyncProcessResponse(IServerResponseChannelSinkStack sinkStack, object state, IMessage msg, ITransportHeaders headers, Stream stream) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         public IServerChannelSink NextChannelSink | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return nextSink; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         #endregion | ||
|  | 
 | ||
|  |         #region IChannelSinkBase Members | ||
|  | 
 | ||
|  |         public IDictionary Properties | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Listens for incoming requests. | ||
|  |         /// </summary> | ||
|  |         internal void Listen()  | ||
|  |         { | ||
|  |             while (true)  | ||
|  |             { | ||
|  |                 try  | ||
|  |                 { | ||
|  |                     NamedPipeListener listener = new NamedPipeListener(portName); | ||
|  |                     NamedPipeSocket socket = listener.Accept(); | ||
|  |                     ThreadPool.QueueUserWorkItem(new WaitCallback(ProcessClient), socket); | ||
|  |                 } | ||
|  |                 catch (NamedPipeException)  | ||
|  |                 { | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void ProcessClient(object state)  | ||
|  |         { | ||
|  |             try  | ||
|  |             { | ||
|  |                 NamedPipeSocket socket = (NamedPipeSocket) state; | ||
|  | 
 | ||
|  |                 ITransportHeaders requestHeaders; | ||
|  |                 Stream requestStream; | ||
|  | 
 | ||
|  |                 IpcTransport t = new IpcTransport(socket); | ||
|  |                 t.Read(out requestHeaders, out requestStream); | ||
|  | 
 | ||
|  |                 // parse the RequestUri | ||
|  |                 string objectUri; | ||
|  |                 string uri = (string) requestHeaders[CommonTransportKeys.RequestUri]; | ||
|  |                 IpcChannelHelper.Parse(uri, out objectUri); | ||
|  |                 if (objectUri == null) objectUri = uri; | ||
|  |                 requestHeaders[CommonTransportKeys.RequestUri] = objectUri; | ||
|  | 
 | ||
|  |                 ServerChannelSinkStack stack = new ServerChannelSinkStack(); | ||
|  |                 stack.Push(this, null); | ||
|  | 
 | ||
|  |                 IMessage responseMsg; | ||
|  |                 ITransportHeaders responseHeaders; | ||
|  |                 Stream responseStream; | ||
|  | 
 | ||
|  |                 requestHeaders["__CustomErrorsEnabled"] = false; | ||
|  | 
 | ||
|  |                 if (impersonate)  | ||
|  |                 { | ||
|  |                     // TODO: Impersonate might throw exceptions. What to do with them? | ||
|  |                     socket.Impersonate(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 ServerProcessing op = nextSink.ProcessMessage( | ||
|  |                     stack, | ||
|  |                     null, | ||
|  |                     requestHeaders, | ||
|  |                     requestStream, | ||
|  |                     out responseMsg, | ||
|  |                     out responseHeaders, | ||
|  |                     out responseStream | ||
|  |                     ); | ||
|  | 
 | ||
|  |                 if (impersonate)  | ||
|  |                 { | ||
|  |                     NamedPipeSocket.RevertToSelf(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 switch (op)  | ||
|  |                 { | ||
|  |                     case ServerProcessing.Complete: | ||
|  |                         stack.Pop(this); | ||
|  |                         // send the response headers and the response data to the client | ||
|  |                         t.Write(responseHeaders, responseStream); | ||
|  |                         break; | ||
|  | 
 | ||
|  |                     case ServerProcessing.Async: | ||
|  |                         stack.StoreAndDispatch(nextSink, null); | ||
|  |                         break; | ||
|  | 
 | ||
|  |                     case ServerProcessing.OneWay: | ||
|  |                         break; | ||
|  |                 } | ||
|  |             } | ||
|  |             catch (Exception) | ||
|  |             { | ||
|  |                 // Console.WriteLine(ex); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         #endregion | ||
|  | 
 | ||
|  |     } | ||
|  | 
 | ||
|  | } | ||
|  | 
 |