在 .NET 环境下访问 SOAP 服务

SOAP 服务有着悠久的历史,目前仍然存在大量的 SOAP 服务,它是基于 HTTP 协议和 XML 技术的简单对象访问协议。

在 .NET Framework 时代,访问 SOAP 服务还是比较方便的,我们既可以使用 wsdl.exe 这个工具,也可以使用集成在 Visual Studio 的项目中的工具来生成服务代理,这个代理需要添加对 System.WebService 程序集的引用,它派生自 System.Web.Services.Protocols.SoapHttpClientProtocol,帮助我们访问 SOAP 服务。

在 .NET 时代,很多时候我们仍然需要访问现存的 SOAP 服务,使用的工具和所基于的库发生了一些变化。

首先,我们需要获取 SOAP 服务的定义 通常我们可以通过访问某个服务端点来获得它,对于大多数的 SOAP 服务来说,它会有一个 WSDL 的部分。例如,微软报表服务的管理端点就是 https://myserver/ReportServer/ReportService2010.asmx?wsdl ,也可以通过访问此类端点并保存获得的 WSDL 到一个文件中。

我们这里就使用这个 Reporting Service 2010 进行说明。

如果你希望使用交互工具的话,Visual Studio 中提供了集成的 使用 WCF Web Service Reference Provider 工具。你可以在项目上点击右键,选择 Add -> Connected Service,然后在 Service Reference (OpenAPI, gRPC, WCF Web Service) 的卡片上点击 + 来添加。后面的步骤与以前的工具就是一致的。

如果你希望自己完成,那请继续和我们一起前进。

第二步,安装用于 .NET 的 dotnet-svcutil NuGet 包

dotnet tool install --global dotnet-svcutil

直接执行它可以看到如下输出

> dotnet-svcutil
Microsoft (R) WCF Service Model Proxy Generation Tool for .Net Core platform
[Microsoft.Tools.ServiceModel.Svcutil, Version 2.1.0]
Copyright (c) Microsoft Corporation. All rights reserved. This tool collects information about how it is used in order to improve the tool. This functionality can be disabled by setting the environment variable "DOTNET_SVCUTIL_TELEMETRY_OPTOUT" to 1. Error: No valid input specified. Specify either a service url or a wsdl file. If you would like more help, type "dotnet-svcutil -h"

and

 minimum supported framework versions are as follows: netcoreapp1.0, netstandard1.3 and net4.5.

执行安装的 dotnet-svcutil 来生成客户端代理。

dotnet-svcutil https://<Server Name>/ReportServer/ReportService2010.asmx?wsdl

第三步,生成了什么

针对我们使用的 ReportingService201 来说,首先工具帮我们生成了访问该服务的接口 ReportingService2010Soap

public interface ReportingService2010Soap
{
// ...
}

实现该接口的类型 ReportingService2010SoapClient

public partial class ReportingService2010SoapClient : System.ServiceModel.ClientBase<ServiceReference.ReportingService2010Soap>, ServiceReference.ReportingService2010Soap
{
// ...
}

需要注意的是该类是一个分部类 partial 类。这意味着我们可以在自己的代码中定义另外一部分,以对生成类进行扩展。

在类定义的第一部分是一个特殊的分部方法 ConfigureEndpoint。方法的注释中这样说:

实现此分部方法以配置此服务端点。

在后面看到的生成的构造函数中,会调用此方法。所以,我们可以在自定义的分部类中,定义实际的此方法来配置该 Web 服务端点。

/// <summary>
/// Implement this partial method to configure the service endpoint.
/// </summary>
/// <param name="serviceEndpoint">The endpoint to configure</param>
/// <param name="clientCredentials">The client credentials</param>
static partial void ConfigureEndpoint(System.ServiceModel.Description.ServiceEndpoint serviceEndpoint, System.ServiceModel.Description.ClientCredentials clientCredentials);

