Неверный запрос HTTP 400 при вызове службы WCF?

#.net #wcf #bad-request

#.net #wcf #неверный запрос

Вопрос:

Когда я вызываю http://localhost/TestService.svc/GetColors я получаю неверный запрос Http (400). Когда я запускаю это в fiddler, я также получаю неверный запрос. Однако, когда я вызываю службу через тестовый клиент wcf, она работает нормально. Что могло вызвать это?

Контракт на обслуживание:

 [ServiceContract]
public interface ITestService
{
    [OperationContract]
    string GetColors();
}
  

Реализация ITestService:

 [ServiceBehavior(InstanceContextMode=InstanceContextMode.Single)]
public class TestService : ITestService
{
    public List<Color> GetColors()
    {
        List<Color> colors= new List<Color>();

        colors.Add(new Color { Name = "Red", Code = "123" });
        colors.Add(new Color { Name = "Blue", Code = "323" });
        colors.Add(new Color { Name = "Green", Code = "3394" });

       return colors;
    }
}
  

Вот мой web.config:

 <?xml version="1.0"?>
<configuration>
   <system.web>
       <compilation debug="true" targetFramework="4.0"/>
   </system.web>
   <system.serviceModel>
      <bindings>
         <wsHttpBinding>
            <binding name="CustomAuthentication">
               <security mode="Message">
                  <message clientCredentialType="UserName"/>
               </security>
            </binding>
         </wsHttpBinding>
      </bindings>
      <serviceHostingEnvironment multipleSiteBindingsEnabled="true"/>
      <services>
          <service name="TestService.TestService"
                behaviorConfiguration="CustomValidator" >
             <endpoint 
                 address="" 
                 binding="wsHttpBinding" 
                 bindingConfiguration="CustomAuthentication"
                 contract="TestService.ITestService">
                 <identity>
                    <dns value="localhost" />
                 </identity>
             </endpoint>
             <endpoint 
                 address="mex" 
                 binding="mexHttpBinding" 
                 contract="IMetadataExchange" />
             <host>
                 <baseAddresses>
                     <add baseAddress="http://localhost/TestService/" />
                 </baseAddresses>
             </host>
          </service>
      </services>
      <behaviors>
          <serviceBehaviors>
             <behavior name="CustomValidator">
                <serviceCredentials>
                   <userNameAuthentication userNamePasswordValidationMode="Custom"
                                customUserNamePasswordValidatorType="TestService.CustomUserNameValidator, TestService"/>
                   <serviceCertificate findValue="Test" storeLocation="LocalMachine" 
                         storeName="My" x509FindType="FindBySubjectName"/>
                </serviceCredentials>
                <serviceMetadata httpGetEnabled="True"/>
             </behavior>
             <behavior>
                <serviceMetadata httpGetEnabled="True"/>
                <serviceDebug includeExceptionDetailInFaults="False"/>
             </behavior>
         </serviceBehaviors>
      </behaviors>
   </system.serviceModel>
   <system.webServer>
       <modules runAllManagedModulesForAllRequests="true"/>
   </system.webServer>
</configuration>
  

Когда я вызываю его, я просто открываю любой браузер и ввожу URL http://localhost/TestService.svc/GetColors . Если я делаю это через среду разработки, я вижу неверный запрос Http 400. Если я делаю это через IIS, я просто вижу пустую страницу.

Вот InnerException:

 <ExceptionString>System.Xml.XmlException: The body of the message cannot be read  
because it is empty.</ExceptionString>
  

Еще один вопрос, касающийся моей пользовательской проверки:

Я реализую пользовательскую проверку с помощью метода Validate UserNamePasswordValidator, но когда я вызываю что-то вроде GetColors через клиент wcf, он не вызывает метод Validate. Единственный способ, которым я могу заставить его вызвать validate, — это вызвать Validate (user, pass) непосредственно в GetColors. Я думал, что он будет вызываться автоматически при каждом вызове службы, если вы правильно настроили его в web.config.

Комментарии:

1. @marc_s, я обновлю свой пост.

Ответ №1:

Похоже, что ваш контракт на обслуживание и реализация сервиса не совпадают….

Контракт на обслуживание определяет один, string который должен быть возвращен из GetColors() :

 [ServiceContract]
public interface ITestService
{
    [OperationContract]
    string GetColors();
}
  

Но реализация возвращает List<Color> вместо:

 [ServiceBehavior(InstanceContextMode=InstanceContextMode.Single)]
public class TestService : ITestService
{
    public List<Color> GetColors()
    {
  

Есть ли вероятность, что вы изменили реализацию своей службы и забыли обновить контракт на обслуживание??

Также: поскольку это служба SOAP с wsHttpBinding , вы не можете просто вызвать метод из браузера, перейдя по его адресу — вам нужно использовать тестовый клиент WCF (который работает, как вы говорите). Вы также не можете просто вызвать его из Fiddler — вам пришлось бы вручную создать всю оболочку SOAP с заголовком и телом и всем остальным — совсем непростая задача.

Что вы можете попробовать, так это перейти к http://localhost/TestService.svc?wsdl , чтобы проверить, получите ли вы обратно правильный WSDL для службы.

Комментарии:

1. Извините, marc_s, я просто забыл отредактировать его здесь. Мой фактический код — List<Color> GetColors(); в моем интерфейсе.

2. marc_s, спасибо за помощь на данный момент. Я получаю правильный wsdl. Есть ли у вас какие-либо идеи о проблеме с валидатором имени пользователя и пароля?

3. Используйте SoapUI для тестирования, он создает для вас пустые запросы, затем вы заполняете параметры и отправляете его

Ответ №2:

Причиной этого может быть множество причин. Я предлагаю вам сначала включить трассировку WCF, обычно это очень полезно, поскольку выдает вам на стороне сервера информацию о том, что происходит на самом деле. Вот ссылка, которая должна помочь: Как включить трассировку WCF

РЕДАКТИРОВАТЬ: Следует отметить одну маленькую деталь с WCF: в отличие от автоматически сгенерированного веб-сервиса ole ‘ ASMX, он не позволяет вам перейти к базовому адресу с помощью стандартного браузера. Вы можете перейти к адресу метаданных (mex), но не к корню. Это просто задумано (я полагаю, что метод HTTP не поддерживается).

Комментарии:

1. хорошо, я включил трассировку. Как мне теперь извлечь из этого что-то значимое?

2. Вам нужно использовать средство просмотра трассировки службы: msdn.microsoft.com/en-us/library/ms732023.aspx

3. Я нашел это, но, похоже, не вижу никаких проблем. Я не вижу никаких значков ошибок. Есть ли что-нибудь, на что я должен обратить внимание?

4. @xaisoft — Я обновил свой ответ. Я думаю, вы пытаетесь просмотреть базовый адрес. Вы просто не можете этого сделать.

5. Саймон, ты прав. Я пытаюсь перейти к базовому адресу GetColors. Значит ли это, что все на самом деле работает так, как должно работать, и эта ошибка на самом деле является тем, что ожидается, когда я пытаюсь вызвать операцию (GetColors) через URL браузера.