在《C# 开发 BIMFACE 系列文章》中介绍了模型转换、模型对比接口。这2个功能接口比较特殊,发起请求后,逻辑处理是在BIMFACE云端进行的,通常需要5~10分钟。当逻辑处理完成后,BIMFACE通过回调机制通知对比结果。

  BIMFACE支持回调机制。在调用方发起模型转换、模型集成、模型对比、生成离线数据包等操作时,可以通过传入参数callback的方式来启用回调机制。 在BIMFACE处理完相应操作后,根据调用方传入的回调地址通知调用方相应操作的结果。

URL参数:

signature(签名):为了确保回调消息是由BIMFACE发出的,调用方在收到回调消息后,须验证签名。签名的计算方式:MD5(``appKey:appSecret:compareId:status:nonce''),如果调用方计算的签名与BIMFACE返回的签名一致,则证明该消息是安全可靠的。

应用收到回调后,须向BIMFace发送回执,回执消息:HTTP STATUS 200

Callbak示例:

* 调用方对文件1685236328506848发起了模型转换,并且传入的回调地址是:https://my.app.com/callback。

* BIMFACE在模型转换任务处理完成后,会发送一个get请求到调用方的callback地址:
https://my.app.com/callback?fileId=1685236328506848&status=success&thumbnail=38044a282f55cb26e3704643dccd2b55/thumbnail/96.png,38044a282f55cb26e3704643dccd2b55/thumbnail/256.png&reason=&signature=99a6fccb1894dfdb4cce48fd5ec58110&nonce=123abc * 调用方接收到这条请求后,可以进行signature的验证,并发送回执消息。

特别说明

  BIMFACE的回调机制与微信公众号或者小程序开发类似,需要开发者提供开发者服务器,且有正式合法域名或者外网IP,对外公布一个地址,BIMFACE服务器能访问到该地址才可以。

  如果无法提供有效的回调地址,则只能通过手动调用 模型转换、模型集成、模型对比、生成离线数据包等操作的其他API来获取对应的处理结果。

  在.NET平台下实现该功能可以使用 WebService、一般处理程序、WebAPI等技术方式实现。下面介绍在一般处理程序中实现的思路与步骤。

 1、配置BIMACE开发者账号信息。

  在web.config 或者 app.config 文件中配置开发者账号信息,供验证消息签名时使用。

 2、获取BIMFace服务器发送的回调请求参数。

 long fileId = context.Request.QueryString["fileId"].ToLong();  // 文件ID
string status = context.Request.QueryString["status"]; // 转换的结果
string reason = context.Request.QueryString["reason"]; // 若转换失败,则返回失败原因
string thumbnail = context.Request.QueryString["thumbnail"]; // 缩略图地址
string nonce = context.Request.QueryString["nonce"]; // 回调随机数
string signature = context.Request.QueryString["signature"]; // BIMFACE的加密签名

 3、根据请求参数计算签名。

  签名的计算方式:MD5(``appKey:appSecret:compareId:status:nonce'')

 /// <summary>
/// 根据回调的参数计算签名
/// </summary>
/// <param name="appKey">开发者秘钥</param>
/// <param name="appSecret">开发者密码</param>
/// <param name="fileId">BIMFace发出的回调信息:文件ID</param>
/// <param name="status">BIMFace发出的回调信息:转换的结果</param>
/// <param name="nonce">BIMFace发出的回调信息:回调随机数</param>
/// <returns></returns>
public static string GetCallbackSignature(string appKey, string appSecret, long fileId, string status, string nonce)
{
return string.Format("{0}:{1}:{2}:{3}:{4}", appKey, appSecret, fileId, status, nonce).EncryptByMD5();
}

其中使用到的扩展方法 EncryptByMD5() 实现如下:

 /// <summary>
/// 自定义扩展方法:使用 MD5(不可逆加密) 算法加密字符串。返回二进制形式的字符串。字符串的编码方式为UTF8。
/// </summary>
/// <param name="this">扩展对象。字符串。编码方式为UTF8</param>
/// <param name="caseType">字符串大小写。默认小写</param>
/// <returns></returns>
public static string EncryptByMD5(this string @this, CaseType caseType = CaseType.Lower)
{
using (MD5 md5 = MD5.Create())
{
var sb = new StringBuilder();
byte[] hashBytes = md5.ComputeHash(Encoding.UTF8.GetBytes(@this));
foreach (byte bytes in hashBytes)
{
sb.Append(bytes.ToString("X2"));//X2 表示二进制
} return caseType == CaseType.Upper ? sb.ToString() : sb.ToString().ToLower();
}
}

 4、验证签名。

  将步骤3中的计算结果与BIMFace发出的回调消息签名做对比,如果签名一直则证明该消息是安全可靠的。

 /// <summary>
