#.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 браузера.