如何将用户名/密码凭据从 php 客户端传递到自托管 wcf 服务?
-
15-11-2019 - |
题
我有一个自托管 wcf 服务,只需添加 2 个数字并返回值。它工作正常,但我不确定如何通过 php 客户端发送用户名和密码,因此它将根据我的 CustomUserNamePasswordValidator 进行验证。下面是 Add 方法的实现:
public class MathService : IMathService
{
public double Add(double x, double y)
{
return x + y;
}
}
这是我当前的 App.Config:
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<system.serviceModel>
<services>
<service behaviorConfiguration="MyServiceBehavior" name="WcfWithPhp.MathService">
<endpoint address="" binding="basicHttpBinding" contract="WcfWithPhp.IMathService">
<identity>
<dns value="localhost" />
</identity>
</endpoint>
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
<host>
<baseAddresses>
<add baseAddress="http://localhost:8731/MathService" />
</baseAddresses>
</host>
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior name="MyServiceBehavior">
<serviceMetadata httpGetEnabled="True"/>
<serviceDebug includeExceptionDetailInFaults="False" />
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
</configuration>
我像这样启动服务:
static void Main(string[] args)
{
ServiceHost host = new ServiceHost(typeof(WcfWithPhp.MathService));
host.Open();
Console.WriteLine("Math Service Host");
Console.WriteLine("Service Started!");
foreach (Uri address in host.BaseAddresses)
{
Console.WriteLine("Listening on " + address);
}
Console.WriteLine("Press any key to close the host...");
Console.ReadLine();
host.Close();
}
对于 php 客户端,我正在做:
<?php
header('Content-Type: text/plain');
echo "WCF Test\r\n\r\n";
// Create a new soap client based on the service's metadata (WSDL)
$client = new SoapClient("http://localhost:8731/MathService?wsdl");
$obj->x = 2.5;
$obj->y = 3.5;
$retval = $client->Add($obj);
echo "2.5 + 3.5 = " . $retval->AddResult;
?>
上面的代码无需身份验证即可正常工作,但我希望能够从 phpclient 验证用户名和密码。当他们尝试访问我的服务时,我希望通过 UserNamePasswordValidator 的重写 Validate 方法来验证用户名和密码,该方法当前定义为:
public override void Validate(string userName, string password)
{
if (string.IsNullOrEmpty(userName))
throw new ArgumentNullException("userName");
if (string.IsNullOrEmpty(password))
throw new ArgumentNullException("password");
// check if the user is not test
if (userName != "test" || password != "test")
throw new FaultException("Username and Password Failed");
}
我只是使用 test 和 test 作为用户名和密码的示例。我知道我必须设置修改行为配置并进行绑定配置,因此该服务将使用 CustomUserNamePasswordValidator,但由于我不了解 PHP,所以我不确定如何将凭据从 php 发送到 wcf 服务,发送凭据后,我不知道如何在 wcf 服务中设置它。我没有创建 wcf 服务。我以为我能做到 client.ClientCredentials.UserName.UserName
和 client.ClientCredentials.UserName.Password
, ,但这仅适用于我正在创建 .NET 客户端(而我不是)的情况。
我的另一个问题是,如果客户端是 php 客户端,我是否只能使用 basicHttpBinding?
另外,理想情况下我想做的是将肥皂请求从 php 客户端发送到 wcf 服务,所以如果有人能为我指出正确的方向,那就太好了。
我刚刚尝试了以下操作,但它不起作用(添加被调用,但未经过身份验证)
$sh_param = array('userName' => 'test', 'passWord' => 'test2');
$headers = new SoapHeader('http://localhost:8731/MathService.svc','UserCredentials',
$sh_param,false);
$client->__setSoapHeaders(array($headers));
更新:我的 PHP Soap 客户端初始化现在是:
$client = new SoapClient('http://localhost:8731/MathService?wsdl',
array('login' => "test2",
'password' => "test",
'trace'=>1));
通过执行上述操作,它在请求中添加了以下内容:
`Authorization: Basic dGVzdDI6dGVzdA==`
但是,我的 wcf 服务托管在控制台应用程序中,没有获取此授权,我有一个自定义用户名验证器,它具有硬编码的用户名 test 值和密码 test 值,但是当我尝试“test2 " 对于登录,仍然是调用该方法。我正在使用 TransportWithCredentialOnly 和 Message="UserName"
解决方案
尝试 SoapClient 构造函数重载:
$client = new SoapClient("some.wsdl", array('login' => "some_name",
'password' => "some_password"));
这是文档: http://www.php.net/manual/pl/soapclient.soapclient.php