Время ожидания wsDualHttpBinding всегда, в то время как другие службы работают как положено
В настоящее время я разрабатываю несколько сервисов (WCF) для работы с TFS 2010.
Некоторые из них используют инструмент подписки на события, в то время как другие используются через портал sharepoint, небольшие приложения wpf и т. Д.
Я работаю над приложением для администрирования некоторых вещей на другом сервере, таких как отправка запроса на сброс IIS на сервер, и я использую wsDualHttpBinding, чтобы я мог сообщать пользователю о прогрессе, как все происходит, через обратные вызовы.
Но я даже не могу вызвать метод в сервисе, потому что всякий раз, когда канал пытается открыть, я получаю исключения тайм-аута, независимо от того, какой у меня OpenTimeout.
Это мой конфиг (на стороне клиента)
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<system.net>
<connectionManagement>
<add maxconnection="200" address="*" />
</connectionManagement>
</system.net>
<system.serviceModel>
<bindings>
<wsDualHttpBinding>
<binding name="WSDualHttpBinding_IReset" closeTimeout="00:01:00"
openTimeout="00:02:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
bypassProxyOnLocal="false" transactionFlow="false" hostNameComparisonMode="StrongWildcard"
maxBufferPoolSize="524288" maxReceivedMessageSize="65536"
messageEncoding="Text" textEncoding="utf-8" useDefaultWebProxy="true">
<readerQuotas maxDepth="32" maxStringContentLength="8192" maxArrayLength="16384"
maxBytesPerRead="4096" maxNameTableCharCount="16384" />
<reliableSession ordered="true" inactivityTimeout="00:10:00" />
<security mode="None">
<message clientCredentialType="Windows" negotiateServiceCredential="true" />
</security>
</binding>
</wsDualHttpBinding>
<wsHttpBinding>
<binding name="WSHttpBinding_IDeploy" closeTimeout="00:01:00"
openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
bypassProxyOnLocal="false" transactionFlow="false" hostNameComparisonMode="StrongWildcard"
maxBufferPoolSize="1524288" maxReceivedMessageSize="1279748152"
messageEncoding="Text" textEncoding="utf-8" useDefaultWebProxy="true"
allowCookies="false">
<readerQuotas maxDepth="32" maxStringContentLength="81925" maxArrayLength="163848"
maxBytesPerRead="9192" maxNameTableCharCount="16384" />
<reliableSession ordered="true" inactivityTimeout="00:10:00"
enabled="false" />
<security mode="None">
<transport clientCredentialType="Windows" proxyCredentialType="None"
realm="" />
<message clientCredentialType="Windows" negotiateServiceCredential="true" />
</security>
</binding>
</wsHttpBinding>
</bindings>
<client>
<endpoint address="http://tfsserver:8080/TFSFacade/DeployFacade.svc"
binding="wsHttpBinding" bindingConfiguration="WSHttpBinding_IDeploy"
contract="DeployFacade.IDeploy" name="WSHttpBinding_IDeploy" />
<endpoint address="http://tfsserver:8080/DeployService/ResetService.svc"
binding="wsDualHttpBinding" bindingConfiguration="WSDualHttpBinding_IReset"
contract="ResetService.IReset" name="WSDualHttpBinding_IReset" />
</client>
</system.serviceModel>
<appSettings>
<add key="CopiaLocalRollback" value="true"/>
<add key="CopiaLocalPublish" value="true"/>
<add key="ModoDiagnostico" value="false" />
<add key="TempoTimeout" value="300000" />
</appSettings>
</configuration>
Серверный:
<?xml version="1.0"?>
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.0" />
</system.web>
<system.serviceModel>
<serviceHostingEnvironment multipleSiteBindingsEnabled="true" aspNetCompatibilityEnabled="true">
<baseAddressPrefixFilters>
<add prefix="http://tfsserver:8080"/>
<add prefix="http://tfsservices:8080"/>
<add prefix="http://tfsservices:1001"/>
</baseAddressPrefixFilters>
</serviceHostingEnvironment>
<bindings>
<wsDualHttpBinding>
<binding name="WSDualHttpBinding_ResetService" closeTimeout="00:01:00"
openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
bypassProxyOnLocal="false" transactionFlow="false" hostNameComparisonMode="StrongWildcard"
maxBufferPoolSize="524288" maxReceivedMessageSize="65536"
messageEncoding="Text" textEncoding="utf-8" useDefaultWebProxy="true">
<readerQuotas maxDepth="32" maxStringContentLength="8192" maxArrayLength="16384"
maxBytesPerRead="4096" maxNameTableCharCount="16384" />
<reliableSession ordered="true" inactivityTimeout="00:10:00" />
<security mode="None">
<message clientCredentialType="Windows" negotiateServiceCredential="true" />
</security>
</binding>
</wsDualHttpBinding>
</bindings>
<behaviors>
<serviceBehaviors>
<behavior name="EventServiceBehavior">
<serviceMetadata httpGetEnabled="true" />
<serviceDebug includeExceptionDetailInFaults="true" />
</behavior>
<behavior name="">
<serviceMetadata httpGetEnabled="true" />
<serviceDebug includeExceptionDetailInFaults="false" />
</behavior>
</serviceBehaviors>
</behaviors>
<services>
<service behaviorConfiguration="EventServiceBehavior" name="DeployService.ResetService">
<endpoint address="" binding="wsDualHttpBinding" bindingConfiguration="WSDualHttpBinding_ResetService"
contract="DeployService.Contracts.IReset" />
</service>
</services>
</system.serviceModel>
<system.webServer>
<modules runAllManagedModulesForAllRequests="true"/>
</system.webServer>
</configuration>
Контракты:
[ServiceContract(SessionMode = SessionMode.Required,
CallbackContract = typeof(IResetCallback))]
public interface IReset
{
[OperationContract]
void ExecutarIISReset();
}
[ServiceContract]
public interface IResetCallback
{
[OperationContract(IsOneWay = true)]
void PumpMessage(string message);
[OperationContract(IsOneWay = true)]
void PumpResponsiveMessage(ResponsiveMessage message);
[OperationContract(IsOneWay = true)]
void PumpLogDeployBEMessage(LogDeployBE message);
bool Confirmar();
ServidorModel[] GetServidores();
}
Служба (не публикует код, так как проблема возникает до ее достижения):
[AspNetCompatibilityRequirementsAttribute(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(ConcurrencyMode = ConcurrencyMode.Multiple)]
public class ResetService : IReset
{
//Stuff here
}
В Fiddler я получаю это на стороне клиента:
7 202 HTTP tfsserver:8080 /DeployService/ResetService.svc 0 private resetman.vshost:9000
Я получаю 202 несколько раз, потом проходит время и происходит тайм-аут
Запустив скрипку на сервере, я абсолютно ничего не получаю.
Что может происходить?
РЕДАКТИРОВАТЬ: странно, localhost работает. Нужны ли для wsDualHttpBinding специальные разрешения или что-то еще? Ничего не нашел в интернете
3 ответа
Так как вы используете wsDualHttpBinding
тогда и ваш сервер, и ваш клиент должны быть доступны на портах, указанных в вашей конфигурации. По этой причине он работает на вашем локальном компьютере, так как у вас нет проблем с портом. Тайм-аут соответствует блокировке брандмауэра или невозможности перенаправления сервером вашего клиента.
Если ваш клиент может быть легко доступен, добавив брандмауэр или правило переадресации портов, wsDualHttpBinding
может все еще быть подходящим - иначе использование wsDualHttpBinding, вероятно, не идеально. В этом случае вам может потребоваться опрос клиентов на наличие сообщений вместо получения их с помощью обратного вызова (т. Е. Клиент извлекает сообщения вместо того, чтобы отправлять их с помощью обратного вызова)
Пытаться
ConcurrencyMode = ConcurrencyMode.Reentrant
InstanceContextMode = InstanceContextMode.Single
- Удалить
SessionMode.Required
на вашем служебном поведении
Я застрял с той же проблемой. Я решил проблему, передав конфигурацию привязки в службу с помощью следующего кода в клиентском приложении. Это даст вам представление о том, как достичь своей цели. Это работало для меня в большинстве моих систем, но в некоторых системах оно все еще дает проблему тайм-аута. Возможная причина: невозможно привязать порт.
Service objService = new Service(objClass, "WSDualHttpBinding_ISendChatService");
WSDualHttpBinding binding = objService.Endpoint.Binding as WSDualHttpBinding;
int portNumber = FreeTcpPort();
string hostName = Dns.GetHostName();
binding.ClientBaseAddress = new Uri("http://"+Dns.GetHostByName(hostName).AddressList[0].ToString()+":" + portNumber + "/");
objService.Start(name);
static int FreeTcpPort()
{
TcpListener l = new TcpListener(IPAddress.Loopback, 0);
l.Start();
int port = ((IPEndPoint)l.LocalEndpoint).Port;
l.Stop();
return port;
}