camunda之——错误事件(Error Events)

错误事件(Error Events):

  • 错误事件是一个被定义好的错误触发的事件
业务错误VS技术错误(Business Errors vs. Technical Errors):

     BPMN的错误是业务错误,这与技术错误是不一样的,所以它与Java的异常是不一样的

一、定义一个错误(Defining an Error):

     错误事件定义会引用一个error元素,下面就是一个error end事件的例子,引用了一个error申明:

<definitions>
  <error id="myError" errorCode="ERROR-OCCURED" name="ERROR-OCCURED"/>
  <!-- ... -->
  <process>
    <!-- ... -->
    <endEvent id="myErrorEndEvent">
      <errorEventDefinition errorRef="myError" />
    </endEvent>
  </process>
</definitions>

     你可以通过抛出一个在流程中定义的错误事件来触发这个错误事件,或者通过委派给代码实现,详情见用户指南
     另一种定义错误的方式是设置Java异常的类名,如下例:

<definitions>
  <error id="myException" errorCode="com.company.MyBusinessException" name="myBusinessException"/>
  <!-- ... -->
  <process>
    <!-- ... -->
    <endEvent id="myErrorEndEvent">
      <errorEventDefinition errorRef="myException" />
    </endEvent>
  </process>
</definitions>
  • 这个异常应该只被用作业务异常,而不应该被用作技术异常
  • 错误事件handler引用了相同的错误元素,以表明它是来捕获处理这个错误的

     错误事件也可以通过camunda:errorMessage属性定义一个错误消息,来扩展错误元素,以提供有关错误的进一步信息。这个引用错误事件的定义必须指明camunda:errorMessageVariable去接收错误消息,这个错误消息也可以包含表达式。

<definitions>
  <error id="myError" errorCode="ERROR-OCCURED" name="ERROR-OCCURED"  camunda:errorMessage="Something went wrong: ${errorCause}" />
  <!-- ... -->
  <process>
    <!-- ... -->
    <endEvent id="myErrorEndEvent">
      <errorEventDefinition errorRef="myError" camunda:errorMessageVariable="err"/>
    </endEvent>
  </process>
</definitions>

     当错误事件抛出的错误被捕获到,会创建一个名叫err的流程变量(process variable),用以保存解析出来的消息。

二、错误开始事件(Error Start Event):

  • 错误开始事件只能被用作触发一个事件子流程,它不能被用作发起一个流程实例,错误开始事件是可中断的
图1:错误开始事件

     错误开始事件有三个可选属性:errorRef、camunda:errorCodeVariable和camunda:errorMessageVariable

<definitions>
  <error id="myException" errorCode="com.company.MyBusinessException" name="myBusinessException"/>
  ...
  <process>
    ...
    <subProcess id="SubProcess_1" triggeredByEvent="true">>
      <startEvent id="myErrorStartEvent">
        <errorEventDefinition errorRef="myException" camunda:errorCodeVariable="myErrorVariable" camunda:errorMessageVariable="myErrorMessageVariable" />
      </startEvent>
    ...
    </subProcess>
  ...
  </process>
</definitions>

1.如果errorRef属性被省略掉,那么任何一个错误事件发生,都会导致一个错误子流程被发起
2.camunda:errorCodeVariable可以包含一个由错误指定的错误码
3.camunda:errorMessageVariable可以包含一个由错误指定的错误消息

     如果流程中设置了camunda:errorCodeVariable和camunda:errorMessageVariable属性,那么在流程中可以像获取其他变量一样获取这两个属性

三、错误结束事件(Error End Event):

     当流程执行到错误结束事件时,当前路径执行结束,并且向外抛出换一个错误,这个错误可以被匹配的中间错误边界事件(intermediate error boundary event)捕获,如果没有匹配的错误边界事件,这个执行的语义跟默认跟空结束事件的语义类似(啥也不干)

四、错误边界事件(Error Boundary Event):

     依附于一个活动的边界上的中间捕获错误事件或者简称为错误边界事件,用于捕获定义于这个活动作用域内的错误。
     大多数场景下定义一个错误边界事件是为了包含一个子流程或者调用一个活动,作为一个子流程,会为所有包含在子流程内的活动创建一个作用域。错误会被错误结束事件抛出,这个错误会传播到它的父作用域,直到有一个匹配的错误边界事件。
     当错误事件被捕获,错误边界事件就会被销毁,也会销毁所有当前包含在子流程里的活动,流程会沿着错误边界事件所在的路径进行

图2:错误边界事件

     错误边界事件时一个典型的边界事件,和其他错误事件一样,errorRef引用一个定义在process元素之外的error

<definitions>
  <error id="myError" errorCode="ERROR-OCCURED" name="name of error"/>
  <!-- ... -->
  <process>
    <!-- ... -->
    <subProcess id="mySubProcess">
      <!-- ... -->
    </subProcess>
    <boundaryEvent id="catchError" attachedToRef="mySubProcess">
      <errorEventDefinition errorRef="myError" camunda:errorCodeVariable="myErrorVariable"
        camunda:errorMessageVariable="myErrorMessageVariable" />
    </boundaryEvent>
  </process>
</definitions>

errorCode被用作匹配其他被捕获到的错误:
     1.如果errorRef被忽略掉,错误边界事件会捕获所有的错误事件,不管是什么错误的errorCode
     2.如果errorRef有,并且它引用一个存在的错误,边界事件将只会捕获定义过error code的错误
     3.如果errorCodeVariable被设置过,error code可以被提取出来用作变量
     4.如果errorMessageVariable被设置,error message可以被提取出来用作变量

五、没有处理的BPMN错误(Unhandled BPMN Error):

     如果发生的错误事件没有被定义错误边界事件,在这种情况下,默认行为是打印log日志并结束当前任务,这个行为可以通过设置enableExceptionsAfterUnhandledBpmnError属性为true来改变,如果没有处理的BPMN错误发生,一个流程引擎错误将会被抛出

六、捕获并再次抛出模式(Catch and Re-Throw Pattern):

     错误事件可以在错误开始事件中被handle,子流程可以抛出跟它handle的相同的错误到更高级的作用域里
图3:捕获并再次抛出相同的错误
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342