You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			299 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			299 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | // System.Runtime.Remoting.Channels.Tcp.TcpMessageIO.cs | ||
|  | // | ||
|  | // Author: Lluis Sanchez Gual (lluis@ideary.com) | ||
|  | // | ||
|  | // (C) 2002 Lluis Sanchez Gual | ||
|  | 
 | ||
|  | // | ||
|  | // 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.Runtime.Serialization; | ||
|  | using System.Runtime.Serialization.Formatters.Binary; | ||
|  | using System.Collections; | ||
|  | using System.IO; | ||
|  | using System.Text; | ||
|  | using System.Net.Sockets; | ||
|  | 
 | ||
|  | namespace System.Runtime.Remoting.Channels.Tcp  | ||
|  | { | ||
|  | 	enum MessageStatus { MethodMessage = 0, CancelSignal = 1, Unknown = 10} | ||
|  | 
 | ||
|  | 	internal class TcpMessageIO | ||
|  | 	{ | ||
|  | 		static byte[][] _msgHeaders =  | ||
|  | 			  { | ||
|  | 				  new byte[] { (byte)'.', (byte)'N', (byte)'E', (byte)'T', 1, 0 }, | ||
|  | 				  new byte[] { 255, 255, 255, 255, 255, 255 } | ||
|  | 			  }; | ||
|  | 							 | ||
|  | 		public static int DefaultStreamBufferSize = 1000; | ||
|  | 
 | ||
|  | 		// Identifies an incoming message | ||
|  | 		public static MessageStatus ReceiveMessageStatus (Stream networkStream, byte[] buffer) | ||
|  | 		{ | ||
|  | 			try { | ||
|  | 				StreamRead (networkStream, buffer, 6); | ||
|  | 			} catch (Exception ex) { | ||
|  | 				throw new RemotingException ("Tcp transport error.", ex); | ||
|  | 			} | ||
|  | 
 | ||
|  | 			try | ||
|  | 			{ | ||
|  | 				bool[] isOnTrack = new bool[_msgHeaders.Length]; | ||
|  | 				bool atLeastOneOnTrack = true; | ||
|  | 				int i = 0; | ||
|  | 
 | ||
|  | 				while (atLeastOneOnTrack) | ||
|  | 				{ | ||
|  | 					atLeastOneOnTrack = false; | ||
|  | 					byte c = buffer [i]; | ||
|  | 					for (int n = 0; n<_msgHeaders.Length; n++) | ||
|  | 					{ | ||
|  | 						if (i > 0 && !isOnTrack[n]) continue; | ||
|  | 
 | ||
|  | 						isOnTrack[n] = (c == _msgHeaders[n][i]); | ||
|  | 						if (isOnTrack[n] && (i == _msgHeaders[n].Length-1)) return (MessageStatus) n; | ||
|  | 						atLeastOneOnTrack = atLeastOneOnTrack || isOnTrack[n]; | ||
|  | 					} | ||
|  | 					i++; | ||
|  | 				} | ||
|  | 				return MessageStatus.Unknown; | ||
|  | 			} | ||
|  | 			catch (Exception ex) { | ||
|  | 				throw new RemotingException ("Tcp transport error.", ex); | ||
|  | 			} | ||
|  | 		} | ||
|  | 		 | ||
|  | 		static bool StreamRead (Stream networkStream, byte[] buffer, int count) | ||
|  | 		{ | ||
|  | 			int nr = 0; | ||
|  | 			do { | ||
|  | 				int pr = networkStream.Read (buffer, nr, count - nr); | ||
|  | 				if (pr == 0) | ||
|  | 					throw new RemotingException ("Connection closed"); | ||
|  | 				nr += pr; | ||
|  | 			} while (nr < count); | ||
|  | 			return true; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public static void SendMessageStream (Stream networkStream, Stream data, ITransportHeaders requestHeaders, byte[] buffer) | ||
|  | 		{ | ||
|  | 			SendMessageStream (networkStream, data, requestHeaders, buffer, false); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public static void SendMessageStream (Stream networkStream, Stream data, ITransportHeaders requestHeaders, byte[] buffer, bool isOneWay) | ||
|  | 		{ | ||
|  | 			if (buffer == null) buffer = new byte[DefaultStreamBufferSize]; | ||
|  | 
 | ||
|  | 			// Writes the message start header | ||
|  | 			byte[] dotnetHeader = _msgHeaders[(int) MessageStatus.MethodMessage]; | ||
|  | 			networkStream.Write(dotnetHeader, 0, dotnetHeader.Length); | ||
|  | 
 | ||
|  | 			// Writes the header tag | ||
|  | 			// 0x0000 - request stream | ||
|  | 			// 0x0001 - OneWay request stream | ||
|  | 			// 0x0002 - response stream | ||
|  | 			if(requestHeaders[CommonTransportKeys.RequestUri]!=null) { | ||
|  | 				buffer [0] = isOneWay ? (byte) 1 : (byte) 0; | ||
|  | 			} | ||
|  | 			else { | ||
|  | 				buffer[0] = (byte) 2; | ||
|  | 			} | ||
|  | 			buffer [1] = (byte) 0 ; | ||
|  | 
 | ||
|  | 			// Writes ID | ||
|  | 			buffer [2] = (byte) 0; | ||
|  | 
 | ||
|  | 			// Writes assemblyID???? | ||
|  | 			buffer [3] = (byte) 0; | ||
|  | 
 | ||
|  | 			// Writes the length of the stream being sent (not including the headers) | ||
|  | 			int num = (int)data.Length; | ||
|  | 			buffer [4] = (byte) num; | ||
|  | 			buffer [5] = (byte) (num >> 8); | ||
|  | 			buffer [6] = (byte) (num >> 16); | ||
|  | 			buffer [7] = (byte) (num >> 24); | ||
|  | 			networkStream.Write(buffer, 0, 8); | ||
|  | 	 | ||
|  | 			// Writes the message headers | ||
|  | 			SendHeaders (networkStream, requestHeaders, buffer); | ||
|  | 
 | ||
|  | 			if (data.Length == 0) | ||
|  | 				return; | ||
|  | 
 | ||
|  | 			// Writes the stream | ||
|  | 			if (data is MemoryStream) | ||
|  | 			{ | ||
|  | 				// The copy of the stream can be optimized. The internal | ||
|  | 				// buffer of MemoryStream can be used. | ||
|  | 				MemoryStream memStream = (MemoryStream)data; | ||
|  | 				networkStream.Write (memStream.GetBuffer(), 0, (int)memStream.Length); | ||
|  | 			} | ||
|  | 			else | ||
|  | 			{ | ||
|  | 				int nread = data.Read (buffer, 0, buffer.Length); | ||
|  | 				while (nread > 0) | ||
|  | 				{ | ||
|  | 					networkStream.Write (buffer, 0, nread); | ||
|  | 					nread = data.Read (buffer, 0, buffer.Length); | ||
|  | 				} | ||
|  | 			} | ||
|  | 		} | ||
|  | 		 | ||
|  | 		static byte[] msgUriTransportKey = new byte[] { 4, 0, 1, 1 }; | ||
|  | 		static byte[] msgContentTypeTransportKey = new byte[] { 6, 0, 1, 1 }; | ||
|  | 		static byte[] msgDefaultTransportKey = new byte[] { 1, 0, 1 }; | ||
|  | 		static byte[] msgHeaderTerminator = new byte[] { 0, 0 }; | ||
|  | 
 | ||
|  | 		private static void SendHeaders(Stream networkStream, ITransportHeaders requestHeaders, byte[] buffer) | ||
|  | 		{ | ||
|  | 			// Writes the headers as a sequence of strings | ||
|  | 			if (networkStream != null) | ||
|  | 			{ | ||
|  | 				IEnumerator e = requestHeaders.GetEnumerator(); | ||
|  | 				while (e.MoveNext()) | ||
|  | 				{ | ||
|  | 					DictionaryEntry hdr = (DictionaryEntry)e.Current; | ||
|  | 					switch (hdr.Key.ToString()) | ||
|  | 					{ | ||
|  | 						case CommonTransportKeys.RequestUri:  | ||
|  | 							networkStream.Write (msgUriTransportKey, 0, 4); | ||
|  | 							break; | ||
|  | 						case "Content-Type":  | ||
|  | 							networkStream.Write (msgContentTypeTransportKey, 0, 4); | ||
|  | 							break; | ||
|  | 						default:  | ||
|  | 							networkStream.Write (msgDefaultTransportKey, 0, 3); | ||
|  | 							SendString (networkStream, hdr.Key.ToString(), buffer); | ||
|  | 							networkStream.WriteByte (1); | ||
|  | 							break; | ||
|  | 					} | ||
|  | 					SendString (networkStream, hdr.Value.ToString(), buffer); | ||
|  | 				} | ||
|  | 			} | ||
|  | 			networkStream.Write (msgHeaderTerminator, 0, 2);	// End of headers | ||
|  | 		} | ||
|  | 		 | ||
|  | 		public static ITransportHeaders ReceiveHeaders (Stream networkStream, byte[] buffer) | ||
|  | 		{ | ||
|  | 			StreamRead (networkStream, buffer, 2); | ||
|  | 			 | ||
|  | 			byte headerType = buffer [0]; | ||
|  | 			TransportHeaders headers = new TransportHeaders (); | ||
|  | 
 | ||
|  | 			while (headerType != 0) | ||
|  | 			{ | ||
|  | 				string key; | ||
|  | 				StreamRead (networkStream, buffer, 1);	// byte 1 | ||
|  | 				switch (headerType) | ||
|  | 				{ | ||
|  | 					case 4: key = CommonTransportKeys.RequestUri; break; | ||
|  | 					case 6: key = "Content-Type"; break; | ||
|  | 					case 1: key = ReceiveString (networkStream, buffer); break; | ||
|  | 					default: throw new NotSupportedException ("Unknown header code: " + headerType); | ||
|  | 				} | ||
|  | 				StreamRead (networkStream, buffer, 1);	// byte 1 | ||
|  | 				headers[key] = ReceiveString (networkStream, buffer); | ||
|  | 
 | ||
|  | 				StreamRead (networkStream, buffer, 2); | ||
|  | 				headerType = buffer [0]; | ||
|  | 			} | ||
|  | 
 | ||
|  | 			return headers; | ||
|  | 		} | ||
|  | 		 | ||
|  | 		public static Stream ReceiveMessageStream (Stream networkStream, out ITransportHeaders headers, byte[] buffer) | ||
|  | 		{ | ||
|  | 			headers = null; | ||
|  | 
 | ||
|  | 			if (buffer == null) buffer = new byte[DefaultStreamBufferSize]; | ||
|  | 
 | ||
|  | 			// Reads header tag:  0 -> Stream with headers or 2 -> Response Stream | ||
|  | 			// + | ||
|  | 			// Gets the length of the data stream | ||
|  | 			StreamRead (networkStream, buffer, 8); | ||
|  | 
 | ||
|  | 			int byteCount = (buffer [4] | (buffer [5] << 8) | | ||
|  | 				(buffer [6] << 16) | (buffer [7] << 24)); | ||
|  | 
 | ||
|  | 			// Reads the headers | ||
|  | 			headers = ReceiveHeaders (networkStream, buffer); | ||
|  | 
 | ||
|  | 			if (byteCount > 0) { | ||
|  | 				byte[] resultBuffer = new byte[byteCount]; | ||
|  | 				StreamRead (networkStream, resultBuffer, byteCount); | ||
|  | 				return new MemoryStream (resultBuffer); | ||
|  | 			} else { | ||
|  | 				return new MemoryStream (); | ||
|  | 			} | ||
|  | 		}		 | ||
|  | 
 | ||
|  | 		private static void SendString (Stream networkStream, string str, byte[] buffer) | ||
|  | 		{ | ||
|  | 			// Allocates a buffer. Use the internal buffer if it is  | ||
|  | 			// big enough. If not, create a new one. | ||
|  | 
 | ||
|  | 			int maxBytes = Encoding.UTF8.GetMaxByteCount(str.Length)+4;	//+4 bytes for storing the string length | ||
|  | 			if (maxBytes > buffer.Length) | ||
|  | 				buffer = new byte[maxBytes]; | ||
|  | 
 | ||
|  | 			int num = Encoding.UTF8.GetBytes (str, 0, str.Length, buffer, 4); | ||
|  | 
 | ||
|  | 			// store number of bytes (not number of chars!) | ||
|  | 
 | ||
|  | 			buffer [0] = (byte) num; | ||
|  | 			buffer [1] = (byte) (num >> 8); | ||
|  | 			buffer [2] = (byte) (num >> 16); | ||
|  | 			buffer [3] = (byte) (num >> 24); | ||
|  | 
 | ||
|  | 			// Write the string bytes | ||
|  | 			networkStream.Write (buffer, 0, num + 4); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private static string ReceiveString (Stream networkStream, byte[] buffer) | ||
|  | 		{ | ||
|  | 			StreamRead (networkStream, buffer, 4); | ||
|  | 
 | ||
|  | 			// Reads the number of bytes (not chars!) | ||
|  | 
 | ||
|  | 			int byteCount = (buffer [0] | (buffer [1] << 8) | | ||
|  | 				(buffer [2] << 16) | (buffer [3] << 24)); | ||
|  | 
 | ||
|  | 			if (byteCount == 0) return string.Empty; | ||
|  | 
 | ||
|  | 			// Allocates a buffer of the correct size. Use the | ||
|  | 			// internal buffer if it is big enough | ||
|  | 
 | ||
|  | 			if (byteCount > buffer.Length) | ||
|  | 				buffer = new byte[byteCount]; | ||
|  | 
 | ||
|  | 			// Reads the string | ||
|  | 
 | ||
|  | 			StreamRead (networkStream, buffer, byteCount); | ||
|  | 			char[] chars = Encoding.UTF8.GetChars (buffer, 0, byteCount); | ||
|  | 	 | ||
|  | 			return new string (chars); | ||
|  | 		} | ||
|  | 		 | ||
|  | 	} | ||
|  | } |