同时,该类还提供了 4 个构造函数。其中前面的 3 个都会调用这个 ConfigureEndpoint() 方法来配置端点。但是,最后一个没有调用这个方法。

public ReportingService2010SoapClient(EndpointConfiguration endpointConfiguration) :
base(ReportingService2010SoapClient.GetBindingForEndpoint(endpointConfiguration), ReportingService2010SoapClient.GetEndpointAddress(endpointConfiguration))
{
this.Endpoint.Name = endpointConfiguration.ToString();
ConfigureEndpoint(this.Endpoint, this.ClientCredentials);
} public ReportingService2010SoapClient(EndpointConfiguration endpointConfiguration, string remoteAddress) :
base(ReportingService2010SoapClient.GetBindingForEndpoint(endpointConfiguration), new System.ServiceModel.EndpointAddress(remoteAddress))
{
this.Endpoint.Name = endpointConfiguration.ToString();
ConfigureEndpoint(this.Endpoint, this.ClientCredentials);
} public ReportingService2010SoapClient(EndpointConfiguration endpointConfiguration, System.ServiceModel.EndpointAddress remoteAddress) :
base(ReportingService2010SoapClient.GetBindingForEndpoint(endpointConfiguration), remoteAddress)
{
this.Endpoint.Name = endpointConfiguration.ToString();
ConfigureEndpoint(this.Endpoint, this.ClientCredentials);
} public ReportingService2010SoapClient(System.ServiceModel.Channels.Binding binding, System.ServiceModel.EndpointAddress remoteAddress) :
base(binding, remoteAddress)
{
}

在项目中,需要添加下面的 NuGet 包

<ItemGroup>
<PackageReference Include="System.ServiceModel.Http" Version="4.10.*" />
<PackageReference Include="System.ServiceModel.Security" Version="4.10.*" />
</ItemGroup>

第 4 步,使用生成的代理

我们通过使用第 4 个构造函数来访问这个服务。

Reporting Service 使用的是基本的 SOAP 服务,同时使用 Windows 验证进行保护。所以使用 BasicHttpBinding,并配置 Ntlm 认证协议。

BasicHttpBinding binding = new BasicHttpBinding();
binding.Security.Mode = BasicHttpSecurityMode.TransportCredentialOnly;
binding.Security.Transport.ClientCredentialType = HttpClientCredentialType.Ntlm; TimeSpan timeout = TimeSpan.FromMinutes(5);
binding.OpenTimeout = timeout;
binding.CloseTimeout = timeout;
binding.SendTimeout = timeout;
binding.ReceiveTimeout = timeout;

配置访问端点

String endpointurl = "http://myserver/ReportServer/reportservice2010.asmx";
EndpointAddress endpoint = new EndpointAddress(endpointurl); var service = new ReportingService2010SoapClient(binding, endpoint);

配置客户端的 Windows 认证账号

service.ClientCredentials.Windows.ClientCredential.UserName = "username";
service.ClientCredentials.Windows.ClientCredential.Password = "password";
service.ClientCredentials.Windows.ClientCredential.Domain = "domain";

好了,现在就可以访问服务了。

第 5 步,使用自定义的分部类来创建 Soap 的客户端

通过上面的介绍,可以看到创建一个客户端实例还是要做一些细致的准备的。我们可以通过自定义的分部类来封装这一部分,通过提供一个扩展出来的方法 GetSoapClientInstance() 来直接获得此实例。

public partial class ReportingService2010SoapClient
{
public static ReportingService2010SoapClient GetSoapClientInstance() {
String endpointurl = "http://apxse07.advent.com/ReportServer/reportservice2010.asmx"; BasicHttpBinding binding = new BasicHttpBinding();
binding.Security.Mode = BasicHttpSecurityMode.TransportCredentialOnly;
binding.Security.Transport.ClientCredentialType = HttpClientCredentialType.Ntlm; TimeSpan timeout = TimeSpan.FromMinutes(5);
binding.OpenTimeout = timeout;
binding.CloseTimeout = timeout;
binding.SendTimeout = timeout;
binding.ReceiveTimeout = timeout; EndpointAddress endpoint = new EndpointAddress(endpointurl); var service = new ReportingService2010SoapClient(binding, endpoint); service.ClientCredentials.Windows.ClientCredential.UserName = "Axyssu";
service.ClientCredentials.Windows.ClientCredential.Password = "June25!!";
service.ClientCredentials.Windows.ClientCredential.Domain = "Advent"; return service;
}
}

