- 1. ArkFBP-JavaScript
- 2. 简介 Introduction
- 3. 快速上手 QuickStart
- 4. 基础节点用法
ArkFBP-JavaScript是一个ArkFBP的JS的实现。
ArkFBP是一个基于Flow-Based Programming的一个开发框架,目前有JavaScript,Python,Golang等几个不同版本。
-
下载arkfbp预编译文件 Download precompiled arkfbp cli binary
wget https://github.com/arkfbp/arkfbp/releases/download/0.0.3-rc/arkfbp-darwin
-
给二进制文件赋予执行权限 Give the execution permission
chmod +x arkfbp
-
将文件移动到系统PATH目录下 Move the arkfbp cli to the OS Path
move arkfbp-darwin /usr/local/bin/arkfbp
-
下载扩展文件 Download VSCode Extension File
https://github.com/arkfbp/arkfbp/releases/download/0.0.3-rc/arkfbp-0.0.3.vsix
-
安装扩展文件 Install the arkfbp Extension
https://vscode-docs.readthedocs.io/en/stable/extensions/install-extension/
arkfbp create --type server --language javascript --name helloworld
用arkfbp cli工具(https://github.com/longguikeji/arkfbp)创建一个js项目后,项目主体代码均在src目录中。
src //源代码根目录,@
databases //数据库的定义
flows //流,所有流存放的目录
models //ORM的数据库配置,由CLI工具自动生成
routes //路由,定义所有开放的API接口
testFlows //测试流,所有测试流存放的目录
所有flow都应该创建在flows目录中。
- 每一个flow都由一个目录组成,目录名即为流的名字。
- 在flow的根目录下的index.js文件中,定义了该流的『图』。
- flow目录下有nodes目录,其中存放所有该流所包含的节点定义。
- 每个节点都是一个js文件,用面向对象的方式继承自某个基础节点。
import { Flow } from 'arkfbp/lib/flow'
import { Graph } from 'arkfbp/lib/graph'
import { StartNode } from 'arkfbp/lib/StartNode'
import { StopNode } from 'arkfbp/lib/StopNode'
import { AddVendorNode } from './nodes/addVendorNode'
export class Main extends Flow {
createNodes() {
return [
{
cls: StartNode,
id: '1',
next: '2',
},
{
cls: AddVendorNode,
id: '2',
next: '3',
},
{
cls: StopNode,
id: '3',
},
]
}
createGraph() {
const g = new Graph()
g.nodes = this.createNodes()
return g
}
}
在createNodes
方法中,return的数据中包含了图的定义。个我们统一定义为 flow chart params,节点的流程图参数。
参数名 | 类型 | 含义 | 示例 |
---|---|---|---|
cls | Class | 节点类型 | cls: StartNode cls: xxxxNode(xxxNode为某个Node的子类) |
id | String | 当前节点id | id: '1' id: 'start' |
next | String | 下一个节点的id | next: 'stop' next: '1' |
一些特殊节点会有特殊的参数,如LoopNode的body。参见每个节点说明。
一个能在流内任意节点读写的变量,this.$state
. 该变量不能直接使用,通过各种方法来进行读写。
主要用来跨节点传输数据
函数 | 参数 | 功能 | 示例 |
---|---|---|---|
commit() | function | 修改state变量 | this.$state.commit(state => { state.a = 1; state.b = 2; return state; } ) |
fetch() | 无 | 获取最新的state变量 | const a = this.$state.fetch().a |
- 节点的用法,需要先继承基础节点,然后确定该节点的参数与重载函数,最后在流程图中定义好该节点的流程图参数即可。
- 所有节点都有一个共同的重载函数
run
,作为节点运行的主函数。 run
函数的返回值会成为该节点的outputs
,即下一个节点的inputs
。- 节点参数仅支持静态值,对节点内部如this.inputs或this.$state等值的访问都只能在函数中进行。
[cls,id,next]
无
函数名 | 描述 | 是否必须重载 | 示例 |
---|---|---|---|
run() | 作为节点运行主函数,默认功能是将inputs直接返回 | 否 | 一般不用,可在此函数中做一些对数据的预处理。 参考[FunctionNode] |
[cls,id,next]
无
函数名 | 描述 | 是否必须重载 | 示例 |
---|---|---|---|
run() | 作为节点运行主函数,默认功能是将inputs直接返回 | 否 | 一般不用,可在此函数中完成对流的返回值的最后整理。 参考[FunctionNode] |
[cls,id,next]
无
函数名 | 描述 | 是否必须重载 |
---|---|---|
run() | 作为节点运行主函数 | 是 |
示例:
import { FunctionNode } from 'arkfbp/lib/functionNode'
export class CaseNode extends FunctionNode {
async run() {
const data = this.inputs.a
data.name = this.$state.fetch().name
this.$state.commit( state=>{
state.count ++;
return state;
})
return data
}
}
[cls,id]
注意:IFNode没有next
参数名 | 类型 | 含义 | 示例 |
---|---|---|---|
positiveNext | String | 当expression返回值为true时,会运行的下一个节点id | positiveNext:'4' |
negativeNext | String | 当expression返回值为false时,会运行的下一个节点id | negativeNext: 'stop' |
无
函数名 | 描述 | 是否必须重载 |
---|---|---|
run() | 作为节点运行主函数 | 是 |
condition() | 条件,返回bool值, 默认返回true | 否 |
positive() | 条件满足的时候执行 | 否 |
negative() | 条件不满足的时候执行 | 否 |
示例:
import { IFNode } from 'arkfbp/lib/ifNode'
export class MyIFNode extends IFNode {
condition(){
return this.inputs.a === 1
}
positive() {
return {'error': 0}
}
negative() {
return {'error': -1}
}
}
实现多条件分支的逻辑,满足任一条件后,剩余的部分将不会被执行
cls,id
注意:SwitchNode没有next
参数名 | 类型 | 含义 | 示例 |
---|---|---|---|
route | [] | 是个数组,每一项即为一个条件 | { |
```cls: MySwitchNode, | |||
id: "1", | |||
route: [ | |||
{ | |||
'condition': 'condition1', | |||
'positive': 'positive1', | |||
'negative': 'negative1', | |||
'next': '2', | |||
}, | |||
{ | |||
'condition': 'condition2', | |||
'positive': 'positive2', | |||
'negative': 'negative2', | |||
'next': '3', | |||
}, | |||
{ | |||
'condition': 'condition3', | |||
'positive': 'positive3', | |||
'negative': 'negative3', | |||
'next': '4', | |||
}, | |||
] | |||
}``` |
无
示例:
import { SwitchNode } from 'arkfbp/lib/switchNode'
export class MySwitchNode extends IFNode {
condition1() {
return this.inputs.a === 1
}
positive1() {
// put any logic here
}
negative1() {
// put any logic here
}
condition2() {
return this.inputs.a === 2
}
positive2() {
// put any logic here
}
negative2() {
// put any logic here
}
condition3() {
return this.inputs.a === 3
}
positive3() {
// put any logic here
}
negative3() {
// put any logic here
}
}
[cls,id,next]
参数名 | 类型 | 是否必须 | 描述 | 示例 |
---|---|---|---|---|
mode | string | 是 | 请求方式,可选值:direct,proxy, 默认值为direct | mode: proxy(通过ArkOS的APIserver发送请求) mode: direct(直接向目标发送请求) |
url | string | 是 | 请求地址 | url: 'https://www.x.com/api' |
method | string | 是 | 请求方式,可选值: post,get等Http协议支持的选项 |
method:POST method:GET |
auth | null | 否 | 签名,保留字段,预计会用于用户验证 | |
headers | any | null | 是 | http请求的headers | headers = { 'Accept': 'application/json;charset=utf-8', 'X-Requested-With': 'XMLHttpRequest', 'User-Agent': 'Mozilla/5.0', 'Content-Type': 'application/x-www-form-urlencoded;charset=utf-8', } |
params | any | null | 否 | http请求的参数,可由buildParams()函数代替 | params: 'a=1&b=2' |
函数名 | 描述 | 是否必须重载 |
---|---|---|
run() | 作为节点运行主函数 | 否 |
buildParams() | 构建参数 | 否,优先级比params高,注意不要重复使用 |
示例:
import { APINode } from 'arkfbp/lib/apiNode'
export class PostAPINode extends APINode {
mode = 'direct'
url = 'https://sms.yunpian.com/v2/sms/tpl_batch_send.json'
method = 'post'
headers = {
'Accept': 'application/json;charset=utf-8',
'X-Requested-With': 'XMLHttpRequest',
'User-Agent': 'Mozilla/5.0',
'Content-Type': 'application/x-www-form-urlencoded;charset=utf-8',
}
buildParams() {
console.log('PostApiNode:',this.inputs)
return this.inputs
}
}
[cls,id,next]
参数名 | 类型 | 含义 | 示例 |
---|---|---|---|
body | Number | String | 循环体的下一个节点,从这个节点开始的后续节点都是在循环体内 |
无
函数名 | 描述 | 是否必须重载 |
---|---|---|
initStatement() | 初始化状态(i = 0) | 是 |
conditionStatement() | 条件判断(i<10) | 是 |
postStatement() | 末尾状态(i++) | 是 |
process() | 循环程序,返回值会被作为body循环体的输入 | 是 |
示例:
import { LoopNode } from 'arkfbp/lib/loopNode'
export class LoopSendNode extends LoopNode {
_i = 0
initStatement() {
this._i = 0
}
conditionStatement() {
return this._i < this.inputs.mobiles.length
}
postStatement() {
this._i++
}
process() {
const data = 'mobile='+this.inputs.mobiles[this._i]
return data
}
}
[cls,id,next]
参数名 | 类型 | 是否必须 | 描述 | 示例 |
---|---|---|---|---|
flow | Flow | 是 | 需要运行的流 |
函数名 | 描述 | 是否必须重载 |
---|---|---|
buildInputs() | 构建运行flow时的输入 | 否 |
示例:
import { TriggerFlowNode } from 'arkfbp/lib/triggerFlowNode'
import {Main as FooBarFlow} from '@/flows/foo/bar'
export class RunFooBarFlowNode extends TriggerFlowNode {
flow = FooBarFlow
buildInputs(){
return {a: 1}
}
}
[cls,id,next]
参数名 | 类型 | 是否必须 | 描述 | 示例 |
---|---|---|---|---|
flow | Flow | 是 | 需要测试的流的类 | |
start | String | 否 | 测试目标流的起始节点id ,为空则代表从流的第一个节点开始 |
start: '5' |
stop | String | 否 | 测试目标流的结束节点id ,为空则代表直到流结束 |
stop:'6' |
函数名 | 描述 | 是否必须重载 |
---|---|---|
setUp() | 在所有case执行前执行,可以用来准备测试数据 | 否 |
tearDown() | 在所有case执行完后执行,可以用来清理测试数据,比如删除操作 | 否 |
test{CaseName}() | case,测试用例 | - testA(){} |
test{CaseName}SetUp() | 在对应case之前执行,用来准备测试数据 | - testASetup() |
test{CaseName}TearDown() | 在对应case之后执行,用来清理测试数据 | - testATearDown() |
示例:
import { TestNode } from 'arkfbp/lib/testNode'
import {Main as RegisterFlow} from '@/flows/main/register'
import {Config} from '@/models/config'
import assert from 'assert'
export class Node1 extends TestNode {
flow = RegisterFlow
setUp() {
}
tearDown() {
//在所有case执行完后执行
console.info('tearDown')
}
testASetUp(){
// 在testA之前执行
//在所有case执行前执行
this.inputs = {
appid : 'demoapp',
keys: [{
key: 'testkey1',
discription: 'testkey1 description11'
},{
key: 'testkey2',
discription: 'testkey2 description22'
}]
}
}
testATearDown(){
// 在testA之后执行
}
async testA() {
const config = await Config.findOne({
where: { appid: 'demoapp', key: 'testkey1' }
})
assert.strictEqual('testkey1',config.key)
}
async testB() {
const config = await Config.findOne({
where: { appid: 'demoapp', key: 'testkey2' }
})
assert.strictEqual('testkey2',config.key)
}
}
什么也不做
无
无