/// 验证BIMFace发出的回调消息签名信息是否安全可靠
/// </summary>
/// <param name="appKey">开发者秘钥</param>
/// <param name="appSecret">开发者密码</param>
/// <param name="fileId">BIMFace发出的回调信息:文件ID</param>
/// <param name="status">BIMFace发出的回调信息:转换的结果</param>
/// <param name="nonce">BIMFace发出的回调信息:回调随机数</param>
/// <param name="signature">BIMFace发出的回调信息:签名</param>
/// <param name="custCalcSignature">输出参数:根据BIMFACE平台的加密规则计算出来的签名信息</param>
/// <returns></returns>
public static bool CheckCallbackSignature(string appKey, string appSecret, long fileId, string status, string nonce, string signature, out string custCalcSignature)
{
/* signature(签名):为了确保回调消息是由BIMFace发出的,应用在收到回调消息后,须验证签名。
* 签名的计算方式:MD5("appKey:appSecret:fileId:status:nonce"),如果应用计算的签名与BIMFace返回的签名一致,则证明该消息是安全可靠的。
*/
custCalcSignature = GetCallbackSignature(appKey, appSecret, fileId, status, nonce); return custCalcSignature == signature;
}

 5、根据签名验证结果做出回执响应消息。

如果验证签名成功则可以将模型转换、模型集成、模型对比、生成离线数据包等操作的处理结果写入数据库保存供后续其他业务逻辑使用。

   签名成功后,须向BIMFace发送回执,回执消息:HTTP STATUS 200。

 bool checkSignature = CallbackUtils.CheckCallbackSignature(appKey, appSecret, fileId, status, nonce, signature, out custCalcSignature);
if (checkSignature)
{
tip = "[BIMFace发出的回调信息签名验证成功!]"
+ Environment.NewLine
+ callbackResponse;
LogUtility.Info(tip); //Todo 此处可以根据fileId把相关的信息写入数据库中 // 回执消息:应用收到回调后,须向BIMFace发送回执,回执消息:HTTP STATUS 200
context.Response.Write("HTTP STATUS 200");
}
else
{
tip = "[BIMFace发出的回调信息签名验证不通过!]"
+ Environment.NewLine
+ callbackResponse
+ Environment.NewLine
+ "自定义计算签名 custCalcSignature:" + custCalcSignature; LogUtility.Error(tip); context.Response.Write(tip);
}

如果签名验证失败,则需要将签名信息写入文本日志供分析原因使用。此时通过编码方式实现邮件、短信、微信消息等方式通知开发者回调程序处理结果不正确,使其及时知道业务系统的运行状况。

6、发布程序并使用该回调地址。

 程序完成后发布到开发者服务器。在模型转换、模型集成、模型对比、生成离线数据包等操作的API接口参数中使用该回调地址。

完整的代码如下:

 /// <summary>
/// BimFace回调处理
/// </summary>
public class BimFaceHandler : IHttpHandler
{
public void ProcessRequest(HttpContext context)
{
context.Response.ContentType = "text/plain";
context.Response.ContentEncoding = Encoding.UTF8; string appKey = ConfigUtility.GetAppSettingValue("BIMFACE_AppKey");
string appSecret = ConfigUtility.GetAppSettingValue("BIMFACE_AppSecret");
string uid = context.Request.QueryString["uid"]; // SparkBimFace #region 校验
if (appKey.IsNullOrWhiteSpace())
{
LogUtility.Error("BIMFace appKey 配置项没有配置!"); return;
} if (appSecret.IsNullOrWhiteSpace())
{
LogUtility.Error("BIMFace appSecret 配置项没有配置!"); return;
} if (uid.IsNullOrWhiteSpace())
{
LogUtility.Error("[非法请求]回调地址Url链接中的参数 uid 没有配置或者配置的值为空!"); return;
}
#endregion long fileId = context.Request.QueryString["fileId"].ToLong(); // 文件ID
string status = context.Request.QueryString["status"]; // 转换的结果
string reason = context.Request.QueryString["reason"]; // 若转换失败,则返回失败原因
string thumbnail = context.Request.QueryString["thumbnail"]; // 缩略图地址
string nonce = context.Request.QueryString["nonce"]; // 回调随机数
string signature = context.Request.QueryString["signature"]; // BIMFACE的加密签名 string callbackResponse = string.Format("fileId:{0},\r\nstatus:{1},\r\nreason:{2},\r\nthumbnail:{3},\r\nnonce:{4},\r\nsignature:{5}",
fileId, status, reason, thumbnail, nonce, signature);
string tip;
string custCalcSignature; bool checkSignature = CallbackUtils.CheckCallbackSignature(appKey, appSecret, fileId, status, nonce, signature, out custCalcSignature);
if (checkSignature)
{
tip = "[BIMFace发出的回调信息签名验证成功!]"
+ Environment.NewLine
+ callbackResponse;
LogUtility.Info(tip); //Todo 此处可以根据fileId把相关的信息写入数据库中 // 回执消息:应用收到回调后,须向BIMFace发送回执,回执消息:HTTP STATUS 200
context.Response.Write("HTTP STATUS 200");
}
else
{
tip = "[BIMFace发出的回调信息签名验证不通过!]"
+ Environment.NewLine
+ callbackResponse
+ Environment.NewLine
+ "自定义计算签名 custCalcSignature:" + custCalcSignature; LogUtility.Error(tip); context.Response.Write(tip);
} context.Response.End();
} /// <summary>
/// 该属性获得一个布尔值,指示另一个请求是否可以使用该HTTP处理程序的实例。
/// <para>如果设置为true,能提高性能,但要注意线程之间安全性问题。如果设置为false,则线程是安全的</para>
/// </summary>
public bool IsReusable
{
get
{
return false;
}
}
}