这样,我们可以使用下面的代码来得到一个客户端代理的实例:

var soapClient = ReportingService2010SoapClient.GetSoapClientInstance();

第 6 步,访问 Reporting Service 2010 获得服务器上的文件列表。

在 Reporting Service 上包含多种资源,Report 只是其中的一种。Reporting Service 的 SOAP 服务提供了通用的方法 FindItems(),可以定义一个通用的方法来支持获得资源的列表。

方法需要两个参数:一个是资源的类型,一个是目标文件夹。

在生成的代码中,已经包含了请求 SOAP 服务的 Request 定义和 Response 定义,我们需要使用它们来访问 SOAP 服务。

private string[] GetAllOfType(string typeName, string folder)
{
SearchCondition[] conditions = new SearchCondition[0];
Property[] options = new Property[0]; FindItemsRequest request = new FindItemsRequest();
request.Folder = folder;
request.BooleanOperator = BooleanOperatorEnum.And;
request.SearchOptions = options;
request.SearchConditions = conditions; FindItemsResponse response = this.FindItems(request);
CatalogItem[] items = response.Items; List<string> itemNames = new List<string>(); if (items != null)
foreach (CatalogItem ci in items)
if (ci.TypeName == typeName)
itemNames.Add(ci.Name); return itemNames.ToArray();
}

在创建了 GetAllOfType() 方法之后,我们可以编写获得服务器上部署的报表列表了。

public string[] GetAllReports(string folder)
{
return GetAllOfType("Report", folder);
}

参考资料

在 .NET 环境下访问 SOAP 服务的更多相关文章

  1. Windows环境下访问NFS(33篇Storage的文章)

    Windows环境下访问NFS 使用Solaris时,如果想在两台Solaris之间共享数据,那么你想到的最省事.最方便的方法肯定是nfs.但是现在的学生们的桌面,估计99%以上都是Windows,W ...

  2. 在Mac OS环境下安装MySQL服务

    在Mac OS环境下安装MySQL服务 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我之前介绍过window环境下安装mysql服务,以及在Linux环境下安装mysql服务,今 ...

  3. ubuntu 14.04LTS 环境下配置NFS服务

    简言之,NFS(Network FileSystem,网络文件系统)用于在不同机器,不同操作系统之间通过网络互相分享各自的文件.NFS设计之初就是为了在不同的系统间使用,所以它的通讯协议设计与主机及操 ...

  4. 【高并发】高并发环境下构建缓存服务需要注意哪些问题?我和阿里P9聊了很久!

    写在前面 周末,跟阿里的一个朋友(去年晋升为P9了)聊了很久,聊的内容几乎全是技术,当然了,两个技术男聊得最多的话题当然就是技术了.从基础到架构,从算法到AI,无所不谈.中间又穿插着不少天马行空的想象 ...

  5. 【swoole】如何在docker+swoole环境下测试udp服务

    前面几篇文章讲了使用 docker+swoole 环境来测试tcp服务以及如何测试,但是当我开始学习udp服务那块的时候,发现使用原来的方式在 docker+swoole 环境下行不通啦,后来发现如果 ...

  6. 项目部署到liunx环境下访问接口返回异常

    1.访问接口返回异常 已经连续踩了两次这个坑了.所以记下来了.方便下次搜索! 项目在window下运行正常,无任何异常! 但是部署到liunx环境下的服务器上就有问题 访问静态页面毫无问题,一旦涉及到 ...

  7. Python 访问soap服务

    使用库:subs soap服务信息: 网址:http://mobile.bjmemc.com.cn/AirService/Service.asmx 功能:使用其中的GetData服务获取北京各个地点的 ...

  8. LNMP环境下搭建SVN服务

    最近自己买了个服务器,试着在上面搭建了LNMP环境,因为以前在本地用MAMP Pro搭建过LAMP环境,所以基本上还算是轻车熟路,第一次搭建LNMP,使用的是一键安装,过程是顺利的,后来在使用过程中遇 ...

  9. Docker容器启动lnmp环境下的mysql服务时报"MySQL server PID file could not be found"错误解决办法

    我在自己的mac笔记本上装了一个docker,并在docker容器中安装了lnmp环境,经常会遇到在使用"lnmp restart"命令启动lnmp服务的时候,mysql服务启动失 ...

  10. Centos7 环境下开机 自启动服务(service) 设置的改变 (命令systemctl 和 chkconfig用法区别比较)

    参考文章:  <Linux 设置程序开机自启动 (命令systemctl 和 chkconfig用法区别比较)> http://blog.csdn.net/kenhins/article/ ...

