forked from BitVM/rust-bitcoin-scriptexec
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwasm.rs
103 lines (95 loc) · 3.41 KB
/
wasm.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
use bitcoin::hashes::Hash;
use bitcoin::hex::{DisplayHex, FromHex};
use bitcoin::taproot::TapLeafHash;
use bitcoin::{ScriptBuf, Transaction};
use serde_json::json;
use wasm_bindgen::prelude::*;
use crate::*;
/// Compile ASM into script hex.
#[wasm_bindgen]
pub fn script_asm_to_hex(script_asm: &str) -> Result<String, JsValue> {
let script =
ScriptBuf::parse_asm(script_asm).map_err(|e| format!("error parsing script: {:?}", e))?;
Ok(script.as_bytes().as_hex().to_string())
}
/// Decode compiled script hex into ASM.
#[wasm_bindgen]
pub fn script_hex_to_asm(script_hex: &str) -> Result<String, JsValue> {
let script = ScriptBuf::from_hex(script_hex).map_err(|e| format!("invalid hex: {}", e))?;
Ok(script.to_asm_string())
}
/// Run the given script.
///
/// Fields on the return value are:
/// - success: bool
/// - final_stack: list of hex stack items after execution
/// - error: (optional) error that caused execution halt
/// - last_opcode: (optional) last opcode run before error produced
/// - stats: execution runtime statistics with following fields:
/// - max_nb_stack_items
/// - max_stack_size
/// - max_stack_item_size
/// - start_validation_weight
/// - validation_weight
#[wasm_bindgen]
pub fn run_script(script_hex: &str, script_witness: Box<[JsValue]>) -> Result<JsValue, JsValue> {
console_error_panic_hook::set_once();
let script =
ScriptBuf::from_hex(script_hex).map_err(|e| format!("invalid hex script: {:?}", e))?;
let witness = {
let mut ret = Vec::with_capacity(script_witness.len());
for item in script_witness.iter() {
let hex = item
.as_string()
.ok_or("script witness must be list of hex strings")?;
let bytes = Vec::from_hex(&hex).map_err(|_| "invalid hex in script witness")?;
ret.push(bytes);
}
ret
};
let mut exec = Exec::new(
ExecCtx::Tapscript,
Options::default(),
TxTemplate {
tx: Transaction {
version: bitcoin::transaction::Version::TWO,
lock_time: bitcoin::locktime::absolute::LockTime::ZERO,
input: vec![],
output: vec![],
},
prevouts: vec![],
input_idx: 0,
taproot_annex_scriptleaf: Some((TapLeafHash::all_zeros(), None)),
},
script,
witness,
)
.map_err(|e| format!("error creating exec: {:?}", e))?;
loop {
if let Err(res) = exec.exec_next() {
let res = res.clone();
let mut ret = json!({
"success": res.success,
"final_stack": res.final_stack.iter_str()
.map(|i| i.as_hex().to_string())
.collect::<Vec<_>>(),
"stats": serde_json::to_value(&exec.stats()).unwrap(),
});
if !res.success {
let obj = ret.as_object_mut().unwrap();
obj.insert(
"last_opcode".into(),
res.opcode.map(|o| o.to_string()).unwrap_or_default().into(),
);
obj.insert(
"error".into(),
res.error
.map(|o| format!("{:?}", o))
.unwrap_or_default()
.into(),
);
}
return Ok(serde_wasm_bindgen::to_value(&ret).unwrap());
}
}
}