C#开发BIMFACE系列36 服务端API之:回调机制的更多相关文章

  1. C#开发BIMFACE系列40 服务端API之模型集成

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 随着建筑信息化模型技术的发展,越来越多的人选择在云端浏览建筑模型.现阶段的云端模型浏览大多是基于文件级别,一次只可以浏览一 ...

  2. C#开发BIMFACE系列41 服务端API之模型对比

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 在建筑施工图审查系统中,设计单位提交设计完成的模型/图纸,审查专家审查模型/图纸.审查过程中如果发现不符合规范的地方,则流 ...

  3. C#开发BIMFACE系列42 服务端API之图纸对比

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 在我的前一篇博客<C#开发BIMFACE系列42 服务端API之图纸对比>中详细介绍了BIMFACE服务端接口 ...

  4. C#开发BIMFACE系列43 服务端API之图纸拆分

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 在上一篇博客<C#开发BIMFACE系列42 服务端API之图纸对比>的最后留了一个问题,在常规业务场景下,一 ...

  5. C#开发BIMFACE系列44 服务端API之计算图纸对比差异项来源自哪个图框

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 在前两篇博客<C#开发BIMFACE系列42 服务端API之图纸对比>.<C#开发BIMFACE系列43 ...

  6. C#开发BIMFACE系列46 服务端API之离线数据包下载及结构详解

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 在前一篇博客<C#开发BIMFACE系列45 服务端API之创建离线数据包>中通过调用接口成功的创建一个离线数 ...

  7. C#开发BIMFACE系列6 服务端API之获取文件信息

    在<C#开发BIMFACE系列4 服务端API之源上传文件>.<C#开发BIMFACE系列5 服务端API之文件直传>两篇文章中详细介绍了如何将本地文件上传到BIMFACE服务 ...

  8. C#开发BIMFACE系列4 服务端API之源上传文件

    在注册成为BIMFACE的应用开发者后,要能在浏览器里浏览你的模型或者获取你模型内的BIM数据, 首先需要把你的模型文件上传到BIMFACE.根据不同场景,BIMFACE提供了丰富的文件相关的接口. ...

  9. C#开发BIMFACE系列3 服务端API之获取应用访问凭证AccessToken

    系列目录     [已更新最新开发文章,点击查看详细] BIMFACE 平台为开发者提供了大量的服务器端 API 与 JavaScript API,用于二次开发 BIM 的相关应用. BIMFACE ...

随机推荐

  1. Notes_STL_List_And_Map

    //Description: 使用STL遇到的问题 //Create Date: 2019-07-08 09:19:15 //Author: channy Notes_STL_List_And_Map ...

  2. 51)PHP,一个数据库操作类的代码

    <?php //类名,也习惯上(推荐)使用跟文件名相似的名字 //定义一个mysql连接类,该类可以连接mysql数据库 //并实现其单例模式 //该类的功能还能够完成如下基本mysql操作: ...

  3. linux文件系统与链接

    Linux的文件属性图1 图1   linux的文件属性 ls -lhi -l 长格式 -h 人性化 -i inodo -d 看目录自己的信息 inode 源自于文件系统 分区 平面设计图 格式化 施 ...

  4. TPO4-1 Deer Populations of the Puget Sound

    The causes of this population rebound are consequences of other human actions. First, the major pred ...

  5. java高并发之线程池

    Java高并发之线程池详解   线程池优势 在业务场景中, 如果一个对象创建销毁开销比较大, 那么此时建议池化对象进行管理. 例如线程, jdbc连接等等, 在高并发场景中, 如果可以复用之前销毁的对 ...

  6. ARM7探究

    1.流水线:三级流水线 预取.译码.执行.三级并行发生 2.什么是哈佛结构? 哈佛结构是一种存储器结构,是一种并行体系结构,它的主要特点是将程序和数据存储在不同的存储空间中,即程序存储器和数据存储器是 ...

  7. TZOJ-STL系列题

    C++实验:STL之vector #include <bits/stdc++.h> using namespace std; void Input(vector<int>&am ...

  8. 1、简述在java网络编程中,服务端程序与客户端程序的具体开发步骤?

    网络编程分为UDP通信和TCP通信 UDP协议: 发送端:1.创建DatagramSocket对象.2.创建DatagramPacket对象,并封装数据.3.发送数据.4.释放 资源. 接收端:1.创 ...

  9. Spring Boot Admin最佳实践

    本文不进行Spring Boot Admin入门知识点说明 在Spring Boot Actuator中提供很多像health.metrics等实时监控接口,可以方便我们随时跟踪服务的性能指标.Spr ...

  10. [LC] 557. Reverse Words in a String III

    Given a string, you need to reverse the order of characters in each word within a sentence while sti ...