AS2协议本身比较复杂,我们不需要了解其中太多细节,只需要知道一些重要概念就行了。

AS2是基于HTTP/HTTPS的,消息的格式使用MIME,就是邮件的格式,使用SHA1或SHA2加RSA进行签名,使用S/MIME进行加密。S/MIME加密本质就是3DES或者AES加RSA加密,有了之前的基础应该就很清楚了。

AS2的通信双方称为partner,每个partner有个partnerId,通信双方需要交互彼此的公钥。

自己实现S/MIME加密比较复杂,我们引入以下包来加密,jdk15on支持jdk1.5-jdk1.8,其他jdk版可去maven上搜索对应的版本

    <dependency><groupId>org.bouncycastle</groupId><artifactId>bcmail-jdk15on</artifactId><version>1.70</version></dependency>

接下来是实现AS2发送文件的JAVA版本

@Test
public void testAS2() throws Exception {//注册证书提供者BCSecurity.addProvider(new org.bouncycastle.jce.provider.BouncyCastleProvider());String password = "testas2";//生成mime消息体,放入待发送文件MimeBodyPart finalMessage = new MimeBodyPart();File file = new File("D:\\workspace\\data\\a.xml");finalMessage.setDataHandler(new DataHandler(new FileDataSource(file)));finalMessage.setHeader("Content-Type", "application/xml");finalMessage.setHeader("Content-Transfer-Encoding", "base64");finalMessage.setFileName(file.getName());//加载证书FileInputStream fis = new FileInputStream(new File("D:\\workspace\\OpenAs2App\\Server\\src\\config\\as2_certs.p12"));KeyStore keyStore = getKeyStore(fis, password, "PKCS12", "BC");PrivateKey privateKey = getPrivateKeyFromKeyStore(keyStore, "mycompany", "password");//签名X509Certificate signCert = getCertificate(keyStore, "mycompany");List certList = new ArrayList();certList.add(signCert);Store certs = new JcaCertStore(certList);SMIMESignedGenerator signer = new SMIMESignedGenerator();signer.setContentTransferEncoding("base64");//使用SHA1进行签名signer.addSignerInfoGenerator(new JcaSimpleSignerInfoGeneratorBuilder().setProvider("BC").build("SHA1WITHRSA", privateKey, signCert));signer.addCertificates(certs);MimeMultipart signedMimeMultipart = signer.generate(finalMessage);finalMessage = new MimeBodyPart();finalMessage.setContent(signedMimeMultipart);finalMessage.setHeader("Content-Type", signedMimeMultipart.getContentType());//加密// 加载partner的数字证书X509Certificate cert = getCertificate(keyStore, "partnera");// 创建加密器SMIMEEnvelopedGenerator encryptor = new SMIMEEnvelopedGenerator();encryptor.addRecipientInfoGenerator(new JceKeyTransRecipientInfoGenerator(cert).setProvider("BC"));encryptor.setContentTransferEncoding("base64");//3DES加密JceCMSContentEncryptorBuilder jceCMSContentEncryptorBuilder =new JceCMSContentEncryptorBuilder(new ASN1ObjectIdentifier(SMIMEEnvelopedGenerator.DES_EDE3_CBC)).setProvider("BC");jceCMSContentEncryptorBuilder.setSecureRandom(new SecureRandom());// 进行加密MimeBodyPart encryptedPart = encryptor.generate(finalMessage, jceCMSContentEncryptorBuilder.build());//准备头字段InternetHeaders ih = new InternetHeaders();ih.addHeader("Connection", "close, TE");ih.addHeader("Message-ID", UUID.randomUUID().toString());ih.addHeader("Mime-Version", "1.0");ih.addHeader("Content-Type", encryptedPart.getContentType());ih.addHeader("AS2-To", "PartnerA_OID");ih.addHeader("AS2-From", "MyCompany_OID");ih.addHeader("Subject", "Subject: File a.xml sent from MyCompany to PartnerA");ih.addHeader("Disposition-Notification-To", "edi@myCompany.com");ih.addHeader("Content-Disposition", "attachment");String url = "http://localhost:10080";execRequest("POST", url, ih.getAllHeaders(), null, encryptedPart.getInputStream());
}private static X509Certificate getCertificate(KeyStore keyStore,String alias) throws Exception {Certificate certificate = keyStore.getCertificate(alias);return (X509Certificate) certificate;
}

发送http请求的代码

