var async = require("async"); var buster = require("buster"); var Amount = require("../src/js/amount.js").Amount; var Remote = require("../src/js/remote.js").Remote; var Server = require("./server.js").Server; var testutils = require("./testutils.js"); require("../src/js/amount.js").config = require("./config.js"); require("../src/js/remote.js").config = require("./config.js"); buster.testRunner.timeout = 5000; buster.testCase("Offer tests", { 'setUp' : testutils.build_setup(), 'tearDown' : testutils.build_teardown(), "offer create then cancel in one ledger" : function (done) { var self = this; var final_create; async.waterfall([ function (callback) { self.remote.transaction() .offer_create("root", "500", "100/USD/root") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS', m); }) .on('final', function (m) { // console.log("FINAL: offer_create: %s", JSON.stringify(m)); buster.assert.equals('tesSUCCESS', m.metadata.TransactionResult); final_create = m; }) .submit(); }, function (m, callback) { self.remote.transaction() .offer_cancel("root", m.tx_json.Sequence) .on('proposed', function (m) { // console.log("PROPOSED: offer_cancel: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS', m); }) .on('final', function (m) { // console.log("FINAL: offer_cancel: %s", JSON.stringify(m)); buster.assert.equals('tesSUCCESS', m.metadata.TransactionResult); buster.assert(final_create); done(); }) .submit(); }, function (m, callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: %d: %s", ledger_index, ledger_hash); }) .ledger_accept(); } ], function (error) { // console.log("result: error=%s", error); buster.refute(error); if (error) done(); }); }, "offer_create then ledger_accept then offer_cancel then ledger_accept." : function (done) { var self = this; var final_create; var offer_seq; async.waterfall([ function (callback) { self.remote.transaction() .offer_create("root", "500", "100/USD/root") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); offer_seq = m.tx_json.Sequence; callback(m.result !== 'tesSUCCESS'); }) .on('final', function (m) { // console.log("FINAL: offer_create: %s", JSON.stringify(m)); buster.assert.equals('tesSUCCESS', m.metadata.TransactionResult); final_create = m; callback(); }) .submit(); }, function (callback) { if (!final_create) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: %d: %s", ledger_index, ledger_hash); }) .ledger_accept(); } else { callback(); } }, function (callback) { // console.log("CANCEL: offer_cancel: %d", offer_seq); self.remote.transaction() .offer_cancel("root", offer_seq) .on('proposed', function (m) { // console.log("PROPOSED: offer_cancel: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .on('final', function (m) { // console.log("FINAL: offer_cancel: %s", JSON.stringify(m)); buster.assert.equals('tesSUCCESS', m.metadata.TransactionResult); buster.assert(final_create); done(); }) .submit(); }, // See if ledger_accept will crash. function (callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: A: %d: %s", ledger_index, ledger_hash); callback(); }) .ledger_accept(); }, function (callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: B: %d: %s", ledger_index, ledger_hash); callback(); }) .ledger_accept(); }, ], function (error) { // console.log("result: error=%s", error); buster.refute(error); if (error) done(); }); }, "new user offer_create then ledger_accept then offer_cancel then ledger_accept." : function (done) { var self = this; var final_create; var offer_seq; async.waterfall([ function (callback) { self.remote.transaction() .payment('root', 'alice', "1000") .set_flags('CreateAccount') .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); buster.assert.equals(m.result, 'tesSUCCESS'); callback(); }) .submit() }, function (callback) { self.remote.transaction() .offer_create("alice", "500", "100/USD/alice") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); offer_seq = m.tx_json.Sequence; callback(m.result !== 'tesSUCCESS'); }) .on('final', function (m) { // console.log("FINAL: offer_create: %s", JSON.stringify(m)); buster.assert.equals('tesSUCCESS', m.metadata.TransactionResult); final_create = m; callback(); }) .submit(); }, function (callback) { if (!final_create) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: %d: %s", ledger_index, ledger_hash); }) .ledger_accept(); } else { callback(); } }, function (callback) { // console.log("CANCEL: offer_cancel: %d", offer_seq); self.remote.transaction() .offer_cancel("alice", offer_seq) .on('proposed', function (m) { // console.log("PROPOSED: offer_cancel: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .on('final', function (m) { // console.log("FINAL: offer_cancel: %s", JSON.stringify(m)); buster.assert.equals('tesSUCCESS', m.metadata.TransactionResult); buster.assert(final_create); done(); }) .submit(); }, // See if ledger_accept will crash. function (callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: A: %d: %s", ledger_index, ledger_hash); callback(); }) .ledger_accept(); }, function (callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: B: %d: %s", ledger_index, ledger_hash); callback(); }) .ledger_accept(); }, ], function (error) { // console.log("result: error=%s", error); buster.refute(error); if (error) done(); }); }, "offer cancel past and future sequence" : function (done) { var self = this; var final_create; async.waterfall([ function (callback) { self.remote.transaction() .payment('root', 'alice', Amount.from_json("10000")) .set_flags('CreateAccount') .on('proposed', function (m) { // console.log("PROPOSED: CreateAccount: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS', m); }) .on('error', function(m) { // console.log("error: %s", m); buster.assert(false); callback(m); }) .submit(); }, // Past sequence but wrong function (m, callback) { self.remote.transaction() .offer_cancel("root", m.tx_json.Sequence) .on('proposed', function (m) { // console.log("PROPOSED: offer_cancel past: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS', m); }) .submit(); }, // Same sequence function (m, callback) { self.remote.transaction() .offer_cancel("root", m.tx_json.Sequence+1) .on('proposed', function (m) { // console.log("PROPOSED: offer_cancel same: %s", JSON.stringify(m)); callback(m.result !== 'temBAD_SEQUENCE', m); }) .submit(); }, // Future sequence function (m, callback) { // After a malformed transaction, need to recover correct sequence. self.remote.set_account_seq("root", self.remote.account_seq("root")-1); self.remote.transaction() .offer_cancel("root", m.tx_json.Sequence+2) .on('proposed', function (m) { // console.log("ERROR: offer_cancel future: %s", JSON.stringify(m)); callback(m.result !== 'temBAD_SEQUENCE'); }) .submit(); }, // See if ledger_accept will crash. function (callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: A: %d: %s", ledger_index, ledger_hash); callback(); }) .ledger_accept(); }, function (callback) { self.remote .once('ledger_closed', function (ledger_hash, ledger_index) { // console.log("LEDGER_CLOSED: B: %d: %s", ledger_index, ledger_hash); callback(); }) .ledger_accept(); }, function (callback) { callback(); } ], function (error) { // console.log("result: error=%s", error); buster.refute(error); done(); }); }, "ripple currency conversion : entire offer" : function (done) { var self = this; var seq; async.waterfall([ function (callback) { self.what = "Create accounts."; testutils.create_accounts(self.remote, "root", "10000", ["alice", "bob", "mtgox"], callback); }, function (callback) { self.what = "Set limits."; testutils.credit_limits(self.remote, { "alice" : "100/USD/mtgox", "bob" : "1000/USD/mtgox" }, callback); }, function (callback) { self.what = "Distribute funds."; testutils.payments(self.remote, { "mtgox" : "100/USD/alice" }, callback); }, function (callback) { self.what = "Create offer."; self.remote.transaction() .offer_create("bob", "100/USD/mtgox", "500") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); seq = m.tx_json.Sequence; }) .submit(); }, function (callback) { self.what = "Verify offer balance."; testutils.verify_offer(self.remote, "bob", seq, "500", "100/USD/mtgox", callback); }, function (callback) { self.what = "Alice converts USD to XRP."; self.remote.transaction() .payment("alice", "alice", "500") .send_max("100/USD/mtgox") .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .submit(); }, function (callback) { self.what = "Verify balances."; testutils.verify_balances(self.remote, { "alice" : [ "0/USD/mtgox", String(10000+500-2*(Remote.fees['default'].to_number())) ], "bob" : "100/USD/mtgox", }, callback); }, function (callback) { self.what = "Verify offer consumed."; testutils.verify_offer_not_found(self.remote, "bob", seq, callback); }, ], function (error) { buster.refute(error, self.what); done(); }); }, "ripple currency conversion : in parts" : function (done) { var self = this; var seq; async.waterfall([ function (callback) { self.what = "Create accounts."; testutils.create_accounts(self.remote, "root", "10000", ["alice", "bob", "mtgox"], callback); }, function (callback) { self.what = "Set limits."; testutils.credit_limits(self.remote, { "alice" : "200/USD/mtgox", "bob" : "1000/USD/mtgox" }, callback); }, function (callback) { self.what = "Distribute funds."; testutils.payments(self.remote, { "mtgox" : "200/USD/alice" }, callback); }, function (callback) { self.what = "Create offer."; self.remote.transaction() .offer_create("bob", "100/USD/mtgox", "500") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); seq = m.tx_json.Sequence; }) .submit(); }, function (callback) { self.what = "Alice converts USD to XRP."; self.remote.transaction() .payment("alice", "alice", "200") .send_max("100/USD/mtgox") .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .submit(); }, function (callback) { self.what = "Verify offer balance."; testutils.verify_offer(self.remote, "bob", seq, "300", "60/USD/mtgox", callback); }, function (callback) { self.what = "Verify balances."; testutils.verify_balances(self.remote, { "alice" : [ "160/USD/mtgox", String(10000+200-2*(Remote.fees['default'].to_number())) ], "bob" : "40/USD/mtgox", }, callback); }, function (callback) { self.what = "Alice converts USD to XRP should fail due to PartialPayment."; self.remote.transaction() .payment("alice", "alice", "600") .send_max("100/USD/mtgox") .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tepPATH_PARTIAL'); }) .submit(); }, function (callback) { self.what = "Alice converts USD to XRP."; self.remote.transaction() .payment("alice", "alice", "600") .send_max("100/USD/mtgox") .set_flags('PartialPayment') .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .submit(); }, function (callback) { self.what = "Verify offer consumed."; testutils.verify_offer_not_found(self.remote, "bob", seq, callback); }, function (callback) { self.what = "Verify balances."; testutils.verify_balances(self.remote, { "alice" : [ "100/USD/mtgox", String(10000+200+300-4*(Remote.fees['default'].to_number())) ], "bob" : "100/USD/mtgox", }, callback); }, ], function (error) { buster.refute(error, self.what); done(); }); }, }); buster.testCase("Offer cross currency", { 'setUp' : testutils.build_setup(), 'tearDown' : testutils.build_teardown(), "ripple cross currency payment - start with XRP" : // alice --> [XRP --> carol --> USD/mtgox] --> bob function (done) { var self = this; var seq; // self.remote.set_trace(); async.waterfall([ function (callback) { self.what = "Create accounts."; testutils.create_accounts(self.remote, "root", "10000", ["alice", "bob", "carol", "mtgox"], callback); }, function (callback) { self.what = "Set limits."; testutils.credit_limits(self.remote, { "carol" : "1000/USD/mtgox", "bob" : "2000/USD/mtgox" }, callback); }, function (callback) { self.what = "Distribute funds."; testutils.payments(self.remote, { "mtgox" : "500/USD/carol" }, callback); }, function (callback) { self.what = "Create offer."; self.remote.transaction() .offer_create("carol", "500", "50/USD/mtgox") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); seq = m.tx_json.Sequence; }) .submit(); }, function (callback) { self.what = "Alice send USD/mtgox converting from XRP."; self.remote.transaction() .payment("alice", "bob", "25/USD/mtgox") .send_max("333") .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .submit(); }, function (callback) { self.what = "Verify balances."; testutils.verify_balances(self.remote, { // "alice" : [ "500" ], "bob" : "25/USD/mtgox", "carol" : "475/USD/mtgox", }, callback); }, function (callback) { self.what = "Verify offer consumed."; testutils.verify_offer_not_found(self.remote, "bob", seq, callback); }, ], function (error) { buster.refute(error, self.what); done(); }); }, "ripple cross currency payment - end with XRP" : // alice --> [USD/mtgox --> carol --> XRP] --> bob function (done) { var self = this; var seq; // self.remote.set_trace(); async.waterfall([ function (callback) { self.what = "Create accounts."; testutils.create_accounts(self.remote, "root", "10000", ["alice", "bob", "carol", "mtgox"], callback); }, function (callback) { self.what = "Set limits."; testutils.credit_limits(self.remote, { "alice" : "1000/USD/mtgox", "carol" : "2000/USD/mtgox" }, callback); }, function (callback) { self.what = "Distribute funds."; testutils.payments(self.remote, { "mtgox" : "500/USD/alice" }, callback); }, function (callback) { self.what = "Create offer."; self.remote.transaction() .offer_create("carol", "50/USD/mtgox", "500") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); seq = m.tx_json.Sequence; }) .submit(); }, function (callback) { self.what = "Alice send XRP to bob converting from USD/mtgox."; self.remote.transaction() .payment("alice", "bob", "250") .send_max("333/USD/mtgox") .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .submit(); }, function (callback) { self.what = "Verify balances."; testutils.verify_balances(self.remote, { "alice" : "475/USD/mtgox", "bob" : "10250", "carol" : "25/USD/mtgox", }, callback); }, function (callback) { self.what = "Verify offer partially consumed."; testutils.verify_offer(self.remote, "carol", seq, "250", "25/USD/mtgox", callback); }, ], function (error) { buster.refute(error, self.what); done(); }); }, "// ripple cross currency bridged payment" : // alice --> [USD/mtgox --> carol --> XRP] --> [XRP --> dan --> EUR/bitstamp] --> bob function (done) { var self = this; var seq_carol; var seq_dan; // self.remote.set_trace(); async.waterfall([ function (callback) { self.what = "Create accounts."; testutils.create_accounts(self.remote, "root", "10000", ["alice", "bob", "carol", "dan", "bitstamp", "mtgox"], callback); }, function (callback) { self.what = "Set limits."; testutils.credit_limits(self.remote, { "alice" : "1000/USD/mtgox", "bob" : "1000/EUR/bitstamp", "carol" : "1000/USD/mtgox", "dan" : "1000/EUR/bitstamp" }, callback); }, function (callback) { self.what = "Distribute funds."; testutils.payments(self.remote, { "bitstamp" : "400/EUR/dan", "mtgox" : "500/USD/alice", }, callback); }, function (callback) { self.what = "Create offer carol."; self.remote.transaction() .offer_create("carol", "50/USD/mtgox", "500") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); seq_carol = m.tx_json.Sequence; }) .submit(); }, function (callback) { self.what = "Create offer dan."; self.remote.transaction() .offer_create("dan", "500", "50/EUR/bitstamp") .on('proposed', function (m) { // console.log("PROPOSED: offer_create: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); seq_dan = m.tx_json.Sequence; }) .submit(); }, function (callback) { self.what = "Alice send EUR/bitstamp to bob converting from USD/mtgox."; self.remote.transaction() .payment("alice", "bob", "30/EUR/bitstamp") .send_max("333/USD/mtgox") .on('proposed', function (m) { // console.log("proposed: %s", JSON.stringify(m)); callback(m.result !== 'tesSUCCESS'); }) .submit(); }, // function (callback) { // self.what = "Verify balances."; // // testutils.verify_balances(self.remote, // { // "alice" : "470/USD/mtgox", // "bob" : "30/EUR/bitstamp", // "carol" : "30/USD/mtgox", // "dan" : "370/EUR/bitstamp", // }, // callback); // }, // function (callback) { // self.what = "Verify carol offer partially consumed."; // // testutils.verify_offer(self.remote, "carol", seq_carol, "250", "25/USD/mtgox", callback); // }, // function (callback) { // self.what = "Verify dan offer partially consumed."; // // testutils.verify_offer(self.remote, "dan", seq_dan, "250", "25/USD/mtgox", callback); // }, ], function (error) { buster.refute(error, self.what); done(); }); }, }); // vim:sw=2:sts=2:ts=8