运行迁移

运行迁移

迁移脚本是使用 JavaScript 编写的文件,用于帮助你发布智能合约到以太坊网络。
这些文件的职责就是分阶段的部署任务。所有历史的迁移都会保存在一个特殊的 Migrations
智能合约中。详细如下:

命令行

执行迁移命令:

$ truffle migrate

这个命令将会执行所有在 migrations 目录下的迁移操作。迁移操作只会执行新添加的迁移命令,如果没有新添加的迁移任务将不会有任何变化。

迁移文件

一个迁移文件的例子如下:

文件名: 4_example_migration.js

var MyContract = artifacts.require("MyContract");

module.exports = function(deployer) {
  // deployment steps
  deployer.deploy(MyContract);
};

文件名的前缀是数字开头,这个数字就是迁移脚本执行的顺序,文件名的后半部分尽量让人能读懂是什么操作。

ARTIFACTS.REQUIRE()

在迁移脚本的开头我们可以使用 artifacts.require() 方法来告诉 truffle 我们要交互的智能合约。
这个方法和 Node.jsrequire 方法类似,但是在我们的示例中,它特别返回一个抽象合约,
我们可以在部署脚本的其余部分中使用它。

如果一个 .sol 文件中有多个合约,我们应该这样引用合约:

// filename: Contracts.sol

contract ContractOne{
    // ...
}

contract ContractTwo{
    // ...
}
var ContractOne = artifacts.require("ContractOne");
var ContractTwo = artifacts.require("ContractTwo");

module.exports

所有的迁移操作必须导出 module.exports 。迁移脚本中的 exports 的方法中至少包含一个参数 deployer
还有一些额外的参数可配置,下面会详细讲解。

初始化迁移操作

为了使用迁移特性 Truffle 需要你提供一个 Migrations 合约。这个合约必须包含一个特殊的接口,
你也可以根据你的意愿修改这个合约。对于大多数工程而言,这个合约在初始化的时候迁移一次,然后不会再更新。
当你创建一个 truffle init 空工程的时候,truffle 也会为你创建这个合约。

文件名: contracts/Migrations.sol

pragma solidity ^0.4.8;

contract Migrations {
  address public owner;

  // A function with the signature `last_completed_migration()`, returning a uint, is required.
  uint public last_completed_migration;

  modifier restricted() {
    if (msg.sender == owner) _;
  }

  function Migrations() {
    owner = msg.sender;
  }

  // A function with the signature `setCompleted(uint)` is required.
  function setCompleted(uint completed) restricted {
    last_completed_migration = completed;
  }

  function upgrade(address new_address) restricted {
    Migrations upgraded = Migrations(new_address);
    upgraded.setCompleted(last_completed_migration);
  }
}

您必须在第一次迁移中部署此合约,以便利用迁移特性。为此,创建以下迁移:

文件名: migrations/1_initial_migration.js

var Migrations = artifacts.require("Migrations");

module.exports = function(deployer) {
  // Deploy the Migrations contract as our only task
  deployer.deploy(Migrations);
};

从这里开始,您可以创建带有编号前缀的新迁移,以部署其他契约并执行进一步的部署步骤。
部署流程会根据部署脚本的前缀的数字依次执行。

DEPLOYER

您的迁移文件将使用部署程序执行部署任务。因此,您可以同步地编写部署任务,它们将按照正确的顺序执行:

// Stage deploying A before B
deployer.deploy(A);
deployer.deploy(B);

另外,部署人员上的每个功能都可以用 Promise ,根据前一个任务的执行对部署任务进行排队:

// Deploy A, then deploy B, passing in A's newly deployed address
deployer.deploy(A).then(function() {
  return deployer.deploy(B, A.address);
});

如果您发现语法更清晰的话,可以将部署编写为一个单一的 Promise 链。部署程序API在本页的底部进行了讨论。

考虑网络配置

可以根据不同的网络进行部署操作。这是一个高级特性,所以在继续之前先看一下网络部分。

为了条件部署,在写迁移脚本的时候要接受第二个参数,称为 network

例子:

module.exports = function(deployer, network) {
  if (network == "live") {
    // Do something specific to the network named "live".
  } else {
    // Perform a different step otherwise.
  }
}

可用账户

迁移也会通过以太坊客户端和 web3 提供的帐户列表,供您在部署过程中使用。这和 web3.eth.getAccounts() 返回的相同的帐户列表。

module.exports = function(deployer, network, accounts) {
  // Use the accounts within your migrations.
}

DEPLOYER API

deployer 有很多功能可以简化你的迁移操作。

DEPLOYER.DEPLOY(CONTRACT, ARGS…, OPTIONS)

使用可选的构造函数参数部署合约。这对于单例合约很有用,这类合约只存在一个 dapp 的实例。
部署后会重新设置的合约地址(即: contact.address 等于新部署的地址),并且覆盖之前存储的地址。

你可以选择传递一组合约或数组,以加速部署多个合约。另外,最后一个参数是一个可选对象,它可以包含名为 overwrite 和其他交易参数,如 gas 和 from 。
overwrite 被设置为false,同时已经部署了该合约,deployer 将不会重新部署该合约。这对于由外部依赖提供合约地址的某些情况很有用。

在调用 deploy 命令发布你的合约之前你需要发布并连接依赖的库。下面会详细介绍 link 方法。

更多信息查看 truffle-contact 文档

例子:

// Deploy a single contract without constructor arguments
deployer.deploy(A);

// Deploy a single contract with constructor arguments
deployer.deploy(A, arg1, arg2, ...);

// Don't deploy this contract if it has already been deployed
deployer.deploy(A, {overwrite: false});

// Set a maximum amount of gas and `from` address for the deployment
deployer.deploy(A, {gas: 4612388, from: "0x...."});

// Deploy multiple contracts, some with arguments and some without.
// This is quicker than writing three `deployer.deploy()` statements as the deployer
// can perform the deployment as a single batched request.
deployer.deploy([
  [A, arg1, arg2, ...],
  B,
  [C, arg1]
]);

// External dependency example:
//
// For this example, our dependency provides an address when we're deploying to the
// live network, but not for any other networks like testing and development.
// When we're deploying to the live network we want it to use that address, but in
// testing and development we need to deploy a version of our own. Instead of writing
// a bunch of conditionals, we can simply use the `overwrite` key.
deployer.deploy(SomeDependency, {overwrite: false});

DEPLOYER.LINK(LIBRARY, DESTINATIONS)

连接一个已经发布的库合约或者多个合约。 destinations 可以是一个或者多个合约。如果任何一个合约的 destination
都没有依赖已经连接的库合约,那么这个合约库将被忽略。

例子:

// Deploy library LibA, then link LibA to contract B, then deploy B.
deployer.deploy(LibA);
deployer.link(LibA, B);
deployer.deploy(B);

// Link LibA to many contracts
deployer.link(LibA, [B, C, D]);

DEPLOYER.THEN(FUNCTION() {…})

Promise 类似,用来运行部署步骤。使用它在迁移期间调用特定的合约函数来添加、编辑和重组合约数据。

例子:

var a, b;
deployer.then(function() {
  // Create a new version of A
  return A.new();
}).then(function(instance) {
  a = instance;
  // Get the deployed instance of B
  return B.deployed();
}).then(function(instance) {
  b = instance;
  // Set the new instance of A's address on B via B's setA() function.
  return b.setA(a.address);
});
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容