You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
151 lines
8.0 KiB
151 lines
8.0 KiB
10 years ago
|
# > `bitcore.Script`
|
||
|
|
||
|
## Description
|
||
10 years ago
|
|
||
10 years ago
|
All bitcoin transactions have scripts embedded into its inputs and outputs. The scripts use a very simple programming language, which is evaluated from left to right using a stack. The language is designed such that it guarantees all scripts will execute in a limited amount of time (it is not Turing-Complete).
|
||
10 years ago
|
|
||
10 years ago
|
When a transaction is validated, the input scripts are concatenated with the output scripts and evaluated. To be valid, all transaction scripts must evaluate to true. A good analogy for how this works is that the output scripts are puzzles that specify in which conditions can those bitcoins be spent. The input scripts provide the correct data to make those output scripts evaluate to true.
|
||
10 years ago
|
|
||
|
|
||
10 years ago
|
For more detailed information about the bitcoin scripting language, check the online reference [on bitcoin's wiki](https://en.bitcoin.it/wiki/Script).
|
||
10 years ago
|
|
||
10 years ago
|
The `Script` object provides an interface to construct, parse, and identify bitcoin scripts. It also gives simple interfaces to create most common script types. This class is useful if you want to create custom input or output scripts. In other case, you should probably use `Transaction`.
|
||
10 years ago
|
|
||
|
## Script creation
|
||
|
|
||
|
Here's how to use `Script` to create the five most common script types:
|
||
|
|
||
|
### Pay to Public Key Hash (p2pkh)
|
||
|
|
||
10 years ago
|
This is the most commonly used transaction output script. It's used to pay to a bitcoin address (a bitcoin address is a public key hash encoded in base58check)
|
||
10 years ago
|
|
||
|
```javascript
|
||
|
// create a new p2pkh paying to a specific address
|
||
|
var address = Address.fromString('1NaTVwXDDUJaXDQajoa9MqHhz4uTxtgK14');
|
||
10 years ago
|
var script = Script.buildPublicKeyHashOut(address);
|
||
|
assert(script.toString() === 'OP_DUP OP_HASH160 20 0xecae7d092947b7ee4998e254aa48900d26d2ce1d OP_EQUALVERIFY OP_CHECKSIG');
|
||
10 years ago
|
```
|
||
|
### Pay to Public Key (p2pk)
|
||
|
|
||
10 years ago
|
Pay to public key scripts are a simplified form of the p2pkh, but aren’t commonly used in new transactions anymore, because p2pkh scripts are more secure (the public key is not revealed until the output is spent).
|
||
10 years ago
|
|
||
|
```javascript
|
||
|
// create a new p2pk paying to a specific public key
|
||
|
var pubkey = new PublicKey('022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da');
|
||
10 years ago
|
var script = Script.buildPublicKeyOut(pubkey);
|
||
|
assert(script.toString() === '33 0x022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da OP_CHECKSIG');
|
||
10 years ago
|
```
|
||
|
|
||
|
### Pay to Multisig (p2ms)
|
||
|
|
||
10 years ago
|
Multisig outputs allow to share control of bitcoins between several keys. When creating the script, one specifies the public keys that control the funds, and how many of those keys are required to sign off spending transactions to be valid. An output with N public keys of which M are required is called an m-of-n output (For example, 2-of-3, 3-of-5, 4-of-4, etc.)
|
||
10 years ago
|
|
||
10 years ago
|
Note that regular multisig outputs are rarely used nowadays. The best practice is to use a p2sh multisig output (See Script#toScriptHashOut()).
|
||
10 years ago
|
|
||
|
```javascript
|
||
|
// create a new 2-of-3 multisig output from 3 given public keys
|
||
|
var pubkeys = [
|
||
|
new PublicKey('022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da'),
|
||
|
new PublicKey('03e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e9'),
|
||
|
new PublicKey('021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc18'),
|
||
|
];
|
||
10 years ago
|
var threshold = 2;
|
||
|
var script = Script.buildMultisigOut(pubkeys, threshold);
|
||
|
assert(script.toString() === 'OP_2 33 0x022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da'
|
||
|
+ ' 33 0x03e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e9'
|
||
|
+ ' 33 0x021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc18 OP_3 OP_CHECKMULTISIG');
|
||
10 years ago
|
```
|
||
|
|
||
|
### Pay to Script Hash (p2sh)
|
||
|
|
||
10 years ago
|
Pay to script hash outputs are scripts that contain the hash of another script, called `redeemScript`. To spend bitcoins sent in a p2sh output, the spending transaction must provide a script matching the script hash and data which makes the script evaluate to true. This allows to defer revealing the spending conditions to the moment of spending. It also makes it possible for the receiver to set the conditions to spend those bitcoins.
|
||
10 years ago
|
|
||
|
Most multisig transactions today use p2sh outputs where the redeemScript is a multisig output.
|
||
|
|
||
|
```javascript
|
||
|
// create a p2sh multisig output
|
||
|
var pubkeys = [
|
||
|
new PublicKey('022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da'),
|
||
|
new PublicKey('03e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e9'),
|
||
|
new PublicKey('021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc18'),
|
||
|
];
|
||
|
var redeemScript = Script.buildMultisigOut(pubkeys, 2);
|
||
10 years ago
|
var script = redeemScript.toScriptHashOut();
|
||
|
assert(script.toString() === 'OP_HASH160 20 0x620a6eeaf538ec9eb89b6ae83f2ed8ef98566a03 OP_EQUAL');
|
||
10 years ago
|
```
|
||
|
### Data output
|
||
|
|
||
10 years ago
|
Data outputs are used to push data into the blockchain. Up to 40 bytes can be pushed in a standard way, but more data can be used, if a miner decides to accept the transaction.
|
||
10 years ago
|
|
||
|
```javascript
|
||
|
var data = 'hello world!!!';
|
||
10 years ago
|
var script = Script.buildDataOut(data);
|
||
|
assert(script.toString() === 'OP_RETURN 14 0x68656c6c6f20776f726c64212121'
|
||
10 years ago
|
```
|
||
|
|
||
|
### Custom scripts
|
||
|
|
||
10 years ago
|
To create a custom `Script` instance, you must rely on the lower-level methods `add` and `prepend`. Both methods accept the same parameter types, and insert an opcode or data at the beginning (`prepend`) or end (`add`) of the `Script`.
|
||
10 years ago
|
|
||
|
```
|
||
10 years ago
|
var script = Script()
|
||
|
.add('OP_IF') // add an opcode by name
|
||
|
.prepend(114) // add OP_2SWAP by code
|
||
|
.add(Opcode.OP_NOT) // add an opcode object
|
||
|
.add(new Buffer('bacacafe', 'hex')) // add a data buffer (will append the size of the push operation first)
|
||
|
|
||
|
assert(script.toString() === 'OP_2SWAP OP_IF OP_NOT 4 0xbacacafe');
|
||
10 years ago
|
```
|
||
|
|
||
|
|
||
|
## Script parsing and identification
|
||
|
|
||
10 years ago
|
`Script` has an easy interface to parse raw scripts from the newtwork or bitcoind, and to extract useful information. An illustrative example (for more options check the API reference)
|
||
|
|
||
10 years ago
|
```javascript
|
||
10 years ago
|
var raw_script = new Buffer('5221022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da2103e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e921021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc1853ae', 'hex');
|
||
|
var s = new Script(raw_script);
|
||
|
console.log(s.toString());
|
||
|
// 'OP_2 33 0x022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da 33 0x03e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e9 33 0x021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc18 OP_3 OP_CHECKMULTISIG'
|
||
|
|
||
|
s.isPublicKeyHashOut() // false
|
||
|
s.isScriptHashOut() // false
|
||
|
s.isMultisigOut() // true
|
||
|
```
|
||
10 years ago
|
|
||
|
|
||
|
## Script interpreting and validation
|
||
|
|
||
|
To validate a transaction, the bitcoin network validates all of its inputs and outputs. To validate an input, the input's script is concatenated with the referenced output script, and the result is executed. If at the end of execution the stack contains a 'true' value, then the transaction is valid.
|
||
|
You can do this in `bitcore` by using the `Interpreter` class. The entry point (and probably the only interface you'll need for most applications) is the method `Interpreter#verify()`.
|
||
|
|
||
|
You can use it like this:
|
||
|
|
||
|
```
|
||
|
var inputScript = Script('OP_1');
|
||
|
var outputScript = Script('OP_15 OP_ADD OP_16 OP_EQUAL');
|
||
|
|
||
|
var verified = Interpreter().verify(inputScript, outputScript);
|
||
|
// verified will be true
|
||
|
```
|
||
|
|
||
|
Note that `verify` expects two scripts: one is the input script (scriptSig) and the other is the output script (scriptPubkey). This is because different conditions are checked for each.
|
||
|
|
||
|
It also accepts some optional parameters, assuming defaults if not provided:
|
||
|
```
|
||
|
// first we create a transaction
|
||
|
var tx = new Transaction()
|
||
|
.from(utxo)
|
||
|
.to(toAddress, 100000)
|
||
|
.sign(privateKey);
|
||
|
|
||
|
// we then extract the signature from the first input
|
||
|
var inputIndex = 0;
|
||
|
var signature = tx.getSignatures(privateKey)[inputIndex].signature;
|
||
|
|
||
|
var scriptSig = Script.buildPublicKeyHashIn(publicKey, signature);
|
||
|
var flags = Interpreter.SCRIPT_VERIFY_P2SH | Interpreter.SCRIPT_VERIFY_STRICTENC;
|
||
|
var verified = Interpreter().verify(scriptSig, scriptPubkey, tx, inputIndex, flags);
|
||
|
```
|
||
|
|