94b2861243
Former-commit-id: 5f9c6ae75f295e057a7d2971f3a6df4656fa8850
2128 lines
82 KiB
C#
2128 lines
82 KiB
C#
//-----------------------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//-----------------------------------------------------------------------------
|
|
|
|
namespace System.ServiceModel
|
|
{
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.ObjectModel;
|
|
using System.ComponentModel;
|
|
using System.Diagnostics;
|
|
using System.Globalization;
|
|
using System.Net;
|
|
using System.Runtime;
|
|
using System.Security;
|
|
using System.ServiceModel.Administration;
|
|
using System.ServiceModel.Channels;
|
|
using System.ServiceModel.Configuration;
|
|
using System.ServiceModel.Description;
|
|
using System.ServiceModel.Diagnostics;
|
|
using System.ServiceModel.Dispatcher;
|
|
using System.Text;
|
|
using System.Runtime.Diagnostics;
|
|
using System.Threading;
|
|
using System.ServiceModel.Activation;
|
|
using System.ServiceModel.Diagnostics.Application;
|
|
using System.Reflection;
|
|
using System.Linq.Expressions;
|
|
|
|
public abstract class ServiceHostBase : CommunicationObject, IExtensibleObject<ServiceHostBase>, IDisposable
|
|
{
|
|
internal static readonly Uri EmptyUri = new Uri(string.Empty, UriKind.RelativeOrAbsolute);
|
|
|
|
bool initializeDescriptionHasFinished;
|
|
UriSchemeKeyedCollection baseAddresses;
|
|
ChannelDispatcherCollection channelDispatchers;
|
|
TimeSpan closeTimeout = ServiceDefaults.ServiceHostCloseTimeout;
|
|
ServiceDescription description;
|
|
ExtensionCollection<ServiceHostBase> extensions;
|
|
ReadOnlyCollection<Uri> externalBaseAddresses;
|
|
IDictionary<string, ContractDescription> implementedContracts;
|
|
IInstanceContextManager instances;
|
|
TimeSpan openTimeout = ServiceDefaults.OpenTimeout;
|
|
ServicePerformanceCountersBase servicePerformanceCounters;
|
|
DefaultPerformanceCounters defaultPerformanceCounters;
|
|
ServiceThrottle serviceThrottle;
|
|
ServiceCredentials readOnlyCredentials;
|
|
ServiceAuthorizationBehavior readOnlyAuthorization;
|
|
ServiceAuthenticationBehavior readOnlyAuthentication;
|
|
Dictionary<DispatcherBuilder.ListenUriInfo, Collection<ServiceEndpoint>> endpointsByListenUriInfo;
|
|
int busyCount;
|
|
EventTraceActivity eventTraceActivity;
|
|
|
|
internal event EventHandler BusyCountIncremented;
|
|
|
|
public event EventHandler<UnknownMessageReceivedEventArgs> UnknownMessageReceived;
|
|
|
|
protected ServiceHostBase()
|
|
{
|
|
TraceUtility.SetEtwProviderId();
|
|
this.baseAddresses = new UriSchemeKeyedCollection(this.ThisLock);
|
|
this.channelDispatchers = new ChannelDispatcherCollection(this, this.ThisLock);
|
|
this.extensions = new ExtensionCollection<ServiceHostBase>(this, this.ThisLock);
|
|
this.instances = new InstanceContextManager(this.ThisLock);
|
|
this.serviceThrottle = new ServiceThrottle(this);
|
|
this.TraceOpenAndClose = true;
|
|
this.Faulted += new EventHandler(OnServiceHostFaulted);
|
|
}
|
|
|
|
|
|
internal EventTraceActivity EventTraceActivity
|
|
{
|
|
get
|
|
{
|
|
if (this.eventTraceActivity == null)
|
|
{
|
|
this.eventTraceActivity = new EventTraceActivity();
|
|
}
|
|
|
|
return eventTraceActivity;
|
|
}
|
|
}
|
|
|
|
public ServiceAuthorizationBehavior Authorization
|
|
{
|
|
get
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
return null;
|
|
}
|
|
else if (this.State == CommunicationState.Created || this.State == CommunicationState.Opening)
|
|
{
|
|
return EnsureAuthorization(this.Description);
|
|
}
|
|
else
|
|
{
|
|
return this.readOnlyAuthorization;
|
|
}
|
|
}
|
|
}
|
|
|
|
public ServiceAuthenticationBehavior Authentication
|
|
{
|
|
get
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
return null;
|
|
}
|
|
else if (this.State == CommunicationState.Created || this.State == CommunicationState.Opening)
|
|
{
|
|
return EnsureAuthentication(this.Description);
|
|
}
|
|
else
|
|
{
|
|
return this.readOnlyAuthentication;
|
|
}
|
|
}
|
|
}
|
|
|
|
public ReadOnlyCollection<Uri> BaseAddresses
|
|
{
|
|
get
|
|
{
|
|
externalBaseAddresses = new ReadOnlyCollection<Uri>(new List<Uri>(this.baseAddresses));
|
|
return externalBaseAddresses;
|
|
}
|
|
}
|
|
|
|
public ChannelDispatcherCollection ChannelDispatchers
|
|
{
|
|
get { return this.channelDispatchers; }
|
|
}
|
|
|
|
public TimeSpan CloseTimeout
|
|
{
|
|
get { return this.closeTimeout; }
|
|
set
|
|
{
|
|
if (value < TimeSpan.Zero)
|
|
{
|
|
string message = SR.GetString(SR.SFxTimeoutOutOfRange0);
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("value", message));
|
|
}
|
|
if (TimeoutHelper.IsTooLarge(value))
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("value", SR.GetString(SR.SFxTimeoutOutOfRangeTooBig)));
|
|
}
|
|
|
|
lock (this.ThisLock)
|
|
{
|
|
this.ThrowIfClosedOrOpened();
|
|
this.closeTimeout = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal ServicePerformanceCountersBase Counters
|
|
{
|
|
get
|
|
{
|
|
return this.servicePerformanceCounters;
|
|
}
|
|
set
|
|
{
|
|
this.servicePerformanceCounters = value;
|
|
this.serviceThrottle.SetServicePerformanceCounters(this.servicePerformanceCounters);
|
|
}
|
|
}
|
|
|
|
internal DefaultPerformanceCounters DefaultCounters
|
|
{
|
|
get
|
|
{
|
|
return this.defaultPerformanceCounters;
|
|
}
|
|
set
|
|
{
|
|
this.defaultPerformanceCounters = value;
|
|
}
|
|
}
|
|
|
|
public ServiceCredentials Credentials
|
|
{
|
|
get
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
return null;
|
|
}
|
|
else if (this.State == CommunicationState.Created || this.State == CommunicationState.Opening)
|
|
{
|
|
return EnsureCredentials(this.Description);
|
|
}
|
|
else
|
|
{
|
|
return this.readOnlyCredentials;
|
|
}
|
|
}
|
|
}
|
|
|
|
protected override TimeSpan DefaultCloseTimeout
|
|
{
|
|
get { return this.CloseTimeout; }
|
|
}
|
|
|
|
protected override TimeSpan DefaultOpenTimeout
|
|
{
|
|
get { return this.OpenTimeout; }
|
|
}
|
|
|
|
public ServiceDescription Description
|
|
{
|
|
get { return this.description; }
|
|
}
|
|
|
|
public IExtensionCollection<ServiceHostBase> Extensions
|
|
{
|
|
get { return this.extensions; }
|
|
}
|
|
|
|
protected internal IDictionary<string, ContractDescription> ImplementedContracts
|
|
{
|
|
get { return this.implementedContracts; }
|
|
}
|
|
|
|
internal UriSchemeKeyedCollection InternalBaseAddresses
|
|
{
|
|
get { return this.baseAddresses; }
|
|
}
|
|
|
|
public int ManualFlowControlLimit
|
|
{
|
|
get { return this.ServiceThrottle.ManualFlowControlLimit; }
|
|
set { this.ServiceThrottle.ManualFlowControlLimit = value; }
|
|
}
|
|
|
|
public TimeSpan OpenTimeout
|
|
{
|
|
get { return this.openTimeout; }
|
|
set
|
|
{
|
|
if (value < TimeSpan.Zero)
|
|
{
|
|
string message = SR.GetString(SR.SFxTimeoutOutOfRange0);
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("value", message));
|
|
}
|
|
if (TimeoutHelper.IsTooLarge(value))
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("value", SR.GetString(SR.SFxTimeoutOutOfRangeTooBig)));
|
|
}
|
|
|
|
lock (this.ThisLock)
|
|
{
|
|
this.ThrowIfClosedOrOpened();
|
|
this.openTimeout = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal ServiceThrottle ServiceThrottle
|
|
{
|
|
get
|
|
{
|
|
return this.serviceThrottle;
|
|
}
|
|
}
|
|
|
|
internal virtual object DisposableInstance
|
|
{
|
|
get
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
|
|
internal Dictionary<DispatcherBuilder.ListenUriInfo, Collection<ServiceEndpoint>> EndpointsByListenUriInfo
|
|
{
|
|
get
|
|
{
|
|
if (this.endpointsByListenUriInfo == null)
|
|
{
|
|
this.endpointsByListenUriInfo = this.GetEndpointsByListenUriInfo();
|
|
}
|
|
return this.endpointsByListenUriInfo;
|
|
}
|
|
}
|
|
|
|
Dictionary<DispatcherBuilder.ListenUriInfo, Collection<ServiceEndpoint>> GetEndpointsByListenUriInfo()
|
|
{
|
|
Dictionary<DispatcherBuilder.ListenUriInfo, Collection<ServiceEndpoint>> endpointDictionary = new Dictionary<DispatcherBuilder.ListenUriInfo, Collection<ServiceEndpoint>>();
|
|
foreach (ServiceEndpoint endpoint in this.Description.Endpoints)
|
|
{
|
|
DispatcherBuilder.ListenUriInfo listenUriInfo = DispatcherBuilder.GetListenUriInfoForEndpoint(this, endpoint);
|
|
if (!endpointDictionary.ContainsKey(listenUriInfo))
|
|
{
|
|
endpointDictionary.Add(listenUriInfo, new Collection<ServiceEndpoint>());
|
|
}
|
|
endpointDictionary[listenUriInfo].Add(endpoint);
|
|
}
|
|
return endpointDictionary;
|
|
}
|
|
|
|
protected void AddBaseAddress(Uri baseAddress)
|
|
{
|
|
if (this.initializeDescriptionHasFinished)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(
|
|
SR.GetString(SR.SFxCannotCallAddBaseAddress)));
|
|
}
|
|
this.baseAddresses.Add(baseAddress);
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(string implementedContract, Binding binding, string address)
|
|
{
|
|
return this.AddServiceEndpoint(implementedContract, binding, address, (Uri)null);
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(string implementedContract, Binding binding, string address, Uri listenUri)
|
|
{
|
|
if (address == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("address"));
|
|
}
|
|
|
|
ServiceEndpoint endpoint = this.AddServiceEndpoint(implementedContract, binding, new Uri(address, UriKind.RelativeOrAbsolute));
|
|
if (listenUri != null)
|
|
{
|
|
endpoint.UnresolvedListenUri = listenUri;
|
|
listenUri = MakeAbsoluteUri(listenUri, binding);
|
|
endpoint.ListenUri = listenUri;
|
|
}
|
|
return endpoint;
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(string implementedContract, Binding binding, Uri address)
|
|
{
|
|
return this.AddServiceEndpoint(implementedContract, binding, address, (Uri)null);
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(string implementedContract, Binding binding, Uri address, Uri listenUri)
|
|
{
|
|
if (address == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("address"));
|
|
}
|
|
|
|
if (binding == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("binding"));
|
|
}
|
|
|
|
if (implementedContract == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("implementedContract"));
|
|
}
|
|
|
|
if (this.State != CommunicationState.Created && this.State != CommunicationState.Opening)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotAddEndpointAfterOpen)));
|
|
}
|
|
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotAddEndpointWithoutDescription)));
|
|
}
|
|
|
|
Uri via = this.MakeAbsoluteUri(address, binding);
|
|
|
|
ConfigLoader configLoader = new ConfigLoader(GetContractResolver(this.implementedContracts));
|
|
ContractDescription contract = configLoader.LookupContract(implementedContract, this.Description.Name);
|
|
|
|
ServiceEndpoint serviceEndpoint = new ServiceEndpoint(contract, binding, new EndpointAddress(via));
|
|
this.Description.Endpoints.Add(serviceEndpoint);
|
|
serviceEndpoint.UnresolvedAddress = address;
|
|
|
|
if (listenUri != null)
|
|
{
|
|
serviceEndpoint.UnresolvedListenUri = listenUri;
|
|
listenUri = MakeAbsoluteUri(listenUri, binding);
|
|
serviceEndpoint.ListenUri = listenUri;
|
|
}
|
|
return serviceEndpoint;
|
|
}
|
|
|
|
public virtual void AddServiceEndpoint(ServiceEndpoint endpoint)
|
|
{
|
|
if (endpoint == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("endpoint");
|
|
}
|
|
if (this.State != CommunicationState.Created && this.State != CommunicationState.Opening)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotAddEndpointAfterOpen)));
|
|
}
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotAddEndpointWithoutDescription)));
|
|
}
|
|
if (endpoint.Address == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SFxEndpointAddressNotSpecified));
|
|
}
|
|
if (endpoint.Contract == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SFxEndpointContractNotSpecified));
|
|
}
|
|
if (endpoint.Binding == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SFxEndpointBindingNotSpecified));
|
|
}
|
|
if (!endpoint.IsSystemEndpoint || endpoint.Contract.ContractType == typeof(IMetadataExchange))
|
|
{
|
|
ConfigLoader loader = new ConfigLoader(GetContractResolver(this.implementedContracts));
|
|
loader.LookupContract(endpoint.Contract.ConfigurationName, this.Description.Name);
|
|
}
|
|
this.Description.Endpoints.Add(endpoint);
|
|
}
|
|
|
|
public void SetEndpointAddress(ServiceEndpoint endpoint, string relativeAddress)
|
|
{
|
|
if (endpoint == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("endpoint");
|
|
}
|
|
if (relativeAddress == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("relativeAddress");
|
|
}
|
|
if (endpoint.Binding == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SFxEndpointBindingNotSpecified));
|
|
}
|
|
Uri absoluteUri = MakeAbsoluteUri(new Uri(relativeAddress, UriKind.Relative), endpoint.Binding);
|
|
endpoint.Address = new EndpointAddress(absoluteUri);
|
|
}
|
|
|
|
internal Uri MakeAbsoluteUri(Uri relativeOrAbsoluteUri, Binding binding)
|
|
{
|
|
return MakeAbsoluteUri(relativeOrAbsoluteUri, binding, this.InternalBaseAddresses);
|
|
}
|
|
|
|
internal static Uri MakeAbsoluteUri(Uri relativeOrAbsoluteUri, Binding binding, UriSchemeKeyedCollection baseAddresses)
|
|
{
|
|
Uri result = relativeOrAbsoluteUri;
|
|
if (!result.IsAbsoluteUri)
|
|
{
|
|
if (binding.Scheme == string.Empty)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxCustomBindingWithoutTransport)));
|
|
}
|
|
result = GetVia(binding.Scheme, result, baseAddresses);
|
|
if (result == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxEndpointNoMatchingScheme, binding.Scheme, binding.Name, GetBaseAddressSchemes(baseAddresses))));
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
protected virtual void ApplyConfiguration()
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotApplyConfigurationWithoutDescription)));
|
|
}
|
|
|
|
ConfigLoader configLoader = new ConfigLoader(GetContractResolver(implementedContracts));
|
|
|
|
// Call the overload of LoadConfigurationSectionInternal which looks up the serviceElement from ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None)
|
|
LoadConfigurationSectionInternal(configLoader, this.Description, this.Description.ConfigurationName);
|
|
|
|
EnsureAuthenticationAuthorizationDebug(this.Description);
|
|
}
|
|
|
|
internal void EnsureAuthenticationAuthorizationDebug(ServiceDescription description)
|
|
{
|
|
EnsureAuthentication(description);
|
|
EnsureAuthorization(description);
|
|
EnsureDebug(description);
|
|
}
|
|
|
|
public virtual ReadOnlyCollection<ServiceEndpoint> AddDefaultEndpoints()
|
|
{
|
|
List<ServiceEndpoint> defaultEndpoints = new List<ServiceEndpoint>();
|
|
foreach (Uri baseAddress in this.InternalBaseAddresses)
|
|
{
|
|
ProtocolMappingItem protocolMappingItem = ConfigLoader.LookupProtocolMapping(baseAddress.Scheme);
|
|
if (protocolMappingItem != null)
|
|
{
|
|
Binding defaultBinding = ConfigLoader.LookupBinding(protocolMappingItem.Binding, protocolMappingItem.BindingConfiguration);
|
|
if (defaultBinding != null)
|
|
{
|
|
AddDefaultEndpoints(defaultBinding, defaultEndpoints);
|
|
}
|
|
else
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new Exception(SR.GetString(SR.BindingProtocolMappingNotDefined, baseAddress.Scheme)));
|
|
}
|
|
}
|
|
}
|
|
if (DiagnosticUtility.ShouldTraceInformation && defaultEndpoints.Count > 0)
|
|
{
|
|
Dictionary<string, string> dictionary = new Dictionary<string, string>();
|
|
dictionary["ServiceConfigurationName"] = this.description.ConfigurationName;
|
|
TraceUtility.TraceEvent(TraceEventType.Information, TraceCode.DefaultEndpointsAdded, SR.GetString(SR.TraceCodeDefaultEndpointsAdded), new DictionaryTraceRecord(dictionary));
|
|
}
|
|
return new ReadOnlyCollection<ServiceEndpoint>(defaultEndpoints);
|
|
}
|
|
|
|
internal virtual void AddDefaultEndpoints(Binding defaultBinding, List<ServiceEndpoint> defaultEndpoints)
|
|
{
|
|
}
|
|
|
|
internal virtual void BindInstance(InstanceContext instance)
|
|
{
|
|
this.instances.Add(instance);
|
|
if (null != this.servicePerformanceCounters)
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
if (null != this.servicePerformanceCounters)
|
|
{
|
|
this.servicePerformanceCounters.ServiceInstanceCreated();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void IDisposable.Dispose()
|
|
{
|
|
Close();
|
|
}
|
|
|
|
protected abstract ServiceDescription CreateDescription(out IDictionary<string, ContractDescription> implementedContracts);
|
|
|
|
protected virtual void InitializeRuntime()
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotInitializeRuntimeWithoutDescription)));
|
|
}
|
|
|
|
if (this.Description.Endpoints.Count == 0)
|
|
{
|
|
this.AddDefaultEndpoints();
|
|
}
|
|
|
|
this.EnsureAuthenticationSchemes();
|
|
|
|
DispatcherBuilder dispatcherBuilder = new DispatcherBuilder();
|
|
dispatcherBuilder.InitializeServiceHost(description, this);
|
|
|
|
SecurityValidationBehavior.Instance.AfterBuildTimeValidation(description);
|
|
}
|
|
|
|
internal virtual void AfterInitializeRuntime(TimeSpan timeout)
|
|
{
|
|
}
|
|
|
|
internal virtual IAsyncResult BeginAfterInitializeRuntime(TimeSpan timeout, AsyncCallback callback, object state)
|
|
{
|
|
return new CompletedAsyncResult(callback, state);
|
|
}
|
|
|
|
internal virtual void EndAfterInitializeRuntime(IAsyncResult result)
|
|
{
|
|
CompletedAsyncResult.End(result);
|
|
}
|
|
|
|
ServiceAuthorizationBehavior EnsureAuthorization(ServiceDescription description)
|
|
{
|
|
Fx.Assert(this.State == CommunicationState.Created || this.State == CommunicationState.Opening, "");
|
|
ServiceAuthorizationBehavior a = description.Behaviors.Find<ServiceAuthorizationBehavior>();
|
|
|
|
if (a == null)
|
|
{
|
|
a = new ServiceAuthorizationBehavior();
|
|
description.Behaviors.Add(a);
|
|
}
|
|
|
|
return a;
|
|
}
|
|
|
|
ServiceAuthenticationBehavior EnsureAuthentication(ServiceDescription description)
|
|
{
|
|
Fx.Assert(this.State == CommunicationState.Created || this.State == CommunicationState.Opening, "");
|
|
ServiceAuthenticationBehavior a = description.Behaviors.Find<ServiceAuthenticationBehavior>();
|
|
|
|
if (a == null)
|
|
{
|
|
a = new ServiceAuthenticationBehavior();
|
|
description.Behaviors.Add(a);
|
|
}
|
|
return a;
|
|
}
|
|
|
|
ServiceDebugBehavior EnsureDebug(ServiceDescription description)
|
|
{
|
|
Fx.Assert(this.State == CommunicationState.Created || this.State == CommunicationState.Opening, "");
|
|
ServiceDebugBehavior m = description.Behaviors.Find<ServiceDebugBehavior>();
|
|
|
|
if (m == null)
|
|
{
|
|
m = new ServiceDebugBehavior();
|
|
description.Behaviors.Add(m);
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
ServiceCredentials EnsureCredentials(ServiceDescription description)
|
|
{
|
|
Fx.Assert(this.State == CommunicationState.Created || this.State == CommunicationState.Opening, "");
|
|
ServiceCredentials c = description.Behaviors.Find<ServiceCredentials>();
|
|
|
|
if (c == null)
|
|
{
|
|
c = new ServiceCredentials();
|
|
description.Behaviors.Add(c);
|
|
}
|
|
|
|
return c;
|
|
}
|
|
|
|
internal void FaultInternal()
|
|
{
|
|
this.Fault();
|
|
}
|
|
|
|
internal string GetBaseAddressSchemes()
|
|
{
|
|
return GetBaseAddressSchemes(baseAddresses);
|
|
}
|
|
|
|
internal static String GetBaseAddressSchemes(UriSchemeKeyedCollection uriSchemeKeyedCollection)
|
|
{
|
|
StringBuilder buffer = new StringBuilder();
|
|
bool firstScheme = true;
|
|
foreach (Uri address in uriSchemeKeyedCollection)
|
|
{
|
|
if (firstScheme)
|
|
{
|
|
buffer.Append(address.Scheme);
|
|
firstScheme = false;
|
|
}
|
|
else
|
|
{
|
|
buffer.Append(CultureInfo.CurrentCulture.TextInfo.ListSeparator).Append(address.Scheme);
|
|
}
|
|
}
|
|
return buffer.ToString();
|
|
}
|
|
|
|
internal BindingParameterCollection GetBindingParameters()
|
|
{
|
|
return DispatcherBuilder.GetBindingParameters(this, new Collection<ServiceEndpoint>());
|
|
}
|
|
|
|
internal BindingParameterCollection GetBindingParameters(ServiceEndpoint inputEndpoint)
|
|
{
|
|
Collection<ServiceEndpoint> endpoints;
|
|
if (inputEndpoint == null)
|
|
{
|
|
endpoints = new Collection<ServiceEndpoint>();
|
|
}
|
|
else if (!this.EndpointsByListenUriInfo.TryGetValue(DispatcherBuilder.GetListenUriInfoForEndpoint(this, inputEndpoint), out endpoints) || !endpoints.Contains(inputEndpoint))
|
|
{
|
|
endpoints = new Collection<ServiceEndpoint>();
|
|
endpoints.Add(inputEndpoint);
|
|
}
|
|
|
|
return DispatcherBuilder.GetBindingParameters(this, endpoints);
|
|
}
|
|
|
|
internal BindingParameterCollection GetBindingParameters(Collection<ServiceEndpoint> endpoints)
|
|
{
|
|
return DispatcherBuilder.GetBindingParameters(this, endpoints);
|
|
}
|
|
|
|
internal ReadOnlyCollection<InstanceContext> GetInstanceContexts()
|
|
{
|
|
return Array.AsReadOnly<InstanceContext>(this.instances.ToArray());
|
|
}
|
|
|
|
internal virtual IContractResolver GetContractResolver(IDictionary<string, ContractDescription> implementedContracts)
|
|
{
|
|
ServiceAndBehaviorsContractResolver resolver = new ServiceAndBehaviorsContractResolver(new ImplementedContractsContractResolver(implementedContracts));
|
|
resolver.AddBehaviorContractsToResolver(this.description == null ? null : this.description.Behaviors);
|
|
return resolver;
|
|
}
|
|
|
|
internal static Uri GetUri(Uri baseUri, Uri relativeUri)
|
|
{
|
|
return GetUri(baseUri, relativeUri.OriginalString);
|
|
}
|
|
|
|
internal static Uri GetUri(Uri baseUri, string path)
|
|
{
|
|
if (path.StartsWith("/", StringComparison.Ordinal) || path.StartsWith("\\", StringComparison.Ordinal))
|
|
{
|
|
int i = 1;
|
|
for (; i < path.Length; ++i)
|
|
{
|
|
if (path[i] != '/' && path[i] != '\\')
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
path = path.Substring(i);
|
|
}
|
|
|
|
// VSWhidbey#541152: new Uri(Uri, string.Empty) is broken
|
|
if (path.Length == 0)
|
|
return baseUri;
|
|
|
|
if (!baseUri.AbsoluteUri.EndsWith("/", StringComparison.Ordinal))
|
|
{
|
|
baseUri = new Uri(baseUri.AbsoluteUri + "/");
|
|
}
|
|
return new Uri(baseUri, path);
|
|
}
|
|
|
|
internal Uri GetVia(string scheme, Uri address)
|
|
{
|
|
return ServiceHost.GetVia(scheme, address, InternalBaseAddresses);
|
|
}
|
|
|
|
internal static Uri GetVia(string scheme, Uri address, UriSchemeKeyedCollection baseAddresses)
|
|
{
|
|
Uri via = address;
|
|
if (!via.IsAbsoluteUri)
|
|
{
|
|
if (!baseAddresses.Contains(scheme))
|
|
{
|
|
return null;
|
|
}
|
|
|
|
via = GetUri(baseAddresses[scheme], address);
|
|
}
|
|
return via;
|
|
}
|
|
|
|
public int IncrementManualFlowControlLimit(int incrementBy)
|
|
{
|
|
return this.ServiceThrottle.IncrementManualFlowControlLimit(incrementBy);
|
|
}
|
|
|
|
protected void InitializeDescription(UriSchemeKeyedCollection baseAddresses)
|
|
{
|
|
foreach (Uri baseAddress in baseAddresses)
|
|
{
|
|
this.baseAddresses.Add(baseAddress);
|
|
}
|
|
IDictionary<string, ContractDescription> implementedContracts = null;
|
|
ServiceDescription description = CreateDescription(out implementedContracts);
|
|
this.description = description;
|
|
this.implementedContracts = implementedContracts;
|
|
|
|
ApplyConfiguration();
|
|
this.initializeDescriptionHasFinished = true;
|
|
}
|
|
|
|
protected void LoadConfigurationSection(ServiceElement serviceSection)
|
|
{
|
|
if (serviceSection == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("serviceSection");
|
|
}
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotLoadConfigurationSectionWithoutDescription)));
|
|
}
|
|
|
|
ConfigLoader configLoader = new ConfigLoader(GetContractResolver(this.ImplementedContracts));
|
|
LoadConfigurationSectionInternal(configLoader, this.Description, serviceSection);
|
|
}
|
|
|
|
internal void LoadConfigurationSectionHelper(Uri baseAddress)
|
|
{
|
|
this.AddBaseAddress(baseAddress);
|
|
}
|
|
|
|
[Fx.Tag.SecurityNote(Critical = "Calls LookupService which is critical.",
|
|
Safe = "Doesn't leak ServiceElement out of SecurityCritical code.")]
|
|
[SecuritySafeCritical]
|
|
void LoadConfigurationSectionInternal(ConfigLoader configLoader, ServiceDescription description, string configurationName)
|
|
{
|
|
ServiceElement serviceSection = configLoader.LookupService(configurationName);
|
|
LoadConfigurationSectionInternal(configLoader, description, serviceSection);
|
|
}
|
|
|
|
[Fx.Tag.SecurityNote(Critical = "Handles a ServiceElement, which should not be leaked out of SecurityCritical code.",
|
|
Safe = "Doesn't leak ServiceElement out of SecurityCritical code.")]
|
|
[SecuritySafeCritical]
|
|
void LoadConfigurationSectionInternal(ConfigLoader configLoader, ServiceDescription description, ServiceElement serviceSection)
|
|
{
|
|
// caller must validate arguments before calling
|
|
configLoader.LoadServiceDescription(this, description, serviceSection, this.LoadConfigurationSectionHelper);
|
|
}
|
|
|
|
protected override void OnAbort()
|
|
{
|
|
this.instances.Abort();
|
|
|
|
foreach (ChannelDispatcherBase dispatcher in this.ChannelDispatchers)
|
|
{
|
|
if (dispatcher.Listener != null)
|
|
{
|
|
dispatcher.Listener.Abort();
|
|
}
|
|
dispatcher.Abort();
|
|
}
|
|
ThreadTrace.StopTracing();
|
|
}
|
|
|
|
internal void OnAddChannelDispatcher(ChannelDispatcherBase channelDispatcher)
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
this.ThrowIfClosedOrOpened();
|
|
channelDispatcher.AttachInternal(this);
|
|
channelDispatcher.Faulted += new EventHandler(OnChannelDispatcherFaulted);
|
|
}
|
|
}
|
|
|
|
protected override IAsyncResult OnBeginClose(TimeSpan timeout, AsyncCallback callback, object state)
|
|
{
|
|
return new CloseAsyncResult(timeout, callback, state, this);
|
|
}
|
|
|
|
void OnBeginOpen()
|
|
{
|
|
this.TraceServiceHostOpenStart();
|
|
this.TraceBaseAddresses();
|
|
MessageLogger.EnsureInitialized(); //force config validation instead of waiting for the first message exchange
|
|
InitializeRuntime();
|
|
}
|
|
|
|
protected override IAsyncResult OnBeginOpen(TimeSpan timeout, AsyncCallback callback, object state)
|
|
{
|
|
this.OnBeginOpen();
|
|
return new OpenAsyncResult(this, timeout, callback, state);
|
|
}
|
|
|
|
IAsyncResult BeginOpenChannelDispatchers(TimeSpan timeout, AsyncCallback callback, object state)
|
|
{
|
|
return new OpenCollectionAsyncResult(timeout, callback, state, this.SnapshotChannelDispatchers());
|
|
}
|
|
|
|
protected override void OnClose(TimeSpan timeout)
|
|
{
|
|
try
|
|
{
|
|
TimeoutHelper timeoutHelper = new TimeoutHelper(timeout);
|
|
|
|
if (ManagementExtension.IsEnabled && null != this.Description)
|
|
{
|
|
ManagementExtension.OnServiceClosing(this);
|
|
}
|
|
|
|
for (int i = 0; i < this.ChannelDispatchers.Count; i++)
|
|
{
|
|
ChannelDispatcherBase dispatcher = this.ChannelDispatchers[i];
|
|
if (dispatcher.Listener != null)
|
|
{
|
|
dispatcher.Listener.Close(timeoutHelper.RemainingTime());
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < this.ChannelDispatchers.Count; i++)
|
|
{
|
|
ChannelDispatcherBase dispatcher = this.ChannelDispatchers[i];
|
|
dispatcher.CloseInput(timeoutHelper.RemainingTime());
|
|
}
|
|
|
|
// Wait for existing work to complete
|
|
this.instances.CloseInput(timeoutHelper.RemainingTime());
|
|
|
|
// Close instances (closes contexts/channels)
|
|
this.instances.Close(timeoutHelper.RemainingTime());
|
|
|
|
// Close dispatchers
|
|
for (int i = 0; i < this.ChannelDispatchers.Count; i++)
|
|
{
|
|
ChannelDispatcherBase dispatcher = this.ChannelDispatchers[i];
|
|
dispatcher.Close(timeoutHelper.RemainingTime());
|
|
}
|
|
|
|
this.ReleasePerformanceCounters();
|
|
|
|
this.TraceBaseAddresses();
|
|
ThreadTrace.StopTracing();
|
|
}
|
|
catch (TimeoutException e)
|
|
{
|
|
if (TD.CloseTimeoutIsEnabled())
|
|
{
|
|
TD.CloseTimeout(SR.GetString(SR.TraceCodeServiceHostTimeoutOnClose));
|
|
}
|
|
if (DiagnosticUtility.ShouldTraceWarning)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Warning, TraceCode.ServiceHostTimeoutOnClose, SR.GetString(SR.TraceCodeServiceHostTimeoutOnClose), this, e);
|
|
}
|
|
this.Abort();
|
|
}
|
|
|
|
}
|
|
|
|
protected override void OnClosed()
|
|
{
|
|
try
|
|
{
|
|
for (int i = 0; i < this.ChannelDispatchers.Count; i++)
|
|
{
|
|
ChannelDispatcher dispatcher = this.ChannelDispatchers[i] as ChannelDispatcher;
|
|
if (dispatcher != null)
|
|
{
|
|
dispatcher.ReleasePerformanceCounters();
|
|
}
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
base.OnClosed();
|
|
}
|
|
}
|
|
|
|
void TraceBaseAddresses()
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceInformation && this.baseAddresses != null
|
|
&& this.baseAddresses.Count > 0)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Information,
|
|
TraceCode.ServiceHostBaseAddresses,
|
|
SR.GetString(SR.TraceCodeServiceHostBaseAddresses),
|
|
new CollectionTraceRecord("BaseAddresses", "Address", this.baseAddresses),
|
|
this, null);
|
|
}
|
|
}
|
|
|
|
void TraceServiceHostOpenStart()
|
|
{
|
|
if (TD.ServiceHostOpenStartIsEnabled())
|
|
{
|
|
TD.ServiceHostOpenStart(this.EventTraceActivity);
|
|
}
|
|
}
|
|
|
|
protected override void OnEndClose(IAsyncResult result)
|
|
{
|
|
try
|
|
{
|
|
CloseAsyncResult.End(result);
|
|
this.TraceBaseAddresses();
|
|
ThreadTrace.StopTracing();
|
|
}
|
|
catch (TimeoutException e)
|
|
{
|
|
if (TD.CloseTimeoutIsEnabled())
|
|
{
|
|
TD.CloseTimeout(SR.GetString(SR.TraceCodeServiceHostTimeoutOnClose));
|
|
}
|
|
if (DiagnosticUtility.ShouldTraceWarning)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Warning, TraceCode.ServiceHostTimeoutOnClose,
|
|
SR.GetString(SR.TraceCodeServiceHostTimeoutOnClose), this, e);
|
|
}
|
|
this.Abort();
|
|
}
|
|
}
|
|
|
|
protected override void OnEndOpen(IAsyncResult result)
|
|
{
|
|
OpenAsyncResult.End(result);
|
|
}
|
|
|
|
void EndOpenChannelDispatchers(IAsyncResult result)
|
|
{
|
|
OpenCollectionAsyncResult.End(result);
|
|
}
|
|
|
|
void EnsureAuthenticationSchemes()
|
|
{
|
|
if (this.Authentication == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
//Exit immediately when not hosted in IIS or if VirtualPathExtension is not set. VirtualPathExtension is used as a flag to indicate whether a ServiceHost
|
|
// is webhosted (WsDualHttpBinding-ChannelFactory is using HttpListener instead of IIS even when running in IIS)
|
|
if (!AspNetEnvironment.Enabled ||
|
|
this.Extensions.Find<VirtualPathExtension>() == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
foreach (ServiceEndpoint serviceEndpoint in this.Description.Endpoints)
|
|
{
|
|
if (serviceEndpoint.Binding != null &&
|
|
serviceEndpoint.ListenUri != null &&
|
|
("http".Equals(serviceEndpoint.ListenUri.Scheme, StringComparison.OrdinalIgnoreCase) || "https".Equals(serviceEndpoint.ListenUri.Scheme, StringComparison.OrdinalIgnoreCase)) &&
|
|
this.baseAddresses.Contains(serviceEndpoint.ListenUri.Scheme))
|
|
{
|
|
HttpTransportBindingElement httpTransportBindingElement = serviceEndpoint.Binding.CreateBindingElements().Find<HttpTransportBindingElement>();
|
|
|
|
if (httpTransportBindingElement != null)
|
|
{
|
|
AuthenticationSchemes hostSupportedAuthenticationSchemes = AspNetEnvironment.Current.GetAuthenticationSchemes(this.baseAddresses[serviceEndpoint.ListenUri.Scheme]);
|
|
|
|
if (hostSupportedAuthenticationSchemes != AuthenticationSchemes.None)
|
|
{
|
|
//If no authentication schemes are explicitly defined for the ServiceHost...
|
|
if (this.Authentication.AuthenticationSchemes == AuthenticationSchemes.None)
|
|
{
|
|
//Inherit authentication schemes from IIS
|
|
this.Authentication.AuthenticationSchemes = hostSupportedAuthenticationSchemes;
|
|
}
|
|
else
|
|
{
|
|
// Build intersection between authenticationSchemes on the ServiceHost and in IIS
|
|
this.Authentication.AuthenticationSchemes &= hostSupportedAuthenticationSchemes;
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
protected override void OnOpen(TimeSpan timeout)
|
|
{
|
|
TimeoutHelper timeoutHelper = new TimeoutHelper(timeout);
|
|
this.OnBeginOpen();
|
|
|
|
AfterInitializeRuntime(timeoutHelper.RemainingTime());
|
|
|
|
for (int i = 0; i < this.ChannelDispatchers.Count; i++)
|
|
{
|
|
ChannelDispatcherBase dispatcher = this.ChannelDispatchers[i];
|
|
dispatcher.Open(timeoutHelper.RemainingTime());
|
|
}
|
|
}
|
|
|
|
protected override void OnOpened()
|
|
{
|
|
if (this.Description != null)
|
|
{
|
|
ServiceCredentials c = description.Behaviors.Find<ServiceCredentials>();
|
|
if (c != null)
|
|
{
|
|
ServiceCredentials credentialsCopy = c.Clone();
|
|
credentialsCopy.MakeReadOnly();
|
|
this.readOnlyCredentials = credentialsCopy;
|
|
}
|
|
|
|
ServiceAuthorizationBehavior authorization = description.Behaviors.Find<ServiceAuthorizationBehavior>();
|
|
if (authorization != null)
|
|
{
|
|
ServiceAuthorizationBehavior authorizationCopy = authorization.Clone();
|
|
authorizationCopy.MakeReadOnly();
|
|
this.readOnlyAuthorization = authorizationCopy;
|
|
}
|
|
|
|
ServiceAuthenticationBehavior authentication = description.Behaviors.Find<ServiceAuthenticationBehavior>();
|
|
if (authentication != null)
|
|
{
|
|
ServiceAuthenticationBehavior authenticationCopy = authentication.Clone();
|
|
authentication.MakeReadOnly();
|
|
this.readOnlyAuthentication = authenticationCopy;
|
|
}
|
|
|
|
if (ManagementExtension.IsEnabled)
|
|
{
|
|
ManagementExtension.OnServiceOpened(this);
|
|
}
|
|
|
|
// log telemetry data for the current WCF service.
|
|
TelemetryTraceLogging.LogSeriveKPIData(this.Description);
|
|
}
|
|
base.OnOpened();
|
|
|
|
if (TD.ServiceHostOpenStopIsEnabled())
|
|
{
|
|
TD.ServiceHostOpenStop(this.EventTraceActivity);
|
|
}
|
|
}
|
|
|
|
internal void OnRemoveChannelDispatcher(ChannelDispatcherBase channelDispatcher)
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
this.ThrowIfClosedOrOpened();
|
|
channelDispatcher.DetachInternal(this);
|
|
}
|
|
}
|
|
|
|
void OnChannelDispatcherFaulted(object sender, EventArgs e)
|
|
{
|
|
this.Fault();
|
|
}
|
|
|
|
void OnServiceHostFaulted(object sender, EventArgs args)
|
|
{
|
|
if (TD.ServiceHostFaultedIsEnabled())
|
|
{
|
|
TD.ServiceHostFaulted(this.EventTraceActivity, this);
|
|
}
|
|
|
|
if (DiagnosticUtility.ShouldTraceWarning)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Warning, TraceCode.ServiceHostFaulted,
|
|
SR.GetString(SR.TraceCodeServiceHostFaulted), this);
|
|
}
|
|
|
|
foreach (ICommunicationObject channelDispatcher in this.SnapshotChannelDispatchers())
|
|
{
|
|
if (channelDispatcher.State == CommunicationState.Opened)
|
|
{
|
|
channelDispatcher.Abort();
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void RaiseUnknownMessageReceived(Message message)
|
|
{
|
|
try
|
|
{
|
|
EventHandler<UnknownMessageReceivedEventArgs> handler = UnknownMessageReceived;
|
|
if (handler != null)
|
|
{
|
|
handler(this, new UnknownMessageReceivedEventArgs(message));
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e))
|
|
throw;
|
|
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperCallback(e);
|
|
}
|
|
}
|
|
|
|
protected void ReleasePerformanceCounters()
|
|
{
|
|
if (this.servicePerformanceCounters != null)
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
if (this.servicePerformanceCounters != null)
|
|
{
|
|
this.servicePerformanceCounters.Dispose();
|
|
this.servicePerformanceCounters = null;
|
|
}
|
|
}
|
|
}
|
|
if (this.defaultPerformanceCounters != null)
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
if (this.defaultPerformanceCounters != null)
|
|
{
|
|
this.defaultPerformanceCounters.Dispose();
|
|
this.defaultPerformanceCounters = null;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
ICommunicationObject[] SnapshotChannelDispatchers()
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
ICommunicationObject[] array = new ICommunicationObject[this.ChannelDispatchers.Count];
|
|
for (int i = 0; i < array.Length; i++)
|
|
{
|
|
array[i] = this.ChannelDispatchers[i];
|
|
}
|
|
return array;
|
|
}
|
|
}
|
|
|
|
internal virtual void UnbindInstance(InstanceContext instance)
|
|
{
|
|
this.instances.Remove(instance);
|
|
if (null != this.servicePerformanceCounters)
|
|
{
|
|
lock (this.ThisLock)
|
|
{
|
|
if (null != this.servicePerformanceCounters)
|
|
{
|
|
this.servicePerformanceCounters.ServiceInstanceRemoved();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void IncrementBusyCount()
|
|
{
|
|
if (AspNetEnvironment.Enabled)
|
|
{
|
|
AspNetEnvironment.Current.IncrementBusyCount();
|
|
Interlocked.Increment(ref this.busyCount);
|
|
}
|
|
|
|
EventHandler handler = this.BusyCountIncremented;
|
|
if (handler != null)
|
|
{
|
|
try
|
|
{
|
|
handler(this, EventArgs.Empty);
|
|
}
|
|
catch (Exception exception)
|
|
{
|
|
if (Fx.IsFatal(exception))
|
|
throw;
|
|
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperCallback(exception);
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void DecrementBusyCount()
|
|
{
|
|
if (AspNetEnvironment.Enabled)
|
|
{
|
|
Interlocked.Decrement(ref this.busyCount);
|
|
AspNetEnvironment.Current.DecrementBusyCount();
|
|
}
|
|
}
|
|
|
|
internal int BusyCount
|
|
{
|
|
get
|
|
{
|
|
return this.busyCount;
|
|
}
|
|
}
|
|
|
|
class OpenAsyncResult : AsyncResult
|
|
{
|
|
static AsyncCompletion handleEndAfterInitializeRuntime = new AsyncCompletion(HandleEndAfterInitializeRuntime);
|
|
static AsyncCompletion handleEndOpenChannelDispatchers = new AsyncCompletion(HandleEndOpenChannelDispatchers);
|
|
|
|
TimeoutHelper timeoutHelper;
|
|
ServiceHostBase host;
|
|
|
|
public OpenAsyncResult(ServiceHostBase host, TimeSpan timeout, AsyncCallback callback, object state)
|
|
: base(callback, state)
|
|
{
|
|
this.timeoutHelper = new TimeoutHelper(timeout);
|
|
this.host = host;
|
|
|
|
if (ProcessAfterInitializeRuntime())
|
|
{
|
|
Complete(true);
|
|
}
|
|
}
|
|
|
|
bool ProcessAfterInitializeRuntime()
|
|
{
|
|
IAsyncResult result = this.host.BeginAfterInitializeRuntime(
|
|
this.timeoutHelper.RemainingTime(), PrepareAsyncCompletion(handleEndAfterInitializeRuntime), this);
|
|
|
|
return SyncContinue(result);
|
|
}
|
|
|
|
static bool HandleEndAfterInitializeRuntime(IAsyncResult result)
|
|
{
|
|
OpenAsyncResult thisPtr = (OpenAsyncResult)result.AsyncState;
|
|
thisPtr.host.EndAfterInitializeRuntime(result);
|
|
|
|
return thisPtr.ProcessOpenChannelDispatchers();
|
|
}
|
|
|
|
bool ProcessOpenChannelDispatchers()
|
|
{
|
|
IAsyncResult result = this.host.BeginOpenChannelDispatchers(
|
|
this.timeoutHelper.RemainingTime(), PrepareAsyncCompletion(handleEndOpenChannelDispatchers), this);
|
|
|
|
return SyncContinue(result);
|
|
}
|
|
|
|
static bool HandleEndOpenChannelDispatchers(IAsyncResult result)
|
|
{
|
|
OpenAsyncResult thisPtr = (OpenAsyncResult)result.AsyncState;
|
|
thisPtr.host.EndOpenChannelDispatchers(result);
|
|
|
|
return true;
|
|
}
|
|
|
|
public static void End(IAsyncResult result)
|
|
{
|
|
AsyncResult.End<OpenAsyncResult>(result);
|
|
}
|
|
}
|
|
|
|
class CloseAsyncResult : AsyncResult
|
|
{
|
|
ServiceHostBase serviceHost;
|
|
TimeoutHelper timeoutHelper;
|
|
|
|
public CloseAsyncResult(TimeSpan timeout, AsyncCallback callback, object state, ServiceHostBase serviceHost)
|
|
: base(callback, state)
|
|
{
|
|
this.timeoutHelper = new TimeoutHelper(timeout);
|
|
this.serviceHost = serviceHost;
|
|
|
|
if (ManagementExtension.IsEnabled && null != serviceHost.Description)
|
|
{
|
|
ManagementExtension.OnServiceClosing(serviceHost);
|
|
}
|
|
|
|
this.CloseListeners(true);
|
|
}
|
|
|
|
void CloseListeners(bool completedSynchronously)
|
|
{
|
|
List<ICommunicationObject> listeners = new List<ICommunicationObject>();
|
|
for (int i = 0; i < this.serviceHost.ChannelDispatchers.Count; i++)
|
|
{
|
|
if (this.serviceHost.ChannelDispatchers[i].Listener != null)
|
|
{
|
|
listeners.Add(this.serviceHost.ChannelDispatchers[i].Listener);
|
|
}
|
|
}
|
|
|
|
AsyncCallback callback = Fx.ThunkCallback(this.CloseListenersCallback);
|
|
TimeSpan timeout = this.timeoutHelper.RemainingTime();
|
|
Exception exception = null;
|
|
IAsyncResult result = null;
|
|
try
|
|
{
|
|
result = new CloseCollectionAsyncResult(timeout, callback, this, listeners);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else if (result.CompletedSynchronously)
|
|
{
|
|
this.FinishCloseListeners(result, completedSynchronously);
|
|
}
|
|
}
|
|
|
|
void CloseListenersCallback(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
((CloseAsyncResult)result.AsyncState).FinishCloseListeners(result, false);
|
|
}
|
|
}
|
|
|
|
void FinishCloseListeners(IAsyncResult result, bool completedSynchronously)
|
|
{
|
|
Exception exception = null;
|
|
try
|
|
{
|
|
CloseCollectionAsyncResult.End(result);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else
|
|
{
|
|
this.CloseInput(completedSynchronously);
|
|
}
|
|
}
|
|
|
|
// Wait for existing work to complete
|
|
void CloseInput(bool completedSynchronously)
|
|
{
|
|
AsyncCallback callback = Fx.ThunkCallback(this.CloseInputCallback);
|
|
Exception exception = null;
|
|
IAsyncResult result = null;
|
|
|
|
try
|
|
{
|
|
for (int i = 0; i < this.serviceHost.ChannelDispatchers.Count; i++)
|
|
{
|
|
ChannelDispatcherBase dispatcher = this.serviceHost.ChannelDispatchers[i];
|
|
dispatcher.CloseInput(this.timeoutHelper.RemainingTime());
|
|
}
|
|
|
|
result = this.serviceHost.instances.BeginCloseInput(this.timeoutHelper.RemainingTime(), callback, this);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
// Any exception during async processing causes this
|
|
// async callback to report the error and then relies on
|
|
// Abort to cleanup any unclosed channels or instance contexts.
|
|
FxTrace.Exception.AsWarning(exception);
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else if (result.CompletedSynchronously)
|
|
{
|
|
this.FinishCloseInput(result, completedSynchronously);
|
|
}
|
|
}
|
|
|
|
void CloseInputCallback(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
((CloseAsyncResult)result.AsyncState).FinishCloseInput(result, false);
|
|
}
|
|
}
|
|
|
|
void FinishCloseInput(IAsyncResult result, bool completedSynchronously)
|
|
{
|
|
Exception exception = null;
|
|
try
|
|
{
|
|
serviceHost.instances.EndCloseInput(result);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else
|
|
{
|
|
this.CloseInstances(completedSynchronously);
|
|
}
|
|
}
|
|
|
|
// Close instances (closes contexts/channels)
|
|
void CloseInstances(bool completedSynchronously)
|
|
{
|
|
AsyncCallback callback = Fx.ThunkCallback(this.CloseInstancesCallback);
|
|
TimeSpan timeout = this.timeoutHelper.RemainingTime();
|
|
Exception exception = null;
|
|
IAsyncResult result = null;
|
|
|
|
try
|
|
{
|
|
result = this.serviceHost.instances.BeginClose(timeout, callback, this);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else if (result.CompletedSynchronously)
|
|
{
|
|
this.FinishCloseInstances(result, completedSynchronously);
|
|
}
|
|
}
|
|
|
|
void CloseInstancesCallback(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
((CloseAsyncResult)result.AsyncState).FinishCloseInstances(result, false);
|
|
}
|
|
}
|
|
|
|
void FinishCloseInstances(IAsyncResult result, bool completedSynchronously)
|
|
{
|
|
Exception exception = null;
|
|
try
|
|
{
|
|
this.serviceHost.instances.EndClose(result);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else
|
|
{
|
|
this.CloseChannelDispatchers(completedSynchronously);
|
|
}
|
|
}
|
|
|
|
void CloseChannelDispatchers(bool completedSynchronously)
|
|
{
|
|
IList<ICommunicationObject> channelDispatchers = this.serviceHost.SnapshotChannelDispatchers();
|
|
AsyncCallback callback = Fx.ThunkCallback(this.CloseChannelDispatchersCallback);
|
|
TimeSpan timeout = this.timeoutHelper.RemainingTime();
|
|
Exception exception = null;
|
|
IAsyncResult result = null;
|
|
try
|
|
{
|
|
result = new CloseCollectionAsyncResult(timeout, callback, this, channelDispatchers);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
if (exception != null)
|
|
{
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
else if (result.CompletedSynchronously)
|
|
{
|
|
this.FinishCloseChannelDispatchers(result, completedSynchronously);
|
|
}
|
|
}
|
|
|
|
void CloseChannelDispatchersCallback(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
((CloseAsyncResult)result.AsyncState).FinishCloseChannelDispatchers(result, false);
|
|
}
|
|
}
|
|
|
|
void FinishCloseChannelDispatchers(IAsyncResult result, bool completedSynchronously)
|
|
{
|
|
Exception exception = null;
|
|
try
|
|
{
|
|
CloseCollectionAsyncResult.End(result);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || completedSynchronously)
|
|
{
|
|
throw;
|
|
}
|
|
exception = e;
|
|
}
|
|
|
|
this.CallComplete(completedSynchronously, exception);
|
|
}
|
|
|
|
void CallComplete(bool completedSynchronously, Exception exception)
|
|
{
|
|
this.Complete(completedSynchronously, exception);
|
|
}
|
|
|
|
public static void End(IAsyncResult result)
|
|
{
|
|
AsyncResult.End<CloseAsyncResult>(result);
|
|
}
|
|
}
|
|
|
|
class ImplementedContractsContractResolver : IContractResolver
|
|
{
|
|
IDictionary<string, ContractDescription> implementedContracts;
|
|
|
|
public ImplementedContractsContractResolver(IDictionary<string, ContractDescription> implementedContracts)
|
|
{
|
|
this.implementedContracts = implementedContracts;
|
|
}
|
|
|
|
public ContractDescription ResolveContract(string contractName)
|
|
{
|
|
return this.implementedContracts != null && this.implementedContracts.ContainsKey(contractName) ? this.implementedContracts[contractName] : null;
|
|
}
|
|
}
|
|
|
|
internal class ServiceAndBehaviorsContractResolver : IContractResolver
|
|
{
|
|
IContractResolver serviceResolver;
|
|
Dictionary<string, ContractDescription> behaviorContracts;
|
|
|
|
public Dictionary<string, ContractDescription> BehaviorContracts
|
|
{
|
|
get { return behaviorContracts; }
|
|
}
|
|
|
|
public ServiceAndBehaviorsContractResolver(IContractResolver serviceResolver)
|
|
{
|
|
this.serviceResolver = serviceResolver;
|
|
behaviorContracts = new Dictionary<string, ContractDescription>();
|
|
}
|
|
|
|
public ContractDescription ResolveContract(string contractName)
|
|
{
|
|
ContractDescription contract = serviceResolver.ResolveContract(contractName);
|
|
|
|
if (contract == null)
|
|
{
|
|
contract = this.behaviorContracts.ContainsKey(contractName) ? this.behaviorContracts[contractName] : null;
|
|
}
|
|
|
|
return contract;
|
|
}
|
|
|
|
public void AddBehaviorContractsToResolver(KeyedByTypeCollection<IServiceBehavior> behaviors)
|
|
{
|
|
// It would be nice to make this loop over all Behaviors... someday.
|
|
if (behaviors != null && behaviors.Contains(typeof(ServiceMetadataBehavior)))
|
|
{
|
|
behaviors.Find<ServiceMetadataBehavior>().AddImplementedContracts(this);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public class ServiceHost : ServiceHostBase
|
|
{
|
|
object singletonInstance;
|
|
Type serviceType;
|
|
ReflectedContractCollection reflectedContracts;
|
|
IDisposable disposableInstance;
|
|
|
|
protected ServiceHost()
|
|
{
|
|
}
|
|
|
|
public ServiceHost(Type serviceType, params Uri[] baseAddresses)
|
|
{
|
|
if (serviceType == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("serviceType"));
|
|
}
|
|
|
|
this.serviceType = serviceType;
|
|
using (ServiceModelActivity activity = DiagnosticUtility.ShouldUseActivity ? ServiceModelActivity.CreateBoundedActivity() : null)
|
|
{
|
|
if (DiagnosticUtility.ShouldUseActivity)
|
|
{
|
|
ServiceModelActivity.Start(activity, SR.GetString(SR.ActivityConstructServiceHost, serviceType.FullName), ActivityType.Construct);
|
|
}
|
|
|
|
InitializeDescription(serviceType, new UriSchemeKeyedCollection(baseAddresses));
|
|
}
|
|
}
|
|
|
|
public ServiceHost(object singletonInstance, params Uri[] baseAddresses)
|
|
{
|
|
if (singletonInstance == null)
|
|
{
|
|
throw new ArgumentNullException("singletonInstance");
|
|
}
|
|
|
|
this.singletonInstance = singletonInstance;
|
|
this.serviceType = singletonInstance.GetType();
|
|
using (ServiceModelActivity activity = DiagnosticUtility.ShouldUseActivity ? ServiceModelActivity.CreateBoundedActivity() : null)
|
|
{
|
|
if (DiagnosticUtility.ShouldUseActivity)
|
|
{
|
|
ServiceModelActivity.Start(activity, SR.GetString(SR.ActivityConstructServiceHost, serviceType.FullName), ActivityType.Construct);
|
|
}
|
|
|
|
InitializeDescription(singletonInstance, new UriSchemeKeyedCollection(baseAddresses));
|
|
}
|
|
}
|
|
|
|
public object SingletonInstance
|
|
{
|
|
get
|
|
{
|
|
return this.singletonInstance;
|
|
}
|
|
}
|
|
|
|
internal override object DisposableInstance
|
|
{
|
|
get
|
|
{
|
|
return this.disposableInstance;
|
|
}
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(Type implementedContract, Binding binding, string address)
|
|
{
|
|
return this.AddServiceEndpoint(implementedContract, binding, address, (Uri)null);
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(Type implementedContract, Binding binding, string address, Uri listenUri)
|
|
{
|
|
if (address == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("address"));
|
|
}
|
|
|
|
ServiceEndpoint endpoint = this.AddServiceEndpoint(implementedContract, binding, new Uri(address, UriKind.RelativeOrAbsolute));
|
|
if (listenUri != null)
|
|
{
|
|
listenUri = MakeAbsoluteUri(listenUri, binding);
|
|
endpoint.ListenUri = listenUri;
|
|
}
|
|
return endpoint;
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(Type implementedContract, Binding binding, Uri address)
|
|
{
|
|
return this.AddServiceEndpoint(implementedContract, binding, address, (Uri)null);
|
|
}
|
|
|
|
void ValidateContractType(Type implementedContract, ReflectedAndBehaviorContractCollection reflectedAndBehaviorContracts)
|
|
{
|
|
if (!implementedContract.IsDefined(typeof(ServiceContractAttribute), false))
|
|
{
|
|
#pragma warning suppress 56506 // implementedContract is never null at this point
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SfxServiceContractAttributeNotFound, implementedContract.FullName)));
|
|
}
|
|
if (!reflectedAndBehaviorContracts.Contains(implementedContract))
|
|
{
|
|
if (implementedContract == typeof(IMetadataExchange))
|
|
#pragma warning suppress 56506 // ServiceType is never null at this point
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SfxReflectedContractKeyNotFoundIMetadataExchange, this.serviceType.FullName)));
|
|
else
|
|
#pragma warning suppress 56506 // implementedContract and ServiceType are never null at this point
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SfxReflectedContractKeyNotFound2, implementedContract.FullName, this.serviceType.FullName)));
|
|
}
|
|
}
|
|
|
|
public ServiceEndpoint AddServiceEndpoint(Type implementedContract, Binding binding, Uri address, Uri listenUri)
|
|
{
|
|
if (implementedContract == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("implementedContract"));
|
|
}
|
|
if (this.reflectedContracts == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SfxReflectedContractsNotInitialized1, implementedContract.FullName)));
|
|
}
|
|
ReflectedAndBehaviorContractCollection reflectedAndBehaviorContracts = new ReflectedAndBehaviorContractCollection(this.reflectedContracts, this.Description.Behaviors);
|
|
ValidateContractType(implementedContract, reflectedAndBehaviorContracts);
|
|
ServiceEndpoint endpoint = AddServiceEndpoint(reflectedAndBehaviorContracts.GetConfigKey(implementedContract), binding, address);
|
|
if (listenUri != null)
|
|
{
|
|
listenUri = MakeAbsoluteUri(listenUri, binding);
|
|
endpoint.ListenUri = listenUri;
|
|
}
|
|
return endpoint;
|
|
}
|
|
|
|
internal override void AddDefaultEndpoints(Binding defaultBinding, List<ServiceEndpoint> defaultEndpoints)
|
|
{
|
|
// don't generate endpoints for contracts that serve as the base type for other reflected contracts
|
|
List<ContractDescription> mostSpecificContracts = new List<ContractDescription>();
|
|
for (int i = 0; i < this.reflectedContracts.Count; i++)
|
|
{
|
|
bool addContractEndpoint = true;
|
|
ContractDescription contract = this.reflectedContracts[i];
|
|
Type contractType = contract.ContractType;
|
|
if (contractType != null)
|
|
{
|
|
for (int j = 0; j < this.reflectedContracts.Count; j++)
|
|
{
|
|
ContractDescription otherContract = this.reflectedContracts[j];
|
|
Type otherContractType = otherContract.ContractType;
|
|
if (i == j || otherContractType == null)
|
|
{
|
|
continue;
|
|
}
|
|
if (contractType.IsAssignableFrom(otherContractType))
|
|
{
|
|
addContractEndpoint = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (addContractEndpoint)
|
|
{
|
|
mostSpecificContracts.Add(contract);
|
|
}
|
|
}
|
|
|
|
foreach (ContractDescription contract in mostSpecificContracts)
|
|
{
|
|
ServiceEndpoint endpoint = AddServiceEndpoint(contract.ConfigurationName, defaultBinding, string.Empty);
|
|
ConfigLoader.LoadDefaultEndpointBehaviors(endpoint);
|
|
defaultEndpoints.Add(endpoint);
|
|
}
|
|
}
|
|
|
|
// Run static Configure method on service type if it exists, else load configuration from Web.config/App.config
|
|
protected override void ApplyConfiguration()
|
|
{
|
|
// Load from static Configure method if it exists with the right signature
|
|
Type serviceType = this.Description.ServiceType;
|
|
if (serviceType != null)
|
|
{
|
|
MethodInfo configure = GetConfigureMethod(serviceType);
|
|
if (configure != null)
|
|
{
|
|
// load <host> config
|
|
ConfigLoader configLoader = new ConfigLoader(GetContractResolver(this.ImplementedContracts));
|
|
LoadHostConfigurationInternal(configLoader, this.Description, this.Description.ConfigurationName);
|
|
|
|
// Invoke configure method for service
|
|
ServiceConfiguration configuration = new ServiceConfiguration(this);
|
|
InvokeConfigure(configure, configuration);
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
// else just load from Web.config/App.config
|
|
base.ApplyConfiguration();
|
|
}
|
|
|
|
// Find the Configure method with the required signature, closest to serviceType in the type hierarchy
|
|
static MethodInfo GetConfigureMethod(Type serviceType)
|
|
{
|
|
// Use recursion instead of BindingFlags.FlattenHierarchy because we require return type to be void
|
|
|
|
// base case: all Types are rooted in object eventually
|
|
if (serviceType == typeof(object))
|
|
{
|
|
return null;
|
|
}
|
|
|
|
// signature: "public static void Configure(ServiceConfiguration)"
|
|
MethodInfo configure = serviceType.GetMethod("Configure", BindingFlags.Static | BindingFlags.Public, null, new[] { typeof(ServiceConfiguration) }, null);
|
|
|
|
if (configure != null && configure.ReturnType == typeof(void))
|
|
{
|
|
return configure;
|
|
}
|
|
else
|
|
{
|
|
return GetConfigureMethod(serviceType.BaseType);
|
|
}
|
|
}
|
|
|
|
static void InvokeConfigure(MethodInfo configureMethod, ServiceConfiguration configuration)
|
|
{
|
|
Action<ServiceConfiguration> call = Delegate.CreateDelegate(typeof(Action<ServiceConfiguration>), configureMethod) as Action<ServiceConfiguration>;
|
|
call(configuration);
|
|
}
|
|
|
|
// called from ServiceConfiguration.LoadFromConfiguration()
|
|
internal void LoadFromConfiguration()
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotApplyConfigurationWithoutDescription)));
|
|
}
|
|
|
|
ConfigLoader configLoader = new ConfigLoader(GetContractResolver(this.ImplementedContracts));
|
|
|
|
// Call the overload of LoadConfigurationSectionInternal which looks up the serviceElement from ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None)
|
|
LoadConfigurationSectionExceptHostInternal(configLoader, this.Description, this.Description.ConfigurationName);
|
|
EnsureAuthenticationAuthorizationDebug(this.Description);
|
|
}
|
|
|
|
// called from ServiceConfiguration.LoadFromConfiguration(configuration)
|
|
internal void LoadFromConfiguration(System.Configuration.Configuration configuration)
|
|
{
|
|
if (this.Description == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostBaseCannotApplyConfigurationWithoutDescription)));
|
|
}
|
|
|
|
ConfigLoader configLoader = new ConfigLoader(GetContractResolver(this.ImplementedContracts));
|
|
|
|
// Look up the serviceElement explicitly on configuration, then call the overload of LoadConfigurationSectionInternal that loads the rest of the config from the same configuration as serviceElement
|
|
ServicesSection servicesSection = (ServicesSection)configuration.GetSection(ConfigurationStrings.ServicesSectionPath);
|
|
ServiceElement serviceElement = configLoader.LookupService(this.Description.ConfigurationName, servicesSection);
|
|
configLoader.LoadServiceDescription(this, this.Description, serviceElement, this.LoadConfigurationSectionHelper, skipHost: true);
|
|
|
|
EnsureAuthenticationAuthorizationDebug(this.Description);
|
|
}
|
|
|
|
// Load only "host" section within "service" tag
|
|
[Fx.Tag.SecurityNote(Critical = "Calls LookupService which is critical.",
|
|
Safe = "Doesn't leak ServiceElement out of SecurityCritical code.")]
|
|
[SecuritySafeCritical]
|
|
void LoadHostConfigurationInternal(ConfigLoader configLoader, ServiceDescription description, string configurationName)
|
|
{
|
|
ServiceElement serviceSection = configLoader.LookupService(configurationName);
|
|
if (serviceSection != null)
|
|
{
|
|
configLoader.LoadHostConfig(serviceSection, this, (addr => this.InternalBaseAddresses.Add(addr)));
|
|
}
|
|
}
|
|
|
|
// Load service description for service from config, but skip "host" section within "service" tag
|
|
[Fx.Tag.SecurityNote(Critical = "Calls LookupService which is critical.",
|
|
Safe = "Doesn't leak ServiceElement out of SecurityCritical code.")]
|
|
[SecuritySafeCritical]
|
|
void LoadConfigurationSectionExceptHostInternal(ConfigLoader configLoader, ServiceDescription description, string configurationName)
|
|
{
|
|
ServiceElement serviceSection = configLoader.LookupService(configurationName);
|
|
configLoader.LoadServiceDescription(this, description, serviceSection, this.LoadConfigurationSectionHelper, skipHost: true);
|
|
}
|
|
|
|
internal override string CloseActivityName
|
|
{
|
|
get { return SR.GetString(SR.ActivityCloseServiceHost, this.serviceType.FullName); }
|
|
}
|
|
|
|
internal override string OpenActivityName
|
|
{
|
|
get { return SR.GetString(SR.ActivityOpenServiceHost, this.serviceType.FullName); }
|
|
}
|
|
|
|
protected override ServiceDescription CreateDescription(out IDictionary<string, ContractDescription> implementedContracts)
|
|
{
|
|
if (this.serviceType == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxServiceHostCannotCreateDescriptionWithoutServiceType)));
|
|
}
|
|
|
|
ServiceDescription description;
|
|
if (this.SingletonInstance != null)
|
|
{
|
|
description = ServiceDescription.GetService(this.SingletonInstance);
|
|
}
|
|
else
|
|
{
|
|
description = ServiceDescription.GetService(this.serviceType);
|
|
}
|
|
ServiceBehaviorAttribute serviceBehavior = description.Behaviors.Find<ServiceBehaviorAttribute>();
|
|
object serviceInstanceUsedAsABehavior = serviceBehavior.GetWellKnownSingleton();
|
|
if (serviceInstanceUsedAsABehavior == null)
|
|
{
|
|
serviceInstanceUsedAsABehavior = serviceBehavior.GetHiddenSingleton();
|
|
this.disposableInstance = serviceInstanceUsedAsABehavior as IDisposable;
|
|
}
|
|
|
|
if ((typeof(IServiceBehavior).IsAssignableFrom(this.serviceType) || typeof(IContractBehavior).IsAssignableFrom(this.serviceType))
|
|
&& serviceInstanceUsedAsABehavior == null)
|
|
{
|
|
serviceInstanceUsedAsABehavior = ServiceDescription.CreateImplementation(this.serviceType);
|
|
this.disposableInstance = serviceInstanceUsedAsABehavior as IDisposable;
|
|
}
|
|
|
|
if (this.SingletonInstance == null)
|
|
{
|
|
if (serviceInstanceUsedAsABehavior is IServiceBehavior)
|
|
{
|
|
description.Behaviors.Add((IServiceBehavior)serviceInstanceUsedAsABehavior);
|
|
}
|
|
}
|
|
|
|
ReflectedContractCollection reflectedContracts = new ReflectedContractCollection();
|
|
List<Type> interfaces = ServiceReflector.GetInterfaces(this.serviceType);
|
|
for (int i = 0; i < interfaces.Count; i++)
|
|
{
|
|
Type contractType = interfaces[i];
|
|
if (!reflectedContracts.Contains(contractType))
|
|
{
|
|
ContractDescription contract = null;
|
|
if (serviceInstanceUsedAsABehavior != null)
|
|
{
|
|
contract = ContractDescription.GetContract(contractType, serviceInstanceUsedAsABehavior);
|
|
}
|
|
else
|
|
{
|
|
contract = ContractDescription.GetContract(contractType, this.serviceType);
|
|
}
|
|
|
|
reflectedContracts.Add(contract);
|
|
Collection<ContractDescription> inheritedContracts = contract.GetInheritedContracts();
|
|
for (int j = 0; j < inheritedContracts.Count; j++)
|
|
{
|
|
ContractDescription inheritedContract = inheritedContracts[j];
|
|
if (!reflectedContracts.Contains(inheritedContract.ContractType))
|
|
{
|
|
reflectedContracts.Add(inheritedContract);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
this.reflectedContracts = reflectedContracts;
|
|
|
|
implementedContracts = reflectedContracts.ToImplementedContracts();
|
|
return description;
|
|
}
|
|
|
|
protected void InitializeDescription(object singletonInstance, UriSchemeKeyedCollection baseAddresses)
|
|
{
|
|
if (singletonInstance == null)
|
|
{
|
|
throw new ArgumentNullException("singletonInstance");
|
|
}
|
|
|
|
this.singletonInstance = singletonInstance;
|
|
InitializeDescription(singletonInstance.GetType(), baseAddresses);
|
|
}
|
|
|
|
protected void InitializeDescription(Type serviceType, UriSchemeKeyedCollection baseAddresses)
|
|
{
|
|
if (serviceType == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("serviceType"));
|
|
}
|
|
|
|
this.serviceType = serviceType;
|
|
|
|
base.InitializeDescription(baseAddresses);
|
|
}
|
|
|
|
protected override void OnClosed()
|
|
{
|
|
base.OnClosed();
|
|
if (this.disposableInstance != null)
|
|
{
|
|
this.disposableInstance.Dispose();
|
|
}
|
|
}
|
|
|
|
class ReflectedContractCollection : KeyedCollection<Type, ContractDescription>
|
|
{
|
|
public ReflectedContractCollection()
|
|
: base(null, 4)
|
|
{
|
|
}
|
|
|
|
protected override Type GetKeyForItem(ContractDescription item)
|
|
{
|
|
if (item == null)
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("item");
|
|
|
|
return item.ContractType;
|
|
}
|
|
|
|
public IDictionary<string, ContractDescription> ToImplementedContracts()
|
|
{
|
|
Dictionary<string, ContractDescription> implementedContracts = new Dictionary<string, ContractDescription>();
|
|
foreach (ContractDescription contract in this.Items)
|
|
{
|
|
implementedContracts.Add(GetConfigKey(contract), contract);
|
|
}
|
|
return implementedContracts;
|
|
}
|
|
|
|
internal static string GetConfigKey(ContractDescription contract)
|
|
{
|
|
return contract.ConfigurationName;
|
|
}
|
|
}
|
|
|
|
class ReflectedAndBehaviorContractCollection
|
|
{
|
|
ReflectedContractCollection reflectedContracts;
|
|
KeyedByTypeCollection<IServiceBehavior> behaviors;
|
|
public ReflectedAndBehaviorContractCollection(ReflectedContractCollection reflectedContracts, KeyedByTypeCollection<IServiceBehavior> behaviors)
|
|
{
|
|
this.reflectedContracts = reflectedContracts;
|
|
this.behaviors = behaviors;
|
|
}
|
|
|
|
internal bool Contains(Type implementedContract)
|
|
{
|
|
if (this.reflectedContracts.Contains(implementedContract))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (this.behaviors.Contains(typeof(ServiceMetadataBehavior)) && ServiceMetadataBehavior.IsMetadataImplementedType(implementedContract))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
internal string GetConfigKey(Type implementedContract)
|
|
{
|
|
if (this.reflectedContracts.Contains(implementedContract))
|
|
{
|
|
return ReflectedContractCollection.GetConfigKey(reflectedContracts[implementedContract]);
|
|
}
|
|
|
|
if (this.behaviors.Contains(typeof(ServiceMetadataBehavior)) && ServiceMetadataBehavior.IsMetadataImplementedType(implementedContract))
|
|
{
|
|
return ServiceMetadataBehavior.MexContractName;
|
|
}
|
|
|
|
Fx.Assert("Calls to GetConfigKey are preceeded by calls to Contains.");
|
|
#pragma warning suppress 56506 // implementedContract is never null at this point
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SfxReflectedContractKeyNotFound2, implementedContract.FullName, string.Empty)));
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|