成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Dapp開發(fā)教程三 Asch Dapp Mini DAO

hsluoyz / 1432人閱讀

摘要:前邊兩篇教程可以稱之為熱身,從這里開始,進入正題。這一次,我們要正式創(chuàng)建新的交易類型或者智能合約了。這個的功能是對賬戶進行操作,這個操作包括對數(shù)字的加減法數(shù)組的增刪字符串的設(shè)置等。

前邊兩篇教程可以稱之為熱身,從這里開始,進入正題。 這一次,我們要正式創(chuàng)建新的交易類型或者智能合約了。

1 創(chuàng)建合約

首先要進入dapp所在目錄

cd dapps//

然后執(zhí)行asch-cli的contract子命令

asch-cli contract -a

接下來會提示輸入合約的名字,這里輸入的是"Project"

? Contract file name (without .js) Project
New contract created: ./contracts/Project.js
Updating contracts list
Done

這個命令會幫我們做三件事

新增了合約模板文件modules/contracts/Project.js
在modules/helper/transaction-types.js注冊了交易類型
在modules.full.json中注冊了新的模塊
2 定義實體字段

在實現(xiàn)一個智能合約之前,需要定義好合約執(zhí)行后生成的交易數(shù)據(jù)實體,即最終存儲到區(qū)塊鏈上的是哪些數(shù)據(jù),也就是相當(dāng)于創(chuàng)建關(guān)系數(shù)據(jù)的表格 一個合約類型對應(yīng)一張表格 表格的schema在blockchain.json中進行配置

project類型比較簡單,只包含name和description字段 另外transactionId字段是每個實體表格都需要的,是作為基礎(chǔ)交易transactions的外鍵。

{
        "table": "asset_project",
        "alias": "t_p",
        "type": "table",
        "tableFields": [
            {
                "name": "name",
                "type": "String",
                "length": 16,
                "not_null": true
            },
            {
                "name": "description",
                "type": "Text",
                "not_null": true
            },
            {
                "name": "transactionId",
                "type": "String",
                "length": 21,
                "not_null": true
            }
        ],
        "foreignKeys": [
            {
                "field": "transactionId",
                "table": "transactions",
                "table_field": "id",
                "on_delete": "cascade"
            }
        ]
    }

然后需要在join字段種加入新的配置,還是為了聯(lián)合查詢以及序列化和反序列化時使用

{
                "type": "left outer",
                "table": "asset_project",
                "alias": "t_p",
                "on": {
                    "t.id": "t_p.transactionId"
                }
            }

將來,asch會把這些配置通過自動化的方式生成,開發(fā)者只需要輸入實體字段的名稱和類型即可。

3 實現(xiàn)合約接口

一個合約包含如下接口,有的必須要實現(xiàn),有個則使用默認(rèn)生成的代碼即可

create              # 創(chuàng)建一個交易的數(shù)據(jù)對象,主要是賦值操作
calculateFee        # 設(shè)置交易費,即生成一次交易需要消耗的XAS數(shù)量
verify              # 驗證交易數(shù)據(jù),比如字段是否合法,依賴條件是否滿足等
getBytes            # 返回交易的二進制數(shù)據(jù),類型為Buffer
apply               # 合約的執(zhí)行邏輯,在區(qū)塊打包時調(diào)用,主要是分配和轉(zhuǎn)移交易涉及到的各個賬戶的資產(chǎn),以及賬戶其他字段的設(shè)置等
undo                # apply的相反操作,在區(qū)塊回滾時會調(diào)用
applyUnconfirmed    # 合約的預(yù)執(zhí)行邏輯,與apply類似,但是這個會實時的調(diào)用,就是說區(qū)塊打包前就會調(diào)用,因此涉及到的賬戶操作都是臨時、未確認(rèn)的
undoUnconfirmed     # applyUnconfirmed的相反操作,回滾時使用
ready               # 交易是否準(zhǔn)備完畢,是否滿足打包的條件,這是個高級功能,大部分情況都不需要,以后會多帶帶講解
save                # 交易數(shù)據(jù)的序列化操作,就是將json字段映射到數(shù)據(jù)庫表格字段
dbRead              # 交易的反序列化操作,將數(shù)據(jù)庫表格字段映射到j(luò)son字段
normalize           # 交易數(shù)據(jù)的格式化,把不相關(guān)的對象字段刪除,相關(guān)的對象統(tǒng)一類型,一般情況不需要

上面的接口大部分情況下使用默認(rèn)的就可以了 開發(fā)者需要注意的主要是apply和applyUnconfirmed兩個接口,這是業(yè)務(wù)邏輯的主體部分。

4 實現(xiàn)Project合約

實現(xiàn)create

    trs.recipientId = null;
  // 創(chuàng)建項目只需要發(fā)起者,不需要接收者,所以設(shè)為null

    trs.amount = 0;
  // 也不需要金額,只需要手續(xù)費

    trs.asset.project = {
        name: data.name,
        description: data.description
    }
  // project對象的兩個數(shù)據(jù)字段

    return trs;

