金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

本文为金蝶云星空自定义WebAPI开发的新手教程。内容为技术类文章,而且内容较长。做星空系统二次开发的新手朋友,可以收藏作为参考。非技术职业的朋友,可以直接划走,以免耽误您的宝贵时间。

文章末尾可以找到本教程源码仓库地址。

服务插件和表单插件的教程:

金蝶云星空插件实战开发-服务插件

金蝶云星空插件实战开发-新手保姆级教程-表单插件


阅读对象:云星空二次开发新手
需求场景:使用Postman模拟第三方系统更新采购订单,提交并审核采购订单
开发工具Visual Studio 2019
开发语言:C#
星空版本:7.6.0
插件类型:自定义WebAPI

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

说明:本文前提是开发机已经安装好金蝶云星空系统和金蝶BOS IDE。

金蝶云社区课程详情-课程详情-金蝶云社区官网

星空系统集成有以下三种方式:
1. WebAPI
2. WebService
3. ClientProxy(客户代理接口)

系统集成是个什么意思呢?

翻译一下就是星空系统与第三方系统进行数据对接。
WebAPI常用的API接口:
1. AuthService.ValidateUser(登录验证)
2. View(查看单据)
3. Save(保存单据)
4. BatchSave(批量保存)
5. Submit(提交单据)
6. Audit(审核单据)
7. UnAudit(反审核单据)
8. Delete(删除单据)
9. ExecuteBillQuery(单据数据查询)
10.
自定义WebAPI
11. Draft(暂存单据)
12. Allocate(分配表单数据)
13. Push(下推)
14. GroupSave(分组保存)
使用WebAPI通用的步骤是登录、构建和传递Model数据包。

WebAPI的工作原理通俗来说,就是在模拟录单的过程,根据传入的JSON数据包构造的每个字段值,按顺序逐一模拟录单填充对应的字段值,触发此字段的相关值更新事件、实体服务和插件逻辑。

构造完成整个单据的数据包之后,调用Save(保存)、Submit(提交)、Audit(审核),并对应触发保存、提交、审核的操作校验、操作逻辑、操作单据插件逻辑等整个过程。
既然录单的顺序已经规定了,那么录入顺序也需要按照系统字段列表属性顺序,从前往后,否则就存在一定的相互覆盖值的情况。

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

熟悉MVC结构的研发应该都清楚,.Net框架的基础结构是通过基础管道中构建HttpApplication和HttpContext,对Http.Sys接收到的请求,进行管道处理。到达HttpModule进行拦截过后,通过路由节点转到正确的Controller进行业务处理。

整个管道过程处理认证、授权等一系列问题,最后在星空的KDsvcHandle的HttpHandle拦截处理业务请求。

星空的自定义WebAPI利用的也是这个思路,继承KDBaseService,请求路径是xxx.common.kdsvc,到达KDSvcHandle后,通过反射执行我们的操作服务。

步骤:
1. 创建Visual C#类库
2. 添加星空系统类库的引用
3. 编写自定义WebAPI
4. 编译代码生成dll文件
5. 部署
6. 重启IIS服务
7. 使用Postman模拟请求

1.创建Visual C#类库

打开Visual Studio IDE,在启动窗口中选择“创建新项目”选项。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

在Visual Studio IDE的项目类型列表中找到“类库(.NET Framework)”选项。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

点击“下一步”按钮,配置项目信息。
【重要】项目名称是一个比较重要的配置项,金蝶官方在《二次开发规范》中有说明。

按照规范我们暂时将项目命名定为:
Test.K3Cloud.SCM.WebAPI.ServerExtend

【重要】框架选择 .NET Framework 4
具体配置如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

2.添加星空系统类库的引用

在Visual Studio IDE的解决方案资源管理器窗口中,选择“引用”,点击右键,呼出右键菜单,选择“添加引用”选项。

打开“应用管理器”窗口,选择“Kingdee.BOS.dll“和”Kingdee.BOS.Core.dll”。

如果窗口列表中没有此类库,可以点击窗口下方的“浏览”按钮,在星空系统的安装目录中找到此类库文件。

默认目录为:C:Program Files (x86)KingdeeK3CloudWebSiteBin

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

用同样的方式,我们再引入以下金蝶动态链接库文件:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

以上库文件都存在于C:Program Files (x86)KingdeeK3CloudWebSiteBin

3.编写自定义WebAPI

代码组织:
1. BusinessService.cs
2. Defineds.cs
3. Checker.cs
4. Formater.cs
5. BillHandler.cs

BusinessService.cs 是程序逻辑的入口,进行参数检查和单据方法调用;
Defineds.cs 声明和定义程序需要的常量、请求和返回参数的结构对象;
Checker.cs 参数检查;
Formater.cs 请求参数格式化和组织。将客户端请求的参数对WebAPI请求参数的要求和格式进行适配;
BillHandler.cs 单据的保存、提交和审核操作。

