跳到主要内容

后端JSF函数开发说明

JSF函数采用javascript作为脚本语言, 支持ES 2017语法,只能使用javascript的基本功能,以及端提供的sdk函数。有关sdk函数说明参考附2: backend SDK函数

JSF按使用场景,有不同的上下文变量和返回值, 目前backend仅支持流程调用这一种后端场景函数, 实际在编写JSF函数时,只需要提供函数体即可。

流程场景函数

流程场景函数,是在一个单据执行的工作流程过程中,可以调用的函数,比如单据生效后调用,单据弃审后调用等。

场景函数的模板:

function ${funcName}(userContext, objectType, objectId, variables) {
${jsfContent}
}

场景函数的参数说明

字段名称字段说明类型必填备注
userContext只读当前用户上下文对象objectY当前调用的上下文对象, 不需要用户传递,为只读类型, 它包括的字段参考附1: 上下文对象
objectType当前流程的单据对象类型stringY-
objectId当前流程的单据对象IDstringY-
variables流程中设置的变量object它是和一KV结构, key为变量名,value为变量值对象(比如J: {"var1": {"type": "String", "value": "var1Value"}}

场景函数返回值规则

返回值说明:通常的返回值格式如下:

	{
status: "success|error|warning", //函数是否成功执行的状态
code: "", //如果执行不成功时,提供的错误码
message: "", //如果执行不成功时,提供的错误提示信息
description: "", //如果执行不成功时,提供的错误详细描述
data: null, //成功时需要返回的数据对象
ars: null, //如果出错时,需要返回的其他错误数据
}

返回会是的属性说明:

字段名称字段说明类型必填备注
status执行函数状态stringY函数是否成功执行的状态, 取值为 success/error/warning
code错误码string-如果执行不成功时,提供的错误码
message错误信息string-如果执行不成功时,提供的错误提示信息
description错误描述string-如果执行不成功时,提供的错误详细描述
data返回的数据any-成功时需要返回的数据对象
args错误附加参数map-如果出错时,需要返回的其他错误数据
logs函数中输出的日志信息array-函数中输出的日志信息

如果JSF函数体没有返回数据,或者返回的数据中没有status字段,则默认jsf函数执行成功, 会设置status=success, 并将返回对象当前data, 即:

return {status: "success", data: {id:"123"}}; 等价于: return {id: "123"};');

场景函数示例:

a) 收款单增加额外的逻辑检查,如果收款部门为’AA'并且收款金额>100时,检查不通过,打回单据,并显示错误信息为“XX部门收款不能超过100"。

if (objectType != 'Receipt') {
return;
}
var result = await query(objectType, ['id'], `id = '${objectId}' and ownerDept.name = 'AA' and amount > 100`);
if (result.status != 'success') {
return result;
}
if (result.data.length == 1) {
return {
status: 'error',
message: 'AA部门收款不能超过100'
};
}
return true;

说明: objectType, objectId为参数, query为backend SDK函数

b) 执行一个插入Comment的操作及删除自动插入的评论操作

return create('Comment', {
objectType: objectType,
objectId: objectId,
content: '这是一个自动增加的评论'
});

c) 执行一个删除自动添加的评论的操作

var result = await query('Comment', ['id'], `objectType = '${objectType}' and objectId = '${objectId}' and content='这是一个自动增加的评论'`);
if (result.status != 'success') {
return result;
}
if(result.data.length != 0){
return remove('Comment', result.data[0].id);
}

d) 发票增加额外弃审检查,发票金额>1000,不允许弃审,并提供弃审错误信息“发票金额大于1000,不允许弃审”。

if (objectType != 'Invoice') {
return;
}
var result = await query(objectType, ['id'], `id = '${objectId}' and amount > 1000`);
if (result.status != 'success') {
return result;
}
if (result.data.length == 1) {
return {
status: 'error',
message: '发票金额大于1000,不允许弃审'
};
}
return true;

