在 .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. thinkphp中的or,like,in等联合查询

    数据库中的两个字段有一个字段等于查询条件时,下面两个方法都是可以的: (1) $where['a.goods_name|a.goods_sn'] = array('like',"%$keyW ...

  2. USB协议详解第7讲(补充-USB帧和微帧剖析)

    1.概念 (1)USB2.0帧和微帧属于物理层时间基准的概念,低速和全速下每个帧时长为1ms,高速下每个帧又分为8个微帧,即每个微帧时长为125us. (2)USB主机和设备控制器同步后,每个微帧起始 ...

  3. python中的内置函数zip函数

    关于zip()函数,有几点要讲的. 首先,官方文档中,它是这样描述的: Make an iterator that aggregates elements from each of the itera ...

  4. 容器化部署nacos 1.4.6报错caused: The specified key byte array is 0 bits which is not secure enough for any JWT

    nacos2.0+ 与nacos 1.x区别 nacos在2.0+版本开始使用grpc与客户端通信,并且通过非8848端口通信 主要是有两个端口 端口 与主端口的偏移量 描述 9848 1000 客户 ...

  5. centos7.9 安装 nodejs(包含安装fnm、更换yum源、升级 gcc、make、glibc、libstdc++)

    1.安装fnm (1)压缩包fnm-linux.zip搞到服务器上,我放在root里. (2)解压.设置权限 unzip fnm-linux.zip chmod 777 fnm (3)设置环境变量,添 ...

  6. IKAnalyzer分词工具的spring boot项目整合代码版

    简介 IK Analyzer是什么呢,一个很轻量的中文分词工具,是基于java开发的轻量级的中文分词工具包.它是以开源项目Luence为主体的,结合词典分词和文法分析算法的中文分词组件.IK有很多版本 ...

  7. 微信H5分享外部链接,缩略图不显示

    可关注微信公众号酒酒酒酒查看原文: 前言:最近做了一款推广茶的APP软件,展厅.产品需要分享功能:从APP内分享到H5网页:微信内打开H5网页,点击微信内右上角三个点,可再次分享: 注意:大多数情况下 ...

  8. 狂神说-Docker基础-学习笔记-03 日志、元数据、进程的查看

    狂神说-Docker基础-学习笔记-03 日志.元数据.进程的查看 视频链接:https://www.bilibili.com/video/BV1og4y1q7M4?p=1 1.docker logs ...

  9. 基于Material Design风格开源、免费的WinForms UI控件库

    前言 今天大姚给大家分享一个基于 Google 的 Material Design 风格开源.免费的.NET WinForms UI控件库:MaterialSkin. WinForms介绍 WinFo ...

  10. ABC365(D,E)

    ABC365(D,E) D - AtCoder Janken 3 石头剪刀布,给出对手的出招,问在保证不败的情况下最多能赢多少回 记 \(f_i,{0/1/2}\) 表示第 \(i\) 局出石头/剪刀 ...