自定义WebAPI继承与AbstractWebApiBusinessService

下面的代码是最基础的代码,我们后面的开发都是在这几行代码的基础来展开。

我们首先将“Class1.cs”重命名为“BusinessService”。

当然我们也可以使用其他的名称。代码如下:

using System;using System.Collections.Generic;using System.Linq;using System.Text;using Kingdee.BOS.ServiceFacade.KDServiceFx;using Kingdee.BOS.WebApi.ServicesStub;namespace Test.K3Cloud.SCM.WebAPI.ServerExtend{ public class BusinessService : AbstractWebApiBusinessService { // 构造函数 public BusinessService(KDServiceContext context) : base(context) { } }}

接下来我们新建一个名为“Defineds.cs”的类文件,这个类用于声明程序中需要用到的常量和公用的struct。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

具体代码如下:

using System;using System.Collections.Generic;using System.Linq;using System.Text;using Newtonsoft.JSON.Linq;namespace Test.K3Cloud.SCM.WebAPI.ServerExtend{ public class Defineds { public const string DefaultSuccessMsg = "星空系统单据处理成功"; } /// <summary> /// API返回结果对象 /// </summary> public struct Result { public bool success; // 请求执行状态,成功:true;失败:false; public string msg; // 返回描述信息 public JObject data; // 请求返回的数据 } /// <summary> /// API请求参数的结构体 /// </summary> public struct Parameter { public string LocalFormId; // 上游单据ID public string EntryKey; // 下游单据明细单据单据ID public JObject Model; // 具体参数 } /// <summary> /// 保存接口请求参数 /// </summary> public struct RequestSave { public JArray NeedUpDateFields; // 需要更新的字段,数组类型,格式:[key1,key2,...] (非必录)注(更新单据体字段得加上单据体key public JArray NeedReturnFields; // 需返回结果的字段集合,数组类型,格式:[key,entitykey.key,...](非必录) 注(返回单据体字段格式:entitykey.key) public bool IsDeleteEntry; // 是否删除已存在的分录,布尔类型,默认true(非必录) public string SubSystemId; // 表单所在的子系统内码,字符串类型(非必录) public bool IsVerifyBaseDataField; // 是否验证所有的基础资料有效性,布尔类,默认false(非必录) public bool IsEntryBatchFill; // 是否批量填充分录,默认true(非必录) public bool ValidateFlag; // 是否验证标志,布尔类型,默认true(非必录) public bool NumberSearch; // 是否用编码搜索基础资料,布尔类型,默认true(非必录) public string InterationFlags; // 交互标志集合,字符串类型,分号分隔,格式:"flag1;flag2;..."(非必录) 例如(允许负库存标识:STK_InvCheckResult) public JObject Model; // 表单数据包,JSON类型(必录) } /// <summary> /// 提交接口请求参数 /// </summary> public struct RequestSubmit { public long CreateOrgId; // 创建者组织内码,字符串类型(非必录) public JArray Numbers; // 单据编码集合,数组类型,格式:[No1,No2,...](使用编码时必录) public string Ids; // 单据内码集合,字符串类型,格式:"Id1,Id2,..."(使用内码时必录) public long SelectedPostId; // 工作流发起员工岗位内码,整型(非必录) 注(员工身兼多岗时不传参默认取第一个岗位) public bool NetworkCtrl; // 是否启用网控,布尔类型,默认false(非必录) } /// <summary> /// 审核接口请求参数 /// </summary> public struct RequestAudit { public long CreateOrgId; // 创建者组织内码,字符串类型(非必录) public JArray Numbers; // 单据编码集合,数组类型,格式:[No1,No2,...](使用编码时必录) public string Ids; // 单据内码集合,字符串类型,格式:"Id1,Id2,..."(使用内码时必录) public string InterationFlags; // 交互标志集合,字符串类型,分号分隔,格式:"flag1;flag2;..."(非必录) 例如(允许负库存标识:STK_InvCheckResult) public bool NetworkCtrl; // 是否启用网控,布尔类型,默认false(非必录) } /// <summary> /// 接口返回数据 /// </summary> public struct ResponseData { public ResponseDataResult Result; } /// <summary> /// 返回数据结构 /// </summary> public struct ResponseDataResult { public ResponseDataResultStatus ResponseStatus; public long ID; // 单据内码 public string Number; // 单据编号 public JArray NeedReturnData; // 返回结果的字段集合 } /// <summary> /// 返回结果状态 /// </summary> public struct ResponseDataResultStatus { public bool IsSuccess; // 是否操作成功 public long ErrorCode; public List<ResponseDataResultErrors> Errors; // 失败错误信息 public List<ResponseDataResultSuccessEntitys> SuccessEntitys; // 操作成功的实体信息 public List<ResponseDataResultSuccessMessages> SuccessMessages; // 成功消息 public long MsgCode; // 消息代码 } /// <summary> /// 返回结果中错误信息结构 /// </summary> public struct ResponseDataResultErrors { public string FieldName; public string Message; public long DIndex; } /// <summary> /// 返回结果中成功的明细 /// </summary> public struct ResponseDataResultSuccessEntitys { public long Id; public string Number; public long DIndex; } /// <summary> /// 返回结果中成功的信息 /// </summary> public struct ResponseDataResultSuccessMessages { public string FieldName; public string Message; public long DIndex; } /// <summary> /// 返回结果,包含执行状态 /// </summary> public struct ResponseDataResultConvertStatus { public bool IsSuccess; // 是否操作成功 public long ErrorCode; public List<ResponseDataResultErrors> Errors; // 失败错误信息 public List<ResponseDataResultSuccessEntitys> SuccessEntitys; // 操作成功的实体信息 public List<ResponseDataResultSuccessMessages> SuccessMessages; // 成功消息 public long MsgCode; // 消息代码 }}

在场景需求中,我们要对“采购订单”进行更新、提交和审核的操作,所以上面的代码中,我们先定义了保存、提交和审核对应的WebAPI动作的参数结构。

同时我们也同时定义了客户端请求参数和处理结果的结构。

保存、提交和审核的WebAPI请求参数的字段,代码中都有注释,当然我们也可以在官方文档中查阅到。

在客户端请求参数的结构中“LocalFormId”是需要处理的单据的FormId。

“Model”是单据的实际数据。那么问题来了,这个FormId我们在哪里能找到呢?

获取单据的FormId有两种比较直接和方便的方式:
1. 在官方文档中查找
2. 在BOS中查看

本例中,我们需要操作的是“采购订单”,那么我们先从官方文档中找到FormId。

如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

在BOS中找到FormId,如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

我们可以看到,“采购订单”FormId的值是“PUR_PurchaseOrder”,这里我们先记下来,后面模拟客户端请求时,需要用到它。

EntryKey 是单据数据结构的明细的熟悉Key。

也可以在官方文档和BOS中找到。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

这里我们可以看到,明细属性的Key的值为“FPOOrderEntry”。

这里我们先记下来,后面模拟客户端请求时,需要用到它。

接下来,我们开始处理业务逻辑。

处理业务逻辑,首先要做的就是对客户端传递过来的参数进行检查校验。

我们用上面同样的方式,新建一个名为“Checker.cs”的类文件。

我们将客户端传递过来的参数的校验的方法,放到这个类文件中。代码如下:

using System;using System.Collections.Generic;using System.Linq;using System.Text;using Kingdee.BOS.Util;namespace Test.K3Cloud.SCM.WebAPI.ServerExtend{ public class Checker { /// <summary> /// 检查客户端传递过来的参数。 /// 客户端参数为JSON格式 /// </summary> /// <param name="parameters"></param> /// <returns></returns> public static Result clientParameterChecker(string parameters) { Result result = new Result(); if (parameters == null) { result.msg = "接口请求参数不能为空"; return result; } if (parameters.IsEmpty()) { result.msg = "接口请求参数不能为空"; return result; } result.success = true; return result; } /// <summary> /// 检查 Model /// </summary> /// <param name="parameter"></param> /// <returns></returns> public static Result checkModel(Parameter parameter) { Result result = new Result(); if (parameter.Model == null) { result.msg = "请求参数中固定参数 Model 不能为空"; return result; } result.success = true; return result; } /// <summary> /// 检查业务参数中的 FID 和 FBillNo /// </summary> /// <param name="parameter"></param> /// <returns></returns> public static Result checkFIDAndBillNo(Parameter parameter) { bool hasFID = false; bool hasFBillNo = false; if (parameter.Model.Property("FID") != null || Convert.ToInt64(parameter.Model["FID"]) > 0) { hasFID = true; } if (parameter.Model.Property("FBillNo") != null || Convert.ToString(parameter.Model["FBillNo"]).Length > 0) { hasFBillNo = true; } Result result = new Result(); if (!hasFID && !hasFBillNo) { result.msg = "单据内码(FID)或者单据编码(FBillNo)不能同时为空"; return result; } result.success = true; return result; } }}

本教程是入门教程,目的是为了帮助我们新手们了解自定义WebAPI的基本原理和流程。

因此,参数的校验没有做得很严格。

在实际的项目中需要对客户端传递的参数进行严格地校验,切记!切记!

接下来,我们编写WebAPI保存、提交和审核三个动作的API的请求参数的格式化方法。

用于后面调用系统标准WebAPI。

新建一个名为“Formater.cs”的类文件。代码如下:

using System;using System.Collections.Generic;using System.Linq;using System.Text;using Kingdee.BOS.Util;using Newtonsoft.Json;using Newtonsoft.Json.Linq;namespace Test.K3Cloud.SCM.WebAPI.ServerExtend{ public class Formater { /// <summary> /// 更新 /// </summary> /// <param name="parameter"></param> /// <returns></returns> public static string Update(Parameter parameter) { RequestSave param = new RequestSave(); param.NeedUpDateFields = new JArray(); param.NeedReturnFields = new JArray(); param.IsDeleteEntry = false; param.SubSystemId = ""; param.IsVerifyBaseDataField = false; param.IsEntryBatchFill = true; param.ValidateFlag = true; param.NumberSearch = true; param.Model = new JObject(); if ((parameter.Model.Property("FID") == null || parameter.Model["FID"] == null) && (parameter.Model.Property("FBillNo") == null || parameter.Model["FBillNo"] == null)) { return JsonConvert.SerializeObject(param); } if (parameter.Model.Property(parameter.EntryKey) == null || parameter.Model[parameter.EntryKey] == null) { return JsonConvert.SerializeObject(param); } param.NeedReturnFields = NeedReturnFields(); // 组织需要更新的字段 JArray needUpDateFields = new JArray(); JArray entryData = parameter.Model[parameter.EntryKey] as JArray; foreach (JProperty item in parameter.Model.Properties()) { if (item.Name != "FID" && item.Name != "FBillNo" && item.Name != parameter.EntryKey) { needUpDateFields.Add(item.Name); } } needUpDateFields.Add(parameter.EntryKey); JObject entryDataZero = entryData[0] as JObject; foreach (JProperty item in entryDataZero.Properties()) { if (item.Name != "FEntryID") { needUpDateFields.Add(item.Name); } } param.NeedUpDateFields = needUpDateFields; param.Model = parameter.Model; return JsonConvert.SerializeObject(param); } /// <summary> /// 提交 /// </summary> /// <param name="parameter"></param> /// <returns></returns> public static string Submit(Parameter parameter) { RequestSubmit param = new RequestSubmit(); param.CreateOrgId = 0; param.Numbers = new JArray(); param.Ids = ""; param.SelectedPostId = 0; param.NetworkCtrl = true; param.Numbers = formatNumbers(parameter); if (param.Numbers.Count() == 0) { param.Ids = formatFID(parameter); } return JsonConvert.SerializeObject(param); } /// <summary> /// 审核 /// </summary> /// <param name="parameter"></param> /// <returns></returns> public static string Audit(Parameter parameter) { RequestAudit param = new RequestAudit(); param.CreateOrgId = 0; param.Numbers = new JArray(); param.Ids = ""; param.InterationFlags = ""; param.NetworkCtrl = true; param.Numbers = formatNumbers(parameter); if (param.Numbers.Count() == 0) { param.Ids = formatFID(parameter); } return JsonConvert.SerializeObject(param); } /// <summary> /// 格式化单据编号 /// </summary> /// <param name="parameter"></param> /// <returns></returns> private static JArray formatNumbers(Parameter parameter) { JArray jNumbers = new JArray(); if (parameter.Model.Property("FBillNo") == null && parameter.Model["FBillNo"] == null) { return jNumbers; } string numbers = Convert.ToString(parameter.Model["FBillNo"]); if (numbers.IsEmpty()) { return jNumbers; } if (numbers.IndexOf(",") > 0) { string[] numberList = numbers.Split(','); foreach (string number in numberList) { jNumbers.Add(new JValue(number.Trim())); } return jNumbers; } jNumbers.Add(new JValue(numbers)); return jNumbers; } /// <summary> /// 格式化单据内码 /// </summary> /// <param name="parameter"></param> /// <returns></returns> private static string formatFID(Parameter parameter) { string ids = ""; if (parameter.Model.Property("FID") == null && parameter.Model["FID"] == null) { return ids; } return Convert.ToString(parameter.Model["FID"]); } /// <summary> /// 需要返回的字段 /// </summary> /// <returns></returns> private static JArray NeedReturnFields() { JArray needReturnFields = new JArray { new JValue("FID"), new JValue("FBillNo") }; return needReturnFields; } /// <summary> /// 格式化WebAPI返回结果 /// </summary> /// <param name="res"></param> /// <returns></returns> public static ResponseData ConvertWebAPIResult(object res) { Dictionary<string, object> _res = res as Dictionary<string, object>; Dictionary<string, object> result = _res["Result"] as Dictionary<string, object>; ResponseData responseData = new ResponseData(); responseData.Result = ConvertWebAPIResultResponseDataResult(result); return responseData; } /// <summary> /// 格式化WebAPI返回结果中的Result /// </summary> /// <param name="res"></param> /// <returns></returns> public static ResponseDataResult ConvertWebAPIResultResponseDataResult(object res) { Dictionary<string, object> result = res as Dictionary<string, object>; ResponseDataResult responseDataResult = new ResponseDataResult(); if (result.ContainsKey("Id") && result["Id"] != null) { responseDataResult.ID = Convert.ToInt64(result["Id"]); } if (result.ContainsKey("Number") && result["Number"] != null) { responseDataResult.Number = Convert.ToString(result["Number"]); } responseDataResult.ResponseStatus = ConvertWebAPIResultData(result["ResponseStatus"]); if (result.ContainsKey("NeedReturnData") && result["NeedReturnData"] != null) { responseDataResult.NeedReturnData = result["NeedReturnData"] as JArray; } return responseDataResult; } /// <summary> /// 格式化WebAPI返回结果中的结果数据 /// </summary> /// <param name="res"></param> /// <returns></returns> public static ResponseDataResultStatus ConvertWebAPIResultData(object res) { Dictionary<string, object> result = res as Dictionary<string, object>; ResponseDataResultStatus responseDataResult = new ResponseDataResultStatus(); if (result.ContainsKey("IsSuccess")) { responseDataResult.IsSuccess = Convert.ToBoolean(result["IsSuccess"]); } else { responseDataResult.IsSuccess = false; } if (result.ContainsKey("MsgCode")) { responseDataResult.MsgCode = Convert.ToInt64(result["MsgCode"]); } else { responseDataResult.MsgCode = 0; } responseDataResult.Errors = ConvertWebAPIResultErrors(result["Errors"]); responseDataResult.SuccessEntitys = ConvertWebAPIResultSuccessEntitys(result["SuccessEntitys"]); responseDataResult.SuccessMessages = ConvertWebAPIResultSuccessMessages(result["SuccessMessages"]); return responseDataResult; } /// <summary> /// 格式化WebAPI返回结果中的错误信息 /// </summary> /// <param name="res"></param> /// <returns></returns> public static List<ResponseDataResultErrors> ConvertWebAPIResultErrors(object res) { if (res == null) { return null; } List<object> result = res as List<object>; List<ResponseDataResultErrors> responseDataResultErrors = new List<ResponseDataResultErrors>(); foreach (object item in result) { ResponseDataResultErrors value = new ResponseDataResultErrors(); Dictionary<string, object> _item = item as Dictionary<string, object>; value.DIndex = Convert.ToInt64(_item["DIndex"]); value.FieldName = Convert.ToString(_item["FieldName"]); value.Message = Convert.ToString(_item["Message"]); responseDataResultErrors.Add(value); } return responseDataResultErrors; } /// <summary> /// 格式化WebAPI返回结果中的成功的明细 /// </summary> /// <param name="res"></param> /// <returns></returns> public static List<ResponseDataResultSuccessEntitys> ConvertWebAPIResultSuccessEntitys(object res) { if (res == null) { return null; } List<object> result = res as List<object>; List<ResponseDataResultSuccessEntitys> responseDataResultSuccessEntitys = new List<ResponseDataResultSuccessEntitys>(); foreach (object item in result) { ResponseDataResultSuccessEntitys value = new ResponseDataResultSuccessEntitys(); Dictionary<string, object> _item = item as Dictionary<string, object>; value.DIndex = Convert.ToInt64(_item["DIndex"]); value.Id = Convert.ToInt64(_item["Id"]); value.Number = Convert.ToString(_item["Number"]); responseDataResultSuccessEntitys.Add(value); } return responseDataResultSuccessEntitys; } /// <summary> /// 格式化WebAPI返回结果中的成功的信息 /// </summary> /// <param name="res"></param> /// <returns></returns> public static List<ResponseDataResultSuccessMessages> ConvertWebAPIResultSuccessMessages(object res) { if (res == null) { return null; } List<object> result = res as List<object>; List<ResponseDataResultSuccessMessages> responseDataResultSuccessMessages = new List<ResponseDataResultSuccessMessages>(); foreach (object item in result) { ResponseDataResultSuccessMessages value = new ResponseDataResultSuccessMessages(); Dictionary<string, object> _item = item as Dictionary<string, object>; value.DIndex = Convert.ToInt64(_item["DIndex"]); value.FieldName = Convert.ToString(_item["FieldName"]); value.Message = Convert.ToString(_item["Message"]); responseDataResultSuccessMessages.Add(value); } return responseDataResultSuccessMessages; } }}

更新动作中,我们需要组织需要更新的字段。

我们无法确定哪个字段需要更新,所以在处理的时候,除了单据的FID(单据内码,换句话说就是单据的流水号)、单据的FBillNo(单据编号)、明细数据属性Key(采购订单明细Key为FPOOrderEntry)、明细数据的FEntryID(明细内码,明细数据流水号)。

其他的字段都作为更新的字段。具体处理代码为:

// 组织需要更新的字段JArray needUpDateFields = new JArray();JArray entryData = parameter.Model[parameter.EntryKey] as JArray;foreach (JProperty item in parameter.Model.Properties()){ if (item.Name != "FID" && item.Name != "FBillNo" && item.Name != parameter.EntryKey) { needUpDateFields.Add(item.Name); }}needUpDateFields.Add(parameter.EntryKey);JObject entryDataZero = entryData[0] as JObject;foreach (JProperty item in entryDataZero.Properties()){ if (item.Name != "FEntryID") { needUpDateFields.Add(item.Name); }}

在Formater.cs类文件中,我们使用一下函数方法来解析WebAPI返回的结果数据:

  • ConvertWebAPIResult
  • ConvertWebAPIResultResponseDataResult
  • ConvertWebAPIResultData
  • ConvertWebAPIResultErrors
  • ConvertWebAPIResultSuccessEntitys
  • ConvertWebAPIResultSuccessMessages

WebAPI返回结果数据的结构也可以在官方文档中查阅到。

当然也可以用其他方式来解析WebAPI返回结果,这不是本教程讨论的重点。

接下来,我们编写单据处理函数的方法。

我们再新建一个名为“BillHandler.cs”的类文件。

单据的处理方法都放到这个类文件中,当前我们只做一个方法,如果需要扩展其他的对单据的处理方法,也可以放在这个文件中。

代码组织方面的问题,我们不在这里讨论和展开。代码如下:

using System;using System.Collections.Generic;using System.Linq;using System.Text;using Kingdee.BOS;using Kingdee.BOS.WebApi.FormService;using Newtonsoft.Json;using Newtonsoft.Json.Linq;namespace Test.K3Cloud.SCM.WebAPI.ServerExtend{ public class BillHandler { /// <summary> /// 保存和更新 /// </summary> /// <param name="context"></param> /// <param name="parameter"></param> /// <returns></returns> public static Result Save(Context context, Parameter parameter) { // 更新 var rs = WebApiServiceCall.Save(context, parameter.LocalFormId, Formater.Update(parameter)); ResponseData res = Formater.ConvertWebAPIResult(rs); Result result = new Result(); if (!res.Result.ResponseStatus.IsSuccess) { result.msg = JsonConvert.SerializeObject(res.Result.ResponseStatus.Errors) "1"; return result; } result.data = new JObject { new JProperty("FID", res.Result.ID), new JProperty("Number", res.Result.Number) }; // 提交 if (parameter.Model.Property("FID") == null && Convert.ToInt64(parameter.Model["FID"]) == 0) { parameter.Model.Add(new JProperty("FID", res.Result.ID)); } if (parameter.Model.Property("FBillNo") == null && Convert.ToInt64(parameter.Model["FBillNo"]) == 0) { parameter.Model.Add(new JProperty("FBillNo", res.Result.Number)); } var rsSubmit = WebApiServiceCall.Submit(context, parameter.LocalFormId, Formater.Submit(parameter)); ResponseData rsSubmitData = Formater.ConvertWebAPIResult(rsSubmit); if (!rsSubmitData.Result.ResponseStatus.IsSuccess) { result.success = false; result.msg = JsonConvert.SerializeObject(rsSubmitData.Result.ResponseStatus.Errors) "2"; return result; } // 审核 var rsAudit = WebApiServiceCall.Audit(context, parameter.LocalFormId, Formater.Audit(parameter)); ResponseData rsAuditData = Formater.ConvertWebAPIResult(rsAudit); if (!rsAuditData.Result.ResponseStatus.IsSuccess) { result.success = false; result.msg = JsonConvert.SerializeObject(rsAuditData.Result.ResponseStatus.Errors) "3"; return result; } result.success = true; result.msg = Defineds.DefaultSuccessMsg; return result; } }}

上面代码中Save方法中,对单据做了保存、提交和审核的操作。

WebApiServiceCall 类中方法中实现了对单据的各种操作,在其他需求场景中,大家可以跟踪进去查看。

当然,对单据的操作还有其他方式,本教程为了方便大家对应官方文档理解,使用了WebApiServiceCall。

调用WebApiServiceCall 当然也是有缺陷的。

星空系统对单个的单据操作是有事务保护的,例如单据保存,如果在保存的操作中,出现了校验不通过的情况下,会触发事务回滚。

但是上面代码中保存、提交和审核这连续的三个动作,没有事务保护。

这种方式的优点是开发代码量少,开发效率高而且容易理解。

接下来我们在 BusinessService.cs类中调用Save方法,来更新、提交和审核采购订单。代码如下:

using System;using System.Collections.Generic;using System.Linq;using System.Text;using Kingdee.BOS;using Kingdee.BOS.ServiceFacade.KDServiceFx;using Kingdee.BOS.WebApi.ServicesStub;using Newtonsoft.Json;namespace Test.K3Cloud.SCM.WebAPI.ServerExtend{ public class BusinessService : AbstractWebApiBusinessService { public BusinessService(KDServiceContext context) : base(context) { } /// <summary> /// 执行接口请求 /// </summary> /// <param name="parameters"></param> /// <returns></returns> public Result ExcuteService(string parameters) { parameters = parameters.Trim(); Result result = Checker.clientParameterChecker(parameters); if (!result.success) { return result; } return ServiceRouter(parameters); } /// <summary> /// 接口请求分发 /// </summary> /// <param name="parameters"></param> /// <returns></returns> private Result ServiceRouter(string parameters) { // 反序列化客户端传递的JSON数据 Parameter parameter = JsonConvert.DeserializeObject<Parameter>(parameters); Result checkRs = Checker.checkModel(parameter); // 业务参数不能为空 if (!checkRs.success) { return checkRs; } Context context = KDContext.Session.AppContext; return BillHandler.Save(context, parameter); } }}

ServiceRouter 为实际调用单据操作的方法。

因为我们还可以根据其他参数来处理不同的场景需求,所以命名为:ServiceRouter 。

当前我们教程中只调用了Save方法。

4.编译代码生成dll文件

在Visual Studio IDE菜单栏中选择“项目”选项,选择 “Test.K3Cloud.SCM.WebAPI.ServerExtend属性”选项,打开“属性配置”窗口。

选择窗口右侧的“生成”菜单选项,将输出路径设置为金蝶的安装目录的Websitebin目录。

默认路径为:C:Program Files (x86)KingdeeK3CloudWebSiteBin。

配置完成之后,点击“保存”。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

点击菜单中的“生成”选项,在下拉菜单中选择“生成Test.K3Cloud.SCM.WebAPI.ServerExtend”选项。

生成“Test.K3Cloud.SCM.WebAPI.ServerExtend”动态链接库文件。

在Visual Studio IDE下方的输出窗口中显示生成成功,则说明Test.K3Cloud.SCM.WebAPI.ServerExtend.dll文件生成成功。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

5.部署

我们在上面的项目属性设置中看到,配置了Test.K3Cloud.SCM.WebAPI.ServerExtend.dll文件的生成目录。

所以,当我们成功生成Test.K3Cloud.SCM.WebAPI.ServerExtend.dll文件之后就已经完成了部署工作。

如果没有做这个配置,我们可以将生成的这个文件复制到星空系统的安装目录。

默认目录为:C:Program Files (x86)KingdeeK3CloudWebSiteBin。

6.重启IIS服务

自定义WebAPI需要重启IIS服务。

打开IIS管理器。选择站点,然后点击右侧窗口中的“重新启动”按钮。重启IIS服务。

每次代码更新,编译之后,都需要重启IIS服务。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

7.使用Postman模拟请求

在模拟请求之前,我们需要了解自定义WebAPI是对星空系统标准WebAPI的扩展,所以同样需要通过星空的身份验证。所以第一步需要进行登录操作。

登录需要的地址为:

http://127.0.0.1/K3Cloud/Test.K3Cloud.SCM.WebAPI.ServerExtend.BusinessService.ExcuteService,Test.K3Cloud.SCM.WebAPI.ServerExtend.common.kdsvc

登录时需要提交参数有:Acctid(账套ID)、userName(用户名)、password(密码)、lcid(语言ID)。

账套ID可以在星空客户端WebAPI测试工具界面找到。如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

第二步是测试请求我们的WebAPI。

根据WebAPI处理的方式我们组织的请求报文如下:

{ "parameters": [{ "LocalFormId": "PUR_PurchaseOrder", "EntryKey": "FPOOrderEntry", "Model": { "FID": 100056, "FPOOrderEntry": [{ "FEntryID": 100107, "FQty": "150", "FEntryNote": "测试明细备注1111" }] } }]}

上面报文中有以下几个涉及到单据的字段:
FID:单据内码
FEntryID:单据明细内码
FQty:采购数量
FEntryNote:明细备注

的FID和FEntryID可以在数据库中查询到。

采购订单记录表为“t_PUR_POOrder”,

明细表为“t_PUR_POOrderEntry”。

备注说明:如果是使用金蝶官方给出的SDK来发起请求的话,就不需要{"parameters":[]}这一层。

自定义WebAPI地址为:
http://127.0.0.1/K3Cloud/Test.K3Cloud.SCM.WebAPI.ServerExtend.BusinessService.ExcuteService,Test.K3Cloud.SCM.WebAPI.ServerExtend.common.kdsvc

自定义接口URL地址格式说明:

http://ServerIp/K3Cloud/接口命名空间.接口实现类名.方法,组件名.common.kdsvc

本例中接口

接口命名空间:Test.K3Cloud.SCM.WebAPI.ServerExtend

接口类名:BusinessService

方法名称:ExcuteService

组件名:编译后的dll名称,这里是Test.K3Cloud.SCM.WebAPI.ServerExtend

首先我们在星空系统中创建一个采购订单,如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

接下来,我们使用Postman来模拟客户端请求。登录请求如下图所示:

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

如果返回结果与截图中一样,则说明登录成功。

Postman会自动维护登录Session会话,所以,登录成功之后,我们可以直接进行自定义WebAPI接口的测试。

金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)金蝶云星空插件实战开发-新手保姆级入门教程-自定义WebAPI(金蝶云星空api接口)

如果返回结果中success为true,则说明执行成功。

教程内容较长,比前面的几篇教程复杂很多。大家可以在自己的开发环境中进行测试练习。

在学习的过程中,如果遇到了什么问题,可以留言。

对于新手来说,一次就成功完成教程,也有一定的难度,大家边做边思考边解决问题,这样我们的收获才是最大的。

虽然我给出了源代码,但是不建议大家直接拿来用,最好是自己敲一遍代码,这样才能发现自己的盲区,以便在今后的开发工作中顺利完成任务。

本教程源码地址:

https://gitee.com/hsg4ok_admin/kingdee_documents/tree/master/示例代码/Test.K3Cloud.SCM.WebAPI.ServerExtend


更多精彩内容发布于公众号:代码乾坤 (CoderLand)

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

(0)
上一篇 2024年1月3日 上午8:35
下一篇 2024年1月3日 上午8:47

相关推荐

  • 科研项目经费支出预算表,管理高层意见不一如何处理(科研项目经费预算说明)

    这也是你确定是否存在对项目前景持有不同意见的管理高层的最佳机会,因为这些持有不同意见的高管可能会成为减慢项目进展的障碍。持有不同意见的高层管理人员,是导致项目失败的主要原因,因此通…

    科研百科 2024年4月13日
    85
  • 律所项目管理

    律所项目管理 律所项目管理是律师工作中非常重要的一部分。一个高效的律所项目管理可以帮助律师团队更有效地管理时间和资源,提高律师工作效率,降低成本,同时也可以提高律所的声誉和竞争力。…

    科研百科 2024年5月28日
    48
  • 开源工程项目管理软件(开源工程项目管理系统)

    开源工程项目管理系统开源工程项目管理系统开源工程是指进入量轮、从起始到终点的工程项目。目前世界范围内,全球范围内,平均每年有3亿多的位数和数字。最受欢迎的,也是量轮的最受欢迎的,有…

    科研百科 2024年9月27日
    25
  • 上海协同办公

    上海协同办公:从城市中心到云端的数字化转型 随着数字化时代的到来,上海协同办公市场也在不断变化和演进。在过去几十年中,上海一直是中国数字化转型的先锋城市之一。随着云计算、大数据、人…

    科研百科 2024年8月21日
    34
  • hbuilder项目管理器

    Hbuilder项目管理器: 让软件开发更高效 Hbuilder项目管理器是一款功能强大的软件开发工具,可以帮助开发人员更有效地管理项目进度、资源、任务和风险。本文将介绍Hbuil…

    科研百科 2024年9月15日
    27
  • 疫情下应该如何开发生活服务APP- 有何亮点?(疫情下应该如何开发生活服务app- 有何亮点和特色)

    生活服务APP软件有什么特点?应该如何开发?在移动互联网技术不断发展、进步的时代之下,各种类型的APP软件不断地涌现出来,为有所需要的用户提供了方便快捷的功能服务。在这其中,生活服…

    科研百科 2023年3月27日
    173
  • 研究课题层次

    研究课题层次: 人类意识的本质与研究方法 人类意识的本质一直是哲学,神经科学和心理学等领域的研究课题。人类意识的本质是什么?它是如何形成的?它与其他认知过程有何不同?这些问题一直吸…

    科研百科 2024年11月9日
    0
  • ERP系统,让你爱恨情仇的那些事

    牵头人不懂业务,懂ERP的人又没权限,吃力不讨好,相信不少系统实施者都有过类似想法吧。 此ERP,别人家的 ERP系统应用成功率高,为企业节省成本,能实现企业精确内控管理,保证物料…

    科研百科 2023年3月24日
    208
  • 软件项目管理系统模块

    软件项目管理系统模块: 软件开发过程中的关键组成部分 随着软件开发的日益复杂和快速发展,软件项目管理系统模块已经成为了软件开发过程中必不可少的关键组成部分。软件项目管理系统模块可以…

    科研百科 2024年12月13日
    0
  • 人事管理软件

    人事管理软件是一种用于管理组织内员工信息的软件,能够帮助组织更好地规划和管理员工工作,提高员工工作效率和生产力。本文将介绍人事管理软件的一些常见功能和优点,以及如何选择合适的人事管…

    科研百科 2024年8月22日
    49