// // EndpointDispatcher.cs // // Author: // Atsushi Enomoto // // 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 actions = new List (); 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 ()); } } }