設(shè)置交易費

這個項目不希望與XAS對接,那么就把交易費設(shè)置為0就行了

Project.prototype.calculateFee = function (trs) {
    return 0;
}

數(shù)據(jù)檢驗

這個沒啥可解釋的

Project.prototype.verify = function (trs, sender, cb, scope) {
    if (trs.recipientId) {
        return cb("Recipient should not exist");
    }
    if (trs.amount != 0) {
        return cb("Amount should be zero");
    }
    if (!trs.asset.project.name) {
        return cb("Project must have a name");
    }
    if (trs.asset.project.name.length > 16) {
        return cb("Project name must be 16 characters or less");
    }
    if (!trs.asset.project.description) {
        return cb("Invalid project description");
    }
    if (trs.asset.project.description.length > 1024) {
        return cb("Project description must be 1024 characters or less");
    }
    cb(null, trs);
}

獲取二進制數(shù)據(jù)

二進制數(shù)據(jù)主要是為了生成簽名數(shù)據(jù),所以只需要把交易的實體數(shù)據(jù)組合起來打包成Buffer就可以了。 組合的方式可以隨便,比如,可以通過bytebuffer,也可以通過簡單的字符串連接。

Project.prototype.getBytes = function (trs) {
    try {
        var buf = new Buffer(trs.asset.project.name + trs.asset.project.description, "utf8");
    } catch (e) {
        throw Error(e.toString());
    }

    return buf;
}

合約執(zhí)行邏輯

先看未確認(rèn)合約的執(zhí)行

Project.prototype.applyUnconfirmed = function (trs, sender, cb, scope) {
    if (sender.u_balance["POINTS"] < BURN_POINTS) {
        return setImmediate(cb, "Account does not have enough POINTS: " + trs.id);
    }
    if (private.uProjects[trs.asset.project.name]){
        return setImmediate(cb, "Project already exists");
    }
    modules.blockchain.accounts.mergeAccountAndGet({
        address: sender.address,
        u_balance: { "POINTS": -BURN_POINTS }
    }, function (err, accounts) {
        if (!err) {
            private.uProjects[trs.asset.project.name] = trs;
        }
        cb(err, accounts);
    }, scope);
}

在這一步,檢查用戶的余額是否足夠,否則拒絕執(zhí)行, 接著判斷是否已經(jīng)存在相同的項目名稱, 最后會看到一個dapp開發(fā)中最重要的api,即modules.blockchain.accounts.mergeAccountAndGet。

這個api的功能是對賬戶進行操作,這個操作包括對數(shù)字的加減法、數(shù)組的增刪、字符串的設(shè)置等。 這里對賬戶余額執(zhí)行了減法操作,即把u_balance中的POINTS資產(chǎn),減去BURN_POINTS。 這里取名BURN_POINTS主要是為了表達(dá)這個合約的執(zhí)行需要燃燒一定數(shù)量的資產(chǎn),因為沒有指定被消耗掉的資產(chǎn)的去向,那么這些被消耗的資產(chǎn)就只有消失了,也就是被燃燒了。 這里只是為了簡單起見,如果業(yè)務(wù)邏輯不希望燃燒,可以把這些資產(chǎn)作為手續(xù)費,轉(zhuǎn)給應(yīng)用的開發(fā)者或者節(jié)點運營者,或者轉(zhuǎn)移到一個基金賬戶中,用作將來的開發(fā)經(jīng)費,完全由你自己決定。

接下來再看看確認(rèn)合約的執(zhí)行代碼

Project.prototype.apply = function (trs, sender, cb, scope) {
    modules.blockchain.accounts.mergeAccountAndGet({
        address: sender.address,
        balance: {"POINTS": -BURN_POINTS}
    }, cb, scope);
}

非常簡單,只有一個操作,僅僅是對賬戶資產(chǎn)進行一個減法操作。 大部分情況下, applyUnconfirmed是比apply要復(fù)雜的,特別是涉及到資產(chǎn)的減法操作時,因為前者要比后者執(zhí)行的更早,后者就沒必要做多余的條件檢查了。 我們要注意到,apply修改的是balance字段,applyUnconfirmed修改的是u_balance字段,

所以如果u_balance滿足條件(即有足夠的剩余資產(chǎn)),那么balance一定也會滿足條件,所以就沒必要進行進一步檢查了。

接下來的save, dbRead就沒必要解釋了,開發(fā)者可以自己發(fā)現(xiàn)其中的規(guī)律,直接套用即可。

5 實現(xiàn)http接口

在上一個步驟,已經(jīng)定義了一個project合約的所有邏輯了。 在這一步,我們需要增加兩個接口,都是為客戶端或前端服務(wù)的,一個是用于創(chuàng)建交易,一個是用于查詢交易歷史。

幾乎所有的交易創(chuàng)建都是類似的,一般可以分解成一下幾步