随机推荐

  1. Vue Cli 创建项目在 GitHub 部署 history 路由模式

    1.修改打包路径 在 vue.config.js 中添加  publicPath  配置,其中 teambition-vue 是你项目的 github 名字.否则会找不到资源. module.expo ...

  2. 高通pmic voter机制

    前不久在高通 SDM450 平台接触了 voter 机制(投票机制).最近终于得空,结合一个问题简单研究了一下.现将研究流程简单记录一下,由于时间有限,所以是实用为目的,没有做详细的分析,不过结合着这 ...

  3. MySQL精品学习资源合集 | 含学习教程笔记、运维技巧、图书推荐

    MySQL凭借开源.免费.低门槛.稳定等优势,成为了当前最流行的关系型数据库之一.从1998年发行第一版以来,通过不断地更新迭代,MySQL被越来越多的公司使用,已然成为当下潮流. 为了帮助大家更好地 ...

  4. Gradio 5 稳定版正式发布

    在过去的几个月里,我们一直在努力工作,今天,我们想向大家展示成果:Gradio 5 稳定版现已发布. 有了 Gradio 5,开发者可以构建 生产级的机器学习 Web 应用,这些应用不仅性能优越.可扩 ...

  5. 什么是SPA页面,它的优缺点?

    什么是SPA spa是单页面应用(single page web application),浏览器会一开始就加载所需要的html.css.javascript.一旦页面加载完毕,整个页面就不会因为用户 ...

  6. 深入浅出 Kubernetes 项目网关与应用路由

    KubeSphere 项目网关与应用路由提供了一种聚合服务的方式,将集群的内部服务通过一个外部可访问的 IP 地址以 HTTP 或 HTTPs 暴露给集群外部.应用路由定义了这些服务的访问规则,用户可 ...

  7. WebUploader 文件上传,兼容ios和安卓

    var upImg = WebUploader.create({ auto: true, swf: 'webuploader-0.1.5/Uploader.swf', // 图片接收服务端. serv ...

  8. Dbeaver24.2.2安装和使用教程(免费的数据库管理工具)

    前言 DBeaver是免费和开源(GPL)为开发人员和数据库管理员通用数据库工具. DBeaver 通过 JDBC 连接到数据库,可以支持几乎所有的数据库产品,包括:MySQL.PostgreSQL. ...

  9. python翻译词典实例

    #!/usr/bin/python # -*- coding:utf-8 -*- #通过有道翻译来进行内容翻译 import urllib2 import urllib import json #-- ...

  10. c++设计模式:设计原则

    c++设计八大原则(降低改变带来的代码修改) 一.依赖倒置原则(DIP) 1.高层模块(稳定)不应该依赖于低层模块(变化),二者应该依赖于抽象(更稳定) <高层模块 包括 低层模块所依赖的抽象, ...