在 Corda 中实现广播机制

摘要:在 Corda 中的所有资产默认是只有交易的参与方所拥有的,那么在 Corda 中可以将一笔交易信息分享给未参与该笔交易的节点么?答案是可以的,本文就在 Corda 中完成了一个简易版的广播实现。

Corda 是一个基于联盟链的分布式账本技术,在 Corda 中,“隐私”是其核心的设计。因此,如果 A 向 B 买了一辆车,那么在链上只有 A (Party) 和 B(Party) 能够看到在这辆车 (State) 之间发生的交易,其他节点都是不知道这笔交易信息的。但是,在现实生活中,很多情况下,都会有监管机构的存在,我们所编写的 Dapp 还需要满足监管的要求。例如,在 A 向 B 买了一辆车这个例子中,就还需要向车管所进行登记,以让车管所知道这辆车的信息及其所属情况,以便将来关于这辆车发生的任何问题车管所都能知道它的责任人是谁。因此,在这个场景中我们还需要将 A 向 B 买车的这笔交易信息共享给第三方,也就是车管所。

Ledger

(如图为 Corda 的账本模型,A 节点 和 B 节点 之间的交易信息,只会在 A 和 B 各自的账本中是可见的。)

在默认情况下,Corda 是不会将交易信息共享给与该笔交易无关的节点的,这也是 Corda 和其他 DLT 技术相比很大的特点。但这并不意味着 Corda 就没有能力实现这件事了,实际上,Corda 可以在网络中的节点之间发送任何内容。Corda 的这个能力也在其官方文档中有所提及。本文就在此基础上实现 Corda 中的广播。

首先,还是先以上面提到的场景为例,当 A 向 B 买了一辆车后,将该交易信息共享给车管所。我们需要在 BuyVehicleFlow 的最后,调用 subFlow 将交易信息共享给车管所。

object BuyVehicleFlow {
    @InitiatingFlow
    @StartableByRPC
    class Initiator(
        private val vin: String
    ) {
        @Suspendable
        override fun call(): SignedTransaction {
            ...... // 省略无关逻辑 
            val finalTx = subFlow(FinalityFlow(allSignedTransaction))
            val vehicleAdministrator = serviceHub.identityService.partiesFromName("VehicleAdministrator", false).single()
            subFlow(ReportToVehicleAdministratorFlow(vehicleAdministrator, finalTx))
            return finalTx
        }
    }

这里 ReportToVehicleAdministratorFlow 就是为了向车管所 (vehicleAdministrator) 共享该笔交易信息的。那么这个 Flow 的具体实现是什么呢。

@InitiatingFlow
class ReportToVehicleAdministratorFlow(
  private val regulator: Party, 
  private val finalTx: SignedTransaction
) : FlowLogic<Unit>() {
  @Suspendable
  override fun call() {
    subFlow(SendTransactionFlow(initiateFlow(regulator), finalTx))
  }
}

稍微看下这个 flow 就会发现,Corda 几乎做了所有的事情,我们只用告诉 Corda 要向哪个节点 (regulator) 共享哪笔交易的信息 (finalTx) 就好了。在该 flow 中我们和 regulator 建立了一个 Session 连接,并调用了 SendTransactionFlow 发送交易信息。因此,我们还需要相应的响应者 (responder) 来接收并处理 (ReceiveTransactionFlow) 该交易。如下:

@InitiatedBy(ReportToVehicleAdministratorFlow::class)
class ReceiveVehicleAdministratorReportFlow(
  private val otherSideSession: FlowSession
) : FlowLogic<Unit>() {
  @Suspendable
  override fun call() {
    subFlow(ReceiveTransactionFlow(
      otherSideSession = otherSideSession,
      checkSufficientSignatures = true,
      statesToRecord = StatesToRecord.ALL_VISIBLE
    ))
  }
}

ReceiveTransactionFlowSendTransactionFlow 是相对应的,它接收并保存发送给它的交易信息。statesToRecord 属性确定该笔交易中哪些资产 (state) 需要被记录在保险库 (Vault) 中,在该例子中我们希望车管所记录该笔交易中所有的资产信息,所以我们将 statesToRecord 设置为 StatesToRecord.ALL_VISIBLE。此外,根据不同的情景,我们也可以设置响应者存储不同的资产信息,只需将其设置为 StatesToRecord.ONLY_RELEVANT 甚至是 StatesToRecord.NONE 即可。而对于 checkSufficientSignatures 属性,它的目的则是检查分享的这一笔交易中是否所有的参与方都进行了签名,一般都是需要检查的,所有就设置为默认的 true 就好。

但其实,到此为止还只是实现了共享某一笔交易信息给某一个节点,要先实现广播,至少要先能够同时分享交易给多个节点吧,如下:

@InitiatingFlow
class BroadcastTransactionFlow(
  private val recipients: List<Party>,
  private val stx: SignedTransaction
) : FlowLogic<Unit>() {

  @Suspendable
  override fun call() {
    for (recipient in recipients) {
      val session = initiateFlow(recipient)
      subFlow(SendTransactionFlow(session, stx))
    }
  }
}

@InitiatedBy(BroadcastTransactionFlow::class)
class BroadcastTransactionResponder(private val session: FlowSession) : FlowLogic<Unit>() {

  @Suspendable
  override fun call() {
    subFlow(ReceiveTransactionFlow(session, statesToRecord = StatesToRecord.ALL_VISIBLE))
  }
}

稍微看一下 BroadcastTransactionFlow 就会发现,我们其实并没有改多少代码,只是将一个 regulator 改为了 recipients: List<Party>,这样就可以循环分享某一笔交易信息给指定的一些节点了。再进一步,能不能分享给全网所有的节点呢,其实也是可以的,如下:

subFlow(FinalityFlow(stx, sessions)).also {
  // sends to everynode in the corda network
  val broadcastToParties =
    serviceHub.networkMapCache.allNodes.map { 
      node -> node.legalIdentities.first() 
    } - message.recipient - message.sender
  subFlow(BroadcastTransactionFlow(broadcastToParties, it))
}

可以看到这里我们只是从 serviceHub 中找到所有的节点 allNodes,减去这笔交易的发送者 (message.sender) 和接受者 (message.recipient) 不用发送之外,均广播给全网中其他的所有节点。

到了这里,我们已经完成了一个简易版的 Corda 广播实现。但在这里我们还需要额外注意一个问题,我们所广播到的每一个节点都是有权利使用我们分享给它的这笔交易中的资产的,所以我们在做与此类问题相关的设计时,即需要共享交易信息给非参与交易方时,需要额外考虑到这一点,是否需要加入必要的资产保护措施。尽管 Corda 是十分灵活的,但我们一定要对这种灵活性加以控制,以防出现我们所不期望的结果。

参考文档:
· https://docs.corda.net/tutorial-observer-nodes.html
· https://docs.corda.net/key-concepts-ledger.html

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

推荐阅读更多精彩内容