使用客戶端傳過來的secret生成密鑰對keypair
使用公鑰查詢或新建賬戶數(shù)據(jù),通過api modules.blockchain.accounts.getAccount
然后使用客戶端傳過來的交易實體數(shù)據(jù)和賬戶數(shù)據(jù)以及密鑰對,創(chuàng)建一個交易對象,通過api modules.logic.transaction.create
最后是調(diào)用api modules.blockchain.transactions.processUnconfirmedTransaction來處理這個交易
有一點需要注意的是library.sequence.add接口的使用,這個接口可以保證多個交易按先后順序嚴(yán)格執(zhí)行,如果你的合約邏輯中涉及到異步操作,應(yīng)該要使用這個api。

再來看一下list這個查詢接口,熟悉sql的同學(xué)一眼就看出,這只不過是個聯(lián)表查詢操作。

為什么要聯(lián)表查詢呢?

因為transactions和asset_xxx表示的是一個交易的不同部分,前者是數(shù)據(jù)的基礎(chǔ)數(shù)據(jù),所有交易都通用,比如交易的發(fā)起者,交易數(shù)據(jù)的簽名,金額等等, 后者則屬于交易數(shù)據(jù)的擴展部分,是用戶自定義的數(shù)據(jù),與具體的業(yè)務(wù)邏輯相關(guān)。

6 實現(xiàn)投票合約

這個就不逐行解釋了,開發(fā)者可以自己研究asch-mini-dao的源碼,有了上面的基礎(chǔ)后,不難理解。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/81333.html

相關(guān)文章

  • Dapp開發(fā)教程 Asch Dapp Mini DAO

    摘要:前邊兩篇教程可以稱之為熱身,從這里開始,進入正題。這一次,我們要正式創(chuàng)建新的交易類型或者智能合約了。這個的功能是對賬戶進行操作,這個操作包括對數(shù)字的加減法數(shù)組的增刪字符串的設(shè)置等。 前邊兩篇教程可以稱之為熱身,從這里開始,進入正題。 這一次,我們要正式創(chuàng)建新的交易類型或者智能合約了。 1 創(chuàng)建合約 首先要進入dapp所在目錄 cd dapps// 然后執(zhí)行asch-cli的contr...

    qqlcbb 評論0 收藏0
  • Dapp開發(fā)教程Asch Dapp Dice Game

    摘要:但是我覺得在原理上與上一個項目相比,并沒有什么不同。源碼是最好的老師。 這個dice game與上一個mini dao相比,代碼規(guī)模大了許多,功能也復(fù)雜了很多,創(chuàng)建了三個合約類型,彼此之間有依賴關(guān)系,合約的執(zhí)行還要依賴歷史交易數(shù)據(jù)。 但是我覺得在原理上與上一個項目相比,并沒有什么不同。 源碼是最好的老師。

    xiaowugui666 評論0 收藏0
  • Dapp開發(fā)教程Asch Dapp Dice Game

    摘要:但是我覺得在原理上與上一個項目相比,并沒有什么不同。源碼是最好的老師。 這個dice game與上一個mini dao相比,代碼規(guī)模大了許多,功能也復(fù)雜了很多,創(chuàng)建了三個合約類型,彼此之間有依賴關(guān)系,合約的執(zhí)行還要依賴歷史交易數(shù)據(jù)。 但是我覺得在原理上與上一個項目相比,并沒有什么不同。 源碼是最好的老師。

    mtunique 評論0 收藏0
  • Dapp開發(fā)教程Asch Dapp Hello World

    摘要:基本流程有三種,,,,后兩種是發(fā)布到線上的,可通過公網(wǎng)訪問。第一種是運行在本地的只有一個節(jié)點的私鏈,主要是為了方便本地測試和開發(fā)。 1 基本流程 Asch有三種net,localnet,testnet,mainnet,后兩種是發(fā)布到線上的,可通過公網(wǎng)訪問。 第一種localnet是運行在本地的、只有一個節(jié)點的私鏈,主要是為了方便本地測試和開發(fā)。 Dapp的開發(fā)同樣要涉及到這三種網(wǎng)絡(luò),即...

    maxmin 評論0 收藏0
  • Dapp開發(fā)教程Asch Dapp Asset

    摘要:前一篇文章介紹了開發(fā)的基本流程,這一次打算創(chuàng)建一個擁有內(nèi)置資產(chǎn)的,并順便介紹下前后端通訊的協(xié)議和常用接口。我們的程序目前只能創(chuàng)建一種內(nèi)置資產(chǎn),如果有創(chuàng)建多種資產(chǎn)的需求,我們可以考慮開發(fā)。 前一篇文章介紹了asch dapp開發(fā)的基本流程,這一次打算創(chuàng)建一個擁有內(nèi)置資產(chǎn)的dapp,并順便介紹下前后端通訊的協(xié)議和常用接口。 1 創(chuàng)建一個帶內(nèi)置資產(chǎn)的dapp 其實這篇文章有些標(biāo)題黨,因為創(chuàng)建...

    VPointer 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<