|
@@ -4,10 +4,7 @@ const {
|
|
|
shouldBehaveLikeERC777DirectSendBurn,
|
|
|
shouldBehaveLikeERC777OperatorSendBurn,
|
|
|
shouldBehaveLikeERC777UnauthorizedOperatorSendBurn,
|
|
|
- shouldDirectSendTokens,
|
|
|
- shouldDirectBurnTokens,
|
|
|
shouldBehaveLikeERC777InternalMint,
|
|
|
- shouldInternalMintTokens,
|
|
|
shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook,
|
|
|
shouldBehaveLikeERC777SendBurnWithSendHook,
|
|
|
} = require('./ERC777.behavior');
|
|
@@ -30,419 +27,367 @@ contract('ERC777', function ([
|
|
|
this.erc1820 = await singletons.ERC1820Registry(registryFunder);
|
|
|
});
|
|
|
|
|
|
- it('reverts with a granularity of zero', async function () {
|
|
|
- await shouldFail.reverting(ERC777.new(holder, initialSupply, name, symbol, 0, []));
|
|
|
- });
|
|
|
+ context('with default operators', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.token = await ERC777.new(holder, initialSupply, name, symbol, defaultOperators);
|
|
|
+ });
|
|
|
|
|
|
- context('with a granularity of one', function () {
|
|
|
- const granularity = new BN('1');
|
|
|
+ it.skip('does not emit AuthorizedOperator events for default operators', async function () {
|
|
|
+ expectEvent.not.inConstructor(this.token, 'AuthorizedOperator'); // This helper needs to be implemented
|
|
|
+ });
|
|
|
|
|
|
- context('with default operators', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.token = await ERC777.new(holder, initialSupply, name, symbol, granularity, defaultOperators);
|
|
|
+ describe('basic information', function () {
|
|
|
+ it('returns the name', async function () {
|
|
|
+ (await this.token.name()).should.equal(name);
|
|
|
});
|
|
|
|
|
|
- it.skip('does not emit AuthorizedOperator events for default operators', async function () {
|
|
|
- expectEvent.not.inConstructor(this.token, 'AuthorizedOperator'); // This helper needs to be implemented
|
|
|
+ it('returns the symbol', async function () {
|
|
|
+ (await this.token.symbol()).should.equal(symbol);
|
|
|
});
|
|
|
|
|
|
- describe('basic information', function () {
|
|
|
- it('returns the name', async function () {
|
|
|
- (await this.token.name()).should.equal(name);
|
|
|
- });
|
|
|
+ it('has a granularity of 1', async function () {
|
|
|
+ (await this.token.granularity()).should.be.bignumber.equal('1');
|
|
|
+ });
|
|
|
+
|
|
|
+ it('returns the default operators', async function () {
|
|
|
+ (await this.token.defaultOperators()).should.deep.equal(defaultOperators);
|
|
|
+ });
|
|
|
+
|
|
|
+ it('default operators are operators for all accounts', async function () {
|
|
|
+ for (const operator of defaultOperators) {
|
|
|
+ (await this.token.isOperatorFor(operator, anyone)).should.equal(true);
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ it('returns thte total supply', async function () {
|
|
|
+ (await this.token.totalSupply()).should.be.bignumber.equal(initialSupply);
|
|
|
+ });
|
|
|
|
|
|
- it('returns the symbol', async function () {
|
|
|
- (await this.token.symbol()).should.equal(symbol);
|
|
|
+ it('is registered in the registry', async function () {
|
|
|
+ (await this.erc1820.getInterfaceImplementer(this.token.address, web3.utils.soliditySha3('ERC777Token')))
|
|
|
+ .should.equal(this.token.address);
|
|
|
+ });
|
|
|
+ });
|
|
|
+
|
|
|
+ describe('balanceOf', function () {
|
|
|
+ context('for an account with no tokens', function () {
|
|
|
+ it('returns zero', async function () {
|
|
|
+ (await this.token.balanceOf(anyone)).should.be.bignumber.equal('0');
|
|
|
});
|
|
|
+ });
|
|
|
|
|
|
- it('returns the granularity', async function () {
|
|
|
- (await this.token.granularity()).should.be.bignumber.equal(granularity);
|
|
|
+ context('for an account with tokens', function () {
|
|
|
+ it('returns their balance', async function () {
|
|
|
+ (await this.token.balanceOf(holder)).should.be.bignumber.equal(initialSupply);
|
|
|
});
|
|
|
+ });
|
|
|
+ });
|
|
|
|
|
|
- it('returns the default operators', async function () {
|
|
|
- (await this.token.defaultOperators()).should.deep.equal(defaultOperators);
|
|
|
+ context('with no ERC777TokensSender and no ERC777TokensRecipient implementers', function () {
|
|
|
+ describe('send/burn', function () {
|
|
|
+ shouldBehaveLikeERC777DirectSendBurn(holder, anyone, data);
|
|
|
+
|
|
|
+ context('with self operator', function () {
|
|
|
+ shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, holder, data, operatorData);
|
|
|
});
|
|
|
|
|
|
- it('default operators are operators for all accounts', async function () {
|
|
|
- for (const operator of defaultOperators) {
|
|
|
- (await this.token.isOperatorFor(operator, anyone)).should.equal(true);
|
|
|
- }
|
|
|
+ context('with first default operator', function () {
|
|
|
+ shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, defaultOperatorA, data, operatorData);
|
|
|
});
|
|
|
|
|
|
- it('returns thte total supply', async function () {
|
|
|
- (await this.token.totalSupply()).should.be.bignumber.equal(initialSupply);
|
|
|
+ context('with second default operator', function () {
|
|
|
+ shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, defaultOperatorB, data, operatorData);
|
|
|
});
|
|
|
|
|
|
- it('is registered in the registry', async function () {
|
|
|
- (await this.erc1820.getInterfaceImplementer(this.token.address, web3.utils.soliditySha3('ERC777Token')))
|
|
|
- .should.equal(this.token.address);
|
|
|
+ context('before authorizing a new operator', function () {
|
|
|
+ shouldBehaveLikeERC777UnauthorizedOperatorSendBurn(holder, anyone, newOperator, data, operatorData);
|
|
|
});
|
|
|
- });
|
|
|
|
|
|
- describe('balanceOf', function () {
|
|
|
- context('for an account with no tokens', function () {
|
|
|
- it('returns zero', async function () {
|
|
|
- (await this.token.balanceOf(anyone)).should.be.bignumber.equal('0');
|
|
|
+ context('with new authorized operator', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
});
|
|
|
- });
|
|
|
|
|
|
- context('for an account with tokens', function () {
|
|
|
- it('returns their balance', async function () {
|
|
|
- (await this.token.balanceOf(holder)).should.be.bignumber.equal(initialSupply);
|
|
|
+ shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, newOperator, data, operatorData);
|
|
|
+
|
|
|
+ context('with revoked operator', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ await this.token.revokeOperator(newOperator, { from: holder });
|
|
|
+ });
|
|
|
+
|
|
|
+ shouldBehaveLikeERC777UnauthorizedOperatorSendBurn(holder, anyone, newOperator, data, operatorData);
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
- context('with no ERC777TokensSender and no ERC777TokensRecipient implementers', function () {
|
|
|
- describe('send/burn', function () {
|
|
|
- shouldBehaveLikeERC777DirectSendBurn(holder, anyone, data);
|
|
|
+ describe('mint (internal)', function () {
|
|
|
+ const to = anyone;
|
|
|
+ const amount = new BN('5');
|
|
|
|
|
|
- context('with self operator', function () {
|
|
|
- shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, holder, data, operatorData);
|
|
|
- });
|
|
|
+ context('with default operator', function () {
|
|
|
+ const operator = defaultOperatorA;
|
|
|
|
|
|
- context('with first default operator', function () {
|
|
|
- shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, defaultOperatorA, data, operatorData);
|
|
|
- });
|
|
|
+ shouldBehaveLikeERC777InternalMint(to, operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
|
|
|
- context('with second default operator', function () {
|
|
|
- shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, defaultOperatorB, data, operatorData);
|
|
|
- });
|
|
|
+ context('with non operator', function () {
|
|
|
+ const operator = newOperator;
|
|
|
|
|
|
- context('before authorizing a new operator', function () {
|
|
|
- shouldBehaveLikeERC777UnauthorizedOperatorSendBurn(holder, anyone, newOperator, data, operatorData);
|
|
|
- });
|
|
|
+ shouldBehaveLikeERC777InternalMint(to, operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
+ });
|
|
|
+ });
|
|
|
|
|
|
- context('with new authorized operator', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
- });
|
|
|
+ describe('operator management', function () {
|
|
|
+ it('accounts are their own operator', async function () {
|
|
|
+ (await this.token.isOperatorFor(holder, holder)).should.equal(true);
|
|
|
+ });
|
|
|
|
|
|
- shouldBehaveLikeERC777OperatorSendBurn(holder, anyone, newOperator, data, operatorData);
|
|
|
+ it('reverts when self-authorizing', async function () {
|
|
|
+ await shouldFail.reverting(this.token.authorizeOperator(holder, { from: holder }));
|
|
|
+ });
|
|
|
|
|
|
- context('with revoked operator', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- await this.token.revokeOperator(newOperator, { from: holder });
|
|
|
- });
|
|
|
+ it('reverts when self-revoking', async function () {
|
|
|
+ await shouldFail.reverting(this.token.revokeOperator(holder, { from: holder }));
|
|
|
+ });
|
|
|
|
|
|
- shouldBehaveLikeERC777UnauthorizedOperatorSendBurn(holder, anyone, newOperator, data, operatorData);
|
|
|
- });
|
|
|
- });
|
|
|
- });
|
|
|
+ it('non-operators can be revoked', async function () {
|
|
|
+ (await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
|
|
|
- describe('mint (internal)', function () {
|
|
|
- const to = anyone;
|
|
|
- const amount = new BN('5');
|
|
|
+ const { logs } = await this.token.revokeOperator(newOperator, { from: holder });
|
|
|
+ expectEvent.inLogs(logs, 'RevokedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
|
|
|
- context('with default operator', function () {
|
|
|
- const operator = defaultOperatorA;
|
|
|
+ (await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
+ });
|
|
|
|
|
|
- shouldBehaveLikeERC777InternalMint(to, operator, amount, data, operatorData);
|
|
|
- });
|
|
|
+ it('non-operators can be authorized', async function () {
|
|
|
+ (await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
|
|
|
- context('with non operator', function () {
|
|
|
- const operator = newOperator;
|
|
|
+ const { logs } = await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
+ expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
|
|
|
- shouldBehaveLikeERC777InternalMint(to, operator, amount, data, operatorData);
|
|
|
- });
|
|
|
- });
|
|
|
+ (await this.token.isOperatorFor(newOperator, holder)).should.equal(true);
|
|
|
});
|
|
|
|
|
|
- describe('operator management', function () {
|
|
|
- it('accounts are their own operator', async function () {
|
|
|
- (await this.token.isOperatorFor(holder, holder)).should.equal(true);
|
|
|
+ describe('new operators', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
});
|
|
|
|
|
|
- it('reverts when self-authorizing', async function () {
|
|
|
- await shouldFail.reverting(this.token.authorizeOperator(holder, { from: holder }));
|
|
|
+ it('are not added to the default operators list', async function () {
|
|
|
+ (await this.token.defaultOperators()).should.deep.equal(defaultOperators);
|
|
|
});
|
|
|
|
|
|
- it('reverts when self-revoking', async function () {
|
|
|
- await shouldFail.reverting(this.token.revokeOperator(holder, { from: holder }));
|
|
|
- });
|
|
|
+ it('can be re-authorized', async function () {
|
|
|
+ const { logs } = await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
+ expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
|
|
|
- it('non-operators can be revoked', async function () {
|
|
|
- (await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
+ (await this.token.isOperatorFor(newOperator, holder)).should.equal(true);
|
|
|
+ });
|
|
|
|
|
|
+ it('can be revoked', async function () {
|
|
|
const { logs } = await this.token.revokeOperator(newOperator, { from: holder });
|
|
|
expectEvent.inLogs(logs, 'RevokedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
|
|
|
(await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
});
|
|
|
+ });
|
|
|
|
|
|
- it('non-operators can be authorized', async function () {
|
|
|
- (await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
+ describe('default operators', function () {
|
|
|
+ it('can be re-authorized', async function () {
|
|
|
+ const { logs } = await this.token.authorizeOperator(defaultOperatorA, { from: holder });
|
|
|
+ expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: defaultOperatorA, tokenHolder: holder });
|
|
|
|
|
|
- const { logs } = await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
- expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
+ (await this.token.isOperatorFor(defaultOperatorA, holder)).should.equal(true);
|
|
|
+ });
|
|
|
|
|
|
- (await this.token.isOperatorFor(newOperator, holder)).should.equal(true);
|
|
|
+ it('can be revoked', async function () {
|
|
|
+ const { logs } = await this.token.revokeOperator(defaultOperatorA, { from: holder });
|
|
|
+ expectEvent.inLogs(logs, 'RevokedOperator', { operator: defaultOperatorA, tokenHolder: holder });
|
|
|
+
|
|
|
+ (await this.token.isOperatorFor(defaultOperatorA, holder)).should.equal(false);
|
|
|
+ });
|
|
|
+
|
|
|
+ it('cannot be revoked for themselves', async function () {
|
|
|
+ await shouldFail.reverting(this.token.revokeOperator(defaultOperatorA, { from: defaultOperatorA }));
|
|
|
});
|
|
|
|
|
|
- describe('new operators', function () {
|
|
|
+ context('with revoked default operator', function () {
|
|
|
beforeEach(async function () {
|
|
|
- await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
+ await this.token.revokeOperator(defaultOperatorA, { from: holder });
|
|
|
});
|
|
|
|
|
|
- it('are not added to the default operators list', async function () {
|
|
|
- (await this.token.defaultOperators()).should.deep.equal(defaultOperators);
|
|
|
+ it('default operator is not revoked for other holders', async function () {
|
|
|
+ (await this.token.isOperatorFor(defaultOperatorA, anyone)).should.equal(true);
|
|
|
});
|
|
|
|
|
|
- it('can be re-authorized', async function () {
|
|
|
- const { logs } = await this.token.authorizeOperator(newOperator, { from: holder });
|
|
|
- expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
-
|
|
|
- (await this.token.isOperatorFor(newOperator, holder)).should.equal(true);
|
|
|
+ it('other default operators are not revoked', async function () {
|
|
|
+ (await this.token.isOperatorFor(defaultOperatorB, holder)).should.equal(true);
|
|
|
});
|
|
|
|
|
|
- it('can be revoked', async function () {
|
|
|
- const { logs } = await this.token.revokeOperator(newOperator, { from: holder });
|
|
|
- expectEvent.inLogs(logs, 'RevokedOperator', { operator: newOperator, tokenHolder: holder });
|
|
|
-
|
|
|
- (await this.token.isOperatorFor(newOperator, holder)).should.equal(false);
|
|
|
+ it('default operators list is not modified', async function () {
|
|
|
+ (await this.token.defaultOperators()).should.deep.equal(defaultOperators);
|
|
|
});
|
|
|
- });
|
|
|
|
|
|
- describe('default operators', function () {
|
|
|
- it('can be re-authorized', async function () {
|
|
|
+ it('revoked default operator can be re-authorized', async function () {
|
|
|
const { logs } = await this.token.authorizeOperator(defaultOperatorA, { from: holder });
|
|
|
expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: defaultOperatorA, tokenHolder: holder });
|
|
|
|
|
|
(await this.token.isOperatorFor(defaultOperatorA, holder)).should.equal(true);
|
|
|
});
|
|
|
+ });
|
|
|
+ });
|
|
|
+ });
|
|
|
|
|
|
- it('can be revoked', async function () {
|
|
|
- const { logs } = await this.token.revokeOperator(defaultOperatorA, { from: holder });
|
|
|
- expectEvent.inLogs(logs, 'RevokedOperator', { operator: defaultOperatorA, tokenHolder: holder });
|
|
|
-
|
|
|
- (await this.token.isOperatorFor(defaultOperatorA, holder)).should.equal(false);
|
|
|
- });
|
|
|
+ describe('send and receive hooks', function () {
|
|
|
+ const amount = new BN('1');
|
|
|
+ const operator = defaultOperatorA;
|
|
|
+ // sender and recipient are stored inside 'this', since in some tests their addresses are determined dynamically
|
|
|
|
|
|
- it('cannot be revoked for themselves', async function () {
|
|
|
- await shouldFail.reverting(this.token.revokeOperator(defaultOperatorA, { from: defaultOperatorA }));
|
|
|
- });
|
|
|
+ describe('tokensReceived', function () {
|
|
|
+ beforeEach(function () {
|
|
|
+ this.sender = holder;
|
|
|
+ });
|
|
|
|
|
|
- context('with revoked default operator', function () {
|
|
|
+ context('with no ERC777TokensRecipient implementer', function () {
|
|
|
+ context('with contract recipient', function () {
|
|
|
beforeEach(async function () {
|
|
|
- await this.token.revokeOperator(defaultOperatorA, { from: holder });
|
|
|
- });
|
|
|
+ this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ this.recipient = this.tokensRecipientImplementer.address;
|
|
|
|
|
|
- it('default operator is not revoked for other holders', async function () {
|
|
|
- (await this.token.isOperatorFor(defaultOperatorA, anyone)).should.equal(true);
|
|
|
+ // Note that tokensRecipientImplementer doesn't implement the recipient interface for the recipient
|
|
|
});
|
|
|
|
|
|
- it('other default operators are not revoked', async function () {
|
|
|
- (await this.token.isOperatorFor(defaultOperatorB, holder)).should.equal(true);
|
|
|
+ it('send reverts', async function () {
|
|
|
+ await shouldFail.reverting(this.token.send(this.recipient, amount, data));
|
|
|
});
|
|
|
|
|
|
- it('default operators list is not modified', async function () {
|
|
|
- (await this.token.defaultOperators()).should.deep.equal(defaultOperators);
|
|
|
+ it('operatorSend reverts', async function () {
|
|
|
+ await shouldFail.reverting(
|
|
|
+ this.token.operatorSend(this.sender, this.recipient, amount, data, operatorData, { from: operator })
|
|
|
+ );
|
|
|
});
|
|
|
|
|
|
- it('revoked default operator can be re-authorized', async function () {
|
|
|
- const { logs } = await this.token.authorizeOperator(defaultOperatorA, { from: holder });
|
|
|
- expectEvent.inLogs(logs, 'AuthorizedOperator', { operator: defaultOperatorA, tokenHolder: holder });
|
|
|
-
|
|
|
- (await this.token.isOperatorFor(defaultOperatorA, holder)).should.equal(true);
|
|
|
+ it('mint (internal) reverts', async function () {
|
|
|
+ await shouldFail.reverting(
|
|
|
+ this.token.mintInternal(operator, this.recipient, amount, data, operatorData)
|
|
|
+ );
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
- });
|
|
|
|
|
|
- describe('send and receive hooks', function () {
|
|
|
- const amount = new BN('1');
|
|
|
- const operator = defaultOperatorA;
|
|
|
- // sender and recipient are stored inside 'this', since in some tests their addresses are determined dynamically
|
|
|
+ context('with ERC777TokensRecipient implementer', function () {
|
|
|
+ context('with contract as implementer for an externally owned account', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ this.recipient = anyone;
|
|
|
|
|
|
- describe('tokensReceived', function () {
|
|
|
- beforeEach(function () {
|
|
|
- this.sender = holder;
|
|
|
- });
|
|
|
+ await this.tokensRecipientImplementer.recipientFor(this.recipient);
|
|
|
|
|
|
- context('with no ERC777TokensRecipient implementer', function () {
|
|
|
- context('with contract recipient', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- this.recipient = this.tokensRecipientImplementer.address;
|
|
|
-
|
|
|
- // Note that tokensRecipientImplementer doesn't implement the recipient interface for the recipient
|
|
|
- });
|
|
|
-
|
|
|
- it('send reverts', async function () {
|
|
|
- await shouldFail.reverting(this.token.send(this.recipient, amount, data));
|
|
|
- });
|
|
|
-
|
|
|
- it('operatorSend reverts', async function () {
|
|
|
- await shouldFail.reverting(
|
|
|
- this.token.operatorSend(this.sender, this.recipient, amount, data, operatorData, { from: operator })
|
|
|
- );
|
|
|
- });
|
|
|
-
|
|
|
- it('mint (internal) reverts', async function () {
|
|
|
- await shouldFail.reverting(
|
|
|
- this.token.mintInternal(operator, this.recipient, amount, data, operatorData)
|
|
|
- );
|
|
|
- });
|
|
|
+ await this.erc1820.setInterfaceImplementer(
|
|
|
+ this.recipient,
|
|
|
+ web3.utils.soliditySha3('ERC777TokensRecipient'), this.tokensRecipientImplementer.address,
|
|
|
+ { from: this.recipient },
|
|
|
+ );
|
|
|
});
|
|
|
- });
|
|
|
-
|
|
|
- context('with ERC777TokensRecipient implementer', function () {
|
|
|
- context('with contract as implementer for an externally owned account', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- this.recipient = anyone;
|
|
|
|
|
|
- await this.tokensRecipientImplementer.recipientFor(this.recipient);
|
|
|
+ shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook(operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
|
|
|
- await this.erc1820.setInterfaceImplementer(
|
|
|
- this.recipient,
|
|
|
- web3.utils.soliditySha3('ERC777TokensRecipient'), this.tokensRecipientImplementer.address,
|
|
|
- { from: this.recipient },
|
|
|
- );
|
|
|
- });
|
|
|
+ context('with contract as implementer for another contract', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.recipientContract = await ERC777SenderRecipientMock.new();
|
|
|
+ this.recipient = this.recipientContract.address;
|
|
|
|
|
|
- shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook(operator, amount, data, operatorData);
|
|
|
+ this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ await this.tokensRecipientImplementer.recipientFor(this.recipient);
|
|
|
+ await this.recipientContract.registerRecipient(this.tokensRecipientImplementer.address);
|
|
|
});
|
|
|
|
|
|
- context('with contract as implementer for another contract', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.recipientContract = await ERC777SenderRecipientMock.new();
|
|
|
- this.recipient = this.recipientContract.address;
|
|
|
+ shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook(operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
|
|
|
- this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- await this.tokensRecipientImplementer.recipientFor(this.recipient);
|
|
|
- await this.recipientContract.registerRecipient(this.tokensRecipientImplementer.address);
|
|
|
- });
|
|
|
+ context('with contract as implementer for itself', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ this.recipient = this.tokensRecipientImplementer.address;
|
|
|
|
|
|
- shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook(operator, amount, data, operatorData);
|
|
|
+ await this.tokensRecipientImplementer.recipientFor(this.recipient);
|
|
|
});
|
|
|
|
|
|
- context('with contract as implementer for itself', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.tokensRecipientImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- this.recipient = this.tokensRecipientImplementer.address;
|
|
|
-
|
|
|
- await this.tokensRecipientImplementer.recipientFor(this.recipient);
|
|
|
- });
|
|
|
-
|
|
|
- shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook(operator, amount, data, operatorData);
|
|
|
- });
|
|
|
+ shouldBehaveLikeERC777SendBurnMintInternalWithReceiveHook(operator, amount, data, operatorData);
|
|
|
});
|
|
|
});
|
|
|
+ });
|
|
|
|
|
|
- describe('tokensToSend', function () {
|
|
|
- beforeEach(function () {
|
|
|
- this.recipient = anyone;
|
|
|
- });
|
|
|
-
|
|
|
- context('with a contract as implementer for an externally owned account', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.tokensSenderImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- this.sender = holder;
|
|
|
+ describe('tokensToSend', function () {
|
|
|
+ beforeEach(function () {
|
|
|
+ this.recipient = anyone;
|
|
|
+ });
|
|
|
|
|
|
- await this.tokensSenderImplementer.senderFor(this.sender);
|
|
|
+ context('with a contract as implementer for an externally owned account', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.tokensSenderImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ this.sender = holder;
|
|
|
|
|
|
- await this.erc1820.setInterfaceImplementer(
|
|
|
- this.sender,
|
|
|
- web3.utils.soliditySha3('ERC777TokensSender'), this.tokensSenderImplementer.address,
|
|
|
- { from: this.sender },
|
|
|
- );
|
|
|
- });
|
|
|
+ await this.tokensSenderImplementer.senderFor(this.sender);
|
|
|
|
|
|
- shouldBehaveLikeERC777SendBurnWithSendHook(operator, amount, data, operatorData);
|
|
|
+ await this.erc1820.setInterfaceImplementer(
|
|
|
+ this.sender,
|
|
|
+ web3.utils.soliditySha3('ERC777TokensSender'), this.tokensSenderImplementer.address,
|
|
|
+ { from: this.sender },
|
|
|
+ );
|
|
|
});
|
|
|
|
|
|
- context('with contract as implementer for another contract', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.senderContract = await ERC777SenderRecipientMock.new();
|
|
|
- this.sender = this.senderContract.address;
|
|
|
+ shouldBehaveLikeERC777SendBurnWithSendHook(operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
|
|
|
- this.tokensSenderImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- await this.tokensSenderImplementer.senderFor(this.sender);
|
|
|
- await this.senderContract.registerSender(this.tokensSenderImplementer.address);
|
|
|
+ context('with contract as implementer for another contract', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.senderContract = await ERC777SenderRecipientMock.new();
|
|
|
+ this.sender = this.senderContract.address;
|
|
|
|
|
|
- // For the contract to be able to receive tokens (that it can later send), it must also implement the
|
|
|
- // recipient interface.
|
|
|
+ this.tokensSenderImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ await this.tokensSenderImplementer.senderFor(this.sender);
|
|
|
+ await this.senderContract.registerSender(this.tokensSenderImplementer.address);
|
|
|
|
|
|
- await this.senderContract.recipientFor(this.sender);
|
|
|
- await this.token.send(this.sender, amount, data, { from: holder });
|
|
|
- });
|
|
|
+ // For the contract to be able to receive tokens (that it can later send), it must also implement the
|
|
|
+ // recipient interface.
|
|
|
|
|
|
- shouldBehaveLikeERC777SendBurnWithSendHook(operator, amount, data, operatorData);
|
|
|
+ await this.senderContract.recipientFor(this.sender);
|
|
|
+ await this.token.send(this.sender, amount, data, { from: holder });
|
|
|
});
|
|
|
|
|
|
- context('with a contract as implementer for itself', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.tokensSenderImplementer = await ERC777SenderRecipientMock.new();
|
|
|
- this.sender = this.tokensSenderImplementer.address;
|
|
|
+ shouldBehaveLikeERC777SendBurnWithSendHook(operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
|
|
|
- await this.tokensSenderImplementer.senderFor(this.sender);
|
|
|
+ context('with a contract as implementer for itself', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ this.tokensSenderImplementer = await ERC777SenderRecipientMock.new();
|
|
|
+ this.sender = this.tokensSenderImplementer.address;
|
|
|
|
|
|
- // For the contract to be able to receive tokens (that it can later send), it must also implement the
|
|
|
- // recipient interface.
|
|
|
+ await this.tokensSenderImplementer.senderFor(this.sender);
|
|
|
|
|
|
- await this.tokensSenderImplementer.recipientFor(this.sender);
|
|
|
- await this.token.send(this.sender, amount, data, { from: holder });
|
|
|
- });
|
|
|
+ // For the contract to be able to receive tokens (that it can later send), it must also implement the
|
|
|
+ // recipient interface.
|
|
|
|
|
|
- shouldBehaveLikeERC777SendBurnWithSendHook(operator, amount, data, operatorData);
|
|
|
+ await this.tokensSenderImplementer.recipientFor(this.sender);
|
|
|
+ await this.token.send(this.sender, amount, data, { from: holder });
|
|
|
});
|
|
|
- });
|
|
|
- });
|
|
|
- });
|
|
|
-
|
|
|
- context('with no default operators', function () {
|
|
|
- beforeEach(async function () {
|
|
|
- this.token = await ERC777.new(holder, initialSupply, name, symbol, granularity, []);
|
|
|
- });
|
|
|
|
|
|
- it('default operators list is empty', async function () {
|
|
|
- (await this.token.defaultOperators()).should.deep.equal([]);
|
|
|
+ shouldBehaveLikeERC777SendBurnWithSendHook(operator, amount, data, operatorData);
|
|
|
+ });
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
- context('with granularity larger than 1', function () {
|
|
|
- const granularity = new BN('4');
|
|
|
-
|
|
|
+ context('with no default operators', function () {
|
|
|
beforeEach(async function () {
|
|
|
- initialSupply.mod(granularity).should.be.bignumber.equal('0');
|
|
|
-
|
|
|
- this.token = await ERC777.new(holder, initialSupply, name, symbol, granularity, defaultOperators);
|
|
|
- });
|
|
|
-
|
|
|
- it('returns the granularity', async function () {
|
|
|
- (await this.token.granularity()).should.be.bignumber.equal(granularity);
|
|
|
+ this.token = await ERC777.new(holder, initialSupply, name, symbol, []);
|
|
|
});
|
|
|
|
|
|
- context('when the sender has tokens', function () {
|
|
|
- const from = holder;
|
|
|
-
|
|
|
- shouldDirectSendTokens(from, anyone, new BN('0'), data);
|
|
|
- shouldDirectSendTokens(from, anyone, granularity, data);
|
|
|
- shouldDirectSendTokens(from, anyone, granularity.muln(2), data);
|
|
|
-
|
|
|
- it('reverts when sending an amount non-multiple of the granularity', async function () {
|
|
|
- await shouldFail.reverting(this.token.send(anyone, granularity.subn(1), data, { from }));
|
|
|
- });
|
|
|
-
|
|
|
- shouldDirectBurnTokens(from, new BN('0'), data);
|
|
|
- shouldDirectBurnTokens(from, granularity, data);
|
|
|
- shouldDirectBurnTokens(from, granularity.muln(2), data);
|
|
|
-
|
|
|
- it('reverts when burning an amount non-multiple of the granularity', async function () {
|
|
|
- await shouldFail.reverting(this.token.burn(granularity.subn(1), data, { from }));
|
|
|
- });
|
|
|
- });
|
|
|
-
|
|
|
- shouldInternalMintTokens(anyone, defaultOperatorA, new BN('0'), data, operatorData);
|
|
|
- shouldInternalMintTokens(anyone, defaultOperatorA, granularity, data, operatorData);
|
|
|
- shouldInternalMintTokens(anyone, defaultOperatorA, granularity.muln(2), data, operatorData);
|
|
|
-
|
|
|
- it('reverts when minting an amount non-multiple of the granularity', async function () {
|
|
|
- await shouldFail.reverting(
|
|
|
- this.token.mintInternal(defaultOperatorA, anyone, granularity.subn(1), data, operatorData)
|
|
|
- );
|
|
|
+ it('default operators list is empty', async function () {
|
|
|
+ (await this.token.defaultOperators()).should.deep.equal([]);
|
|
|
});
|
|
|
});
|
|
|
});
|