public static void execRequest(String method, String url, Enumeration<Header> headers, NameValuePair[] params, InputStream inputStream) throws Exception {HttpClientBuilder httpBuilder = HttpClientBuilder.create();URL urlObj = new URL(url);/** httpClient is used for this request only,* set a connection manager that manages just one connection.*/if (urlObj.getProtocol().equalsIgnoreCase("https")) {/** Note: registration of a custom SSLSocketFactory via httpBuilder.setSSLSocketFactory is ignored when a connection manager is set.* The custom SSLSocketFactory needs to be registered together with the connection manager.*/SSLConnectionSocketFactory sslCsf = buildSslFactory();httpBuilder.setConnectionManager(new BasicHttpClientConnectionManager(RegistryBuilder.<ConnectionSocketFactory>create().register("http", PlainConnectionSocketFactory.getSocketFactory()).register("https", sslCsf).build()));} else {httpBuilder.setConnectionManager(new BasicHttpClientConnectionManager());}RequestBuilder rb = getRequestBuilder(method, urlObj, params, headers);if (inputStream != null) {InputStreamEntity ise = new InputStreamEntity(inputStream);rb.setEntity(ise);}final HttpUriRequest request = rb.build();try (CloseableHttpClient httpClient = httpBuilder.build()) {try (CloseableHttpResponse response = httpClient.execute(request)) {HttpEntity entity = response.getEntity();String result = EntityUtils.toString(entity);System.out.println(result);}}
}private static SSLConnectionSocketFactory buildSslFactory() throws Exception {boolean overrideSslChecks = true;SSLContext sslcontext;sslcontext = SSLContexts.createSystemDefault();// String [] protocols = Properties.getProperty(HTTP_PROP_SSL_PROTOCOLS,// "TLSv1").split("\\s*,\\s*");HostnameVerifier hnv = SSLConnectionSocketFactory.getDefaultHostnameVerifier();if (overrideSslChecks) {hnv = new HostnameVerifier() {@Overridepublic boolean verify(String hostname, SSLSession session) {return true;}};}SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslcontext, null, null, hnv);return sslsf;
}private static RequestBuilder getRequestBuilder(String method, URL urlObj, NameValuePair[] params, Enumeration<Header> headers) throws URISyntaxException {RequestBuilder req = null;if (method == null || method.equalsIgnoreCase(Method.GET)) {//default getreq = RequestBuilder.get();} else if (method.equalsIgnoreCase(Method.POST)) {req = RequestBuilder.post();} else if (method.equalsIgnoreCase(Method.HEAD)) {req = RequestBuilder.head();} else if (method.equalsIgnoreCase(Method.PUT)) {req = RequestBuilder.put();} else if (method.equalsIgnoreCase(Method.DELETE)) {req = RequestBuilder.delete();} else if (method.equalsIgnoreCase(Method.TRACE)) {req = RequestBuilder.trace();} else {throw new IllegalArgumentException("Illegal HTTP Method: " + method);}req.setUri(urlObj.toURI());if (params != null && params.length > 0) {req.addParameters(params);}if (headers != null) {while (headers.hasMoreElements()) {Header header = headers.nextElement();String headerValue = header.getValue();req.setHeader(header.getName(), headerValue);}}return req;
}public abstract static class Method {public static final String GET = "GET";public static final String HEAD = "HEAD";public static final String POST = "POST";public static final String PUT = "PUT";public static final String DELETE = "DELETE";public static final String TRACE = "TRACE";public static final String CONNECT = "CONNECT";
}

代码讲解

来看这段代码,这段代码是加载证书库,方法上篇给出了,类型是PKCS12,BC是什么鬼?BC是证书提供者,如果是JKS类型的证书,提供者是SUN,就不需要传了,如果是PKCS12类型的证书,就需要传provider。

KeyStore keyStore = getKeyStore(fis, password, "PKCS12", "BC");

生成mime消息体,放入待发送文件那段代码很简单,就是获取待发送文件的输入流然后生成mime消息体,没啥好讲的,加载证书库获取私钥的代码前篇都讲过了,也很简单。

接下来是签名和加密,都是固定写法,唯一有改动的可能是签名的算法是SHA1还是SHA2,加密算法是3DES还是AES,需要告知接收方。这边需要注意的是签名是使用自己的私钥,对方用我们的公钥进行验签,加密是使用对方的公钥,对方用自己的私钥进行解密。

接下去就是准备头字段,填写自己的partnerId和对方的partnerId就行了

其实有现成的代码,集成AS2不难,如果是自己去找协议去实现,难度就相当大了。

AS2协议详解(三)相关推荐

  1. MQTT协议详解 三、MQTT控制包(CONNECT)

    文章目录 系列文章目录 前言 CONNECT(客户端请求连接服务端) 一.固定包头(2字节) 二.可变包头(10字节) 协议名字(6字节) 协议等级(1字节) 连接标识(1字节) Clean Sess ...

  2. HTTPS协议详解(三):PKI 体系

    本文大部分内容摘自:http://www.wosign.com/faq/faq2016-0309-03.htm 尊重知识产权,转载注明Wosign -------------------------- ...

  3. NVMe协议详解(三)

    NVMe协议详解(三) 4.内存数据结构 4.1 SQ与CQ的详细定义 4.1.1 空队列 4.1.2 满队列 4.1.3 队列性质 4.2 仲裁机制 4.2.1 RR 带有优先权的RR 4.2.3 ...

  4. EtherCAT设备协议详解三、EtherCAT CoE

    CoE 是 CANopen on EtherCAT, 在ethercat报文中封装CANopen协议 服务数据对象(SDO)用来访问CANopen对象字典条目的 关于canopen怎么封装到ether ...

  5. AS2协议详解(一)

    在沃尔玛要求其供应商使用该协议之后,AS2(或Applicability Statement 2)快速流行起来.许多其他大型零售商也纷纷效仿,这意味着AS2迅速成为许多行业点对点连接的最流行EDI传输 ...

  6. AS2协议详解(二)

    上一篇中主要讲解了AS2协议加密的理论知识,这篇来上代码. 签名和验签 通过信息摘要算法和非对称加密,可以实现信息的防伪造,防篡改,通过我们的私钥来签名消息,接收方就能通过我们的公钥来校验该消息是否是 ...

  7. SSL/TLS协议详解(三)——证书颁发机构

    目录 证书颁发机构的需求 数字签名的定义 证书颁发机构的技术实现 如果攻击者篡改证书会怎样 信任链 数字签名的数学算法 浏览器如何实际验证给定服务器证书的有效性 TLS加密客户端-服务器通信并阻止中间 ...

  8. SRT协议详解三 传输参数

    4.1. 参数名称解析 这一节,我将逐个向大家介绍会影响SRT传输性能的参数名称,他们包括:Round Trip Time(RTT,往返延时).RTT Multiplier(RTT倍数).Packet ...

  9. MQTT协议详解 二、MQTT控制包格式

    文章目录 系列文章目录 前言 一.MQTT控制包格式 二.固定包头 控制包类型 控制包类型标识 剩余长度 三.可变包头 数据包标识 四.载荷 系列文章目录 MQTT协议详解 一.MQTT简介 MQTT ...

最新文章

  1. 【复盘】如何培养小朋友的编程能力?
  2. python3下载教程-Python3 教程
  3. LeetCode-剑指 Offer 10- II. 青蛙跳台阶问题
  4. EXCEL的下拉列表
  5. nyoj 304(区间dp)
  6. Log4Net ,.net和SQL Server的完美结合
  7. server长时间运行query,Ajax刷新被block
  8. C语言学习之用指针处理,输入a和 b两个整数,按先大后小的顺序输出a和 b
  9. 相对熵(relative entropy或 Kullback-Leibler divergence,KL距离)的java实现(三)
  10. GCPC2014 C Bounty Hunter
  11. Linux下使用源码安装出现的坑
  12. python 匿名函数添加判断_Python之内置函数、匿名函数
  13. 【旅行商问题】基于matlab免疫算法求解旅行商问题【含Matlab源码 195期】
  14. MethodArgumentNotValidException异常拦截提示信息太长?
  15. python爬虫系列第六次笔记之验证码以及代理的使用
  16. 一、恒生电子面试 (校招)
  17. 阿里云code的git
  18. 解决某APP游戏内购
  19. 使用Scratch制作项目《弹珠游戏》
  20. java调用摄像头保存到图库_Java调用摄像头并拍摄保存

热门文章

  1. 南京理工大学在校学生申请学校邮箱账户以及获得Intellij IDEA许可证的流程
  2. dp和px,那些不得不吐槽的故事——Android平台图片文字元素单位浅析 (转)
  3. mysql 表里restrict_数据库的三种状态RESTRICT、QUIESCE和SUSPEND(一)
  4. 默克公司扩大美国生命科学产能,合计投资4000万欧元
  5. 【酷熊科技】工作积累 ----------- Unity3D / c 语言 使用 Stringbuilder 引用头文件 using System.Text...
  6. 适合老人养殖的植物,你知道几个
  7. 简述BEM, OOCSS,AMCSS,SMACSS,SUITCSS,ITCSS
  8. 【单片机】2.6 时钟电路与时序
  9. 次氯酸盐溶液增稠剂对稠度的执著已成癖,但似乎并没有坏处
  10. 同一个世界,同一种和谐