附1: 上下文对象

上下文对象为JSF函数运行时上下文信息对象, 它包括以下内容:

{
rootRequestId: string; //根请求ID, 如果当前请求是别的请求触发的,保存的别的请求的requestId, 以便输出日志时,用来跟踪调用链
requestId: string; //当前请法求ID, 当前请求的唯一标识, 相同的requestId,以同样的参数调用时,后端会进行幂等处理,只响应第一次成功调用结果
tenantId: string; //当前租户ID
userId: string; //当前用户ID
departmentId: string; //当前用户所属部门ID
currentRoleId: string; //当前用户使用的角色ID
currentOrgId: string; //当前的组织ID
isDisableDataScope: boolean; //是否禁用数据权限
roles: Map<string, boolean>; //当前用户具有的角色信息, key为角色ID, value表示该角色是否有管理员身份
}

附2: backend SDK函数

目前backend sdk支持以下函数

  • 查询对象:query(objectType: string, fields: string[], criteriaStr: string, sorts: object, offset: number, limit: number, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>
  • 新增对象:create(objectType: string, data: object, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>
  • 修改对象: update(objectType: string, data: object, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>
  • 删除对象: remove(objectType: string, id: string, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>
  • Gql查询:graphql(gql: string, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>
  • 通用api调用: restapi(appName: string, method: string, url: string, data: object, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>
  • 新增变更单:change((objectType: string, data: object, headers: Map<string, string>): Promise<RestResponse>
  • 异步批量触发jsf函数:batch(objectType: string,execUserId:string,criteriaStr:string,bindVers:Map<String,Object>,batchSize:Integer,jsfDefName:string,remarks:string,headers:Map<string,string>): Promise<RestResponse>

其中返回值 RestResponse对象结构为:

{
status: string;
code: string;
message: string;
description: string;
data: any;
args: any;
}

字段说明如下:

字段名称字段说明类型必填备注
status执行函数状态stringY函数是否成功执行的状态, 取值为 success/error/warning
code错误码string-如果执行不成功时,提供的错误码
message错误信息string-如果执行不成功时,提供的错误提示信息
description错误描述string-如果执行不成功时,提供的错误详细描述
data返回的数据any-成功时需要返回的数据对象
args错误附加参数map-如果出错时,需要返回的其他错误数据

查询对象 query:

query(objectType: string, fields: string[], criteriaStr: string, sorts: object, offset, limit: number, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>

参数说明:

字段名称字段说明类型必填备注
objectType查询对象名stringY
fields查询的字段列表string[]Y至少指定一个字段
criteriaStr查询条件表达式string-指定查询的eql条件表达式, 与postgresql sql条件表达式一致, 只字段使用的是对象属性名,并且支持级联(a.b.c), 不指定条件表示无条件查询
sorts排序object-查询的排序,为要排序的字段列表, 结构示例:[{name: "name"} {name: "createdTime", isDesending: true}]
offset查询起始行integer-从0开始的起始行,不指定表示从0开始
limit返回最大的行数integer-不指定表示返回所有记录,但受系统最大行10000限制
asAdmin以管理员权限运行boolean-如果不指定或者为false,表示以当前用户身份执行
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

成功执行后返回值中data值为查询的对象的数据列表

执行 query('User', ['id', 'name', 'createdTime'], "id='1'") 返回值示例:

{
"status": "success",
"code": null,
"message": null,
"description": null,
"data": [
{
"id": "1",
"name": "小企",
"createdTime": 1626916982389
}
],
"args": null
}

新增对象 create:

create(objectType: string, data: object, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>

参数说明:

字段名称字段说明类型必填备注
objectType新增对象名stringY
data新增对象结构化数据objectY为一个Json对象,包括key为字段名,值为字段值的数据项
asAdmin以管理员权限运行boolean-如果不指定或者为false,表示以当前用户身份执行
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

成功执行后返回值中data值新增加的对象的ID

执行 create('Role', {code: '001', name: 'manager'}) 返回值示例:

{
"status": "success",
"code": null,
"message": null,
"description": null,
"data": "NEP8HV515KG0008",
"args": null
}

执行 create('User', {code: '001', name: 'testuser'}) 执行出错返回值示例:

{
"status": "error",
"code": "0010",
"message": "提交的数据存在问题",
"description": "Object validation found 2 problems, error codes: [0011, baseapp-200108]",
"data": null,
"args": {
"problems": {
"default": {
"ERROR": [
{
"fieldPath": "department",
"type": "error",
"code": "0011",
"message": "字段的值必须指定"
},
{
"fieldPath": "jobRelationships",
"type": "error",
"code": "baseapp-200108",
"message": "用户任职记录中必须包含一条主职记录",
"args": {
"jobRelationships": "用户任职记录中必须包含一条主职记录"
}
}
]
}
}
}
}

修改对象 update:

update(objectType: string, data: object, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>

参数说明:

字段名称字段说明类型必填备注
objectType新增对象名stringY
data修改对象结构化数据objectY为一个Json对象,包括key为字段名,值为字段值的数据项, 其中id必须指定,表示要修改的对象ID
asAdmin以管理员权限运行boolean-如果不指定或者为false,表示以当前用户身份执行
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

执行 update('Role', {id: 'NEP8HV515KG0008', code: '001', name: 'manager2'}) 返回值示例:

{
"status": "success",
"code": null,
"message": null,
"description": null,
"data": "",
"args": null
}

删除对象 remove:

remove(objectType: string, id: string, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>

参数说明:

字段名称字段说明类型必填备注
objectType新增对象名stringY
id删除的对象的IDstringY
asAdmin以管理员权限运行boolean-如果不指定或者为false,表示以当前用户身份执行
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

执行 remove('Role','NEP8HV515KG0008') 返回值示例:

{
"status": "success",
"code": null,
"message": null,
"description": null,
"data": "",
"args": null
}

执行 remove('Role','NEP8HV515KG0008') 出错返回值示例:

{
"status": "error",
"code": "0100",
"message": "对象不存在或已被删除",
"description": "entity Role not found: NEP8HV515KG0008",
"data": null,
"args": {
"objectType": "Role",
"objectId": "NEP8HV515KG0008"
}
}

Gql查询 graphql:

graphql(gql: string, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>

参数说明:

字段名称字段说明类型必填备注
gqlgql查询语法stringY
asAdmin以管理员权限运行boolean-如果不指定或者为false,表示以当前用户身份执行
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

执行 graphql('{User{id,name}}') 返回值示例:

{
"status": "success",
"code": null,
"message": null,
"description": null,
"data": {
"User": [
{
"id": "1",
"name": "小企"
}
]
},
"args": null,
"logs": []
}

执行 graphql('{User{id,name1}} 出错返回值示例:

{
"status": "error",
"code": "400",
"message": "查询出错",
"description": "[{\"message\":\"Validation error of type FieldUndefined: Field 'name1' in type 'UserSchemaType' is undefined\",\"locations\":[{\"line\":1,\"column\":10}],\"description\":\"Field 'name1' in type 'UserSchemaType' is undefined\",\"validationErrorType\":\"FieldUndefined\"}]",
"data": null,
"args": null,
"logs": []
}

通用api调用 restapi:

restapi(appName: string, method: string, url: string, data: object, asAdmin: boolean, headers: Map<string, string>): Promise<RestResponse>

该方法为支持特定的场景使用的方法,可以访问租户服务的api, 调用成功后,返回值的data中为接口实际的返回结果。

参数说明:

字段名称字段说明类型必填备注
appNamerestapi所在的服务名,eg: baseappstringY
method请求类型:get,post,put,deletestringY
urlrestapi相对于服务的url, 比如:"/User/1"stringY
data主求数据string-如果是get提供的是kv值对照,做为url查询参数, 其他方式请求为提供的json数据
asAdmin以管理员权限运行boolean-如果不指定或者为false,表示以当前用户身份执行
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

执行 restapi('graphql', 'post', '/', {query:'{User{id,name}}'}) 返回值示例:

{
"status": "success",
"code": null,
"message": null,
"description": null,
"data": {
"data": {
"User": [
{
"id": "1",
"name": "小企"
}
]
},
"errors": []
},
"args": null,
"logs": []
}

新增变更单 change:

change(objectType: string, data: object, headers: Map<string, string>): Promise<RestResponse>

change 函数支持: 自定义单据新增变更单

参数说明:

字段名称字段说明类型必填备注
objectType新增变更单对象stringY
data对象结构化数据objectY数据以增量形式来提交,其中baseBillId 和baseBillItemId无需指定,系统会自动根据id 补齐,和修改接口参数一致
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

执行 change('CsZiDingYiBianGengDan01', {id: 'NEP8HV515KG0008', code: '001', name: 'manager2'}) 返回示例值:

{
"status":"success",
"code":null,
"message":null,
"description":null,
"data":"SXTF6662XC500EH", // 新增的变更单id
"args":null
}

执行 change('CsZiDingYiBianGengDan01', {id: 'NEP8HV515KG0008', code: '001', name: 'manager2'}) 执行出错返回值示例:

{
"status":"error",
"code":"500",
"message":"服务器端运行您的请求出现错误!",
"description":"post http://inventory/inventory/PickingApply/changeBill: NoSuchElementException No value present",
"data":null,
"args":null
}

异步批量触发jsf函数 batch:

batch(objectType: string,execUserId:string,criteriaStr:string,bindVers:Map<String,Object>,batchSize:Integer,jsfDefName:string,remarks:string,headers:Map<string,string>): Promise<RestResponse>

异步批量触发jsf函数。当jsf 函数批量处理大量数据时,可以考虑使用此函数来拆分,防止jsf 函数执行超时。

参数说明:

字段名称字段说明类型必填备注
objectTypejsf 待处理的对象stringY
execUserId执行人stringY以执行人的身份触发jsf 函数
criteriaStr包含占位符的gql查询条件stringY
headersbindVars查询条件中的占位符绑定变量Map<string,Object>-
batchSize每批数据量大小Integer-默认值10
jsfDefNamejsf 函数名stringY异步批量调用的jsf 函数
remarks备注string-
headers附加的请求头Map<string,string>-可以指定附件的请求头,目前支持: Ignore-Warn=true, 表示忽略警告错误

执行 batch("CsZiDingYiBianGengDan01", "GK63GT50E7U0032", "billFullStatus=:billStatusTest", { billStatusTest: "BillStatus.draft" }, 1, "testJsfDSL", "测试一下", null) 返回示例值:

{
"status":"success",
"code":null,
"message":null,
"description":null,
"data":"J1SF6662XC5007X", // 用于查询异步任务的执行情况
"args":null
}

可以根据返回的J1SF6662XC5007X 使用 GQL 查看异步任务的执行情况

{
JsfBatch(criteriaStr:"id='J1SF6662XC5007X'") {
id
backlogNbr // 按criteriaStr查询出的符合条件的记录总数
asyncTaskId
asyncTask{
beanName
params
isCompleted
successRate
asyncTaskItems{
batchNo
lastErrorMsg
}
}
}
}

执行 batch("CsZiDingYiBianGengDan01", "GK63GT50E7U0032", "billFullStatus=:billStatusTest", { billStatusTest: "BillStatus.draft" }, 1, "testJsfDSL", "测试一下", null) 执行出错返回值示例:

{
"status":"error",
"code":"500",
"message":"服务器端运行您的请求出现错误!",
"description":"post http://baseapp/baseapp/JsfBatch: StringIndexOutOfBoundsException String index out of range: -1",
"data":null,
"args":null
}