You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			209 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			209 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //
 | |
| // EndpointDispatcher.cs
 | |
| //
 | |
| // Author:
 | |
| //	Atsushi Enomoto <atsushi@ximian.com>
 | |
| //
 | |
| // Copyright (C) 2005-2010 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.Generic;
 | |
| using System.Collections.ObjectModel;
 | |
| using System.Reflection;
 | |
| using System.ServiceModel.Description;
 | |
| using System.ServiceModel.Channels;
 | |
| using System.ServiceModel.Dispatcher;
 | |
| using System.ServiceModel.Security.Tokens;
 | |
| using System.Text;
 | |
| 
 | |
| namespace System.ServiceModel.Dispatcher
 | |
| {
 | |
| 	public class EndpointDispatcher
 | |
| 	{
 | |
| 		EndpointAddress address;
 | |
| 		string contract_name, contract_ns;
 | |
| 		ChannelDispatcher channel_dispatcher;
 | |
| 		MessageFilter address_filter;
 | |
| 		MessageFilter contract_filter;
 | |
| 		int filter_priority;
 | |
| 		DispatchRuntime dispatch_runtime;
 | |
| 
 | |
| 		// Umm, this API is ugly, since it or its members will
 | |
| 		// anyways require ServiceEndpoint, those arguments are
 | |
| 		// likely to be replaced by ServiceEndpoint (especially
 | |
| 		// considering about possible EndpointAddress inconsistency).
 | |
| 		public EndpointDispatcher (EndpointAddress address,
 | |
| 			string contractName, string contractNamespace)
 | |
| 		{
 | |
| 			if (contractName == null)
 | |
| 				throw new ArgumentNullException ("contractName");
 | |
| 			if (contractNamespace == null)
 | |
| 				throw new ArgumentNullException ("contractNamespace");
 | |
| 			if (address == null)
 | |
| 				throw new ArgumentNullException ("address");
 | |
| 
 | |
| 			this.address = address;
 | |
| 			contract_name = contractName;
 | |
| 			contract_ns = contractNamespace;
 | |
| 
 | |
| 			dispatch_runtime = new DispatchRuntime (this, null);
 | |
| 
 | |
| 			this.address_filter = new EndpointAddressMessageFilter (address);
 | |
| 		}
 | |
| 
 | |
| 		public DispatchRuntime DispatchRuntime {
 | |
| 			get { return dispatch_runtime; }
 | |
| 		}
 | |
| 
 | |
| 		public string ContractName {
 | |
| 			get { return contract_name; }
 | |
| 		}
 | |
| 
 | |
| 		public string ContractNamespace {
 | |
| 			get { return contract_ns; }
 | |
| 		}
 | |
| 
 | |
| 		public ChannelDispatcher ChannelDispatcher {
 | |
| 			get { return channel_dispatcher; }
 | |
| 			internal set { channel_dispatcher = value; }
 | |
| 		}
 | |
| 
 | |
| 		public MessageFilter AddressFilter {
 | |
| 			get { return address_filter; }
 | |
| 			set {
 | |
| 				if (value == null)
 | |
| 					throw new ArgumentNullException ("value");
 | |
| 				address_filter = value;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public MessageFilter ContractFilter {
 | |
| 			get { return contract_filter ?? (contract_filter = new MatchAllMessageFilter ()); }
 | |
| 			set {
 | |
| 				if (value == null)
 | |
| 					throw new ArgumentNullException ("value");
 | |
| 				contract_filter = value;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public EndpointAddress EndpointAddress {
 | |
| 			get { return address; }
 | |
| 		}
 | |
| 
 | |
| 		public int FilterPriority {
 | |
| 			get { return filter_priority; }
 | |
| 			set { filter_priority = value; }
 | |
| 		}
 | |
| 
 | |
| 		public bool IsSystemEndpoint { get; private set; }
 | |
| 
 | |
| 		internal void InitializeServiceEndpoint (bool isCallback, Type serviceType, ServiceEndpoint se)
 | |
| 		{
 | |
| 			IsSystemEndpoint = se.IsSystemEndpoint;
 | |
| 
 | |
| 			this.ContractFilter = GetContractFilter (se.Contract, isCallback);
 | |
| 
 | |
| 			this.DispatchRuntime.Type = serviceType;
 | |
| 			
 | |
| 			//Build the dispatch operations
 | |
| 			DispatchRuntime db = this.DispatchRuntime;
 | |
| 			if (!isCallback && se.Contract.CallbackContractType != null) {
 | |
| 				var ccd = se.Contract;
 | |
| 				db.CallbackClientRuntime.CallbackClientType = se.Contract.CallbackContractType;
 | |
| 				db.CallbackClientRuntime.ContractClientType = se.Contract.ContractType;
 | |
| 				ccd.FillClientOperations (db.CallbackClientRuntime, true);
 | |
| 			}
 | |
| 			foreach (OperationDescription od in se.Contract.Operations)
 | |
| 				if (od.InCallbackContract == isCallback/* && !db.Operations.Contains (od.Name)*/)
 | |
| 					PopulateDispatchOperation (db, od);
 | |
| 		}
 | |
| 
 | |
| 		void PopulateDispatchOperation (DispatchRuntime db, OperationDescription od) {
 | |
| 			string reqA = null, resA = null;
 | |
| 			foreach (MessageDescription m in od.Messages) {
 | |
| 				if (m.IsRequest)
 | |
| 					reqA = m.Action;
 | |
| 				else
 | |
| 					resA = m.Action;
 | |
| 			}
 | |
| 			DispatchOperation o =
 | |
| 				od.IsOneWay ?
 | |
| 				new DispatchOperation (db, od.Name, reqA) :
 | |
| 				new DispatchOperation (db, od.Name, reqA, resA);
 | |
| 			o.IsTerminating = od.IsTerminating;
 | |
| 			bool no_serialized_reply = od.IsOneWay;
 | |
| 			foreach (MessageDescription md in od.Messages) {
 | |
| 				if (md.IsRequest &&
 | |
| 					md.Body.Parts.Count == 1 &&
 | |
| 					md.Body.Parts [0].Type == typeof (Message))
 | |
| 					o.DeserializeRequest = false;
 | |
| 				if (!md.IsRequest &&
 | |
| 					md.Body.ReturnValue != null) {
 | |
| 					if (md.Body.ReturnValue.Type == typeof (Message))
 | |
| 						o.SerializeReply = false;
 | |
| 					else if (md.Body.ReturnValue.Type == typeof (void))
 | |
| 						no_serialized_reply = true;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			foreach (var fd in od.Faults)
 | |
| 				o.FaultContractInfos.Add (new FaultContractInfo (fd.Action, fd.DetailType));
 | |
| 
 | |
| 			// Setup Invoker
 | |
| 			o.Invoker = new DefaultOperationInvoker (od);
 | |
| 
 | |
| 			// Setup Formater
 | |
| 			// FIXME: this seems to be null at initializing, and should be set after applying all behaviors.
 | |
| 			// I leave this as is to not regress and it's cosmetic compatibility to fix.
 | |
| 			// FIXME: pass correct isRpc, isEncoded
 | |
| 			o.Formatter = new OperationFormatter (od, false, false);
 | |
| 
 | |
| 			if (o.Action == "*" && (o.IsOneWay || o.ReplyAction == "*")) {
 | |
| 				//Signature : Message  (Message)
 | |
| 				//	    : void  (Message)
 | |
| 				//FIXME: void (IChannel)
 | |
| 				if (!o.DeserializeRequest && (!o.SerializeReply || no_serialized_reply)) // what is this double-ish check for?
 | |
| 					db.UnhandledDispatchOperation = o;
 | |
| 			}
 | |
| 
 | |
| 			db.Operations.Add (o);
 | |
| 		}
 | |
| 
 | |
| 		MessageFilter GetContractFilter (ContractDescription cd, bool isCallback)
 | |
| 		{
 | |
| 			List<string> actions = new List<string> ();
 | |
| 			foreach (var od in cd.Operations) {
 | |
| 				foreach (var md in od.Messages)
 | |
| 					// For callback EndpointDispatcher (i.e. for duplex client), it should get "incoming" request for callback operations and "outgoing" response for non-callback operations.
 | |
| 					// For non-callback EndpointDispatcher,  it should get "outgoing" request for non-callback operations and "incoming" response for callback operations.
 | |
| 					if ((od.InCallbackContract == isCallback) == md.IsRequest) {
 | |
| 						if (md.Action == "*")
 | |
| 							return new MatchAllMessageFilter ();
 | |
| 						else
 | |
| 							actions.Add (md.Action);
 | |
| 					}
 | |
| 			}
 | |
| 			return new ActionMessageFilter (actions.ToArray ());
 | |
| 		}
 | |
| 	}
 | |
| }
 |