From 97804d1837e3b556b9881947062833292adf90df Mon Sep 17 00:00:00 2001 From: Carlos Alaniz Date: Fri, 28 Jul 2023 10:30:41 -0500 Subject: [PATCH 01/61] Fix remote RPC wallet commands (#1766) --- src/options.rs | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/src/options.rs b/src/options.rs index a8853ae7a9..20841e911b 100644 --- a/src/options.rs +++ b/src/options.rs @@ -82,13 +82,15 @@ impl Options { } pub(crate) fn rpc_url(&self) -> String { - self.rpc_url.clone().unwrap_or_else(|| { + if let Some(rpc_url) = &self.rpc_url { + format!("{rpc_url}/wallet/{}", self.wallet) + } else { format!( "127.0.0.1:{}/wallet/{}", self.chain().default_rpc_port(), self.wallet ) - }) + } } pub(crate) fn cookie_file(&self) -> Result { @@ -285,7 +287,7 @@ mod tests { .unwrap() .options .rpc_url(), - "127.0.0.1:1234" + "127.0.0.1:1234/wallet/ord" ); } From 2c03829cd44ec2bd575d903d1bec75867eb55615 Mon Sep 17 00:00:00 2001 From: raph Date: Sun, 30 Jul 2023 17:30:52 +0200 Subject: [PATCH 02/61] Select multiple utxos (#2303) Co-authored-by: Greg Martin --- fuzz/Cargo.lock | 1791 +++++++++--------- fuzz/Cargo.toml | 4 - fuzz/fuzz_targets/transaction_builder.rs | 20 +- src/subcommand/wallet/transaction_builder.rs | 257 ++- 4 files changed, 1133 insertions(+), 939 deletions(-) diff --git a/fuzz/Cargo.lock b/fuzz/Cargo.lock index f30058f2a7..d8121a4921 100644 --- a/fuzz/Cargo.lock +++ b/fuzz/Cargo.lock @@ -4,9 +4,9 @@ version = 3 [[package]] name = "addr2line" -version = "0.19.0" +version = "0.20.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a76fd60b23679b7d19bd066031410fb7e458ccc5e958eb5c325888ce4baedc97" +checksum = "f4fa78e18c64fce05e902adecd7a5eed15a5e0a3439f7b0e169f0252214865e3" dependencies = [ "gimli", ] @@ -18,67 +18,34 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" [[package]] -name = "aead" -version = "0.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7fc95d1bdb8e6666b2b217308eeeb09f2d6728d104be3e31916cc74d15420331" -dependencies = [ - "generic-array", -] - -[[package]] -name = "aes" -version = "0.6.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "884391ef1066acaa41e766ba8f596341b96e93ce34f9a43e7d24bf0a0eaf0561" -dependencies = [ - "aes-soft", - "aesni", - "cipher", -] - -[[package]] -name = "aes-gcm" -version = "0.8.0" +name = "aho-corasick" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5278b5fabbb9bd46e24aa69b2fdea62c99088e0a950a9be40e3e0101298f88da" +checksum = "43f6cb1bf222025340178f382c426f13757b2960e89779dfcb319c32542a5a41" dependencies = [ - "aead", - "aes", - "cipher", - "ctr", - "ghash", - "subtle", + "memchr", ] [[package]] -name = "aes-soft" -version = "0.6.4" +name = "alloc-no-stdlib" +version = "2.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "be14c7498ea50828a38d0e24a765ed2effe92a705885b57d029cd67d45744072" -dependencies = [ - "cipher", - "opaque-debug", -] +checksum = "cc7bb162ec39d46ab1ca8c77bf72e890535becd1751bb45f64c597edb4c8c6b3" [[package]] -name = "aesni" -version = "0.10.0" +name = "alloc-stdlib" +version = "0.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ea2e11f5e94c2f7d386164cc2aa1f97823fed6f259e486940a71c174dd01b0ce" +checksum = "94fb8275041c72129eb51b7d0322c29b8387a0386127718b096429201a5d6ece" dependencies = [ - "cipher", - "opaque-debug", + "alloc-no-stdlib", ] [[package]] -name = "aho-corasick" -version = "0.7.20" +name = "android-tzdata" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cc936419f96fa211c1b9166887b38e5e40b19958e5b895be7c1f93adec7071ac" -dependencies = [ - "memchr", -] +checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0" [[package]] name = "android_system_properties" @@ -91,18 +58,18 @@ dependencies = [ [[package]] name = "anyhow" -version = "1.0.68" +version = "1.0.72" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2cb2f989d18dd141ab8ae82f64d1a8cdd37e0840f73a406896cf5e99502fab61" +checksum = "3b13c32d80ecc7ab747b80c3784bce54ee8a7a0cc4fbda9bf4cda2cf6fe90854" dependencies = [ "backtrace", ] [[package]] name = "arbitrary" -version = "1.2.2" +version = "1.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b0224938f92e7aef515fac2ff2d18bd1115c1394ddf4a092e0c87e8be9499ee5" +checksum = "e2d098ff73c1ca148721f37baad5ea6a465a13f9573aba8641fbbbae8164a54e" dependencies = [ "derive_arbitrary", ] @@ -126,7 +93,7 @@ dependencies = [ "num-traits", "rusticata-macros", "thiserror", - "time 0.3.17", + "time 0.3.23", ] [[package]] @@ -137,7 +104,7 @@ checksum = "db8b7511298d5b7784b40b092d9e9dcd3a627a5707e4b5e507931ab0d44eeebf" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 1.0.109", "synstructure", ] @@ -149,20 +116,34 @@ checksum = "2777730b2039ac0f95f093556e61b6d26cebed5393ca6f152717777cec3a42ed" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 1.0.109", ] [[package]] name = "async-channel" -version = "1.8.0" +version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cf46fee83e5ccffc220104713af3292ff9bc7c64c7de289f66dae8e38d826833" +checksum = "81953c529336010edd6d8e358f886d9581267795c61b19475b71314bffa46d35" dependencies = [ "concurrent-queue", "event-listener", "futures-core", ] +[[package]] +name = "async-compression" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62b74f44609f0f91493e3082d3734d98497e094777144380ea4db9f9905dd5b6" +dependencies = [ + "brotli", + "flate2", + "futures-core", + "memchr", + "pin-project-lite", + "tokio", +] + [[package]] name = "async-dup" version = "1.2.2" @@ -175,14 +156,14 @@ dependencies = [ [[package]] name = "async-executor" -version = "1.5.0" +version = "1.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "17adb73da160dfb475c183343c8cccd80721ea5a605d3eb57125f0a7b7a92d0b" +checksum = "6fa3dc5f2a8564f07759c008b9109dc0d39de92a88d5588b8a5036d286383afb" dependencies = [ "async-lock", "async-task", "concurrent-queue", - "fastrand", + "fastrand 1.9.0", "futures-lite", "slab", ] @@ -232,32 +213,31 @@ dependencies = [ [[package]] name = "async-io" -version = "1.12.0" +version = "1.13.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8c374dda1ed3e7d8f0d9ba58715f924862c63eae6849c92d3a18e7fbde9e2794" +checksum = "0fc5b45d93ef0529756f812ca52e44c221b35341892d3dcc34132ac02f3dd2af" dependencies = [ "async-lock", "autocfg", + "cfg-if", "concurrent-queue", "futures-lite", - "libc", "log", "parking", "polling", + "rustix 0.37.23", "slab", "socket2", "waker-fn", - "windows-sys", ] [[package]] name = "async-lock" -version = "2.6.0" +version = "2.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8101efe8695a6c17e02911402145357e718ac92d3ff88ae8419e84b1707b685" +checksum = "fa24f727524730b077666307f2734b4a1a1c57acb79193127dcc8914d5242dd7" dependencies = [ "event-listener", - "futures-lite", ] [[package]] @@ -274,9 +254,9 @@ dependencies = [ [[package]] name = "async-process" -version = "1.6.0" +version = "1.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6381ead98388605d0d9ff86371043b5aa922a3905824244de40dc263a14fcba4" +checksum = "7a9d28b1d97e08915212e2e45310d47854eafa69600756fc735fb788f75199c9" dependencies = [ "async-io", "async-lock", @@ -285,9 +265,9 @@ dependencies = [ "cfg-if", "event-listener", "futures-lite", - "libc", + "rustix 0.37.23", "signal-hook", - "windows-sys", + "windows-sys 0.48.0", ] [[package]] @@ -318,26 +298,26 @@ dependencies = [ [[package]] name = "async-task" -version = "4.3.0" +version = "4.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a40729d2133846d9ed0ea60a8b9541bccddab49cd30f0715a1da672fe9a2524" +checksum = "ecc7ab41815b3c653ccd2978ec3255c81349336702dfdf62ee6f7069b12a3aae" [[package]] name = "async-trait" -version = "0.1.61" +version = "0.1.72" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "705339e0e4a9690e2908d2b3d049d85682cf19fbd5782494498fbf7003a6a282" +checksum = "cc6dde6e4ed435a4c1ee4e73592f5ba9da2151af10076cc04858746af9352d09" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] name = "atom_syndication" -version = "0.12.0" +version = "0.12.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "91a85f2ee28cbd1ecf91288460f6dc74661fd99b4e9a559836a667ccf63aa38c" +checksum = "571832dcff775e26562e8e6930cd483de5587301d40d3a3b85d532b6383e15a7" dependencies = [ "chrono", "derive_builder", @@ -348,9 +328,9 @@ dependencies = [ [[package]] name = "atomic-waker" -version = "1.1.0" +version = "1.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "debc29dde2e69f9e47506b525f639ed42300fc014a3e007832592448fa8e4599" +checksum = "1181e1e0d1fce796a03db1ae795d67167da795f9cf4a39c37589e85ef57f26d3" [[package]] name = "atty" @@ -371,15 +351,16 @@ checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" [[package]] name = "axum" -version = "0.6.2" +version = "0.6.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1304eab461cf02bd70b083ed8273388f9724c549b316ba3d1e213ce0e9e7fb7e" +checksum = "a6a1de45611fdb535bfde7b7de4fd54f4fd2b17b1737c0a59b69bf9b92074b8c" dependencies = [ "async-trait", "axum-core", - "bitflags", + "bitflags 1.3.2", "bytes", "futures-util", + "headers", "http", "http-body", "hyper", @@ -397,16 +378,15 @@ dependencies = [ "sync_wrapper", "tokio", "tower", - "tower-http", "tower-layer", "tower-service", ] [[package]] name = "axum-core" -version = "0.3.1" +version = "0.3.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f487e40dc9daee24d8a1779df88522f159a54a980f99cfbe43db0be0bd3444a8" +checksum = "759fa577a247914fd3f7f76d62972792636412fbfd634cd452f6a385a74d2d2c" dependencies = [ "async-trait", "bytes", @@ -421,9 +401,9 @@ dependencies = [ [[package]] name = "axum-server" -version = "0.4.4" +version = "0.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8456dab8f11484979a86651da8e619b355ede5d61a160755155f6c344bd18c47" +checksum = "447f28c85900215cc1bea282f32d4a2f22d55c5a300afdfbc661c8d6a632e063" dependencies = [ "arc-swap", "bytes", @@ -441,9 +421,9 @@ dependencies = [ [[package]] name = "backtrace" -version = "0.3.67" +version = "0.3.68" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "233d376d6d185f2a3093e58f283f60f880315b6c60075b01f36b3b85154564ca" +checksum = "4319208da049c43661739c5fade2ba182f09d1dc2299b32298d3a31692b17e12" dependencies = [ "addr2line", "cc", @@ -454,12 +434,6 @@ dependencies = [ "rustc-demangle", ] -[[package]] -name = "base-x" -version = "0.2.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4cbbc9d0964165b47557570cce6c952866c2678457aca742aafc9fb771d30270" - [[package]] name = "base64" version = "0.13.1" @@ -468,9 +442,9 @@ checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" [[package]] name = "base64" -version = "0.21.0" +version = "0.21.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a4a4ddaa51a5bc52a6948f74c06d20aaaddb71924eab79b8c97a8c556e942d6a" +checksum = "604178f6c5c21f02dc555784810edfb88d34ac2c73b2eae109655649ee73ce3d" [[package]] name = "bech32" @@ -480,40 +454,48 @@ checksum = "d86b93f97252c47b41663388e6d155714a9d0c398b99f1005cbc5f978b29f445" [[package]] name = "bip39" -version = "1.0.1" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b9e89470017230c38e52b82b3ee3f530db1856ba1d434e3a67a3456a8a8dec5f" +checksum = "93f2635620bf0b9d4576eb7bb9a38a55df78bd1205d26fa994b25911a69f212f" dependencies = [ - "bitcoin_hashes 0.9.7", - "rand_core 0.4.2", + "bitcoin_hashes 0.11.0", "serde", "unicode-normalization", ] [[package]] name = "bitcoin" -version = "0.29.2" +version = "0.30.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0694ea59225b0c5f3cb405ff3f670e4828358ed26aec49dc352f730f0cb1a8a3" +checksum = "4e99ff7289b20a7385f66a0feda78af2fc119d28fb56aea8886a9cd0a4abdd75" dependencies = [ "bech32", - "bitcoin_hashes 0.11.0", + "bitcoin-private", + "bitcoin_hashes 0.12.0", + "hex_lit", "secp256k1", "serde", ] [[package]] -name = "bitcoin_hashes" -version = "0.9.7" +name = "bitcoin-private" +version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7ce18265ec2324ad075345d5814fbeed4f41f0a660055dc78840b74d19b874b1" +checksum = "73290177011694f38ec25e165d0387ab7ea749a4b81cd4c80dae5988229f7a57" [[package]] name = "bitcoin_hashes" version = "0.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "90064b8dee6815a6470d60bad07bbbaee885c0e12d04177138fa3291a01b7bc4" + +[[package]] +name = "bitcoin_hashes" +version = "0.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d7066118b13d4b20b23645932dfb3a81ce7e29f95726c2036fa33cd7b092501" dependencies = [ + "bitcoin-private", "serde", ] @@ -524,35 +506,33 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" [[package]] -name = "block-buffer" -version = "0.9.0" +name = "bitflags" +version = "2.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4" -dependencies = [ - "generic-array", -] +checksum = "630be753d4e58660abd17930c71b647fe46c27ea6b63cc59e1e3851406972e42" [[package]] name = "block-buffer" -version = "0.10.3" +version = "0.10.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "69cce20737498f97b993470a6e536b8523f0af7892a4f928cceb1ac5e52ebe7e" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" dependencies = [ "generic-array", ] [[package]] name = "blocking" -version = "1.3.0" +version = "1.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3c67b173a56acffd6d2326fb7ab938ba0b00a71480e14902b2591c87bc5741e8" +checksum = "77231a1c8f801696fc0123ec6150ce92cffb8e164a02afb9c8ddee0e9b65ad65" dependencies = [ "async-channel", "async-lock", "async-task", "atomic-waker", - "fastrand", + "fastrand 1.9.0", "futures-lite", + "log", ] [[package]] @@ -566,26 +546,53 @@ dependencies = [ "new_mime_guess", "proc-macro2", "quote", - "syn", + "syn 1.0.109", +] + +[[package]] +name = "brotli" +version = "3.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1a0b1dbcc8ae29329621f8d4f0d835787c1c38bb1401979b49d13b0b305ff68" +dependencies = [ + "alloc-no-stdlib", + "alloc-stdlib", + "brotli-decompressor", +] + +[[package]] +name = "brotli-decompressor" +version = "2.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b6561fd3f895a11e8f72af2cb7d22e08366bebc2b6b57f7744c4bda27034744" +dependencies = [ + "alloc-no-stdlib", + "alloc-stdlib", ] [[package]] name = "bumpalo" -version = "3.12.0" +version = "3.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3e2c3daef883ecc1b5d58c15adae93470a91d425f3532ba1695849656af3fc1" + +[[package]] +name = "byteorder" +version = "1.4.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0d261e256854913907f67ed06efbc3338dfe6179796deefc1ff763fc1aee5535" +checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" [[package]] name = "bytes" -version = "1.3.0" +version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dfb24e866b15a1af2a1b663f10c6b6b8f397a84aadb828f12e5b289ec23a3a3c" +checksum = "89b2fd2a0dcf38d7971e2194b6b6eebab45ae01067456a7fd93d5547a61b70be" [[package]] name = "cc" -version = "1.0.78" +version = "1.0.79" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a20104e2335ce8a659d6dd92a51a767a0c062599c73b343fd152cb401e828c3d" +checksum = "50d30906286121d95be3d479533b458f87493b30a4b5f79a607db8f5d11aa91f" dependencies = [ "jobserver", ] @@ -598,39 +605,30 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" [[package]] name = "chrono" -version = "0.4.23" +version = "0.4.26" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "16b0a3d9ed01224b22057780a37bb8c5dbfe1be8ba48678e7bf57ec4b385411f" +checksum = "ec837a71355b28f6556dbd569b37b3f363091c0bd4b2e735674521b4c5fd9bc5" dependencies = [ + "android-tzdata", "iana-time-zone", "js-sys", - "num-integer", "num-traits", "time 0.1.45", "wasm-bindgen", "winapi", ] -[[package]] -name = "cipher" -version = "0.2.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "12f8e7987cbd042a63249497f41aed09f8e65add917ea6566effbc56578d6801" -dependencies = [ - "generic-array", -] - [[package]] name = "clap" -version = "3.2.23" +version = "3.2.25" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "71655c45cb9845d3270c9d6df84ebe72b4dad3c2ba3f7023ad47c144e4e473a5" +checksum = "4ea181bf566f71cb9a5d17a59e1871af638180a18fb0035c92ae62b705207123" dependencies = [ "atty", - "bitflags", + "bitflags 1.3.2", "clap_derive", "clap_lex", - "indexmap", + "indexmap 1.9.3", "once_cell", "strsim", "termcolor", @@ -639,15 +637,15 @@ dependencies = [ [[package]] name = "clap_derive" -version = "3.2.18" +version = "3.2.25" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ea0c8bce528c4be4da13ea6fead8965e95b6073585a2f05204bd8f4119f82a65" +checksum = "ae6371b8bdc8b7d3959e9cf7b22d4435ef3e79e138688421ec654acf8c81b008" dependencies = [ "heck", "proc-macro-error", "proc-macro2", "quote", - "syn", + "syn 1.0.109", ] [[package]] @@ -659,195 +657,126 @@ dependencies = [ "os_str_bytes", ] -[[package]] -name = "codespan-reporting" -version = "0.11.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3538270d33cc669650c4b093848450d380def10c331d38c768e34cac80576e6e" -dependencies = [ - "termcolor", - "unicode-width", -] - [[package]] name = "concurrent-queue" -version = "2.1.0" +version = "2.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c278839b831783b70278b14df4d45e1beb1aad306c07bb796637de9a0e323e8e" +checksum = "62ec6771ecfa0762d24683ee5a32ad78487a3d3afdc0fb8cae19d2c5deb50b7c" dependencies = [ "crossbeam-utils", ] [[package]] name = "console" -version = "0.15.5" +version = "0.15.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c3d79fbe8970a77e3e34151cc13d3b3e248aa0faaecb9f6091fa07ebefe5ad60" +checksum = "c926e00cc70edefdc64d3a5ff31cc65bb97a3460097762bd23afb4d8145fccf8" dependencies = [ "encode_unicode", "lazy_static", "libc", "unicode-width", - "windows-sys", + "windows-sys 0.45.0", ] -[[package]] -name = "const_fn" -version = "0.4.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fbdcdcb6d86f71c5e97409ad45898af11cbc995b4ee8112d59095a28d376c935" - [[package]] name = "convert_case" version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" -[[package]] -name = "cookie" -version = "0.14.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "03a5d7b21829bc7b4bf4754a978a241ae54ea55a40f92bb20216e54096f4b951" -dependencies = [ - "aes-gcm", - "base64 0.13.1", - "hkdf", - "hmac", - "percent-encoding", - "rand 0.8.5", - "sha2 0.9.9", - "time 0.2.27", - "version_check", -] - [[package]] name = "core-foundation-sys" -version = "0.8.3" +version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5827cebf4670468b8772dd191856768aedcb1b0278a04f989f7766351917b9dc" +checksum = "e496a50fda8aacccc86d7529e2c1e0892dbd0f898a6b5645b5561b89c3210efa" [[package]] name = "cpufeatures" -version = "0.2.5" +version = "0.2.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "28d997bd5e24a5928dd43e46dc529867e207907fe0b239c3477d924f7f2ca320" +checksum = "a17b76ff3a4162b0b27f354a0c87015ddad39d35f9c0c36607a3bdd175dde1f1" dependencies = [ "libc", ] [[package]] -name = "cpuid-bool" -version = "0.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dcb25d077389e53838a8158c8e99174c5a9d902dee4904320db714f3c653ffba" - -[[package]] -name = "crossbeam-utils" -version = "0.8.14" +name = "crc32fast" +version = "1.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4fb766fa798726286dbbb842f174001dab8abc7b627a1dd86e0b7222a95d929f" +checksum = "b540bd8bc810d3885c6ea91e2018302f68baba2129ab3e88f32389ee9370880d" dependencies = [ "cfg-if", ] [[package]] -name = "crypto-common" -version = "0.1.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" -dependencies = [ - "generic-array", - "typenum", -] - -[[package]] -name = "crypto-mac" -version = "0.10.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bff07008ec701e8028e2ceb8f83f0e4274ee62bd2dbdc4fefff2e9a91824081a" -dependencies = [ - "generic-array", - "subtle", -] - -[[package]] -name = "ctor" -version = "0.1.26" +name = "crossbeam-channel" +version = "0.5.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6d2301688392eb071b0bf1a37be05c469d3cc4dbbd95df672fe28ab021e6a096" +checksum = "a33c2bf77f2df06183c3aa30d1e96c0695a313d4f9c453cc3762a6db39f99200" dependencies = [ - "quote", - "syn", + "cfg-if", + "crossbeam-utils", ] [[package]] -name = "ctr" -version = "0.6.0" +name = "crossbeam-deque" +version = "0.8.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fb4a30d54f7443bf3d6191dcd486aca19e67cb3c49fa7a06a319966346707e7f" +checksum = "ce6fd6f855243022dcecf8702fef0c297d4338e226845fe067f6341ad9fa0cef" dependencies = [ - "cipher", + "cfg-if", + "crossbeam-epoch", + "crossbeam-utils", ] [[package]] -name = "ctrlc" -version = "3.2.4" +name = "crossbeam-epoch" +version = "0.9.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1631ca6e3c59112501a9d87fd86f21591ff77acd31331e8a73f8d80a65bbdd71" +checksum = "ae211234986c545741a7dc064309f67ee1e5ad243d0e48335adc0484d960bcc7" dependencies = [ - "nix", - "windows-sys", + "autocfg", + "cfg-if", + "crossbeam-utils", + "memoffset", + "scopeguard", ] [[package]] -name = "cxx" -version = "1.0.86" +name = "crossbeam-utils" +version = "0.8.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "51d1075c37807dcf850c379432f0df05ba52cc30f279c5cfc43cc221ce7f8579" +checksum = "5a22b2d63d4d1dc0b7f1b6b2747dd0088008a9be28b6ddf0b1e7d335e3037294" dependencies = [ - "cc", - "cxxbridge-flags", - "cxxbridge-macro", - "link-cplusplus", + "cfg-if", ] [[package]] -name = "cxx-build" -version = "1.0.86" +name = "crypto-common" +version = "0.1.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5044281f61b27bc598f2f6647d480aed48d2bf52d6eb0b627d84c0361b17aa70" +checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" dependencies = [ - "cc", - "codespan-reporting", - "once_cell", - "proc-macro2", - "quote", - "scratch", - "syn", + "generic-array", + "typenum", ] [[package]] -name = "cxxbridge-flags" -version = "1.0.86" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61b50bc93ba22c27b0d31128d2d130a0a6b3d267ae27ef7e4fae2167dfe8781c" - -[[package]] -name = "cxxbridge-macro" -version = "1.0.86" +name = "ctrlc" +version = "3.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "39e61fda7e62115119469c7b3591fd913ecca96fb766cfd3f2e2502ab7bc87a5" +checksum = "2a011bbe2c35ce9c1f143b7af6f94f29a167beb4cd1d29e6740ce836f723120e" dependencies = [ - "proc-macro2", - "quote", - "syn", + "nix", + "windows-sys 0.48.0", ] [[package]] name = "darling" -version = "0.14.2" +version = "0.14.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b0dd3cd20dc6b5a876612a6e5accfe7f3dd883db6d07acfbf14c128f61550dfa" +checksum = "7b750cb3417fd1b327431a470f388520309479ab0bf5e323505daf0290cd3850" dependencies = [ "darling_core", "darling_macro", @@ -855,34 +784,34 @@ dependencies = [ [[package]] name = "darling_core" -version = "0.14.2" +version = "0.14.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a784d2ccaf7c98501746bf0be29b2022ba41fd62a2e622af997a03e9f972859f" +checksum = "109c1ca6e6b7f82cc233a97004ea8ed7ca123a9af07a8230878fcfda9b158bf0" dependencies = [ "fnv", "ident_case", "proc-macro2", "quote", "strsim", - "syn", + "syn 1.0.109", ] [[package]] name = "darling_macro" -version = "0.14.2" +version = "0.14.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7618812407e9402654622dd402b0a89dff9ba93badd6540781526117b92aab7e" +checksum = "a4aab4dbc9f7611d8b55048a3a16d2d010c2c8334e46304b40ac1cc14bf3b48e" dependencies = [ "darling_core", "quote", - "syn", + "syn 1.0.109", ] [[package]] name = "data-encoding" -version = "2.3.3" +version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "23d8666cb01533c39dde32bcbab8e227b4ed6679b2c925eba05feabea39508fb" +checksum = "c2e66c9d817f1720209181c316d28635c050fa304f9c79e47a520882661b7308" [[package]] name = "der-parser" @@ -900,13 +829,13 @@ dependencies = [ [[package]] name = "derive_arbitrary" -version = "1.2.2" +version = "1.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cf460bbff5f571bfc762da5102729f59f338be7db17a21fade44c5c4f5005350" +checksum = "53e0efad4403bfc52dc201159c4b842a246a14b98c64b55dfd0f2d89729dfeb8" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] @@ -927,7 +856,7 @@ dependencies = [ "darling", "proc-macro2", "quote", - "syn", + "syn 1.0.109", ] [[package]] @@ -937,7 +866,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ebcda35c7a396850a55ffeac740804b40ffec779b98fffbb1738f4033f0ee79e" dependencies = [ "derive_builder_core", - "syn", + "syn 1.0.109", ] [[package]] @@ -949,26 +878,17 @@ dependencies = [ "convert_case", "proc-macro2", "quote", - "rustc_version 0.4.0", - "syn", -] - -[[package]] -name = "digest" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066" -dependencies = [ - "generic-array", + "rustc_version", + "syn 1.0.109", ] [[package]] name = "digest" -version = "0.10.6" +version = "0.10.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8168378f4e5023e7218c89c891c0fd8ecdb5e5e4f18cb78f38cf245dd021e76f" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" dependencies = [ - "block-buffer 0.10.3", + "block-buffer", "crypto-common", ] @@ -983,41 +903,42 @@ dependencies = [ [[package]] name = "dirs" -version = "4.0.0" +version = "5.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ca3aa72a6f96ea37bbc5aa912f6788242832f75369bdfdadcb0e38423f100059" +checksum = "44c45a9d03d6676652bcb5e724c7e988de1acad23a711b5217ab9cbecbec2225" dependencies = [ "dirs-sys", ] [[package]] name = "dirs-sys" -version = "0.3.7" +version = "0.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1b1d1d91c932ef41c0f2663aa8b0ca0342d444d842c06914aa0a7e352d0bada6" +checksum = "520f05a5cbd335fae5a99ff7a6ab8627577660ee5cfd6a94a6a929b52ff0321c" dependencies = [ "libc", + "option-ext", "redox_users", - "winapi", + "windows-sys 0.48.0", ] -[[package]] -name = "discard" -version = "1.0.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "212d0f5754cb6769937f4501cc0e67f4f4483c8d2c3e1e922ee9edbe4ab4c7c0" - [[package]] name = "displaydoc" -version = "0.2.3" +version = "0.2.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3bf95dc3f046b9da4f2d51833c0d3547d8564ef6910f5c1ed130306a75b92886" +checksum = "487585f4d0c6655fe74905e2504d8ad6908e4db67f744eb140876906c2f3175d" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] +[[package]] +name = "either" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a26ae43d7bcc3b814de94796a5e736d4029efb0ee900c12e2d54c993ad1a1e07" + [[package]] name = "encode_unicode" version = "0.3.6" @@ -1026,9 +947,9 @@ checksum = "a357d28ed41a50f9c765dbfe56cbc04a64e53e5fc58ba79fbc34c10ef3df831f" [[package]] name = "encoding_rs" -version = "0.8.31" +version = "0.8.32" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9852635589dc9f9ea1b6fe9f05b50ef208c85c834a562f0c6abb1c475736ec2b" +checksum = "071a31f4ee85403370b58aca746f01041ede6f0da2730960ad001edc2b71b394" dependencies = [ "cfg-if", ] @@ -1046,15 +967,21 @@ dependencies = [ "termcolor", ] +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + [[package]] name = "errno" -version = "0.2.8" +version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f639046355ee4f37944e44f60642c6f3a7efa3cf6b78c78a0d989a8ce6c396a1" +checksum = "4bcfec3a70f97c962c307b2d2c56e358cf1d00b558d74262b5f929ee8cc7e73a" dependencies = [ "errno-dragonfly", "libc", - "winapi", + "windows-sys 0.48.0", ] [[package]] @@ -1075,13 +1002,29 @@ checksum = "0206175f82b8d6bf6652ff7d71a1e27fd2e4efde587fd368662814d6ec1d9ce0" [[package]] name = "fastrand" -version = "1.8.0" +version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a7a407cfaa3385c4ae6b23e84623d48c2798d06e3e6a1878f7f59f17b3f86499" +checksum = "e51093e27b0797c359783294ca4f0a911c270184cb10f85783b118614a1501be" dependencies = [ "instant", ] +[[package]] +name = "fastrand" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6999dc1837253364c2ebb0704ba97994bd874e8f195d665c50b7548f6ea92764" + +[[package]] +name = "flate2" +version = "1.0.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b9429470923de8e8cbd4d2dc513535400b4b3fef0319fb5c4e1f520a7bef743" +dependencies = [ + "crc32fast", + "miniz_oxide", +] + [[package]] name = "fnv" version = "1.0.7" @@ -1090,18 +1033,18 @@ checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" [[package]] name = "form_urlencoded" -version = "1.1.0" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a9c384f161156f5260c24a097c56119f9be8c798586aecc13afbcbe7b7e26bf8" +checksum = "a62bc1cf6f830c2ec14a513a9fb124d0a213a629668a4186f329db21fe045652" dependencies = [ "percent-encoding", ] [[package]] name = "futures" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38390104763dc37a5145a53c29c63c1290b5d316d6086ec32c293f6736051bb0" +checksum = "23342abe12aba583913b2e62f22225ff9c950774065e4bfb61a19cd9770fec40" dependencies = [ "futures-channel", "futures-core", @@ -1114,9 +1057,9 @@ dependencies = [ [[package]] name = "futures-channel" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "52ba265a92256105f45b719605a571ffe2d1f0fea3807304b522c1d778f79eed" +checksum = "955518d47e09b25bbebc7a18df10b81f0c766eaf4c4f1cccef2fca5f2a4fb5f2" dependencies = [ "futures-core", "futures-sink", @@ -1124,15 +1067,15 @@ dependencies = [ [[package]] name = "futures-core" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04909a7a7e4633ae6c4a9ab280aeb86da1236243a77b694a49eacd659a4bd3ac" +checksum = "4bca583b7e26f571124fe5b7561d49cb2868d79116cfa0eefce955557c6fee8c" [[package]] name = "futures-executor" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7acc85df6714c176ab5edf386123fafe217be88c0840ec11f199441134a074e2" +checksum = "ccecee823288125bd88b4d7f565c9e58e41858e47ab72e8ea2d64e93624386e0" dependencies = [ "futures-core", "futures-task", @@ -1141,17 +1084,17 @@ dependencies = [ [[package]] name = "futures-io" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "00f5fb52a06bdcadeb54e8d3671f8888a39697dcb0b81b23b55174030427f4eb" +checksum = "4fff74096e71ed47f8e023204cfd0aa1289cd54ae5430a9523be060cdb849964" [[package]] name = "futures-lite" -version = "1.12.0" +version = "1.13.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7694489acd39452c77daa48516b894c153f192c3578d5a839b62c58099fcbf48" +checksum = "49a9d51ce47660b1e808d3c990b4709f2f415d928835a17dfd16991515c46bce" dependencies = [ - "fastrand", + "fastrand 1.9.0", "futures-core", "futures-io", "memchr", @@ -1162,43 +1105,42 @@ dependencies = [ [[package]] name = "futures-macro" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bdfb8ce053d86b91919aad980c220b1fb8401a9394410e1c289ed7e66b61835d" +checksum = "89ca545a94061b6365f2c7355b4b32bd20df3ff95f02da9329b34ccc3bd6ee72" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] name = "futures-rustls" -version = "0.22.2" +version = "0.24.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d2411eed028cdf8c8034eaf21f9915f956b6c3abec4d4c7949ee67f0721127bd" +checksum = "35bd3cf68c183738046838e300353e4716c674dc5e56890de4826801a6622a28" dependencies = [ "futures-io", "rustls", - "webpki 0.22.0", ] [[package]] name = "futures-sink" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "39c15cf1a4aa79df40f1bb462fb39676d0ad9e366c2a33b590d7c66f4f81fcf9" +checksum = "f43be4fe21a13b9781a69afa4985b0f6ee0e1afab2c6f454a8cf30e2b2237b6e" [[package]] name = "futures-task" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2ffb393ac5d9a6eaa9d3fdf37ae2776656b706e200c8e16b1bdb227f5198e6ea" +checksum = "76d3d132be6c0e6aa1534069c705a74a5997a356c0dc2f86a47765e5617c5b65" [[package]] name = "futures-util" -version = "0.3.25" +version = "0.3.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "197676987abd2f9cadff84926f410af1c183608d36641465df73ae8211dc65d6" +checksum = "26b01e40b772d54cf6c6d721c1d1abd0647a0106a12ecaa1c186273392a69533" dependencies = [ "futures-channel", "futures-core", @@ -1224,9 +1166,9 @@ dependencies = [ [[package]] name = "generic-array" -version = "0.14.6" +version = "0.14.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bff49e947297f3312447abdca79f45f4738097cc82b06e72054d2223f601f1b9" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" dependencies = [ "typenum", "version_check", @@ -1254,36 +1196,26 @@ dependencies = [ [[package]] name = "getrandom" -version = "0.2.8" +version = "0.2.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c05aeb6a22b8f62540c194aac980f2115af067bfe15a0734d7277a768d396b31" +checksum = "be4136b2a15dd319360be1c07d9933517ccf0be8f16bf62a3bee4f0d618df427" dependencies = [ "cfg-if", "libc", "wasi 0.11.0+wasi-snapshot-preview1", ] -[[package]] -name = "ghash" -version = "0.3.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "97304e4cd182c3846f7575ced3890c53012ce534ad9114046b0a9e00bb30a375" -dependencies = [ - "opaque-debug", - "polyval", -] - [[package]] name = "gimli" -version = "0.27.0" +version = "0.27.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dec7af912d60cdbd3677c1af9352ebae6fb8394d165568a2234df0fa00f87793" +checksum = "b6c80984affa11d98d1b88b66ac8853f143217b399d3c74116778ff8fdb4ed2e" [[package]] name = "gloo-timers" -version = "0.2.5" +version = "0.2.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "98c4a8d6391675c6b2ee1a6c8d06e8e2d03605c44cec1270675985a4c2a5500b" +checksum = "9b995a66bb87bebce9a0f4a95aed01daca4872c050bfcb21653361c03bc35e5c" dependencies = [ "futures-channel", "futures-core", @@ -1293,9 +1225,9 @@ dependencies = [ [[package]] name = "h2" -version = "0.3.15" +version = "0.3.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f9f29bc9dda355256b2916cf526ab02ce0aeaaaf2bad60d65ef3f12f11dd0f4" +checksum = "97ec8491ebaf99c8eaa73058b045fe58073cd6be7f596ac993ced0b0a0c01049" dependencies = [ "bytes", "fnv", @@ -1303,7 +1235,7 @@ dependencies = [ "futures-sink", "futures-util", "http", - "indexmap", + "indexmap 1.9.3", "slab", "tokio", "tokio-util", @@ -1317,54 +1249,68 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" [[package]] -name = "heck" -version = "0.4.0" +name = "hashbrown" +version = "0.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2540771e65fc8cb83cd6e8a237f70c319bd5c29f78ed1084ba5d50eeac86f7f9" +checksum = "2c6201b9ff9fd90a5a3bac2e56a830d0caa509576f0e503818ee82c181b3437a" [[package]] -name = "hermit-abi" -version = "0.1.19" +name = "headers" +version = "0.3.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" +checksum = "f3e372db8e5c0d213e0cd0b9be18be2aca3d44cf2fe30a9d46a65581cd454584" dependencies = [ - "libc", + "base64 0.13.1", + "bitflags 1.3.2", + "bytes", + "headers-core", + "http", + "httpdate", + "mime", + "sha1", ] [[package]] -name = "hermit-abi" -version = "0.2.6" +name = "headers-core" +version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ee512640fe35acbfb4bb779db6f0d80704c2cacfa2e39b601ef3e3f47d1ae4c7" +checksum = "e7f66481bfee273957b1f20485a4ff3362987f85b2c236580d81b4eb7a326429" dependencies = [ - "libc", + "http", ] [[package]] -name = "hex" -version = "0.4.3" +name = "heck" +version = "0.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" +checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8" [[package]] -name = "hkdf" -version = "0.10.0" +name = "hermit-abi" +version = "0.1.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "51ab2f639c231793c5f6114bdb9bbe50a7dbbfcd7c7c6bd8475dec2d991e964f" +checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" dependencies = [ - "digest 0.9.0", - "hmac", + "libc", ] [[package]] -name = "hmac" -version = "0.10.1" +name = "hermit-abi" +version = "0.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c1441c6b1e930e2817404b5046f1f989899143a12bf92de603b69f4e0aee1e15" -dependencies = [ - "crypto-mac", - "digest 0.9.0", -] +checksum = "443144c8cdadd93ebf52ddb4056d257f5b52c04d3c804e657d19eb73fc33668b" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "hex_lit" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3011d1213f159867b13cfd6ac92d2cd5f1345762c63be3554e84092d85a50bbd" [[package]] name = "html-escaper" @@ -1374,9 +1320,9 @@ checksum = "459a0ca33ee92551e0a3bb1774f2d3bdd1c09fb6341845736662dd25e1fcb52a" [[package]] name = "http" -version = "0.2.8" +version = "0.2.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "75f43d41e26995c17e71ee126451dd3941010b0514a81a9d11f3b341debc2399" +checksum = "bd6effc99afb63425aff9b05836f029929e345a6148a14b7ecd5ab67af944482" dependencies = [ "bytes", "fnv", @@ -1396,9 +1342,9 @@ dependencies = [ [[package]] name = "http-range-header" -version = "0.3.0" +version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0bfe8eed0a9285ef776bb792479ea3834e8b94e13d615c2f66d03dd50a435a29" +checksum = "add0ab9360ddbd88cfeb3bd9574a1d85cfdfa14db10b3e21d3700dbc4328758f" [[package]] name = "http-types" @@ -1408,9 +1354,7 @@ checksum = "6e9b187a72d63adbfba487f48095306ac823049cb504ee195541e91c7775f5ad" dependencies = [ "anyhow", "async-channel", - "async-std", "base64 0.13.1", - "cookie", "futures-lite", "infer", "pin-project-lite", @@ -1442,9 +1386,9 @@ checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" [[package]] name = "hyper" -version = "0.14.23" +version = "0.14.27" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "034711faac9d2166cb1baf1a2fb0b60b1f277f8492fd72176c17f3515e1abd3c" +checksum = "ffb1cfd654a8219eaef89881fdb3bb3b1cdc5fa75ded05d6933b2b382e395468" dependencies = [ "bytes", "futures-channel", @@ -1466,26 +1410,25 @@ dependencies = [ [[package]] name = "iana-time-zone" -version = "0.1.53" +version = "0.1.57" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "64c122667b287044802d6ce17ee2ddf13207ed924c712de9a66a5814d5b64765" +checksum = "2fad5b825842d2b38bd206f3e81d6957625fd7f0a361e345c30e01a0ae2dd613" dependencies = [ "android_system_properties", "core-foundation-sys", "iana-time-zone-haiku", "js-sys", "wasm-bindgen", - "winapi", + "windows", ] [[package]] name = "iana-time-zone-haiku" -version = "0.1.1" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0703ae284fc167426161c2e3f1da3ea71d94b21bedbcc9494e92b28e334e3dca" +checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f" dependencies = [ - "cxx", - "cxx-build", + "cc", ] [[package]] @@ -1496,32 +1439,42 @@ checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" [[package]] name = "idna" -version = "0.2.1" +version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "de910d521f7cc3135c4de8db1cb910e0b5ed1dc6f57c381cd07e8e661ce10094" +checksum = "7d20d6b07bfbc108882d88ed8e37d39636dcc260e15e30c45e6ba089610b917c" dependencies = [ - "matches", "unicode-bidi", "unicode-normalization", ] [[package]] name = "indexmap" -version = "1.9.2" +version = "1.9.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1885e79c1fc4b10f0e172c475f458b7f7b93061064d98c3293e98c5ba0c8b399" +checksum = "bd070e393353796e801d209ad339e89596eb4c8d430d18ede6a1cced8fafbd99" dependencies = [ "autocfg", - "hashbrown", + "hashbrown 0.12.3", +] + +[[package]] +name = "indexmap" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d5477fe2230a79769d8dc68e0eabf5437907c0457a5614a9e8dddb67f65eb65d" +dependencies = [ + "equivalent", + "hashbrown 0.14.0", ] [[package]] name = "indicatif" -version = "0.17.3" +version = "0.17.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cef509aa9bc73864d6756f0d34d35504af3cf0844373afe9b8669a5b8005a729" +checksum = "8ff8cc23a7393a397ed1d7f56e6365cba772aba9f9912ab968b03043c395d057" dependencies = [ "console", + "instant", "number_prefix", "portable-atomic", "unicode-width", @@ -1544,54 +1497,55 @@ dependencies = [ [[package]] name = "io-lifetimes" -version = "1.0.4" +version = "1.0.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e7d6c6f8c91b4b9ed43484ad1a938e393caf35960fce7f82a040497207bd8e9e" +checksum = "eae7b9aee968036d54dce06cebaefd919e4472e753296daccd6d344e3e2df0c2" dependencies = [ + "hermit-abi 0.3.2", "libc", - "windows-sys", + "windows-sys 0.48.0", ] [[package]] name = "is-terminal" -version = "0.4.2" +version = "0.4.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "28dfb6c8100ccc63462345b67d1bbc3679177c75ee4bf59bf29c8b1d110b8189" +checksum = "cb0889898416213fab133e1d33a0e5858a48177452750691bde3666d0fdbaf8b" dependencies = [ - "hermit-abi 0.2.6", - "io-lifetimes", - "rustix", - "windows-sys", + "hermit-abi 0.3.2", + "rustix 0.38.4", + "windows-sys 0.48.0", ] [[package]] name = "itoa" -version = "1.0.5" +version = "1.0.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fad582f4b9e86b6caa621cabeb0963332d92eea04729ab12892c2533951e6440" +checksum = "af150ab688ff2122fcef229be89cb50dd66af9e01a4ff320cc137eecc9bacc38" [[package]] name = "jobserver" -version = "0.1.25" +version = "0.1.26" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "068b1ee6743e4d11fb9c6a1e6064b3693a1b600e7f5f5988047d98b3dc9fb90b" +checksum = "936cfd212a0155903bcbc060e316fb6cc7cbf2e1907329391ebadc1fe0ce77c2" dependencies = [ "libc", ] [[package]] name = "js-sys" -version = "0.3.60" +version = "0.3.64" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "49409df3e3bf0856b916e2ceaca09ee28e6871cf7d9ce97a692cacfdb2a25a47" +checksum = "c5f195fe497f702db0f318b07fdd68edb16955aed830df8363d837542f8f935a" dependencies = [ "wasm-bindgen", ] [[package]] name = "jsonrpc" -version = "0.14.0" -source = "git+https://github.com/apoelstra/rust-jsonrpc.git?rev=64b58797dd517c4de0cec769ff5652220801fe18#64b58797dd517c4de0cec769ff5652220801fe18" +version = "0.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8128f36b47411cd3f044be8c1f5cc0c9e24d1d1bfdc45f0a57897b32513053f2" dependencies = [ "base64 0.13.1", "serde", @@ -1615,15 +1569,15 @@ checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" [[package]] name = "libc" -version = "0.2.139" +version = "0.2.147" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "201de327520df007757c1f0adce6e827fe8562fbc28bfd9c15571c66ca1f5f79" +checksum = "b4668fb0ea861c1df094127ac5f1da3409a82116a4ba74fca2e58ef927159bb3" [[package]] name = "libfuzzer-sys" -version = "0.4.5" +version = "0.4.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8fff891139ee62800da71b7fd5b508d570b9ad95e614a53c6f453ca08366038" +checksum = "beb09950ae85a0a94b27676cccf37da5ff13f27076aa1adbc6545dd0d0e1bd4e" dependencies = [ "arbitrary", "cc", @@ -1631,59 +1585,52 @@ dependencies = [ ] [[package]] -name = "link-cplusplus" -version = "1.0.8" +name = "linux-raw-sys" +version = "0.3.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ecd207c9c713c34f95a097a5b029ac2ce6010530c7b49d7fea24d977dede04f5" -dependencies = [ - "cc", -] +checksum = "ef53942eb7bf7ff43a617b3e2c1c4a5ecf5944a7c1bc12d7ee39bbb15e5c1519" [[package]] name = "linux-raw-sys" -version = "0.1.4" +version = "0.4.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f051f77a7c8e6957c0696eac88f26b0117e54f52d3fc682ab19397a8812846a4" +checksum = "09fc20d2ca12cb9f044c93e3bd6d32d523e6e2ec3db4f7b2939cd99026ecd3f0" [[package]] name = "log" -version = "0.4.17" +version = "0.4.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e" +checksum = "b06a4cde4c0f271a446782e3eff8de789548ce57dbc8eca9292c27f4a42004b4" dependencies = [ - "cfg-if", "value-bag", ] -[[package]] -name = "matches" -version = "0.1.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a3e378b66a060d48947b590737b30a1be76706c8dd7b8ba0f2fe3989c68a853f" - [[package]] name = "matchit" version = "0.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b87248edafb776e59e6ee64a79086f65890d3510f2c656c000bf2a7e8a0aea40" -[[package]] -name = "maybe-uninit" -version = "2.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "60302e4db3a61da70c0cb7991976248362f30319e88850c487b9b95bbf059e00" - [[package]] name = "memchr" version = "2.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d" +[[package]] +name = "memoffset" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a634b1c61a95585bd15607c6ab0c4e5b226e695ff2800ba0cdccddf208c406c" +dependencies = [ + "autocfg", +] + [[package]] name = "mime" -version = "0.3.16" +version = "0.3.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2a60c7ce501c71e03a9c9c0d35b861413ae925bd979cc7a4e30d060069aaac8d" +checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" [[package]] name = "mime_guess" @@ -1703,32 +1650,46 @@ checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a" [[package]] name = "miniscript" -version = "9.0.0" +version = "10.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "123a10aae81d0712ecc09b780f6f0ae0b0f506a5c4c912974725760d59ba073e" +checksum = "1eb102b66b2127a872dbcc73095b7b47aeb9d92f7b03c2b2298253ffc82c7594" dependencies = [ "bitcoin", + "bitcoin-private", ] [[package]] name = "miniz_oxide" -version = "0.6.2" +version = "0.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b275950c28b37e794e8c55d88aeb5e139d0ce23fdbbeda68f8d7174abdf9e8fa" +checksum = "e7810e0be55b428ada41041c41f32c9f1a42817901b4ccf45fa3d4b6561e74c7" dependencies = [ "adler", ] [[package]] name = "mio" -version = "0.8.5" +version = "0.8.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e5d732bc30207a6423068df043e3d02e0735b155ad7ce1a6f76fe2baa5b158de" +checksum = "927a765cd3fc26206e66b296465fa9d3e5ab003e651c1b3c060e7956d96b19d2" dependencies = [ "libc", - "log", "wasi 0.11.0+wasi-snapshot-preview1", - "windows-sys", + "windows-sys 0.48.0", +] + +[[package]] +name = "mp4" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "509348cba250e7b852a875100a2ddce7a36ee3abf881a681c756670c1774264d" +dependencies = [ + "byteorder", + "bytes", + "num-rational", + "serde", + "serde_json", + "thiserror", ] [[package]] @@ -1753,7 +1714,7 @@ version = "0.26.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bfdda3d196821d6af13126e40375cdf7da646a96114af134d5f417a9a1dc8e1a" dependencies = [ - "bitflags", + "bitflags 1.3.2", "cfg-if", "libc", "static_assertions", @@ -1769,6 +1730,15 @@ dependencies = [ "minimal-lexical", ] +[[package]] +name = "ntapi" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8a3895c6391c39d7fe7ebc444a87eb2991b2a0bc718fdabd071eec617fc68e4" +dependencies = [ + "winapi", +] + [[package]] name = "num-bigint" version = "0.4.3" @@ -1790,22 +1760,35 @@ dependencies = [ "num-traits", ] +[[package]] +name = "num-rational" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0638a1c9d0a3c0914158145bc76cff373a75a627e6ecbfb71cbe6f453a5a19b0" +dependencies = [ + "autocfg", + "num-bigint", + "num-integer", + "num-traits", + "serde", +] + [[package]] name = "num-traits" -version = "0.2.15" +version = "0.2.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "578ede34cf02f8924ab9447f50c28075b4d3e5b269972345e7e0372b38c6cdcd" +checksum = "f30b0abd723be7e2ffca1272140fac1a2f084c77ec3e123c192b66af1ee9e6c2" dependencies = [ "autocfg", ] [[package]] name = "num_cpus" -version = "1.15.0" +version = "1.16.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0fac9e2da13b5eb447a6ce3d392f23a29d8694bff781bf03a16cd9ac8697593b" +checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" dependencies = [ - "hermit-abi 0.2.6", + "hermit-abi 0.3.2", "libc", ] @@ -1817,9 +1800,9 @@ checksum = "830b246a0e5f20af87141b25c173cd1b609bd7779a4617d6ec582abaf90870f3" [[package]] name = "object" -version = "0.30.2" +version = "0.31.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2b8c786513eb403643f2a88c244c2aaa270ef2153f55094587d0c48a3cf22a83" +checksum = "8bda667d9f2b5051b8833f59f3bf748b28ef54f850f4fcb389a252aa383866d1" dependencies = [ "memchr", ] @@ -1835,23 +1818,24 @@ dependencies = [ [[package]] name = "once_cell" -version = "1.17.0" +version = "1.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6f61fba1741ea2b3d6a1e3178721804bb716a68a6aeba1149b5d52e3d464ea66" +checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d" [[package]] -name = "opaque-debug" -version = "0.3.0" +name = "option-ext" +version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" +checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" [[package]] name = "ord" -version = "0.4.0" +version = "0.8.1" dependencies = [ "anyhow", "axum", "axum-server", + "base64 0.21.2", "bech32", "bip39", "bitcoin", @@ -1866,12 +1850,14 @@ dependencies = [ "hex", "html-escaper", "http", + "hyper", "indicatif", "lazy_static", "log", "mime", "mime_guess", "miniscript", + "mp4", "ord-bitcoincore-rpc", "pulldown-cmark", "redb", @@ -1882,7 +1868,8 @@ dependencies = [ "rustls-acme", "serde", "serde_json", - "sys-info", + "serde_yaml", + "sysinfo", "tempfile", "tokio", "tokio-stream", @@ -1892,9 +1879,11 @@ dependencies = [ [[package]] name = "ord-bitcoincore-rpc" -version = "0.16.4" -source = "git+https://github.com/casey/rust-bitcoincore-rpc?branch=ord#1bc2638f5a3049495a7aa953c4c6dac44fd1524b" +version = "0.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ece3c1158d3391127ddd7615868792a602745cf4f9fbea259c4449c9ed89814a" dependencies = [ + "bitcoin-private", "jsonrpc", "log", "ord-bitcoincore-rpc-json", @@ -1904,25 +1893,27 @@ dependencies = [ [[package]] name = "ord-bitcoincore-rpc-json" -version = "0.16.4" -source = "git+https://github.com/casey/rust-bitcoincore-rpc?branch=ord#1bc2638f5a3049495a7aa953c4c6dac44fd1524b" +version = "0.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45f46eb509cc95759461d3cb483ba51b2adbea0e31747cb1e3f388e71e13a321" dependencies = [ "bitcoin", + "bitcoin-private", "serde", "serde_json", ] [[package]] name = "os_str_bytes" -version = "6.4.1" +version = "6.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b7820b9daea5457c9f21c69448905d723fbd21136ccf521748f23fd49e723ee" +checksum = "4d5d9eb14b174ee9aa2ef96dc2b94637a2d4b6e7cb873c7e171f0c20c6cf3eac" [[package]] name = "parking" -version = "2.0.0" +version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "427c3892f9e783d91cc128285287e70a59e206ca452770ece88a76f7a3eddd72" +checksum = "14f2252c834a40ed9bb5422029649578e63aa341ac401f74e719dd1afda8394e" [[package]] name = "pem" @@ -1935,35 +1926,35 @@ dependencies = [ [[package]] name = "percent-encoding" -version = "2.2.0" +version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "478c572c3d73181ff3c2539045f6eb99e5491218eae919370993b890cdbdd98e" +checksum = "9b2a4787296e9989611394c33f193f676704af1686e70b8f8033ab5ba9a35a94" [[package]] name = "pin-project" -version = "1.0.12" +version = "1.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ad29a609b6bcd67fee905812e544992d216af9d755757c05ed2d0e15a74c6ecc" +checksum = "030ad2bc4db10a8944cb0d837f158bdfec4d4a4873ab701a95046770d11f8842" dependencies = [ "pin-project-internal", ] [[package]] name = "pin-project-internal" -version = "1.0.12" +version = "1.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "069bdb1e05adc7a8990dce9cc75370895fbe4e3d58b9b73bf1aee56359344a55" +checksum = "ec2e072ecce94ec471b13398d5402c188e76ac03cf74dd1a975161b23a3f6d9c" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] name = "pin-project-lite" -version = "0.2.9" +version = "0.2.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e0a7ae3ac2f1173085d398531c705756c94a4c56843785df85a60c1a0afac116" +checksum = "4c40d25201921e5ff0c862a505c6557ea88568a4e3ace775ab55e93f2f4f9d57" [[package]] name = "pin-utils" @@ -1973,34 +1964,25 @@ checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" [[package]] name = "polling" -version = "2.5.2" +version = "2.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22122d5ec4f9fe1b3916419b76be1e80bcb93f618d071d2edf841b137b2a2bd6" +checksum = "4b2d323e8ca7996b3e23126511a523f7e62924d93ecd5ae73b333815b0eb3dce" dependencies = [ "autocfg", + "bitflags 1.3.2", "cfg-if", + "concurrent-queue", "libc", "log", - "wepoll-ffi", - "windows-sys", -] - -[[package]] -name = "polyval" -version = "0.4.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eebcc4aa140b9abd2bc40d9c3f7ccec842679cd79045ac3a7ac698c1a064b7cd" -dependencies = [ - "cpuid-bool", - "opaque-debug", - "universal-hash", + "pin-project-lite", + "windows-sys 0.48.0", ] [[package]] name = "portable-atomic" -version = "0.3.19" +version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "26f6a7b87c2e435a3241addceeeff740ff8b7e76b74c13bf9acb17fa454ea00b" +checksum = "f32154ba0af3a075eefa1eda8bb414ee928f62303a54ea85b8d6638ff1a6ee9e" [[package]] name = "ppv-lite86" @@ -2017,7 +1999,7 @@ dependencies = [ "proc-macro-error-attr", "proc-macro2", "quote", - "syn", + "syn 1.0.109", "version_check", ] @@ -2032,28 +2014,22 @@ dependencies = [ "version_check", ] -[[package]] -name = "proc-macro-hack" -version = "0.5.20+deprecated" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068" - [[package]] name = "proc-macro2" -version = "1.0.50" +version = "1.0.66" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ef7d57beacfaf2d8aee5937dab7b7f28de3cb8b1828479bb5de2a7106f2bae2" +checksum = "18fb31db3f9bddb2ea821cde30a9f70117e3f119938b5ee630b7403aa6e2ead9" dependencies = [ "unicode-ident", ] [[package]] name = "pulldown-cmark" -version = "0.9.2" +version = "0.9.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2d9cc634bc78768157b5cbfe988ffcd1dcba95cd2b2f03a88316c08c6d00ed63" +checksum = "77a1a2f1f0a7ecff9c31abbe177637be0e97a0aef46cf8738ece09327985d998" dependencies = [ - "bitflags", + "bitflags 1.3.2", "getopts", "memchr", "unicase", @@ -2061,9 +2037,9 @@ dependencies = [ [[package]] name = "pyo3-build-config" -version = "0.17.3" +version = "0.19.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "28fcd1e73f06ec85bf3280c48c67e731d8290ad3d730f8be9dc07946923005c8" +checksum = "554db24f0b3c180a9c0b1268f91287ab3f17c162e15b54caaae5a6b3773396b0" dependencies = [ "once_cell", "target-lexicon", @@ -2071,9 +2047,9 @@ dependencies = [ [[package]] name = "quick-xml" -version = "0.27.1" +version = "0.30.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ffc053f057dd768a56f62cd7e434c42c831d296968997e9ac1f76ea7c2d14c41" +checksum = "eff6510e86862b57b210fd8cbe8ed3f0d7d600b9c2863cd4549a2e033c66e956" dependencies = [ "encoding_rs", "memchr", @@ -2081,9 +2057,9 @@ dependencies = [ [[package]] name = "quote" -version = "1.0.23" +version = "1.0.32" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8856d8364d252a14d474036ea1358d63c9e6965c8e5c1885c18f73d70bff9c7b" +checksum = "50f3b39ccfb720540debaa0164757101c08ecb8d326b15358ce76a62c7e85965" dependencies = [ "proc-macro2", ] @@ -2132,12 +2108,6 @@ dependencies = [ "rand_core 0.6.4", ] -[[package]] -name = "rand_core" -version = "0.4.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc" - [[package]] name = "rand_core" version = "0.5.1" @@ -2153,7 +2123,7 @@ version = "0.6.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" dependencies = [ - "getrandom 0.2.8", + "getrandom 0.2.10", ] [[package]] @@ -2165,22 +2135,45 @@ dependencies = [ "rand_core 0.5.1", ] +[[package]] +name = "rayon" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d2df5196e37bcc87abebc0053e20787d73847bb33134a69841207dd0a47f03b" +dependencies = [ + "either", + "rayon-core", +] + +[[package]] +name = "rayon-core" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b8f95bd6966f5c87776639160a66bd8ab9895d9d4ab01ddba9fc60661aebe8d" +dependencies = [ + "crossbeam-channel", + "crossbeam-deque", + "crossbeam-utils", + "num_cpus", +] + [[package]] name = "rcgen" -version = "0.9.3" +version = "0.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6413f3de1edee53342e6138e75b56d32e7bc6e332b3bd62d497b1929d4cfbcdd" +checksum = "ffbe84efe2f38dea12e9bfc1f65377fdf03e53a18cb3b995faedf7934c7e785b" dependencies = [ "pem", "ring", - "time 0.3.17", + "time 0.3.23", "yasna", ] [[package]] name = "redb" -version = "0.11.0" -source = "git+https://github.com/casey/redb.git?branch=ord#826fd633b2e26bb649cf7004406cb469fbb28837" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "717a806693d0e1ed6cc55b392066bf13e703dd835acf5c5888c74740f924d355" dependencies = [ "libc", "pyo3-build-config", @@ -2192,7 +2185,16 @@ version = "0.2.16" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" dependencies = [ - "bitflags", + "bitflags 1.3.2", +] + +[[package]] +name = "redox_syscall" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "567664f262709473930a4bf9e51bf2ebf3348f2e748ccc50dea20646858f8f29" +dependencies = [ + "bitflags 1.3.2", ] [[package]] @@ -2201,36 +2203,39 @@ version = "0.4.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b033d837a7cf162d7993aded9304e30a83213c648b6e389db233191f891e5c2b" dependencies = [ - "getrandom 0.2.8", - "redox_syscall", + "getrandom 0.2.10", + "redox_syscall 0.2.16", "thiserror", ] [[package]] name = "regex" -version = "1.7.1" +version = "1.9.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "48aaa5748ba571fb95cd2c85c09f629215d3a6ece942baa100950af03a34f733" +checksum = "b2eae68fc220f7cf2532e4494aded17545fce192d59cd996e0fe7887f4ceb575" dependencies = [ "aho-corasick", "memchr", + "regex-automata", "regex-syntax", ] [[package]] -name = "regex-syntax" -version = "0.6.28" +name = "regex-automata" +version = "0.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "456c603be3e8d448b072f410900c09faf164fbce2d480456f50eea6e25f9c848" +checksum = "39354c10dd07468c2e73926b23bb9c2caca74c5501e38a35da70406f1d923310" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] [[package]] -name = "remove_dir_all" -version = "0.5.3" +name = "regex-syntax" +version = "0.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3acd125665422973a33ac9d3dd2df85edad0f4ae9b00dafb1a05e43a9f5ef8e7" -dependencies = [ - "winapi", -] +checksum = "e5ea92a5b6195c6ef2a0295ea818b312502c6fc94dde986c5553242e18fd4ce2" [[package]] name = "ring" @@ -2249,9 +2254,9 @@ dependencies = [ [[package]] name = "rss" -version = "2.0.2" +version = "2.0.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "14816ec8b4e58b34a36adba66dcadd3e1e221dcb0fb2fd83e7c5129ea1a72458" +checksum = "32100fb1eea9edc4be79553ae97914add78b55892a517995fe96d1ada50d09d7" dependencies = [ "atom_syndication", "derive_builder", @@ -2261,9 +2266,9 @@ dependencies = [ [[package]] name = "rust-embed" -version = "6.4.2" +version = "6.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "283ffe2f866869428c92e0d61c2f35dfb4355293cdfdc48f49e895c15f1333d1" +checksum = "a36224c3276f8c4ebc8c20f158eca7ca4359c8db89991c4925132aaaf6702661" dependencies = [ "rust-embed-impl", "rust-embed-utils", @@ -2272,41 +2277,32 @@ dependencies = [ [[package]] name = "rust-embed-impl" -version = "6.3.1" +version = "6.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "31ab23d42d71fb9be1b643fe6765d292c5e14d46912d13f3ae2815ca048ea04d" +checksum = "49b94b81e5b2c284684141a2fb9e2a31be90638caf040bf9afbc5a0416afe1ac" dependencies = [ "proc-macro2", "quote", "rust-embed-utils", - "syn", + "syn 2.0.27", "walkdir", ] [[package]] name = "rust-embed-utils" -version = "7.3.0" +version = "7.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c1669d81dfabd1b5f8e2856b8bbe146c6192b0ba22162edc738ac0a5de18f054" +checksum = "9d38ff6bf570dc3bb7100fce9f7b60c33fa71d80e88da3f2580df4ff2bdded74" dependencies = [ - "sha2 0.10.6", + "sha2", "walkdir", ] [[package]] name = "rustc-demangle" -version = "0.1.21" +version = "0.1.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7ef03e0a2b150c7a90d01faf6254c9c48a41e95fb2a8c2ac1c6f0d2b9aefc342" - -[[package]] -name = "rustc_version" -version = "0.2.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "138e3e0acb6c9fb258b19b67cb8abd63c00679d2851805ea151465464fe9030a" -dependencies = [ - "semver 0.9.0", -] +checksum = "d626bb9dae77e28219937af045c257c28bfd3f69333c512553507f5f9798cb76" [[package]] name = "rustc_version" @@ -2314,7 +2310,7 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bfa0f585226d2e68097d4f95d113b15b83a82e819ab25717ec0590d9584ef366" dependencies = [ - "semver 1.0.16", + "semver", ] [[package]] @@ -2328,35 +2324,48 @@ dependencies = [ [[package]] name = "rustix" -version = "0.36.7" +version = "0.37.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d4fdebc4b395b7fbb9ab11e462e20ed9051e7b16e42d24042c776eca0ac81b03" +checksum = "4d69718bf81c6127a49dc64e44a742e8bb9213c0ff8869a22c308f84c1d4ab06" dependencies = [ - "bitflags", + "bitflags 1.3.2", "errno", "io-lifetimes", "libc", - "linux-raw-sys", - "windows-sys", + "linux-raw-sys 0.3.8", + "windows-sys 0.48.0", +] + +[[package]] +name = "rustix" +version = "0.38.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a962918ea88d644592894bc6dc55acc6c0956488adcebbfb6e273506b7fd6e5" +dependencies = [ + "bitflags 2.3.3", + "errno", + "libc", + "linux-raw-sys 0.4.3", + "windows-sys 0.48.0", ] [[package]] name = "rustls" -version = "0.20.8" +version = "0.21.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fff78fc74d175294f4e83b28343315ffcfb114b156f0185e9741cb5570f50e2f" +checksum = "79ea77c539259495ce8ca47f53e66ae0330a8819f67e23ac96ca02f50e7b7d36" dependencies = [ "log", "ring", + "rustls-webpki 0.101.2", "sct", - "webpki 0.22.0", ] [[package]] name = "rustls-acme" -version = "0.5.2" +version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3718e4ae9807cd8ec87578b742c18c2627edca926bc17d0d8dfbf43ca99d38ce" +checksum = "6b252541bcfab8ae3ed2240b9e6d5a4254a40b66597d9adb7f74f9f81b2d4f21" dependencies = [ "async-h1", "async-io", @@ -2369,7 +2378,6 @@ dependencies = [ "http-types", "log", "pem", - "pin-project", "rcgen", "ring", "rustls", @@ -2386,24 +2394,44 @@ dependencies = [ [[package]] name = "rustls-pemfile" -version = "1.0.2" +version = "1.0.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d194b56d58803a43635bdc398cd17e383d6f71f9182b9a192c127ca42494a59b" +checksum = "2d3987094b1d07b653b7dfdc3f70ce9a1da9c51ac18c1b06b662e4f9a0e9f4b2" dependencies = [ - "base64 0.21.0", + "base64 0.21.2", +] + +[[package]] +name = "rustls-webpki" +version = "0.100.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d6207cd5ed3d8dca7816f8f3725513a34609c0c765bf652b8c3cb4cfd87db46b" +dependencies = [ + "ring", + "untrusted", +] + +[[package]] +name = "rustls-webpki" +version = "0.101.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "513722fd73ad80a71f72b61009ea1b584bcfa1483ca93949c8f290298837fa59" +dependencies = [ + "ring", + "untrusted", ] [[package]] name = "rustversion" -version = "1.0.11" +version = "1.0.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5583e89e108996506031660fe09baa5011b9dd0341b89029313006d1fb508d70" +checksum = "7ffc183a10b4478d04cbbbfc96d0873219d962dd5accaff2ffbd4ceb7df837f4" [[package]] name = "ryu" -version = "1.0.12" +version = "1.0.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7b4b9743ed687d4b4bcedf9ff5eaa7398495ae14e61cba0a295704edbc7decde" +checksum = "1ad4cc8da4ef723ed60bced201181d83791ad433213d8c24efffda1eec85d741" [[package]] name = "same-file" @@ -2415,10 +2443,10 @@ dependencies = [ ] [[package]] -name = "scratch" -version = "1.0.3" +name = "scopeguard" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ddccb15bcce173023b3fedd9436f882a0739b8dfb45e4f6b6002bee5929f61b2" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" [[package]] name = "sct" @@ -2432,11 +2460,11 @@ dependencies = [ [[package]] name = "secp256k1" -version = "0.24.3" +version = "0.27.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6b1629c9c557ef9b293568b338dddfc8208c98a18c59d722a9d53f859d9c9b62" +checksum = "25996b82292a7a57ed3508f052cfff8640d38d32018784acd714758b43da9c8f" dependencies = [ - "bitcoin_hashes 0.11.0", + "bitcoin_hashes 0.12.0", "rand 0.8.5", "secp256k1-sys", "serde", @@ -2444,59 +2472,44 @@ dependencies = [ [[package]] name = "secp256k1-sys" -version = "0.6.1" +version = "0.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "83080e2c2fc1006e625be82e5d1eb6a43b7fd9578b617fcc55814daf286bba4b" +checksum = "70a129b9e9efbfb223753b9163c4ab3b13cff7fd9c7f010fbac25ab4099fa07e" dependencies = [ "cc", ] [[package]] name = "semver" -version = "0.9.0" +version = "1.0.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1d7eb9ef2c18661902cc47e535f9bc51b78acd254da71d375c2f6720d9a40403" -dependencies = [ - "semver-parser", -] - -[[package]] -name = "semver" -version = "1.0.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "58bc9567378fc7690d6b2addae4e60ac2eeea07becb2c64b9f218b53865cba2a" - -[[package]] -name = "semver-parser" -version = "0.7.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3" +checksum = "b0293b4b29daaf487284529cc2f5675b8e57c61f70167ba415a463651fd6a918" [[package]] name = "serde" -version = "1.0.152" +version = "1.0.177" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bb7d1f0d3021d347a83e556fc4683dea2ea09d87bccdf88ff5c12545d89d5efb" +checksum = "63ba2516aa6bf82e0b19ca8b50019d52df58455d3cf9bdaf6315225fdd0c560a" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.152" +version = "1.0.177" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "af487d118eecd09402d70a5d72551860e788df87b464af30e5ea6a38c75c541e" +checksum = "401797fe7833d72109fedec6bfcbe67c0eed9b99772f26eb8afd261f0abc6fd3" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] name = "serde_json" -version = "1.0.91" +version = "1.0.104" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "877c235533714907a8c2464236f5c4b2a17262ef1bd71f38f35ea592c8da6883" +checksum = "076066c5f1078eac5b722a31827a8832fe108bed65dfa75e233c89f8206e976c" dependencies = [ "itoa", "ryu", @@ -2505,10 +2518,11 @@ dependencies = [ [[package]] name = "serde_path_to_error" -version = "0.1.9" +version = "0.1.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "26b04f22b563c91331a10074bda3dd5492e3cc39d56bd557e91c0af42b6c7341" +checksum = "4beec8bce849d58d06238cb50db2e1c417cfeafa4c63f692b15c82b7c80f8335" dependencies = [ + "itoa", "serde", ] @@ -2536,49 +2550,45 @@ dependencies = [ ] [[package]] -name = "sha1" -version = "0.6.1" +name = "serde_yaml" +version = "0.9.25" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c1da05c97445caa12d05e848c4a4fcbbea29e748ac28f7e80e9b010392063770" +checksum = "1a49e178e4452f45cb61d0cd8cebc1b0fafd3e41929e996cef79aa3aca91f574" dependencies = [ - "sha1_smol", + "indexmap 2.0.0", + "itoa", + "ryu", + "serde", + "unsafe-libyaml", ] [[package]] -name = "sha1_smol" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae1a47186c03a32177042e55dbc5fd5aee900b8e0069a8d70fba96a9375cd012" - -[[package]] -name = "sha2" -version = "0.9.9" +name = "sha1" +version = "0.10.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800" +checksum = "f04293dc80c3993519f2d7f6f511707ee7094fe0c6d3406feb330cdb3540eba3" dependencies = [ - "block-buffer 0.9.0", "cfg-if", "cpufeatures", - "digest 0.9.0", - "opaque-debug", + "digest", ] [[package]] name = "sha2" -version = "0.10.6" +version = "0.10.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "82e6b795fe2e3b1e845bafcb27aa35405c4d47cdfc92af5fc8d3002f76cebdc0" +checksum = "479fb9d862239e610720565ca91403019f2f00410f1864c5aa7479b950a76ed8" dependencies = [ "cfg-if", "cpufeatures", - "digest 0.10.6", + "digest", ] [[package]] name = "signal-hook" -version = "0.3.14" +version = "0.3.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a253b5e89e2698464fc26b545c9edceb338e18a89effeeecfea192c3025be29d" +checksum = "8621587d4798caf8eb44879d42e56b9a93ea5dcd315a6487c357130095b62801" dependencies = [ "libc", "signal-hook-registry", @@ -2586,9 +2596,9 @@ dependencies = [ [[package]] name = "signal-hook-registry" -version = "1.4.0" +version = "1.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e51e73328dc4ac0c7ccbda3a494dfa03df1de2f46018127f60c693f2648455b0" +checksum = "d8229b473baa5980ac72ef434c4415e70c4b5e71b423043adb4ba059f89c99a1" dependencies = [ "libc", ] @@ -2604,22 +2614,13 @@ dependencies = [ [[package]] name = "slab" -version = "0.4.7" +version = "0.4.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4614a76b2a8be0058caa9dbbaf66d988527d86d003c11a94fbd335d7661edcef" +checksum = "6528351c9bc8ab22353f9d776db39a20288e8d6c37ef8cfe3317cf875eecfc2d" dependencies = [ "autocfg", ] -[[package]] -name = "smallvec" -version = "0.6.14" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b97fcaeba89edba30f044a10c6a3cc39df9c3f17d7cd829dd1446cab35f890e0" -dependencies = [ - "maybe-uninit", -] - [[package]] name = "smol" version = "1.3.0" @@ -2639,9 +2640,9 @@ dependencies = [ [[package]] name = "socket2" -version = "0.4.7" +version = "0.4.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "02e2d2db9033d13a1567121ddd7a095ee144db4e1ca1b1bda3419bc0da294ebd" +checksum = "64a4a911eed85daf18834cfaa86a79b7d266ff93ff5ba14005426219480ed662" dependencies = [ "libc", "winapi", @@ -2653,15 +2654,6 @@ version = "0.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6e63cff320ae2c57904679ba7cb63280a3dc4613885beafb148ee7bf9aa9042d" -[[package]] -name = "standback" -version = "0.2.17" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e113fb6f3de07a243d434a56ec6f186dfd51cb08448239fe7bcae73f87ff28ff" -dependencies = [ - "version_check", -] - [[package]] name = "static_assertions" version = "1.1.0" @@ -2669,71 +2661,27 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" [[package]] -name = "stdweb" -version = "0.4.20" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d022496b16281348b52d0e30ae99e01a73d737b2f45d38fed4edf79f9325a1d5" -dependencies = [ - "discard", - "rustc_version 0.2.3", - "stdweb-derive", - "stdweb-internal-macros", - "stdweb-internal-runtime", - "wasm-bindgen", -] - -[[package]] -name = "stdweb-derive" -version = "0.5.3" +name = "strsim" +version = "0.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c87a60a40fccc84bef0652345bbbbbe20a605bf5d0ce81719fc476f5c03b50ef" -dependencies = [ - "proc-macro2", - "quote", - "serde", - "serde_derive", - "syn", -] +checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623" [[package]] -name = "stdweb-internal-macros" -version = "0.2.9" +name = "syn" +version = "1.0.109" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "58fa5ff6ad0d98d1ffa8cb115892b6e69d67799f6763e162a1c9db421dc22e11" +checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237" dependencies = [ - "base-x", "proc-macro2", "quote", - "serde", - "serde_derive", - "serde_json", - "sha1", - "syn", + "unicode-ident", ] -[[package]] -name = "stdweb-internal-runtime" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "213701ba3370744dcd1a12960caa4843b3d68b4d1c0a5d575e0d65b2ee9d16c0" - -[[package]] -name = "strsim" -version = "0.10.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623" - -[[package]] -name = "subtle" -version = "2.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6bdef32e8150c2a081110b42772ffe7d7c9032b606bc226c8260fd97e0976601" - [[package]] name = "syn" -version = "1.0.107" +version = "2.0.27" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f4064b5b16e03ae50984a5a8ed5d4f8803e6bc1fd170a3cda91a1be4b18e3f5" +checksum = "b60f673f44a8255b9c8c657daf66a596d435f2da81a555b06dc644d080ba45e0" dependencies = [ "proc-macro2", "quote", @@ -2742,9 +2690,9 @@ dependencies = [ [[package]] name = "sync_wrapper" -version = "0.1.1" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "20518fe4a4c9acf048008599e464deb21beeae3d3578418951a189c235a7a9a8" +checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" [[package]] name = "synstructure" @@ -2754,38 +2702,42 @@ checksum = "f36bdaa60a83aca3921b5259d5400cbf5e90fc51931376a9bd4a0eb79aa7210f" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 1.0.109", "unicode-xid", ] [[package]] -name = "sys-info" -version = "0.9.1" +name = "sysinfo" +version = "0.29.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0b3a0d0aba8bf96a0e1ddfdc352fc53b3df7f39318c71854910c3c4b024ae52c" +checksum = "165d6d8539689e3d3bc8b98ac59541e1f21c7de7c85d60dc80e43ae0ed2113db" dependencies = [ - "cc", + "cfg-if", + "core-foundation-sys", "libc", + "ntapi", + "once_cell", + "rayon", + "winapi", ] [[package]] name = "target-lexicon" -version = "0.12.5" +version = "0.12.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9410d0f6853b1d94f0e519fb95df60f29d2c1eff2d921ffdf01a4c8a3b54f12d" +checksum = "1d2faeef5759ab89935255b1a4cd98e0baf99d1085e37d36599c625dac49ae8e" [[package]] name = "tempfile" -version = "3.3.0" +version = "3.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5cdb1ef4eaeeaddc8fbd371e5017057064af0911902ef36b39801f67cc6d79e4" +checksum = "5486094ee78b2e5038a6382ed7645bc084dc2ec433426ca4c3cb61e2007b8998" dependencies = [ "cfg-if", - "fastrand", - "libc", - "redox_syscall", - "remove_dir_all", - "winapi", + "fastrand 2.0.0", + "redox_syscall 0.3.5", + "rustix 0.38.4", + "windows-sys 0.48.0", ] [[package]] @@ -2805,22 +2757,22 @@ checksum = "222a222a5bfe1bba4a77b45ec488a741b3cb8872e5e499451fd7d0129c9c7c3d" [[package]] name = "thiserror" -version = "1.0.38" +version = "1.0.44" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6a9cd18aa97d5c45c6603caea1da6628790b37f7a34b6ca89522331c5180fed0" +checksum = "611040a08a0439f8248d1990b111c95baa9c704c805fa1f62104b39655fd7f90" dependencies = [ "thiserror-impl", ] [[package]] name = "thiserror-impl" -version = "1.0.38" +version = "1.0.44" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1fb327af4685e4d03fa8cbcf1716380da910eeb2bb8be417e7f9fd3fb164f36f" +checksum = "090198534930841fab3a5d1bb637cde49e339654e606195f8d9c76eeb081dc96" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] @@ -2836,114 +2788,90 @@ dependencies = [ [[package]] name = "time" -version = "0.2.27" +version = "0.3.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4752a97f8eebd6854ff91f1c1824cd6160626ac4bd44287f7f4ea2035a02a242" -dependencies = [ - "const_fn", - "libc", - "standback", - "stdweb", - "time-macros 0.1.1", - "version_check", - "winapi", -] - -[[package]] -name = "time" -version = "0.3.17" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a561bf4617eebd33bca6434b988f39ed798e527f51a1e797d0ee4f61c0a38376" +checksum = "59e399c068f43a5d116fedaf73b203fa4f9c519f17e2b34f63221d3792f81446" dependencies = [ "itoa", "serde", "time-core", - "time-macros 0.2.6", + "time-macros", ] [[package]] name = "time-core" -version = "0.1.0" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2e153e1f1acaef8acc537e68b44906d2db6436e2b35ac2c6b42640fff91f00fd" +checksum = "7300fbefb4dadc1af235a9cef3737cea692a9d97e1b9cbcd4ebdae6f8868e6fb" [[package]] name = "time-macros" -version = "0.1.1" +version = "0.2.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "957e9c6e26f12cb6d0dd7fc776bb67a706312e7299aed74c8dd5b17ebb27e2f1" +checksum = "96ba15a897f3c86766b757e5ac7221554c6750054d74d5b28844fce5fb36a6c4" dependencies = [ - "proc-macro-hack", - "time-macros-impl", + "time-core", ] [[package]] -name = "time-macros" -version = "0.2.6" +name = "tinyvec" +version = "1.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d967f99f534ca7e495c575c62638eebc2898a8c84c119b89e250477bc4ba16b2" +checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" dependencies = [ - "time-core", + "tinyvec_macros", ] [[package]] -name = "time-macros-impl" -version = "0.1.2" +name = "tinyvec_macros" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd3c141a1b43194f3f56a1411225df8646c55781d5f26db825b3d98507eb482f" -dependencies = [ - "proc-macro-hack", - "proc-macro2", - "quote", - "standback", - "syn", -] +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" [[package]] name = "tokio" -version = "1.24.2" +version = "1.29.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "597a12a59981d9e3c38d216785b0c37399f6e415e8d0712047620f189371b0bb" +checksum = "532826ff75199d5833b9d2c5fe410f29235e25704ee5f0ef599fb51c21f4a4da" dependencies = [ "autocfg", + "backtrace", "bytes", "libc", - "memchr", "mio", "num_cpus", "pin-project-lite", "socket2", "tokio-macros", - "windows-sys", + "windows-sys 0.48.0", ] [[package]] name = "tokio-macros" -version = "1.8.2" +version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d266c00fde287f55d3f1c3e96c500c362a2b8c695076ec180f27918820bc6df8" +checksum = "630bdcf245f78637c13ec01ffae6187cca34625e8c63150d424b59e55af2675e" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", ] [[package]] name = "tokio-rustls" -version = "0.23.4" +version = "0.24.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c43ee83903113e03984cb9e5cebe6c04a5116269e900e3ddba8f068a62adda59" +checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081" dependencies = [ "rustls", "tokio", - "webpki 0.22.0", ] [[package]] name = "tokio-stream" -version = "0.1.11" +version = "0.1.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d660770404473ccd7bc9f8b28494a811bc18542b915c0855c51e8f419d5223ce" +checksum = "397c988d37662c7dda6d2208364a706264bf3d6138b11d436cbac0ad38832842" dependencies = [ "futures-core", "pin-project-lite", @@ -2952,9 +2880,9 @@ dependencies = [ [[package]] name = "tokio-util" -version = "0.7.4" +version = "0.7.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0bb2e075f03b3d66d8d8785356224ba688d2906a371015e225beeb65ca92c740" +checksum = "806fe8c2c87eccc8b3267cbae29ed3ab2d0bd37fca70ab622e46aaa9375ddb7d" dependencies = [ "bytes", "futures-core", @@ -2983,11 +2911,12 @@ dependencies = [ [[package]] name = "tower-http" -version = "0.3.5" +version = "0.4.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f873044bf02dd1e8239e9c1293ea39dad76dc594ec16185d0a1bf31d8dc8d858" +checksum = "55ae70283aba8d2a8b411c695c437fe25b8b5e44e23e780662002fc72fb47a82" dependencies = [ - "bitflags", + "async-compression", + "bitflags 2.3.3", "bytes", "futures-core", "futures-util", @@ -2995,7 +2924,8 @@ dependencies = [ "http-body", "http-range-header", "pin-project-lite", - "tower", + "tokio", + "tokio-util", "tower-layer", "tower-service", ] @@ -3026,9 +2956,9 @@ dependencies = [ [[package]] name = "tracing-core" -version = "0.1.30" +version = "0.1.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "24eb03ba0eab1fd845050058ce5e616558e8f8d8fca633e6b163fe25c797213a" +checksum = "0955b8137a1df6f1a2e9a37d8a6656291ff0297c1a97c24e0d8425fe2312f79a" dependencies = [ "once_cell", ] @@ -3056,23 +2986,23 @@ dependencies = [ [[package]] name = "unicode-bidi" -version = "0.3.9" +version = "0.3.13" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0046be40136ef78dc325e0edefccf84ccddacd0afcc1ca54103fa3c61bbdab1d" +checksum = "92888ba5573ff080736b3648696b70cafad7d250551175acbaa4e0385b3e1460" [[package]] name = "unicode-ident" -version = "1.0.6" +version = "1.0.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "84a22b9f218b40614adcb3f4ff08b703773ad44fa9423e4e0d346d5db86e4ebc" +checksum = "301abaae475aa91687eb82514b328ab47a211a533026cb25fc3e519b86adfc3c" [[package]] name = "unicode-normalization" -version = "0.1.9" +version = "0.1.22" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "09c8070a9942f5e7cfccd93f490fdebd230ee3c3c9f107cb25bad5351ef671cf" +checksum = "5c5713f0fc4b5db668a2ac63cdb7bb4469d8c9fed047b1d0292cc7b0ce2ba921" dependencies = [ - "smallvec", + "tinyvec", ] [[package]] @@ -3088,14 +3018,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f962df74c8c05a667b5ee8bcf162993134c104e96440b663c8daa176dc772d8c" [[package]] -name = "universal-hash" -version = "0.4.1" +name = "unsafe-libyaml" +version = "0.2.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9f214e8f697e925001e66ec2c6e37a4ef93f0f78c2eed7814394e10c62025b05" -dependencies = [ - "generic-array", - "subtle", -] +checksum = "f28467d3e1d3c6586d8f25fa243f544f5800fec42d97032474e17222c2b75cfa" [[package]] name = "untrusted" @@ -3105,9 +3031,9 @@ checksum = "a156c684c91ea7d62626509bce3cb4e1d9ed5c4d978f7b4352658f96a4c26b4a" [[package]] name = "url" -version = "2.3.0" +version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22fe195a4f217c25b25cb5058ced57059824a678474874038dc88d211bf508d3" +checksum = "50bff7831e19200a85b17131d085c25d7811bc4e186efdaf54bbd132994a88cb" dependencies = [ "form_urlencoded", "idna", @@ -3117,13 +3043,9 @@ dependencies = [ [[package]] name = "value-bag" -version = "1.0.0-alpha.9" +version = "1.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2209b78d1249f7e6f3293657c9779fe31ced465df091bbd433a1cf88e916ec55" -dependencies = [ - "ctor", - "version_check", -] +checksum = "d92ccd67fb88503048c01b59152a04effd0782d035a83a6d256ce6085f08f4a3" [[package]] name = "version_check" @@ -3139,22 +3061,20 @@ checksum = "9d5b2c62b4012a3e1eca5a7e077d13b3bf498c4073e33ccd58626607748ceeca" [[package]] name = "walkdir" -version = "2.3.2" +version = "2.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "808cf2735cd4b6866113f648b791c6adc5714537bc222d9347bb203386ffda56" +checksum = "36df944cda56c7d8d8b7496af378e6b16de9284591917d307c9b4d313c44e698" dependencies = [ "same-file", - "winapi", "winapi-util", ] [[package]] name = "want" -version = "0.3.0" +version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ce8a968cb1cd110d136ff8b819a556d6fb6d919363c61534f6860c7eb172ba0" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" dependencies = [ - "log", "try-lock", ] @@ -3178,9 +3098,9 @@ checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" [[package]] name = "wasm-bindgen" -version = "0.2.83" +version = "0.2.87" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eaf9f5aceeec8be17c128b2e93e031fb8a4d469bb9c4ae2d7dc1888b26887268" +checksum = "7706a72ab36d8cb1f80ffbf0e071533974a60d0a308d01a5d0375bf60499a342" dependencies = [ "cfg-if", "wasm-bindgen-macro", @@ -3188,24 +3108,24 @@ dependencies = [ [[package]] name = "wasm-bindgen-backend" -version = "0.2.83" +version = "0.2.87" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4c8ffb332579b0557b52d268b91feab8df3615f265d5270fec2a8c95b17c1142" +checksum = "5ef2b6d3c510e9625e5fe6f509ab07d66a760f0885d858736483c32ed7809abd" dependencies = [ "bumpalo", "log", "once_cell", "proc-macro2", "quote", - "syn", + "syn 2.0.27", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-futures" -version = "0.4.33" +version = "0.4.37" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "23639446165ca5a5de86ae1d8896b737ae80319560fbaa4c2887b7da6e7ebd7d" +checksum = "c02dbc21516f9f1f04f187958890d7e6026df8d16540b7ad9492bc34a67cea03" dependencies = [ "cfg-if", "js-sys", @@ -3215,9 +3135,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro" -version = "0.2.83" +version = "0.2.87" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "052be0f94026e6cbc75cdefc9bae13fd6052cdcaf532fa6c45e7ae33a1e6c810" +checksum = "dee495e55982a3bd48105a7b947fd2a9b4a8ae3010041b9e0faab3f9cd028f1d" dependencies = [ "quote", "wasm-bindgen-macro-support", @@ -3225,69 +3145,40 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro-support" -version = "0.2.83" +version = "0.2.87" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "07bc0c051dc5f23e307b13285f9d75df86bfdf816c5721e573dec1f9b8aa193c" +checksum = "54681b18a46765f095758388f2d0cf16eb8d4169b639ab575a8f5693af210c7b" dependencies = [ "proc-macro2", "quote", - "syn", + "syn 2.0.27", "wasm-bindgen-backend", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-shared" -version = "0.2.83" +version = "0.2.87" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1c38c045535d93ec4f0b4defec448e4291638ee608530863b1e2ba115d4fff7f" +checksum = "ca6ad05a4870b2bf5fe995117d3728437bd27d7cd5f06f13c17443ef369775a1" [[package]] name = "web-sys" -version = "0.3.60" +version = "0.3.64" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bcda906d8be16e728fd5adc5b729afad4e444e106ab28cd1c7256e54fa61510f" +checksum = "9b85cbef8c220a6abc02aefd892dfc0fc23afb1c6a426316ec33253a3877249b" dependencies = [ "js-sys", "wasm-bindgen", ] -[[package]] -name = "webpki" -version = "0.21.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8e38c0608262c46d4a56202ebabdeb094cef7e560ca7a226c6bf055188aa4ea" -dependencies = [ - "ring", - "untrusted", -] - -[[package]] -name = "webpki" -version = "0.22.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f095d78192e208183081cc07bc5515ef55216397af48b873e5edcd72637fa1bd" -dependencies = [ - "ring", - "untrusted", -] - [[package]] name = "webpki-roots" -version = "0.21.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "aabe153544e473b775453675851ecc86863d2a81d786d741f6b76778f2a48940" -dependencies = [ - "webpki 0.21.4", -] - -[[package]] -name = "wepoll-ffi" -version = "0.1.2" +version = "0.23.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d743fdedc5c64377b5fc2bc036b01c7fd642205a0d96356034ae3404d49eb7fb" +checksum = "b03058f88386e5ff5310d9111d53f48b17d732b401aeb83a8d5190f2ac459338" dependencies = [ - "cc", + "rustls-webpki 0.100.1", ] [[package]] @@ -3321,62 +3212,146 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" +[[package]] +name = "windows" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f" +dependencies = [ + "windows-targets 0.48.1", +] + [[package]] name = "windows-sys" -version = "0.42.0" +version = "0.45.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5a3e1820f08b8513f676f7ab6c1f99ff312fb97b553d30ff4dd86f9f15728aa7" +checksum = "75283be5efb2831d37ea142365f009c02ec203cd29a3ebecbc093d52315b66d0" dependencies = [ - "windows_aarch64_gnullvm", - "windows_aarch64_msvc", - "windows_i686_gnu", - "windows_i686_msvc", - "windows_x86_64_gnu", - "windows_x86_64_gnullvm", - "windows_x86_64_msvc", + "windows-targets 0.42.2", ] +[[package]] +name = "windows-sys" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" +dependencies = [ + "windows-targets 0.48.1", +] + +[[package]] +name = "windows-targets" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e5180c00cd44c9b1c88adb3693291f1cd93605ded80c250a75d472756b4d071" +dependencies = [ + "windows_aarch64_gnullvm 0.42.2", + "windows_aarch64_msvc 0.42.2", + "windows_i686_gnu 0.42.2", + "windows_i686_msvc 0.42.2", + "windows_x86_64_gnu 0.42.2", + "windows_x86_64_gnullvm 0.42.2", + "windows_x86_64_msvc 0.42.2", +] + +[[package]] +name = "windows-targets" +version = "0.48.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05d4b17490f70499f20b9e791dcf6a299785ce8af4d709018206dc5b4953e95f" +dependencies = [ + "windows_aarch64_gnullvm 0.48.0", + "windows_aarch64_msvc 0.48.0", + "windows_i686_gnu 0.48.0", + "windows_i686_msvc 0.48.0", + "windows_x86_64_gnu 0.48.0", + "windows_x86_64_gnullvm 0.48.0", + "windows_x86_64_msvc 0.48.0", +] + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "597a5118570b68bc08d8d59125332c54f1ba9d9adeedeef5b99b02ba2b0698f8" + [[package]] name = "windows_aarch64_gnullvm" -version = "0.42.1" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91ae572e1b79dba883e0d315474df7305d12f569b400fcf90581b06062f7e1bc" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.42.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8c9864e83243fdec7fc9c5444389dcbbfd258f745e7853198f365e3c4968a608" +checksum = "e08e8864a60f06ef0d0ff4ba04124db8b0fb3be5776a5cd47641e942e58c4d43" [[package]] name = "windows_aarch64_msvc" -version = "0.42.1" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2ef27e0d7bdfcfc7b868b317c1d32c641a6fe4629c171b8928c7b08d98d7cf3" + +[[package]] +name = "windows_i686_gnu" +version = "0.42.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4c8b1b673ffc16c47a9ff48570a9d85e25d265735c503681332589af6253c6c7" +checksum = "c61d927d8da41da96a81f029489353e68739737d3beca43145c8afec9a31a84f" [[package]] name = "windows_i686_gnu" -version = "0.42.1" +version = "0.48.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "de3887528ad530ba7bdbb1faa8275ec7a1155a45ffa57c37993960277145d640" +checksum = "622a1962a7db830d6fd0a69683c80a18fda201879f0f447f065a3b7467daa241" [[package]] name = "windows_i686_msvc" -version = "0.42.1" +version = "0.42.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bf4d1122317eddd6ff351aa852118a2418ad4214e6613a50e0191f7004372605" +checksum = "44d840b6ec649f480a41c8d80f9c65108b92d89345dd94027bfe06ac444d1060" + +[[package]] +name = "windows_i686_msvc" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4542c6e364ce21bf45d69fdd2a8e455fa38d316158cfd43b3ac1c5b1b19f8e00" [[package]] name = "windows_x86_64_gnu" -version = "0.42.1" +version = "0.42.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c1040f221285e17ebccbc2591ffdc2d44ee1f9186324dd3e84e99ac68d699c45" +checksum = "8de912b8b8feb55c064867cf047dda097f92d51efad5b491dfb98f6bbb70cb36" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca2b8a661f7628cbd23440e50b05d705db3686f894fc9580820623656af974b1" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26d41b46a36d453748aedef1486d5c7a85db22e56aff34643984ea85514e94a3" [[package]] name = "windows_x86_64_gnullvm" -version = "0.42.1" +version = "0.48.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "628bfdf232daa22b0d64fdb62b09fcc36bb01f05a3939e20ab73aaf9470d0463" +checksum = "7896dbc1f41e08872e9d5e8f8baa8fdd2677f29468c4e156210174edc7f7b953" [[package]] name = "windows_x86_64_msvc" -version = "0.42.1" +version = "0.42.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "447660ad36a13288b1db4d4248e857b510e8c3a225c822ba4fb748c0aafecffd" +checksum = "9aec5da331524158c6d1a4ac0ab1541149c0b9505fde06423b02f5ef0106b9f0" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1a515f5799fe4961cb532f983ce2b23082366b898e52ffbce459c86f67c8378a" [[package]] name = "x509-parser" @@ -3393,14 +3368,14 @@ dependencies = [ "oid-registry", "rusticata-macros", "thiserror", - "time 0.3.17", + "time 0.3.23", ] [[package]] name = "yasna" -version = "0.5.1" +version = "0.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "aed2e7a52e3744ab4d0c05c20aa065258e84c49fd4226f5191b2ed29712710b4" +checksum = "e17bb3549cc1321ae1296b9cdc2698e2b6cb1992adfa19a8c72e5b7a738f44cd" dependencies = [ - "time 0.3.17", + "time 0.3.23", ] diff --git a/fuzz/Cargo.toml b/fuzz/Cargo.toml index 9c56ad2cd3..4725759858 100644 --- a/fuzz/Cargo.toml +++ b/fuzz/Cargo.toml @@ -24,7 +24,3 @@ name = "transaction-builder" path = "fuzz_targets/transaction_builder.rs" test = false doc = false - -[patch.crates-io] -jsonrpc = { git = "https://github.com/apoelstra/rust-jsonrpc.git", rev = "64b58797dd517c4de0cec769ff5652220801fe18" } -redb = { git = "https://github.com/casey/redb.git", branch = "ord" } diff --git a/fuzz/fuzz_targets/transaction_builder.rs b/fuzz/fuzz_targets/transaction_builder.rs index f7b9c41717..ad1be3dcfe 100644 --- a/fuzz/fuzz_targets/transaction_builder.rs +++ b/fuzz/fuzz_targets/transaction_builder.rs @@ -2,7 +2,10 @@ use { arbitrary::Arbitrary, - bitcoin::{Amount, OutPoint}, + bitcoin::{ + address::{Address, NetworkUnchecked}, + Amount, OutPoint, + }, libfuzzer_sys::fuzz_target, ord::{FeeRate, SatPoint, TransactionBuilder}, std::collections::BTreeMap, @@ -44,16 +47,19 @@ fuzz_target!(|input: Input| { } let recipient = "bc1pdqrcrxa8vx6gy75mfdfj84puhxffh4fq46h3gkp6jxdd0vjcsdyspfxcv6" - .parse() - .unwrap(); + .parse::>() + .unwrap() + .assume_checked(); let change = [ "bc1pxwww0ct9ue7e8tdnlmug5m2tamfn7q06sahstg39ys4c9f3340qqxrdu9k" - .parse() - .unwrap(), + .parse::>() + .unwrap() + .assume_checked(), "bc1pxwww0ct9ue7e8tdnlmug5m2tamfn7q06sahstg39ys4c9f3340qqxrdu9k" - .parse() - .unwrap(), + .parse::>() + .unwrap() + .assume_checked(), ]; let Ok(fee_rate) = FeeRate::try_from(input.fee_rate) else { return; }; diff --git a/src/subcommand/wallet/transaction_builder.rs b/src/subcommand/wallet/transaction_builder.rs index 7b82a018c2..5738e0c8ee 100644 --- a/src/subcommand/wallet/transaction_builder.rs +++ b/src/subcommand/wallet/transaction_builder.rs @@ -38,7 +38,10 @@ use { blockdata::{locktime::absolute::LockTime, witness::Witness}, Amount, ScriptBuf, }, - std::collections::{BTreeMap, BTreeSet}, + std::{ + cmp::{max, min}, + collections::{BTreeMap, BTreeSet}, + }, }; #[derive(Debug, PartialEq)] @@ -255,6 +258,7 @@ impl TransactionBuilder { ); let sat_offset = self.calculate_sat_offset(); + if sat_offset == 0 { tprintln!("outgoing is aligned"); } else { @@ -285,16 +289,21 @@ impl TransactionBuilder { .unwrap() .script_pubkey() .dust_value(); + if self.outputs[0].1 >= dust_limit { tprintln!("no padding needed"); } else { - let (utxo, size) = self.select_cardinal_utxo(dust_limit - self.outputs[0].1)?; - self.inputs.insert(0, utxo); - self.outputs[0].1 += size; - tprintln!( - "padded alignment output to {} with additional {size} sat input", - self.outputs[0].1 - ); + while self.outputs[0].1 < dust_limit { + let (utxo, size) = self.select_cardinal_utxo(dust_limit - self.outputs[0].1, true)?; + + self.inputs.insert(0, utxo); + self.outputs[0].1 += size; + + tprintln!( + "padded alignment output to {} with additional {size} sat input", + self.outputs[0].1 + ); + } } } @@ -313,15 +322,31 @@ impl TransactionBuilder { .checked_add(estimated_fee) .ok_or(Error::ValueOverflow)?; - if let Some(deficit) = total.checked_sub(self.outputs.last().unwrap().1) { - if deficit > Amount::ZERO { + if let Some(mut deficit) = total.checked_sub(self.outputs.last().unwrap().1) { + while deficit > Amount::ZERO { + let additional_fee = self.fee_rate.fee(Self::ADDITIONAL_INPUT_VBYTES); + let needed = deficit - .checked_add(self.fee_rate.fee(Self::ADDITIONAL_INPUT_VBYTES)) + .checked_add(additional_fee) .ok_or(Error::ValueOverflow)?; - let (utxo, value) = self.select_cardinal_utxo(needed)?; + + let (utxo, value) = self.select_cardinal_utxo(needed, false)?; + + let benefit = value + .checked_sub(additional_fee) + .ok_or(Error::NotEnoughCardinalUtxos)?; + self.inputs.push(utxo); + self.outputs.last_mut().unwrap().1 += value; - tprintln!("added {value} sat input to cover {deficit} sat deficit"); + + if benefit > deficit { + tprintln!("added {value} sat input to cover {deficit} sat deficit"); + deficit = Amount::ZERO; + } else { + tprintln!("added {value} sat input to reduce {deficit} sat deficit by {benefit} sat"); + deficit -= benefit; + } } } @@ -636,8 +661,19 @@ impl TransactionBuilder { panic!("Could not find outgoing sat in inputs"); } - fn select_cardinal_utxo(&mut self, minimum_value: Amount) -> Result<(OutPoint, Amount)> { - let mut found = None; + /// Cardinal UTXOs are those that contain no inscriptions and can therefore + /// be used to pad transactions. Sometimes multiple of these UTXOs are needed + /// and depending on the context we want to select either ones above or + /// under (when trying to consolidate dust outputs) the target value. + fn select_cardinal_utxo( + &mut self, + target_value: Amount, + prefer_under: bool, + ) -> Result<(OutPoint, Amount)> { + tprintln!( + "looking for {} cardinal worth {target_value}", + if prefer_under { "smaller" } else { "bigger" } + ); let inscribed_utxos = self .inscriptions @@ -645,22 +681,46 @@ impl TransactionBuilder { .map(|satpoint| satpoint.outpoint) .collect::>(); + let mut best_match = None; for utxo in &self.utxos { if inscribed_utxos.contains(utxo) { continue; } - let value = self.amounts[utxo]; + let current_value = self.amounts[utxo]; - if value >= minimum_value { - found = Some((*utxo, value)); - break; + let (_, best_value) = match best_match { + Some(prev) => prev, + None => { + best_match = Some((*utxo, current_value)); + (*utxo, current_value) + } + }; + + let abs_diff = |a: Amount, b: Amount| -> Amount { max(a, b) - min(a, b) }; + let is_closer = abs_diff(current_value, target_value) < abs_diff(best_value, target_value); + + let not_preference_but_closer = if prefer_under { + best_value > target_value && is_closer + } else { + best_value < target_value && is_closer + }; + + let is_preference_and_closer = if prefer_under { + current_value <= target_value && is_closer + } else { + current_value >= target_value && is_closer + }; + + if is_preference_and_closer || not_preference_but_closer { + best_match = Some((*utxo, current_value)) } } - let (utxo, value) = found.ok_or(Error::NotEnoughCardinalUtxos)?; + let (utxo, value) = best_match.ok_or(Error::NotEnoughCardinalUtxos)?; self.utxos.remove(&utxo); + tprintln!("found cardinal worth {}", value); Ok((utxo, value)) } @@ -1624,4 +1684,161 @@ mod tests { }), ); } + + #[test] + fn select_outgoing_can_select_multiple_utxos() { + let mut utxos = vec![ + (outpoint(2), Amount::from_sat(3_006)), // 2. biggest utxo is selected 2nd leaving us needing 4206 more + (outpoint(1), Amount::from_sat(3_003)), // 1. satpoint is selected 1st leaving us needing 7154 more + (outpoint(5), Amount::from_sat(3_004)), + (outpoint(4), Amount::from_sat(3_001)), // 4. smallest utxo >= 1259 is selected 4th, filling deficit + (outpoint(3), Amount::from_sat(3_005)), // 3. next biggest utxo is selected 3rd leaving us needing 1259 more + (outpoint(6), Amount::from_sat(3_002)), + ]; + + let tx_builder = TransactionBuilder::new( + satpoint(1, 0), + BTreeMap::new(), + utxos.clone().into_iter().collect(), + recipient(), + [change(0), change(1)], + FeeRate::try_from(1.0).unwrap(), + Target::Value(Amount::from_sat(10_000)), + ) + .unwrap() + .select_outgoing() + .unwrap() + .add_value() + .unwrap(); + + utxos.remove(4); + utxos.remove(3); + utxos.remove(1); + utxos.remove(0); + assert_eq!( + tx_builder.utxos, + utxos.iter().map(|(outpoint, _ranges)| *outpoint).collect() + ); + assert_eq!( + tx_builder.inputs, + [outpoint(1), outpoint(2), outpoint(3), outpoint(4)] + ); // value inputs are pushed at the end + assert_eq!( + tx_builder.outputs, + [(recipient(), Amount::from_sat(3_003 + 3_006 + 3_005 + 3_001))] + ) + } + + #[test] + fn pad_alignment_output_can_select_multiple_utxos() { + let mut utxos = vec![ + (outpoint(4), Amount::from_sat(101)), // 4. smallest utxo >= 84 is selected 4th, filling deficit + (outpoint(1), Amount::from_sat(20_000)), // 1. satpoint is selected 1st leaving deficit 293 + (outpoint(2), Amount::from_sat(105)), // 2. biggest utxo <= 293 is selected 2nd leaving deficit 188 + (outpoint(5), Amount::from_sat(103)), + (outpoint(6), Amount::from_sat(10_000)), + (outpoint(3), Amount::from_sat(104)), // 3. biggest utxo <= 188 is selected 3rd leaving deficit 84 + (outpoint(7), Amount::from_sat(102)), + ]; + + let tx_builder = TransactionBuilder::new( + satpoint(1, 1), + BTreeMap::new(), + utxos.clone().into_iter().collect(), + recipient(), + [change(0), change(1)], + FeeRate::try_from(1.0).unwrap(), + Target::Value(Amount::from_sat(10_000)), + ) + .unwrap() + .select_outgoing() + .unwrap() + .align_outgoing() + .pad_alignment_output() + .unwrap(); + + utxos.remove(5); + utxos.remove(2); + utxos.remove(1); + utxos.remove(0); + assert_eq!( + tx_builder.utxos, + utxos.iter().map(|(outpoint, _ranges)| *outpoint).collect() + ); + assert_eq!( + tx_builder.inputs, + [outpoint(4), outpoint(3), outpoint(2), outpoint(1)] + ); // padding inputs are inserted at the start + assert_eq!( + tx_builder.outputs, + [ + (change(1), Amount::from_sat(101 + 104 + 105 + 1)), + (recipient(), Amount::from_sat(19_999)) + ] + ) + } + + fn select_cardinal_utxo_prefer_under_helper( + target_value: Amount, + prefer_under: bool, + expected_value: Amount, + ) { + let utxos = vec![ + (outpoint(4), Amount::from_sat(101)), + (outpoint(1), Amount::from_sat(20_000)), + (outpoint(2), Amount::from_sat(105)), + (outpoint(5), Amount::from_sat(103)), + (outpoint(6), Amount::from_sat(10_000)), + (outpoint(3), Amount::from_sat(104)), + (outpoint(7), Amount::from_sat(102)), + ]; + + let mut tx_builder = TransactionBuilder::new( + satpoint(0, 0), + BTreeMap::new(), + utxos.into_iter().collect(), + recipient(), + [change(0), change(1)], + FeeRate::try_from(1.0).unwrap(), + Target::Value(Amount::from_sat(10_000)), + ) + .unwrap(); + + assert_eq!( + tx_builder + .select_cardinal_utxo(target_value, prefer_under) + .unwrap() + .1, + expected_value + ); + } + + #[test] + fn select_cardinal_utxo_prefer_under() { + // select biggest utxo <= 104 + select_cardinal_utxo_prefer_under_helper(Amount::from_sat(104), true, Amount::from_sat(104)); + + // select biggest utxo <= 1_000 + select_cardinal_utxo_prefer_under_helper(Amount::from_sat(1_000), true, Amount::from_sat(105)); + + // select biggest utxo <= 10, else smallest > 10 + select_cardinal_utxo_prefer_under_helper(Amount::from_sat(10), true, Amount::from_sat(101)); + + // select smallest utxo >= 104 + select_cardinal_utxo_prefer_under_helper(Amount::from_sat(104), false, Amount::from_sat(104)); + + // select smallest utxo >= 1_000 + select_cardinal_utxo_prefer_under_helper( + Amount::from_sat(1000), + false, + Amount::from_sat(10_000), + ); + + // select smallest utxo >= 100_000, else biggest < 100_000 + select_cardinal_utxo_prefer_under_helper( + Amount::from_sat(100_000), + false, + Amount::from_sat(20_000), + ); + } } From fb0b1e826df34de0f6d8d6527c617d2e8d32c0ac Mon Sep 17 00:00:00 2001 From: gmart7t2 <49558347+gmart7t2@users.noreply.github.com> Date: Thu, 10 Aug 2023 06:50:26 -0300 Subject: [PATCH 03/61] Only fetch inscriptions that are owned by the ord wallet (#2310) --- src/index.rs | 17 ++++------------- src/subcommand/wallet/balance.rs | 4 +++- src/subcommand/wallet/cardinals.rs | 7 ++++--- src/subcommand/wallet/inscribe.rs | 2 +- src/subcommand/wallet/inscriptions.rs | 2 +- src/subcommand/wallet/send.rs | 2 +- 6 files changed, 14 insertions(+), 20 deletions(-) diff --git a/src/index.rs b/src/index.rs index b2b208c7c7..05092ec223 100644 --- a/src/index.rs +++ b/src/index.rs @@ -826,24 +826,15 @@ impl Index { pub(crate) fn get_inscriptions( &self, - n: Option, + utxos: BTreeMap, ) -> Result> { let rtx = self.database.begin_read()?; let mut result = BTreeMap::new(); - for range_result in rtx - .open_multimap_table(SATPOINT_TO_INSCRIPTION_ID)? - .range::<&[u8; 44]>(&[0; 44]..)? - { - let (satpoint, ids) = range_result?; - for id_result in ids { - let id = id_result?; - result.insert(Entry::load(*satpoint.value()), Entry::load(*id.value())); - } - if result.len() >= n.unwrap_or(usize::MAX) { - break; - } + let table = rtx.open_multimap_table(SATPOINT_TO_INSCRIPTION_ID)?; + for utxo in utxos.keys() { + result.extend(Self::inscriptions_on_output_unordered(&table, *utxo)?); } Ok(result) diff --git a/src/subcommand/wallet/balance.rs b/src/subcommand/wallet/balance.rs index 84ff68d2b9..9813634118 100644 --- a/src/subcommand/wallet/balance.rs +++ b/src/subcommand/wallet/balance.rs @@ -9,8 +9,10 @@ pub(crate) fn run(options: Options) -> Result { let index = Index::open(&options)?; index.update()?; + let unspent_outputs = index.get_unspent_outputs(Wallet::load(&options)?)?; + let inscription_outputs = index - .get_inscriptions(None)? + .get_inscriptions(unspent_outputs)? .keys() .map(|satpoint| satpoint.outpoint) .collect::>(); diff --git a/src/subcommand/wallet/cardinals.rs b/src/subcommand/wallet/cardinals.rs index 32076229a1..4df85204fd 100644 --- a/src/subcommand/wallet/cardinals.rs +++ b/src/subcommand/wallet/cardinals.rs @@ -10,14 +10,15 @@ pub(crate) fn run(options: Options) -> Result { let index = Index::open(&options)?; index.update()?; + let unspent_outputs = index.get_unspent_outputs(Wallet::load(&options)?)?; + let inscribed_utxos = index - .get_inscriptions(None)? + .get_inscriptions(unspent_outputs.clone())? .keys() .map(|satpoint| satpoint.outpoint) .collect::>(); - let cardinal_utxos = index - .get_unspent_outputs(Wallet::load(&options)?)? + let cardinal_utxos = unspent_outputs .iter() .filter_map(|(output, amount)| { if inscribed_utxos.contains(output) { diff --git a/src/subcommand/wallet/inscribe.rs b/src/subcommand/wallet/inscribe.rs index 639be28dc0..c3b8aac51b 100644 --- a/src/subcommand/wallet/inscribe.rs +++ b/src/subcommand/wallet/inscribe.rs @@ -64,7 +64,7 @@ impl Inscribe { let mut utxos = index.get_unspent_outputs(Wallet::load(&options)?)?; - let inscriptions = index.get_inscriptions(None)?; + let inscriptions = index.get_inscriptions(utxos.clone())?; let commit_tx_change = [ get_change_address(&client, &options)?, diff --git a/src/subcommand/wallet/inscriptions.rs b/src/subcommand/wallet/inscriptions.rs index 020ac5ce61..2b9395e378 100644 --- a/src/subcommand/wallet/inscriptions.rs +++ b/src/subcommand/wallet/inscriptions.rs @@ -12,8 +12,8 @@ pub(crate) fn run(options: Options) -> Result { let index = Index::open(&options)?; index.update()?; - let inscriptions = index.get_inscriptions(None)?; let unspent_outputs = index.get_unspent_outputs(Wallet::load(&options)?)?; + let inscriptions = index.get_inscriptions(unspent_outputs.clone())?; let explorer = match options.chain() { Chain::Mainnet => "https://ordinals.com/inscription/", diff --git a/src/subcommand/wallet/send.rs b/src/subcommand/wallet/send.rs index b4f1551d4b..f55c4601e5 100644 --- a/src/subcommand/wallet/send.rs +++ b/src/subcommand/wallet/send.rs @@ -24,7 +24,7 @@ impl Send { let unspent_outputs = index.get_unspent_outputs(Wallet::load(&options)?)?; - let inscriptions = index.get_inscriptions(None)?; + let inscriptions = index.get_inscriptions(unspent_outputs.clone())?; let satpoint = match self.outgoing { Outgoing::SatPoint(satpoint) => { From 13d5c2fcd5e48eda9e1eb2ceb08c0b7312804372 Mon Sep 17 00:00:00 2001 From: Patate <85969303+Mathieu-Be@users.noreply.github.com> Date: Thu, 10 Aug 2023 12:33:19 +0200 Subject: [PATCH 04/61] Add JSON API endpoint `/sat/` (#2250) Co-authored-by: raphjaph --- Cargo.lock | 5 +- Cargo.toml | 1 + src/epoch.rs | 2 +- src/height.rs | 2 +- src/index.rs | 4 ++ src/lib.rs | 8 +-- src/options.rs | 2 + src/subcommand/server.rs | 45 ++++++++++++--- src/subcommand/server/accept_json.rs | 24 ++++++++ src/templates.rs | 4 +- src/templates/sat.rs | 18 ++++++ tests/json_api.rs | 86 ++++++++++++++++++++++++++++ tests/lib.rs | 6 +- tests/test_server.rs | 34 ++++++----- 14 files changed, 205 insertions(+), 36 deletions(-) create mode 100644 src/subcommand/server/accept_json.rs create mode 100644 tests/json_api.rs diff --git a/Cargo.lock b/Cargo.lock index d74ac94a20..0089ed5dda 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -295,9 +295,9 @@ checksum = "ecc7ab41815b3c653ccd2978ec3255c81349336702dfdf62ee6f7069b12a3aae" [[package]] name = "async-trait" -version = "0.1.71" +version = "0.1.72" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a564d521dd56509c4c47480d00b80ee55f7e385ae48db5744c67ad50c92d2ebf" +checksum = "cc6dde6e4ed435a4c1ee4e73592f5ba9da2151af10076cc04858746af9352d09" dependencies = [ "proc-macro2", "quote", @@ -2003,6 +2003,7 @@ name = "ord" version = "0.8.1" dependencies = [ "anyhow", + "async-trait", "axum", "axum-server", "base64 0.21.2", diff --git a/Cargo.toml b/Cargo.toml index e4ea445979..eb4e6b6498 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -15,6 +15,7 @@ members = [".", "test-bitcoincore-rpc"] [dependencies] anyhow = { version = "1.0.56", features = ["backtrace"] } +async-trait = "0.1.72" axum = { version = "0.6.1", features = ["headers"] } axum-server = "0.5.0" base64 = "0.21.0" diff --git a/src/epoch.rs b/src/epoch.rs index 7918191574..328b09af32 100644 --- a/src/epoch.rs +++ b/src/epoch.rs @@ -1,6 +1,6 @@ use super::*; -#[derive(Copy, Clone, Eq, PartialEq, Debug, Display, PartialOrd)] +#[derive(Copy, Clone, Eq, PartialEq, Debug, Display, Serialize, PartialOrd)] pub(crate) struct Epoch(pub(crate) u64); impl Epoch { diff --git a/src/height.rs b/src/height.rs index e2a3e20749..75ac7ec8ab 100644 --- a/src/height.rs +++ b/src/height.rs @@ -1,6 +1,6 @@ use super::*; -#[derive(Copy, Clone, Debug, Display, FromStr, Ord, Eq, PartialEq, PartialOrd)] +#[derive(Copy, Clone, Debug, Display, FromStr, Ord, Eq, Serialize, PartialEq, PartialOrd)] pub(crate) struct Height(pub(crate) u64); impl Height { diff --git a/src/index.rs b/src/index.rs index 05092ec223..1e1cf70405 100644 --- a/src/index.rs +++ b/src/index.rs @@ -464,6 +464,10 @@ impl Index { Ok(()) } + pub(crate) fn is_json_api_enabled(&self) -> bool { + self.options.enable_json_api + } + pub(crate) fn is_reorged(&self) -> bool { self.reorged.load(atomic::Ordering::Relaxed) } diff --git a/src/lib.rs b/src/lib.rs index 64a828a0f0..1ef8d220ab 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -108,19 +108,19 @@ mod fee_rate; mod height; mod index; mod inscription; -mod inscription_id; +pub mod inscription_id; mod media; mod object; mod options; mod outgoing; mod page_config; -mod rarity; +pub mod rarity; mod representation; -mod sat; +pub mod sat; mod sat_point; pub mod subcommand; mod tally; -mod templates; +pub mod templates; mod wallet; type Result = std::result::Result; diff --git a/src/options.rs b/src/options.rs index 20841e911b..7d2e5ef8a2 100644 --- a/src/options.rs +++ b/src/options.rs @@ -54,6 +54,8 @@ pub(crate) struct Options { pub(crate) testnet: bool, #[clap(long, default_value = "ord", help = "Use wallet named .")] pub(crate) wallet: String, + #[clap(long, short, help = "Enable JSON API.")] + pub(crate) enable_json_api: bool, } impl Options { diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 52b3985d76..d33290c4e3 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -1,5 +1,6 @@ use { self::{ + accept_json::AcceptJson, deserialize_from_str::DeserializeFromStr, error::{OptionExt, ServerError, ServerResult}, }, @@ -8,11 +9,11 @@ use { crate::templates::{ BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionsHtml, OutputHtml, PageContent, PageHtml, PreviewAudioHtml, PreviewImageHtml, PreviewPdfHtml, PreviewTextHtml, - PreviewUnknownHtml, PreviewVideoHtml, RangeHtml, RareTxt, SatHtml, TransactionHtml, + PreviewUnknownHtml, PreviewVideoHtml, RangeHtml, RareTxt, SatHtml, SatJson, TransactionHtml, }, axum::{ body, - extract::{Extension, Path, Query}, + extract::{Extension, Json, Path, Query}, headers::UserAgent, http::{header, HeaderMap, HeaderValue, StatusCode, Uri}, response::{IntoResponse, Redirect, Response}, @@ -36,6 +37,7 @@ use { }, }; +mod accept_json; mod error; enum BlockQuery { @@ -382,18 +384,43 @@ impl Server { Extension(page_config): Extension>, Extension(index): Extension>, Path(DeserializeFromStr(sat)): Path>, - ) -> ServerResult> { + accept_json: AcceptJson, + ) -> ServerResult { let satpoint = index.rare_sat_satpoint(sat)?; - - Ok( + let blocktime = index.block_time(sat.height())?; + let inscriptions = index.get_inscription_ids_by_sat(sat)?; + Ok(if accept_json.0 { + if index.is_json_api_enabled() { + Json(SatJson { + number: sat.0, + decimal: sat.decimal().to_string(), + degree: sat.degree().to_string(), + name: sat.name(), + block: sat.height().0, + cycle: sat.cycle(), + epoch: sat.epoch().0, + period: sat.period(), + offset: sat.third(), + rarity: sat.rarity(), + percentile: sat.percentile(), + satpoint, + timestamp: blocktime.timestamp().to_string(), + inscriptions, + }) + .into_response() + } else { + StatusCode::NOT_ACCEPTABLE.into_response() + } + } else { SatHtml { sat, satpoint, - blocktime: index.block_time(sat.height())?, - inscriptions: index.get_inscription_ids_by_sat(sat)?, + blocktime, + inscriptions, } - .page(page_config, index.has_sat_index()?), - ) + .page(page_config, index.has_sat_index()?) + .into_response() + }) } async fn ordinal(Path(sat): Path) -> Redirect { diff --git a/src/subcommand/server/accept_json.rs b/src/subcommand/server/accept_json.rs new file mode 100644 index 0000000000..e01edf29c2 --- /dev/null +++ b/src/subcommand/server/accept_json.rs @@ -0,0 +1,24 @@ +use super::*; + +pub(crate) struct AcceptJson(pub(crate) bool); + +#[async_trait::async_trait] +impl axum::extract::FromRequestParts for AcceptJson +where + S: Send + Sync, +{ + type Rejection = (); + + async fn from_request_parts( + parts: &mut http::request::Parts, + _state: &S, + ) -> Result { + Ok(Self( + parts + .headers + .get("accept") + .map(|value| value == "application/json") + .unwrap_or_default(), + )) + } +} diff --git a/src/templates.rs b/src/templates.rs index 7a8858a869..e4f008f6a8 100644 --- a/src/templates.rs +++ b/src/templates.rs @@ -16,7 +16,7 @@ pub(crate) use { }, range::RangeHtml, rare::RareTxt, - sat::SatHtml, + sat::{SatHtml, SatJson}, transaction::TransactionHtml, }; @@ -31,7 +31,7 @@ mod output; mod preview; mod range; mod rare; -mod sat; +pub mod sat; mod transaction; #[derive(Boilerplate)] diff --git a/src/templates/sat.rs b/src/templates/sat.rs index fd1c5bab7d..c42b839750 100644 --- a/src/templates/sat.rs +++ b/src/templates/sat.rs @@ -8,6 +8,24 @@ pub(crate) struct SatHtml { pub(crate) inscriptions: Vec, } +#[derive(Debug, PartialEq, Serialize, Deserialize)] +pub struct SatJson { + pub number: u64, + pub decimal: String, + pub degree: String, + pub name: String, + pub block: u64, + pub cycle: u64, + pub epoch: u64, + pub period: u64, + pub offset: u64, + pub rarity: Rarity, + pub percentile: String, + pub satpoint: Option, + pub timestamp: String, + pub inscriptions: Vec, +} + impl PageContent for SatHtml { fn title(&self) -> String { format!("Sat {}", self.sat) diff --git a/tests/json_api.rs b/tests/json_api.rs new file mode 100644 index 0000000000..3cbba92188 --- /dev/null +++ b/tests/json_api.rs @@ -0,0 +1,86 @@ +use { + super::*, ord::inscription_id::InscriptionId, ord::rarity::Rarity, ord::templates::sat::SatJson, + ord::SatPoint, +}; + +#[test] +fn get_sat_without_sat_index() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + let response = TestServer::spawn_with_args(&rpc_server, &["--enable-json-api"]) + .json_request("/sat/2099999997689999"); + + assert_eq!(response.status(), StatusCode::OK); + + let mut sat_json: SatJson = serde_json::from_str(&response.text().unwrap()).unwrap(); + + // this is a hack to ignore the timestamp, since it changes for every request + sat_json.timestamp = "".into(); + + pretty_assert_eq!( + sat_json, + SatJson { + number: 2099999997689999, + decimal: "6929999.0".into(), + degree: "5°209999′1007″0‴".into(), + name: "a".into(), + block: 6929999, + cycle: 5, + epoch: 32, + period: 3437, + offset: 0, + rarity: Rarity::Uncommon, + percentile: "100%".into(), + satpoint: None, + timestamp: "".into(), + inscriptions: vec![], + } + ) +} + +#[test] +fn get_sat_with_inscription_and_sat_index() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + create_wallet(&rpc_server); + + let Inscribe { reveal, .. } = inscribe(&rpc_server); + let inscription_id = InscriptionId::from(reveal); + + let response = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]) + .json_request(format!("/sat/{}", 50 * COIN_VALUE)); + + assert_eq!(response.status(), StatusCode::OK); + + let sat_json: SatJson = serde_json::from_str(&response.text().unwrap()).unwrap(); + + pretty_assert_eq!( + sat_json, + SatJson { + number: 50 * COIN_VALUE, + decimal: "1.0".into(), + degree: "0°1′1″0‴".into(), + name: "nvtcsezkbth".into(), + block: 1, + cycle: 0, + epoch: 0, + period: 0, + offset: 0, + rarity: Rarity::Uncommon, + percentile: "0.00023809523835714296%".into(), + satpoint: Some(SatPoint::from_str(&format!("{}:{}:{}", reveal, 0, 0)).unwrap()), + timestamp: "1970-01-01 00:00:01 UTC".into(), + inscriptions: vec![inscription_id], + } + ) +} + +#[test] +fn json_request_fails_when_not_enabled() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + let response = + TestServer::spawn_with_args(&rpc_server, &[]).json_request("/sat/2099999997689999"); + + assert_eq!(response.status(), StatusCode::NOT_ACCEPTABLE); +} diff --git a/tests/lib.rs b/tests/lib.rs index cd9f4dfab1..6ad150a58e 100644 --- a/tests/lib.rs +++ b/tests/lib.rs @@ -74,18 +74,20 @@ fn create_wallet(rpc_server: &test_bitcoincore_rpc::Handle) { } mod command_builder; +mod expected; +mod test_server; + mod core; mod epochs; -mod expected; mod find; mod index; mod info; +mod json_api; mod list; mod parse; mod server; mod subsidy; mod supply; -mod test_server; mod traits; mod version; mod wallet; diff --git a/tests/test_server.rs b/tests/test_server.rs index e068f9b698..098aa51d17 100644 --- a/tests/test_server.rs +++ b/tests/test_server.rs @@ -60,19 +60,7 @@ impl TestServer { } pub(crate) fn assert_response_regex(&self, path: impl AsRef, regex: impl AsRef) { - let client = Client::new(&self.rpc_url, Auth::None).unwrap(); - let chain_block_count = client.get_block_count().unwrap() + 1; - - for i in 0.. { - let response = reqwest::blocking::get(self.url().join("/blockcount").unwrap()).unwrap(); - assert_eq!(response.status(), StatusCode::OK); - if response.text().unwrap().parse::().unwrap() == chain_block_count { - break; - } else if i == 20 { - panic!("index failed to synchronize with chain"); - } - thread::sleep(Duration::from_millis(25)); - } + self.sync_server(); let response = reqwest::blocking::get(self.url().join(path.as_ref()).unwrap()).unwrap(); assert_eq!(response.status(), StatusCode::OK); @@ -80,6 +68,24 @@ impl TestServer { } pub(crate) fn request(&self, path: impl AsRef) -> Response { + self.sync_server(); + + reqwest::blocking::get(self.url().join(path.as_ref()).unwrap()).unwrap() + } + + pub(crate) fn json_request(&self, path: impl AsRef) -> Response { + self.sync_server(); + + let client = reqwest::blocking::Client::new(); + + client + .get(self.url().join(path.as_ref()).unwrap()) + .header(reqwest::header::ACCEPT, "application/json") + .send() + .unwrap() + } + + fn sync_server(&self) { let client = Client::new(&self.rpc_url, Auth::None).unwrap(); let chain_block_count = client.get_block_count().unwrap() + 1; @@ -93,8 +99,6 @@ impl TestServer { } thread::sleep(Duration::from_millis(25)); } - - reqwest::blocking::get(self.url().join(path.as_ref()).unwrap()).unwrap() } } From 0286e79a9fc317558131c1fdf76bab72e09faa49 Mon Sep 17 00:00:00 2001 From: raph Date: Thu, 10 Aug 2023 12:49:51 +0200 Subject: [PATCH 05/61] Add reorg resistance (#2320) Co-authored-by: ordinally --- src/index.rs | 290 ++++++++++++++++++++++++++++++++++----- src/index/reorg.rs | 108 +++++++++++++++ src/index/updater.rs | 107 ++++++--------- src/subcommand/server.rs | 24 ++-- 4 files changed, 424 insertions(+), 105 deletions(-) create mode 100644 src/index/reorg.rs diff --git a/src/index.rs b/src/index.rs index 1e1cf70405..f366f3c899 100644 --- a/src/index.rs +++ b/src/index.rs @@ -4,6 +4,7 @@ use { BlockHashValue, Entry, InscriptionEntry, InscriptionEntryValue, InscriptionIdValue, OutPointValue, SatPointValue, SatRange, }, + reorg::*, updater::Updater, }, super::*, @@ -19,11 +20,11 @@ use { }, std::collections::HashMap, std::io::{BufWriter, Read, Write}, - std::sync::atomic::{self, AtomicBool}, }; mod entry; mod fetcher; +mod reorg; mod rtx; mod updater; @@ -55,18 +56,6 @@ define_table! { SAT_TO_SATPOINT, u64, &SatPointValue } define_table! { STATISTIC_TO_COUNT, u64, u64 } define_table! { WRITE_TRANSACTION_STARTING_BLOCK_COUNT_TO_TIMESTAMP, u64, u128 } -pub(crate) struct Index { - client: Client, - database: Database, - path: PathBuf, - first_inscription_height: u64, - genesis_block_coinbase_transaction: Transaction, - genesis_block_coinbase_txid: Txid, - height_limit: Option, - options: Options, - reorged: AtomicBool, -} - #[derive(Debug, PartialEq)] pub(crate) enum List { Spent, @@ -143,6 +132,18 @@ impl BitcoinCoreRpcResultExt for Result { } } +pub(crate) struct Index { + client: Client, + database: Database, + path: PathBuf, + first_inscription_height: u64, + genesis_block_coinbase_transaction: Transaction, + genesis_block_coinbase_txid: Txid, + height_limit: Option, + options: Options, + unrecoverably_reorged: AtomicBool, +} + impl Index { pub(crate) fn open(options: &Options) -> Result { let client = options.bitcoin_rpc_client()?; @@ -221,11 +222,7 @@ impl Index { let mut tx = database.begin_write()?; - if cfg!(test) { - tx.set_durability(redb::Durability::None); - } else { - tx.set_durability(redb::Durability::Immediate); - }; + tx.set_durability(redb::Durability::Immediate); tx.open_table(HEIGHT_TO_BLOCK_HASH)?; tx.open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY)?; @@ -263,8 +260,8 @@ impl Index { first_inscription_height: options.first_inscription_height(), genesis_block_coinbase_transaction, height_limit: options.height_limit, - reorged: AtomicBool::new(false), options: options.clone(), + unrecoverably_reorged: AtomicBool::new(false), }) } @@ -396,7 +393,31 @@ impl Index { } pub(crate) fn update(&self) -> Result { - Updater::update(self) + let mut updater = Updater::new(self)?; + + loop { + match updater.update_index() { + Ok(ok) => return Ok(ok), + Err(err) => { + log::info!("{}", err.to_string()); + + match err.downcast_ref() { + Some(&ReorgError::Recoverable((height, depth))) => { + Reorg::handle_reorg(self, height, depth)?; + + updater = Updater::new(self)?; + } + Some(&ReorgError::Unrecoverable) => { + self + .unrecoverably_reorged + .store(true, atomic::Ordering::Relaxed); + return Err(anyhow!(ReorgError::Unrecoverable)); + } + _ => return Err(err), + }; + } + } + } } pub(crate) fn export(&self, filename: &String, include_addresses: bool) -> Result { @@ -464,12 +485,12 @@ impl Index { Ok(()) } - pub(crate) fn is_json_api_enabled(&self) -> bool { - self.options.enable_json_api + pub(crate) fn is_unrecoverably_reorged(&self) -> bool { + self.unrecoverably_reorged.load(atomic::Ordering::Relaxed) } - pub(crate) fn is_reorged(&self) -> bool { - self.reorged.load(atomic::Ordering::Relaxed) + pub(crate) fn is_json_api_enabled(&self) -> bool { + self.options.enable_json_api } fn begin_read(&self) -> Result { @@ -477,13 +498,7 @@ impl Index { } fn begin_write(&self) -> Result { - if cfg!(test) { - let mut tx = self.database.begin_write()?; - tx.set_durability(redb::Durability::None); - Ok(tx) - } else { - Ok(self.database.begin_write()?) - } + Ok(self.database.begin_write()?) } fn increment_statistic(wtx: &WriteTransaction, statistic: Statistic, n: u64) -> Result { @@ -1004,6 +1019,65 @@ impl Index { } } + #[cfg(test)] + fn assert_non_existence_of_inscription(&self, inscription_id: InscriptionId) { + let rtx = self.database.begin_read().unwrap(); + + let inscription_id_to_satpoint = rtx.open_table(INSCRIPTION_ID_TO_SATPOINT).unwrap(); + assert!(inscription_id_to_satpoint + .get(&inscription_id.store()) + .unwrap() + .is_none()); + + let inscription_id_to_entry = rtx.open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY).unwrap(); + assert!(inscription_id_to_entry + .get(&inscription_id.store()) + .unwrap() + .is_none()); + + for range in rtx + .open_table(INSCRIPTION_NUMBER_TO_INSCRIPTION_ID) + .unwrap() + .iter() + .into_iter() + { + for entry in range.into_iter() { + let (_number, id) = entry.unwrap(); + assert!(InscriptionId::load(*id.value()) != inscription_id); + } + } + + for range in rtx + .open_multimap_table(SATPOINT_TO_INSCRIPTION_ID) + .unwrap() + .iter() + .into_iter() + { + for entry in range.into_iter() { + let (_satpoint, ids) = entry.unwrap(); + assert!(!ids + .into_iter() + .any(|id| InscriptionId::load(*id.unwrap().value()) == inscription_id)) + } + } + + if self.has_sat_index().unwrap() { + for range in rtx + .open_multimap_table(SAT_TO_INSCRIPTION_ID) + .unwrap() + .iter() + .into_iter() + { + for entry in range.into_iter() { + let (_sat, ids) = entry.unwrap(); + assert!(!ids + .into_iter() + .any(|id| InscriptionId::load(*id.unwrap().value()) == inscription_id)) + } + } + } + } + fn inscriptions_on_output_unordered<'a: 'tx, 'tx>( satpoint_to_id: &'a impl ReadableMultimapTable<&'static SatPointValue, &'static InscriptionIdValue>, outpoint: OutPoint, @@ -3205,4 +3279,158 @@ mod tests { ) } } + + #[test] + fn recover_from_reorg() { + for context in Context::configurations() { + context.mine_blocks(1); + + let txid = context.rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(1, 0, 0)], + witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), + ..Default::default() + }); + let first_id = InscriptionId { txid, index: 0 }; + let first_location = SatPoint { + outpoint: OutPoint { txid, vout: 0 }, + offset: 0, + }; + + context.mine_blocks(6); + + context + .index + .assert_inscription_location(first_id, first_location, Some(50 * COIN_VALUE)); + + let txid = context.rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(2, 0, 0)], + witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), + ..Default::default() + }); + let second_id = InscriptionId { txid, index: 0 }; + let second_location = SatPoint { + outpoint: OutPoint { txid, vout: 0 }, + offset: 0, + }; + + context.mine_blocks(1); + + context + .index + .assert_inscription_location(second_id, second_location, Some(100 * COIN_VALUE)); + + context.rpc_server.invalidate_tip(); + context.mine_blocks(2); + + context + .index + .assert_inscription_location(first_id, first_location, Some(50 * COIN_VALUE)); + + context.index.assert_non_existence_of_inscription(second_id); + } + } + + #[test] + fn recover_from_3_block_deep_and_consecutive_reorg() { + for context in Context::configurations() { + context.mine_blocks(1); + + let txid = context.rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(1, 0, 0)], + witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), + ..Default::default() + }); + let first_id = InscriptionId { txid, index: 0 }; + let first_location = SatPoint { + outpoint: OutPoint { txid, vout: 0 }, + offset: 0, + }; + + context.mine_blocks(10); + + let txid = context.rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(2, 0, 0)], + witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), + ..Default::default() + }); + let second_id = InscriptionId { txid, index: 0 }; + let second_location = SatPoint { + outpoint: OutPoint { txid, vout: 0 }, + offset: 0, + }; + + context.mine_blocks(1); + + context + .index + .assert_inscription_location(second_id, second_location, Some(100 * COIN_VALUE)); + + context.rpc_server.invalidate_tip(); + context.rpc_server.invalidate_tip(); + context.rpc_server.invalidate_tip(); + + context.mine_blocks(4); + + context.index.assert_non_existence_of_inscription(second_id); + + context.rpc_server.invalidate_tip(); + + context.mine_blocks(2); + + context + .index + .assert_inscription_location(first_id, first_location, Some(50 * COIN_VALUE)); + } + } + + #[test] + fn recover_from_very_unlikely_7_block_deep_reorg() { + for context in Context::configurations() { + context.mine_blocks(1); + + let txid = context.rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(1, 0, 0)], + witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), + ..Default::default() + }); + + context.mine_blocks(11); + + let first_id = InscriptionId { txid, index: 0 }; + let first_location = SatPoint { + outpoint: OutPoint { txid, vout: 0 }, + offset: 0, + }; + + let txid = context.rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(2, 0, 0)], + witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), + ..Default::default() + }); + + let second_id = InscriptionId { txid, index: 0 }; + let second_location = SatPoint { + outpoint: OutPoint { txid, vout: 0 }, + offset: 0, + }; + + context.mine_blocks(7); + + context + .index + .assert_inscription_location(second_id, second_location, Some(100 * COIN_VALUE)); + + for _ in 0..7 { + context.rpc_server.invalidate_tip(); + } + + context.mine_blocks(9); + + context.index.assert_non_existence_of_inscription(second_id); + + context + .index + .assert_inscription_location(first_id, first_location, Some(50 * COIN_VALUE)); + } + } } diff --git a/src/index/reorg.rs b/src/index/reorg.rs new file mode 100644 index 0000000000..8de5da973c --- /dev/null +++ b/src/index/reorg.rs @@ -0,0 +1,108 @@ +use {super::*, updater::BlockData}; + +#[derive(Debug, PartialEq)] +pub(crate) enum ReorgError { + Recoverable((u64, u64)), + Unrecoverable, +} + +impl fmt::Display for ReorgError { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + ReorgError::Recoverable((height, depth)) => { + write!(f, "{depth} block deep reorg detected at height {height}") + } + ReorgError::Unrecoverable => write!(f, "unrecoverable reorg detected"), + } + } +} + +impl std::error::Error for ReorgError {} + +const MAX_SAVEPOINTS: usize = 2; +const SAVEPOINT_INTERVAL: u64 = 10; +const CHAIN_TIP_DISTANCE: u64 = 21; + +pub(crate) struct Reorg {} + +impl Reorg { + pub(crate) fn detect_reorg(block: &BlockData, height: u64, index: &Index) -> Result { + let bitcoind_prev_blockhash = block.header.prev_blockhash; + + match index.block_hash(height.checked_sub(1))? { + Some(index_prev_blockhash) if index_prev_blockhash == bitcoind_prev_blockhash => Ok(()), + Some(index_prev_blockhash) if index_prev_blockhash != bitcoind_prev_blockhash => { + let max_recoverable_reorg_depth = + (MAX_SAVEPOINTS as u64 - 1) * SAVEPOINT_INTERVAL + height % SAVEPOINT_INTERVAL; + + for depth in 1..max_recoverable_reorg_depth { + let index_block_hash = index.block_hash(height.checked_sub(depth))?; + let bitcoind_block_hash = index + .client + .get_block_hash(height.saturating_sub(depth)) + .into_option()?; + + if index_block_hash == bitcoind_block_hash { + return Err(anyhow!(ReorgError::Recoverable((depth, height)))); + } + } + + Err(anyhow!(ReorgError::Unrecoverable)) + } + _ => Ok(()), + } + } + + pub(crate) fn handle_reorg(index: &Index, height: u64, depth: u64) -> Result { + log::info!("rolling back database after reorg of depth {depth} at height {height}"); + + let mut wtx = index.begin_write()?; + + let oldest_savepoint = + wtx.get_persistent_savepoint(wtx.list_persistent_savepoints()?.min().unwrap())?; + + wtx.restore_savepoint(&oldest_savepoint)?; + + Index::increment_statistic(&wtx, Statistic::Commits, 1)?; + wtx.commit()?; + + log::info!( + "successfully rolled back database to height {}", + index.block_count()? + ); + + Ok(()) + } + + pub(crate) fn update_savepoints(index: &Index, height: u64) -> Result { + if (height < SAVEPOINT_INTERVAL || height % SAVEPOINT_INTERVAL == 0) + && index + .client + .get_blockchain_info()? + .headers + .saturating_sub(height) + <= CHAIN_TIP_DISTANCE + { + let wtx = index.begin_write()?; + + let savepoints = wtx.list_persistent_savepoints()?.collect::>(); + + if savepoints.len() >= MAX_SAVEPOINTS { + wtx.delete_persistent_savepoint(savepoints.into_iter().min().unwrap())?; + } + + Index::increment_statistic(&wtx, Statistic::Commits, 1)?; + wtx.commit()?; + + let wtx = index.begin_write()?; + + log::debug!("creating savepoint at height {}", height); + wtx.persistent_savepoint()?; + + Index::increment_statistic(&wtx, Statistic::Commits, 1)?; + wtx.commit()?; + } + + Ok(()) + } +} diff --git a/src/index/updater.rs b/src/index/updater.rs index 7b78fb435e..0828f156d3 100644 --- a/src/index/updater.rs +++ b/src/index/updater.rs @@ -8,9 +8,9 @@ use { mod inscription_updater; -struct BlockData { - header: Header, - txdata: Vec<(Transaction, Txid)>, +pub(crate) struct BlockData { + pub(crate) header: Header, + pub(crate) txdata: Vec<(Transaction, Txid)>, } impl From for BlockData { @@ -29,9 +29,10 @@ impl From for BlockData { } } -pub(crate) struct Updater { +pub(crate) struct Updater<'index> { range_cache: HashMap>, height: u64, + index: &'index Index, index_sats: bool, sat_ranges_since_flush: u64, outputs_cached: u64, @@ -39,48 +40,34 @@ pub(crate) struct Updater { outputs_traversed: u64, } -impl Updater { - pub(crate) fn update(index: &Index) -> Result { - let wtx = index.begin_write()?; +impl<'index> Updater<'_> { + pub(crate) fn new(index: &'index Index) -> Result> { + Ok(Updater { + range_cache: HashMap::new(), + height: index.block_count()?, + index, + index_sats: index.has_sat_index()?, + sat_ranges_since_flush: 0, + outputs_cached: 0, + outputs_inserted_since_flush: 0, + outputs_traversed: 0, + }) + } - let height = wtx - .open_table(HEIGHT_TO_BLOCK_HASH)? - .range(0..)? - .next_back() - .and_then(|result| result.ok()) - .map(|(height, _hash)| height.value() + 1) - .unwrap_or(0); + pub(crate) fn update_index(&mut self) -> Result { + let mut wtx = self.index.begin_write()?; + let starting_height = self.index.client.get_block_count()? + 1; wtx .open_table(WRITE_TRANSACTION_STARTING_BLOCK_COUNT_TO_TIMESTAMP)? .insert( - &height, + &self.height, &SystemTime::now() .duration_since(SystemTime::UNIX_EPOCH) .map(|duration| duration.as_millis()) .unwrap_or(0), )?; - let mut updater = Self { - range_cache: HashMap::new(), - height, - index_sats: index.has_sat_index()?, - sat_ranges_since_flush: 0, - outputs_cached: 0, - outputs_inserted_since_flush: 0, - outputs_traversed: 0, - }; - - updater.update_index(index, wtx) - } - - fn update_index<'index>( - &mut self, - index: &'index Index, - mut wtx: WriteTransaction<'index>, - ) -> Result { - let starting_height = index.client.get_block_count()? + 1; - let mut progress_bar = if cfg!(test) || log_enabled!(log::Level::Info) || starting_height <= self.height @@ -96,15 +83,15 @@ impl Updater { Some(progress_bar) }; - let rx = Self::fetch_blocks_from(index, self.height, self.index_sats)?; + let rx = Self::fetch_blocks_from(self.index, self.height, self.index_sats)?; - let (mut outpoint_sender, mut value_receiver) = Self::spawn_fetcher(index)?; + let (mut outpoint_sender, mut value_receiver) = Self::spawn_fetcher(self.index)?; let mut uncommitted = 0; let mut value_cache = HashMap::new(); while let Ok(block) = rx.recv() { self.index_block( - index, + self.index, &mut outpoint_sender, &mut value_receiver, &mut wtx, @@ -116,7 +103,7 @@ impl Updater { progress_bar.inc(1); if progress_bar.position() > progress_bar.length().unwrap() { - if let Ok(count) = index.client.get_block_count() { + if let Ok(count) = self.index.client.get_block_count() { progress_bar.set_length(count + 1); } else { log::warn!("Failed to fetch latest block height"); @@ -130,7 +117,7 @@ impl Updater { self.commit(wtx, value_cache)?; value_cache = HashMap::new(); uncommitted = 0; - wtx = index.begin_write()?; + wtx = self.index.begin_write()?; let height = wtx .open_table(HEIGHT_TO_BLOCK_HASH)? .range(0..)? @@ -330,6 +317,19 @@ impl Updater { block: BlockData, value_cache: &mut HashMap, ) -> Result<()> { + Reorg::detect_reorg(&block, self.height, self.index)?; + + let start = Instant::now(); + let mut sat_ranges_written = 0; + let mut outputs_in_block = 0; + + log::info!( + "Block {} at {} with {} transactions…", + self.height, + timestamp(block.header.time), + block.txdata.len() + ); + // If value_receiver still has values something went wrong with the last block // Could be an assert, shouldn't recover from this and commit the last block let Err(TryRecvError::Empty) = value_receiver.try_recv() else { @@ -375,29 +375,6 @@ impl Updater { } let mut height_to_block_hash = wtx.open_table(HEIGHT_TO_BLOCK_HASH)?; - - let start = Instant::now(); - let mut sat_ranges_written = 0; - let mut outputs_in_block = 0; - - let time = timestamp(block.header.time); - - log::info!( - "Block {} at {} with {} transactions…", - self.height, - time, - block.txdata.len() - ); - - if let Some(prev_height) = self.height.checked_sub(1) { - let prev_hash = height_to_block_hash.get(&prev_height)?.unwrap(); - - if prev_hash.value() != &block.header.prev_blockhash.as_raw_hash().to_byte_array() { - index.reorged.store(true, atomic::Ordering::Relaxed); - return Err(anyhow!("reorg detected at or before {prev_height}")); - } - } - let mut inscription_id_to_inscription_entry = wtx.open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY)?; let mut inscription_id_to_satpoint = wtx.open_table(INSCRIPTION_ID_TO_SATPOINT)?; @@ -655,8 +632,10 @@ impl Updater { Index::increment_statistic(&wtx, Statistic::SatRanges, self.sat_ranges_since_flush)?; self.sat_ranges_since_flush = 0; Index::increment_statistic(&wtx, Statistic::Commits, 1)?; - wtx.commit()?; + + Reorg::update_savepoints(self.index, self.height)?; + Ok(()) } } diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index d33290c4e3..8b1b1476b8 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -569,10 +569,10 @@ impl Server { } async fn status(Extension(index): Extension>) -> (StatusCode, &'static str) { - if index.is_reorged() { + if index.is_unrecoverably_reorged() { ( StatusCode::OK, - "reorg detected, please rebuild the database.", + "unrecoverable reorg detected, please rebuild the database.", ) } else { ( @@ -1944,17 +1944,20 @@ mod tests { } #[test] - fn detect_reorg() { + fn detect_unrecoverable_reorg() { let test_server = TestServer::new(); - test_server.mine_blocks(1); + test_server.mine_blocks(21); test_server.assert_response("/status", StatusCode::OK, "OK"); - test_server.bitcoin_rpc_server.invalidate_tip(); - test_server.bitcoin_rpc_server.mine_blocks(2); + for _ in 0..15 { + test_server.bitcoin_rpc_server.invalidate_tip(); + } + + test_server.bitcoin_rpc_server.mine_blocks(21); - test_server.assert_response_regex("/status", StatusCode::OK, "reorg detected.*"); + test_server.assert_response_regex("/status", StatusCode::OK, "unrecoverable reorg detected.*"); } #[test] @@ -2031,7 +2034,8 @@ mod tests { fn commits_are_tracked() { let server = TestServer::new(); - assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 1); + thread::sleep(Duration::from_millis(25)); + assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 3); let info = server.index.info().unwrap(); assert_eq!(info.transactions.len(), 1); @@ -2039,7 +2043,7 @@ mod tests { server.index.update().unwrap(); - assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 1); + assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 3); let info = server.index.info().unwrap(); assert_eq!(info.transactions.len(), 1); @@ -2050,7 +2054,7 @@ mod tests { thread::sleep(Duration::from_millis(10)); server.index.update().unwrap(); - assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 2); + assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 6); let info = server.index.info().unwrap(); assert_eq!(info.transactions.len(), 2); From 13c75926d392ca3f12fdd96cb408547d275e9d6f Mon Sep 17 00:00:00 2001 From: omahs <73983677+omahs@users.noreply.github.com> Date: Sun, 13 Aug 2023 14:33:24 +0200 Subject: [PATCH 06/61] Fix typos in documentation (#2328) * fix typos --- README.md | 6 +++--- docs/src/contributing.md | 4 ++-- docs/src/donate.md | 2 +- docs/src/faq.md | 4 ++-- docs/src/inscriptions.md | 2 +- docs/src/overview.md | 12 ++++++------ 6 files changed, 15 insertions(+), 15 deletions(-) diff --git a/README.md b/README.md index 93cbf89d79..be2203823b 100644 --- a/README.md +++ b/README.md @@ -40,7 +40,7 @@ This address is 2 of 4 multisig wallet with keys held by [rodarmor](https://twitter.com/rodarmor), and [ordinally](https://twitter.com/veryordinally). -Bitcoin received will go towards funding maintainance and development of `ord`, +Bitcoin received will go towards funding maintenance and development of `ord`, as well as hosting costs for [ordinals.com](https://ordinals.com). Thank you for donating! @@ -141,7 +141,7 @@ cargo test --all cargo test --all -- --ignored ``` -Have look at the [justfile](justfile) to see some more helpful recipes +Have a look at the [justfile](justfile) to see some more helpful recipes (commands). Here are a couple more good ones: ``` @@ -174,7 +174,7 @@ See `ord --help` for details. `bitcoind` RPC Authentication ----------------------------- -`ord` makes RPC calls to `bitcoind`, which usually require a username and +`ord` makes RPC calls to `bitcoind`, which usually requires a username and password. By default, `ord` looks a username and password in the cookie file created by diff --git a/docs/src/contributing.md b/docs/src/contributing.md index bfb24f4e97..b175845725 100644 --- a/docs/src/contributing.md +++ b/docs/src/contributing.md @@ -36,7 +36,7 @@ Ideas for small issues: - Find an issue that needs more research, and do that research and summarize it in a comment - Find an out-of-date issue and comment that it can be closed -- Find an issue that shouldn't be done, and provide constrictive feedback +- Find an issue that shouldn't be done, and provide constructive feedback detailing why you think that is the case Merge early and often @@ -49,7 +49,7 @@ test. Do research or testing, and report on your results. Break a feature into small sub-features, and implement them one at a time. Figuring out how to break down a larger PR into smaller PRs where each can be -merged is a art form well-worth practicing. The hard part is that each PR must +merged is an art form well-worth practicing. The hard part is that each PR must itself be an improvement. I strive to follow this advice myself, and am always better off when I do. diff --git a/docs/src/donate.md b/docs/src/donate.md index bdbf05cef1..1e2842bcc7 100644 --- a/docs/src/donate.md +++ b/docs/src/donate.md @@ -14,7 +14,7 @@ Both addresses are in a 2 of 4 multisig wallet with keys held by [rodarmor](https://twitter.com/rodarmor), and [ordinally](https://twitter.com/veryordinally). -Donations received will go towards funding maintainance and development of `ord`, +Donations received will go towards funding maintenance and development of `ord`, as well as hosting costs for [ordinals.com](https://ordinals.com). Thank you for donating! diff --git a/docs/src/faq.md b/docs/src/faq.md index 62086836b5..72d263425a 100644 --- a/docs/src/faq.md +++ b/docs/src/faq.md @@ -4,7 +4,7 @@ Ordinal Theory FAQ What is ordinal theory? ----------------------- -Ordinal theory is is a protocol for assigning serial numbers to satoshis, the +Ordinal theory is a protocol for assigning serial numbers to satoshis, the smallest subdivision of a bitcoin, and tracking those satoshis as they are spent by transactions. @@ -268,7 +268,7 @@ features like globally unique symbols and enhanced provenance. *Inscriptions do not support on-chain royalties.* This is negative, but only depending on how you look at it. On-chain royalties have been a boon for creators, but have also created a huge amount of confusion in the Ethereum NFT -ecosystem. The ecosystem now grapples with this issue, and is is engaged in a +ecosystem. The ecosystem now grapples with this issue, and is engaged in a race to the bottom, towards a royalties-optional future. Inscriptions have no support for on-chain royalties, because they are technically infeasible. If you choose to create inscriptions, there are many ways you can work around this diff --git a/docs/src/inscriptions.md b/docs/src/inscriptions.md index 94b0d67d41..27ba6971f0 100644 --- a/docs/src/inscriptions.md +++ b/docs/src/inscriptions.md @@ -54,7 +54,7 @@ First the string `ord` is pushed, to disambiguate inscriptions from other uses of envelopes. `OP_PUSH 1` indicates that the next push contains the content type, and `OP_PUSH -0`indicates that subsequent data pushes contain the content itself. Multiple data +0` indicates that subsequent data pushes contain the content itself. Multiple data pushes must be used for large inscriptions, as one of taproot's few restrictions is that individual data pushes may not be larger than 520 bytes. diff --git a/docs/src/overview.md b/docs/src/overview.md index bc51ee07bc..952d3ce66d 100644 --- a/docs/src/overview.md +++ b/docs/src/overview.md @@ -72,7 +72,7 @@ naturally lend themselves to a system of rarity. These periodic events are: - *Cycles*: Every six halvings, something magical happens: the halving and the difficulty adjustment coincide. This is called a conjunction, and the time period between conjunctions a cycle. A conjunction occurs roughly every 24 - years. The first conjunction should happen some time in 2032. + years. The first conjunction should happen sometime in 2032. This gives us the following rarity levels: @@ -102,7 +102,7 @@ Now for some examples. This satoshi is common: ``` 1°1′1″1‴ │ │ │ ╰─ Not first sat in block -│ │ ╰─── Not first block in difficutly adjustment period +│ │ ╰─── Not first block in difficulty adjustment period │ ╰───── Not first block in halving epoch ╰─────── Second cycle ``` @@ -113,7 +113,7 @@ This satoshi is uncommon: ``` 1°1′1″0‴ │ │ │ ╰─ First sat in block -│ │ ╰─── Not first block in difficutly adjustment period +│ │ ╰─── Not first block in difficulty adjustment period │ ╰───── Not first block in halving epoch ╰─────── Second cycle ``` @@ -163,7 +163,7 @@ from above: ``` 1°1′1″ -│ │ ╰─ Not first block in difficutly adjustment period +│ │ ╰─ Not first block in difficulty adjustment period │ ╰─── Not first block in halving epoch ╰───── Second cycle ``` @@ -203,7 +203,7 @@ the unspendable genesis block. As an example, 1905530482684727°'s name is "iaiufjszmoba". The name of the last satoshi to be mined is "a". Every combination of 10 characters or less is out -there, or will be out there, some day. +there, or will be out there, someday. Exotics ------- @@ -253,7 +253,7 @@ ordinals were independently discovered on at least two separate occasions, long before the era of modern NFTs began. On August 21st, 2012, Charlie Lee [posted a proposal to add proof-of-stake to -Bitcoin to the Bitocin Talk +Bitcoin to the Bitcoin Talk forum](https://bitcointalk.org/index.php?topic=102355.0). This wasn't an asset scheme, but did use the ordinal algorithm, and was implemented but never deployed. From f3a5182b5f85e02063d5d10c02f4026b22188e98 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Sun, 13 Aug 2023 06:03:06 -0700 Subject: [PATCH 07/61] Ignore invalid content type header values (#2326) Fixes #2322. --- src/subcommand/server.rs | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 8b1b1476b8..59d46286cf 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -815,9 +815,8 @@ impl Server { header::CONTENT_TYPE, inscription .content_type() - .unwrap_or("application/octet-stream") - .parse() - .unwrap(), + .and_then(|content_type| content_type.parse().ok()) + .unwrap_or(HeaderValue::from_static("application/octet-stream")), ); headers.insert( header::CONTENT_SECURITY_POLICY, @@ -2201,6 +2200,18 @@ mod tests { assert!(body.is_empty()); } + #[test] + fn content_response_bad_content_type() { + let (headers, body) = Server::content_response(Inscription::new( + Some("\n".as_bytes().to_vec()), + Some(Vec::new()), + )) + .unwrap(); + + assert_eq!(headers["content-type"], "application/octet-stream"); + assert!(body.is_empty()); + } + #[test] fn text_preview() { let server = TestServer::new_with_regtest(); From 40e4807d82508d4f346d34e4d353f7b816d59e0f Mon Sep 17 00:00:00 2001 From: ordinally <11798624+veryordinally@users.noreply.github.com> Date: Mon, 14 Aug 2023 14:52:11 +0200 Subject: [PATCH 08/61] JSON API for `/inscription`, `/inscriptions` and `/output` (#2323) --- Cargo.lock | 10 + Cargo.toml | 1 + src/chain.rs | 2 +- src/index.rs | 60 +++-- src/inscription.rs | 2 +- src/inscription_id.rs | 4 +- src/subcommand/server.rs | 181 +++++++++++---- src/subcommand/server/accept_json.rs | 28 ++- src/templates.rs | 12 +- src/templates/inscription.rs | 53 +++++ src/templates/inscriptions.rs | 27 +++ src/templates/output.rs | 35 +++ src/templates/sat.rs | 2 +- tests/json_api.rs | 318 ++++++++++++++++++++++++++- tests/lib.rs | 18 ++ 15 files changed, 667 insertions(+), 86 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 0089ed5dda..30f559314a 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1563,6 +1563,15 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "itertools" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1c173a5686ce8bfa551b3563d0c2170bf24ca44da99c7ca4bfdab5418c3fe57" +dependencies = [ + "either", +] + [[package]] name = "itoa" version = "1.0.9" @@ -2024,6 +2033,7 @@ dependencies = [ "http", "hyper", "indicatif", + "itertools", "lazy_static", "log", "mime", diff --git a/Cargo.toml b/Cargo.toml index eb4e6b6498..e852c4aa04 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -35,6 +35,7 @@ html-escaper = "0.2.0" http = "0.2.6" hyper = { version = "0.14.24", features = ["http1", "client"] } indicatif = "0.17.1" +itertools = "0.11.0" lazy_static = "1.4.0" log = "0.4.14" mime = "0.3.16" diff --git a/src/chain.rs b/src/chain.rs index 1f72115bbb..2d100d6dde 100644 --- a/src/chain.rs +++ b/src/chain.rs @@ -2,7 +2,7 @@ use {super::*, clap::ValueEnum}; #[derive(Default, ValueEnum, Copy, Clone, Debug, PartialEq, Serialize, Deserialize)] #[serde(rename_all = "kebab-case")] -pub(crate) enum Chain { +pub enum Chain { #[default] #[clap(alias("main"))] Mainnet, diff --git a/src/index.rs b/src/index.rs index f366f3c899..19f6f7c513 100644 --- a/src/index.rs +++ b/src/index.rs @@ -13,6 +13,7 @@ use { bitcoincore_rpc::{json::GetBlockHeaderResult, Client}, chrono::SubsecRound, indicatif::{ProgressBar, ProgressStyle}, + itertools::Itertools, log::log_enabled, redb::{ Database, MultimapTable, MultimapTableDefinition, ReadableMultimapTable, ReadableTable, Table, @@ -57,7 +58,7 @@ define_table! { STATISTIC_TO_COUNT, u64, u64 } define_table! { WRITE_TRANSACTION_STARTING_BLOCK_COUNT_TO_TIMESTAMP, u64, u128 } #[derive(Debug, PartialEq)] -pub(crate) enum List { +pub enum List { Spent, Unspent(Vec<(u64, u64)>), } @@ -877,19 +878,23 @@ impl Index { &self, n: usize, from: Option, - ) -> Result<(Vec, Option, Option)> { + ) -> Result<(Vec, Option, Option, i64, i64)> { let rtx = self.database.begin_read()?; let inscription_number_to_inscription_id = rtx.open_table(INSCRIPTION_NUMBER_TO_INSCRIPTION_ID)?; - let latest = match inscription_number_to_inscription_id.iter()?.next_back() { + let highest = match inscription_number_to_inscription_id.iter()?.next_back() { Some(Ok((number, _id))) => number.value(), - Some(Err(_)) => return Ok(Default::default()), - None => return Ok(Default::default()), + Some(Err(_)) | None => return Ok(Default::default()), }; - let from = from.unwrap_or(latest); + let lowest = match inscription_number_to_inscription_id.iter()?.next() { + Some(Ok((number, _id))) => number.value(), + Some(Err(_)) | None => return Ok(Default::default()), + }; + + let from = from.unwrap_or(highest); let prev = if let Some(prev) = from.checked_sub(n.try_into()?) { inscription_number_to_inscription_id @@ -899,12 +904,12 @@ impl Index { None }; - let next = if from < latest { + let next = if from < highest { Some( from .checked_add(n.try_into()?) - .unwrap_or(latest) - .min(latest), + .unwrap_or(highest) + .min(highest), ) } else { None @@ -917,7 +922,32 @@ impl Index { .flat_map(|result| result.map(|(_number, id)| Entry::load(*id.value()))) .collect(); - Ok((inscriptions, prev, next)) + Ok((inscriptions, prev, next, lowest, highest)) + } + + pub(crate) fn get_inscriptions_in_block(&self, block_height: u64) -> Result> { + // This is a naive approach and will require optimization, but we don't have an index by block + let block_inscriptions = self + .database + .begin_read()? + .open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY)? + .iter()? + .filter_map(|result| match result { + Ok((key, entry_value)) => { + let entry = InscriptionEntry::load(entry_value.value()); + if entry.height == block_height { + Some((InscriptionId::load(*key.value()), entry.number)) + } else { + None + } + } + Err(_) => None, + }) + .sorted_by_key(|&(_id, number)| number) + .map(|(id, _)| id) + .collect(); + + Ok(block_inscriptions) } pub(crate) fn get_feed_inscriptions(&self, n: usize) -> Result> { @@ -2447,7 +2477,7 @@ mod tests { context.mine_blocks(1); - let (inscriptions, prev, next) = context + let (inscriptions, prev, next, _, _) = context .index .get_latest_inscriptions_with_prev_and_next(100, None) .unwrap(); @@ -2476,15 +2506,17 @@ mod tests { ids.reverse(); - let (inscriptions, prev, next) = context + let (inscriptions, prev, next, lowest, highest) = context .index .get_latest_inscriptions_with_prev_and_next(100, None) .unwrap(); assert_eq!(inscriptions, &ids[..100]); assert_eq!(prev, Some(2)); assert_eq!(next, None); + assert_eq!(highest, 102); + assert_eq!(lowest, 0); - let (inscriptions, prev, next) = context + let (inscriptions, prev, next, _lowest, _highest) = context .index .get_latest_inscriptions_with_prev_and_next(100, Some(101)) .unwrap(); @@ -2492,7 +2524,7 @@ mod tests { assert_eq!(prev, Some(1)); assert_eq!(next, Some(102)); - let (inscriptions, prev, next) = context + let (inscriptions, prev, next, _lowest, _highest) = context .index .get_latest_inscriptions_with_prev_and_next(100, Some(0)) .unwrap(); diff --git a/src/inscription.rs b/src/inscription.rs index e647999fc6..5506c5f361 100644 --- a/src/inscription.rs +++ b/src/inscription.rs @@ -23,7 +23,7 @@ pub(crate) enum Curse { } #[derive(Debug, PartialEq, Clone)] -pub(crate) struct Inscription { +pub struct Inscription { body: Option>, content_type: Option>, } diff --git a/src/inscription_id.rs b/src/inscription_id.rs index 5ff9a347c4..998faeb3c3 100644 --- a/src/inscription_id.rs +++ b/src/inscription_id.rs @@ -2,8 +2,8 @@ use super::*; #[derive(Debug, PartialEq, Copy, Clone, Hash, Eq)] pub struct InscriptionId { - pub(crate) txid: Txid, - pub(crate) index: u32, + pub txid: Txid, + pub index: u32, } impl<'de> Deserialize<'de> for InscriptionId { diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 59d46286cf..8cc8f672a1 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -7,9 +7,10 @@ use { super::*, crate::page_config::PageConfig, crate::templates::{ - BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionsHtml, OutputHtml, - PageContent, PageHtml, PreviewAudioHtml, PreviewImageHtml, PreviewPdfHtml, PreviewTextHtml, - PreviewUnknownHtml, PreviewVideoHtml, RangeHtml, RareTxt, SatHtml, SatJson, TransactionHtml, + BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionJson, InscriptionsHtml, + InscriptionsJson, OutputHtml, OutputJson, PageContent, PageHtml, PreviewAudioHtml, + PreviewImageHtml, PreviewPdfHtml, PreviewTextHtml, PreviewUnknownHtml, PreviewVideoHtml, + RangeHtml, RareTxt, SatHtml, SatJson, TransactionHtml, }, axum::{ body, @@ -28,7 +29,7 @@ use { caches::DirCache, AcmeConfig, }, - std::{cmp::Ordering, str}, + std::{cmp::Ordering, str, sync::Arc}, tokio_stream::StreamExt, tower_http::{ compression::CompressionLayer, @@ -40,6 +41,11 @@ use { mod accept_json; mod error; +#[derive(Clone)] +pub struct ServerConfig { + pub is_json_api_enabled: bool, +} + enum BlockQuery { Height(u64), Hash(BlockHash), @@ -148,6 +154,10 @@ impl Server { domain: acme_domains.first().cloned(), }); + let server_config = Arc::new(ServerConfig { + is_json_api_enabled: index.is_json_api_enabled(), + }); + let router = Router::new() .route("/", get(Self::home)) .route("/block/:query", get(Self::block)) @@ -165,7 +175,9 @@ impl Server { .route("/input/:block/:transaction/:input", get(Self::input)) .route("/inscription/:inscription_id", get(Self::inscription)) .route("/inscriptions", get(Self::inscriptions)) + .route("/inscriptions/block/:n", get(Self::inscriptions_in_block)) .route("/inscriptions/:from", get(Self::inscriptions_from)) + .route("/inscriptions/:from/:n", get(Self::inscriptions_from_n)) .route("/install.sh", get(Self::install_script)) .route("/ordinal/:sat", get(Self::ordinal)) .route("/output/:output", get(Self::output)) @@ -194,7 +206,8 @@ impl Server { .allow_methods([http::Method::GET]) .allow_origin(Any), ) - .layer(CompressionLayer::new()); + .layer(CompressionLayer::new()) + .with_state(server_config); match (self.http_port(), self.https_port()) { (Some(http_port), None) => { @@ -386,31 +399,34 @@ impl Server { Path(DeserializeFromStr(sat)): Path>, accept_json: AcceptJson, ) -> ServerResult { - let satpoint = index.rare_sat_satpoint(sat)?; - let blocktime = index.block_time(sat.height())?; let inscriptions = index.get_inscription_ids_by_sat(sat)?; + let satpoint = index.rare_sat_satpoint(sat)?.or_else(|| { + inscriptions.first().and_then(|&first_inscription_id| { + index + .get_inscription_satpoint_by_id(first_inscription_id) + .ok() + .flatten() + }) + }); + let blocktime = index.block_time(sat.height())?; Ok(if accept_json.0 { - if index.is_json_api_enabled() { - Json(SatJson { - number: sat.0, - decimal: sat.decimal().to_string(), - degree: sat.degree().to_string(), - name: sat.name(), - block: sat.height().0, - cycle: sat.cycle(), - epoch: sat.epoch().0, - period: sat.period(), - offset: sat.third(), - rarity: sat.rarity(), - percentile: sat.percentile(), - satpoint, - timestamp: blocktime.timestamp().to_string(), - inscriptions, - }) - .into_response() - } else { - StatusCode::NOT_ACCEPTABLE.into_response() - } + Json(SatJson { + number: sat.0, + decimal: sat.decimal().to_string(), + degree: sat.degree().to_string(), + name: sat.name(), + block: sat.height().0, + cycle: sat.cycle(), + epoch: sat.epoch().0, + period: sat.period(), + offset: sat.third(), + rarity: sat.rarity(), + percentile: sat.percentile(), + satpoint, + timestamp: blocktime.timestamp().timestamp(), + inscriptions, + }) + .into_response() } else { SatHtml { sat, @@ -431,7 +447,8 @@ impl Server { Extension(page_config): Extension>, Extension(index): Extension>, Path(outpoint): Path, - ) -> ServerResult> { + accept_json: AcceptJson, + ) -> ServerResult { let list = if index.has_sat_index()? { index.list(outpoint)? } else { @@ -463,7 +480,16 @@ impl Server { let inscriptions = index.get_inscriptions_on_output(outpoint)?; - Ok( + Ok(if accept_json.0 { + Json(OutputJson::new( + outpoint, + list, + page_config.chain, + output, + inscriptions, + )) + .into_response() + } else { OutputHtml { outpoint, inscriptions, @@ -471,8 +497,9 @@ impl Server { chain: page_config.chain, output, } - .page(page_config, index.has_sat_index()?), - ) + .page(page_config, index.has_sat_index()?) + .into_response() + }) } async fn range( @@ -901,7 +928,8 @@ impl Server { Extension(page_config): Extension>, Extension(index): Extension>, Path(inscription_id): Path, - ) -> ServerResult> { + accept_json: AcceptJson, + ) -> ServerResult { let entry = index .get_inscription_entry(inscription_id)? .ok_or_not_found(|| format!("inscription {inscription_id}"))?; @@ -932,7 +960,23 @@ impl Server { let next = index.get_inscription_id_by_inscription_number(entry.number + 1)?; - Ok( + Ok(if accept_json.0 { + Json(InscriptionJson::new( + page_config.chain, + entry.fee, + entry.height, + inscription, + inscription_id, + next, + entry.number, + output, + previous, + entry.sat, + satpoint, + timestamp(entry.timestamp), + )) + .into_response() + } else { InscriptionHtml { chain: page_config.chain, genesis_fee: entry.fee, @@ -947,39 +991,84 @@ impl Server { satpoint, timestamp: timestamp(entry.timestamp), } - .page(page_config, index.has_sat_index()?), - ) + .page(page_config, index.has_sat_index()?) + .into_response() + }) } async fn inscriptions( Extension(page_config): Extension>, Extension(index): Extension>, - ) -> ServerResult> { - Self::inscriptions_inner(page_config, index, None).await + accept_json: AcceptJson, + ) -> ServerResult { + Self::inscriptions_inner(page_config, index, None, 100, accept_json).await + } + + async fn inscriptions_in_block( + Extension(page_config): Extension>, + Extension(index): Extension>, + Path(block_height): Path, + accept_json: AcceptJson, + ) -> ServerResult { + let inscriptions = index.get_inscriptions_in_block(block_height)?; + Ok(if accept_json.0 { + Json(InscriptionsJson::new(inscriptions, None, None, None, None)).into_response() + } else { + InscriptionsHtml { + inscriptions, + prev: None, + next: None, + } + .page(page_config, index.has_sat_index()?) + .into_response() + }) } async fn inscriptions_from( Extension(page_config): Extension>, Extension(index): Extension>, Path(from): Path, - ) -> ServerResult> { - Self::inscriptions_inner(page_config, index, Some(from)).await + accept_json: AcceptJson, + ) -> ServerResult { + Self::inscriptions_inner(page_config, index, Some(from), 100, accept_json).await + } + + async fn inscriptions_from_n( + Extension(page_config): Extension>, + Extension(index): Extension>, + Path((from, n)): Path<(i64, usize)>, + accept_json: AcceptJson, + ) -> ServerResult { + Self::inscriptions_inner(page_config, index, Some(from), n, accept_json).await } async fn inscriptions_inner( page_config: Arc, index: Arc, from: Option, - ) -> ServerResult> { - let (inscriptions, prev, next) = index.get_latest_inscriptions_with_prev_and_next(100, from)?; - Ok( + n: usize, + accept_json: AcceptJson, + ) -> ServerResult { + let (inscriptions, prev, next, lowest, highest) = + index.get_latest_inscriptions_with_prev_and_next(n, from)?; + Ok(if accept_json.0 { + Json(InscriptionsJson::new( + inscriptions, + prev, + next, + Some(lowest), + Some(highest), + )) + .into_response() + } else { InscriptionsHtml { inscriptions, next, prev, } - .page(page_config, index.has_sat_index()?), - ) + .page(page_config, index.has_sat_index()?) + .into_response() + }) } async fn redirect_http_to_https( @@ -2033,7 +2122,7 @@ mod tests { fn commits_are_tracked() { let server = TestServer::new(); - thread::sleep(Duration::from_millis(25)); + thread::sleep(Duration::from_millis(100)); assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 3); let info = server.index.info().unwrap(); diff --git a/src/subcommand/server/accept_json.rs b/src/subcommand/server/accept_json.rs index e01edf29c2..a471c3d049 100644 --- a/src/subcommand/server/accept_json.rs +++ b/src/subcommand/server/accept_json.rs @@ -1,24 +1,32 @@ -use super::*; +use {super::*, axum::extract::FromRef}; pub(crate) struct AcceptJson(pub(crate) bool); #[async_trait::async_trait] impl axum::extract::FromRequestParts for AcceptJson where + Arc: FromRef, S: Send + Sync, { - type Rejection = (); + type Rejection = (StatusCode, &'static str); async fn from_request_parts( parts: &mut http::request::Parts, - _state: &S, + state: &S, ) -> Result { - Ok(Self( - parts - .headers - .get("accept") - .map(|value| value == "application/json") - .unwrap_or_default(), - )) + let state = Arc::from_ref(state); + let json_api_enabled = state.is_json_api_enabled; + let json_header = parts + .headers + .get("accept") + .map(|value| value == "application/json") + .unwrap_or_default(); + if json_header && json_api_enabled { + Ok(Self(true)) + } else if json_header && !json_api_enabled { + Err((StatusCode::NOT_ACCEPTABLE, "JSON API not enabled")) + } else { + Ok(Self(false)) + } } } diff --git a/src/templates.rs b/src/templates.rs index e4f008f6a8..259ad50370 100644 --- a/src/templates.rs +++ b/src/templates.rs @@ -6,9 +6,9 @@ pub(crate) use { home::HomeHtml, iframe::Iframe, input::InputHtml, - inscription::InscriptionHtml, - inscriptions::InscriptionsHtml, - output::OutputHtml, + inscription::{InscriptionHtml, InscriptionJson}, + inscriptions::{InscriptionsHtml, InscriptionsJson}, + output::{OutputHtml, OutputJson}, page_config::PageConfig, preview::{ PreviewAudioHtml, PreviewImageHtml, PreviewPdfHtml, PreviewTextHtml, PreviewUnknownHtml, @@ -25,9 +25,9 @@ mod clock; mod home; mod iframe; mod input; -mod inscription; -mod inscriptions; -mod output; +pub mod inscription; +pub mod inscriptions; +pub mod output; mod preview; mod range; mod rare; diff --git a/src/templates/inscription.rs b/src/templates/inscription.rs index c19e5bf4b2..8700e82a9c 100644 --- a/src/templates/inscription.rs +++ b/src/templates/inscription.rs @@ -16,6 +16,59 @@ pub(crate) struct InscriptionHtml { pub(crate) timestamp: DateTime, } +#[derive(Debug, PartialEq, Serialize, Deserialize)] +pub struct InscriptionJson { + pub inscription_id: InscriptionId, + pub number: i64, + pub genesis_height: u64, + pub genesis_fee: u64, + pub output_value: Option, + pub address: Option, + pub sat: Option, + pub satpoint: SatPoint, + pub content_type: Option, + pub content_length: Option, + pub timestamp: i64, + pub previous: Option, + pub next: Option, +} + +impl InscriptionJson { + pub fn new( + chain: Chain, + genesis_fee: u64, + genesis_height: u64, + inscription: Inscription, + inscription_id: InscriptionId, + next: Option, + number: i64, + output: Option, + previous: Option, + sat: Option, + satpoint: SatPoint, + timestamp: DateTime, + ) -> Self { + Self { + inscription_id, + number, + genesis_height, + genesis_fee, + output_value: output.as_ref().map(|o| o.value), + address: output + .as_ref() + .and_then(|o| chain.address_from_script(&o.script_pubkey).ok()) + .map(|address| address.to_string()), + sat, + satpoint, + content_type: inscription.content_type().map(|s| s.to_string()), + content_length: inscription.content_length(), + timestamp: timestamp.timestamp(), + previous, + next, + } + } +} + impl PageContent for InscriptionHtml { fn title(&self) -> String { format!("Inscription {}", self.number) diff --git a/src/templates/inscriptions.rs b/src/templates/inscriptions.rs index 8399e8fa59..eb2a0e091f 100644 --- a/src/templates/inscriptions.rs +++ b/src/templates/inscriptions.rs @@ -7,6 +7,33 @@ pub(crate) struct InscriptionsHtml { pub(crate) next: Option, } +#[derive(Debug, PartialEq, Serialize, Deserialize)] +pub struct InscriptionsJson { + pub inscriptions: Vec, + pub prev: Option, + pub next: Option, + pub lowest: Option, + pub highest: Option, +} + +impl InscriptionsJson { + pub fn new( + inscriptions: Vec, + prev: Option, + next: Option, + lowest: Option, + highest: Option, + ) -> Self { + Self { + inscriptions, + prev, + next, + lowest, + highest, + } + } +} + impl PageContent for InscriptionsHtml { fn title(&self) -> String { "Inscriptions".into() diff --git a/src/templates/output.rs b/src/templates/output.rs index 2172795b18..fac57cc1d7 100644 --- a/src/templates/output.rs +++ b/src/templates/output.rs @@ -9,6 +9,41 @@ pub(crate) struct OutputHtml { pub(crate) inscriptions: Vec, } +#[derive(Debug, PartialEq, Serialize, Deserialize)] +pub struct OutputJson { + pub value: u64, + pub script_pubkey: String, + pub address: Option, + pub transaction: String, + pub sat_ranges: Option>, + pub inscriptions: Vec, +} + +impl OutputJson { + pub fn new( + outpoint: OutPoint, + list: Option, + chain: Chain, + output: TxOut, + inscriptions: Vec, + ) -> Self { + Self { + value: output.value, + script_pubkey: output.script_pubkey.to_asm_string(), + address: chain + .address_from_script(&output.script_pubkey) + .ok() + .map(|address| address.to_string()), + transaction: outpoint.txid.to_string(), + sat_ranges: match list { + Some(List::Unspent(ranges)) => Some(ranges), + _ => None, + }, + inscriptions, + } + } +} + impl PageContent for OutputHtml { fn title(&self) -> String { format!("Output {}", self.outpoint) diff --git a/src/templates/sat.rs b/src/templates/sat.rs index c42b839750..c79c717749 100644 --- a/src/templates/sat.rs +++ b/src/templates/sat.rs @@ -22,7 +22,7 @@ pub struct SatJson { pub rarity: Rarity, pub percentile: String, pub satpoint: Option, - pub timestamp: String, + pub timestamp: i64, pub inscriptions: Vec, } diff --git a/tests/json_api.rs b/tests/json_api.rs index 3cbba92188..893ede9327 100644 --- a/tests/json_api.rs +++ b/tests/json_api.rs @@ -1,6 +1,8 @@ use { - super::*, ord::inscription_id::InscriptionId, ord::rarity::Rarity, ord::templates::sat::SatJson, - ord::SatPoint, + super::*, ord::inscription_id::InscriptionId, ord::rarity::Rarity, + ord::templates::inscription::InscriptionJson, ord::templates::inscriptions::InscriptionsJson, + ord::templates::output::OutputJson, ord::templates::sat::SatJson, ord::SatPoint, + test_bitcoincore_rpc::TransactionTemplate, }; #[test] @@ -15,7 +17,7 @@ fn get_sat_without_sat_index() { let mut sat_json: SatJson = serde_json::from_str(&response.text().unwrap()).unwrap(); // this is a hack to ignore the timestamp, since it changes for every request - sat_json.timestamp = "".into(); + sat_json.timestamp = 0; pretty_assert_eq!( sat_json, @@ -32,7 +34,7 @@ fn get_sat_without_sat_index() { rarity: Rarity::Uncommon, percentile: "100%".into(), satpoint: None, - timestamp: "".into(), + timestamp: 0, inscriptions: vec![], } ) @@ -69,12 +71,318 @@ fn get_sat_with_inscription_and_sat_index() { rarity: Rarity::Uncommon, percentile: "0.00023809523835714296%".into(), satpoint: Some(SatPoint::from_str(&format!("{}:{}:{}", reveal, 0, 0)).unwrap()), - timestamp: "1970-01-01 00:00:01 UTC".into(), + timestamp: 1, inscriptions: vec![inscription_id], } ) } +#[test] +fn get_sat_with_inscription_on_common_sat_and_more_inscriptions() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + create_wallet(&rpc_server); + + inscribe(&rpc_server); + + let txid = rpc_server.mine_blocks(1)[0].txdata[0].txid(); + + let Inscribe { reveal, .. } = CommandBuilder::new(format!( + "wallet inscribe --satpoint {}:0:1 --fee-rate 1 foo.txt", + txid + )) + .write("foo.txt", "FOO") + .rpc_server(&rpc_server) + .run_and_check_output(); + + rpc_server.mine_blocks(1); + let inscription_id = InscriptionId::from(reveal); + + let response = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]) + .json_request(format!("/sat/{}", 3 * 50 * COIN_VALUE + 1)); + + assert_eq!(response.status(), StatusCode::OK); + + let sat_json: SatJson = serde_json::from_str(&response.text().unwrap()).unwrap(); + + pretty_assert_eq!( + sat_json, + SatJson { + number: 3 * 50 * COIN_VALUE + 1, + decimal: "3.1".into(), + degree: "0°3′3″1‴".into(), + name: "nvtblvikkiq".into(), + block: 3, + cycle: 0, + epoch: 0, + period: 0, + offset: 1, + rarity: Rarity::Common, + percentile: "0.000714285715119048%".into(), + satpoint: Some(SatPoint::from_str(&format!("{}:{}:{}", reveal, 0, 0)).unwrap()), + timestamp: 3, + inscriptions: vec![inscription_id], + } + ) +} + +#[test] +fn get_inscription() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + create_wallet(&rpc_server); + + let Inscribe { reveal, .. } = inscribe(&rpc_server); + let inscription_id = InscriptionId::from(reveal); + + let response = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]) + .json_request(format!("/inscription/{}", inscription_id)); + + assert_eq!(response.status(), StatusCode::OK); + + let mut inscription_json: InscriptionJson = + serde_json::from_str(&response.text().unwrap()).unwrap(); + assert_regex_match!(inscription_json.address.unwrap(), r"bc1p.*"); + inscription_json.address = None; + + pretty_assert_eq!( + inscription_json, + InscriptionJson { + inscription_id, + number: 0, + genesis_height: 2, + genesis_fee: 138, + output_value: Some(10000), + address: None, + sat: Some(ord::Sat(50 * COIN_VALUE)), + satpoint: SatPoint::from_str(&format!("{}:{}:{}", reveal, 0, 0)).unwrap(), + content_type: Some("text/plain;charset=utf-8".to_string()), + content_length: Some(3), + timestamp: 2, + previous: None, + next: None + } + ) +} + +fn create_210_inscriptions( + rpc_server: &test_bitcoincore_rpc::Handle, +) -> (Vec, Vec) { + let witness = envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]); + + let mut blessed_inscriptions = Vec::new(); + let mut cursed_inscriptions = Vec::new(); + + // Create 150 inscriptions, 50 non-cursed and 100 cursed + for i in 0..50 { + rpc_server.mine_blocks(1); + rpc_server.mine_blocks(1); + rpc_server.mine_blocks(1); + + let txid = rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(i * 3 + 1, 0, 0), (i * 3 + 2, 0, 0), (i * 3 + 3, 0, 0)], + witness: witness.clone(), + ..Default::default() + }); + + blessed_inscriptions.push(InscriptionId { txid, index: 0 }); + cursed_inscriptions.push(InscriptionId { txid, index: 1 }); + cursed_inscriptions.push(InscriptionId { txid, index: 2 }); + } + + rpc_server.mine_blocks(1); + + // Create another 60 non cursed + for _ in 0..60 { + let Inscribe { reveal, .. } = CommandBuilder::new("wallet inscribe --fee-rate 1 foo.txt") + .write("foo.txt", "FOO") + .rpc_server(rpc_server) + .run_and_check_output(); + rpc_server.mine_blocks(1); + blessed_inscriptions.push(InscriptionId::from(reveal)); + } + + rpc_server.mine_blocks(1); + + (blessed_inscriptions, cursed_inscriptions) +} + +#[test] +fn get_inscriptions() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + create_wallet(&rpc_server); + let (blessed_inscriptions, cursed_inscriptions) = create_210_inscriptions(&rpc_server); + + let server = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]); + + let response = server.json_request("/inscriptions"); + assert_eq!(response.status(), StatusCode::OK); + let inscriptions_json: InscriptionsJson = + serde_json::from_str(&response.text().unwrap()).unwrap(); + + // 100 latest (blessed) inscriptions + assert_eq!(inscriptions_json.inscriptions.len(), 100); + pretty_assert_eq!( + inscriptions_json, + InscriptionsJson { + inscriptions: blessed_inscriptions[10..110] + .iter() + .cloned() + .rev() + .collect(), + prev: Some(9), + next: None, + lowest: Some(-100), + highest: Some(109), + } + ); + + // get all inscriptions + let response = server.json_request(format!("/inscriptions/{}/{}", 200, 400)); + assert_eq!(response.status(), StatusCode::OK); + + let inscriptions_json: InscriptionsJson = + serde_json::from_str(&response.text().unwrap()).unwrap(); + + assert_eq!( + inscriptions_json.inscriptions.len(), + blessed_inscriptions.len() + cursed_inscriptions.len() + ); + pretty_assert_eq!( + inscriptions_json.inscriptions, + blessed_inscriptions + .iter() + .cloned() + .rev() + .chain(cursed_inscriptions.clone()) + .collect::>() + ); + + // iterate over all inscriptions 1 by 1 + let all_inscriptions = cursed_inscriptions + .clone() + .iter() + .cloned() + .rev() + .chain(blessed_inscriptions.clone()) + .collect::>(); // from lowest to highest inscription number + + let (lowest, highest) = ( + inscriptions_json.lowest.unwrap(), + inscriptions_json.highest.unwrap(), + ); + for i in lowest..=highest { + let response = server.json_request(format!("/inscriptions/{}/1", i)); + assert_eq!(response.status(), StatusCode::OK); + + let inscriptions_json: InscriptionsJson = + serde_json::from_str(&response.text().unwrap()).unwrap(); + + assert_eq!(inscriptions_json.inscriptions.len(), 1); + assert_eq!( + inscriptions_json.inscriptions[0], + all_inscriptions[(i - lowest) as usize] + ); + + let response = server.json_request(format!( + "/inscription/{}", + inscriptions_json.inscriptions[0] + )); + assert_eq!(response.status(), StatusCode::OK); + + let inscription_json: InscriptionJson = + serde_json::from_str(&response.text().unwrap()).unwrap(); + + assert_eq!( + inscription_json.inscription_id, + inscriptions_json.inscriptions[0] + ); + assert_eq!(inscription_json.number, i); + } +} + +#[test] +fn get_inscriptions_in_block() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + create_wallet(&rpc_server); + rpc_server.mine_blocks(10); + + let txid = rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(1, 0, 0), (2, 0, 0), (3, 0, 0)], + witness: envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]), + ..Default::default() + }); + rpc_server.mine_blocks(1); + + for _ in 0..10 { + inscribe(&rpc_server); + } + rpc_server.mine_blocks(1); + + let server = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]); + + // get all inscriptions from block 11 + let response = server.json_request(format!("/inscriptions/block/{}", 11)); + assert_eq!(response.status(), StatusCode::OK); + + let inscriptions_json: InscriptionsJson = + serde_json::from_str(&response.text().unwrap()).unwrap(); + + assert_eq!(inscriptions_json.inscriptions.len(), 3); + pretty_assert_eq!( + inscriptions_json.inscriptions, + vec![ + InscriptionId { txid, index: 2 }, + InscriptionId { txid, index: 1 }, + InscriptionId { txid, index: 0 } + ] + ); +} + +#[test] +fn get_output() { + let rpc_server = test_bitcoincore_rpc::spawn(); + + create_wallet(&rpc_server); + rpc_server.mine_blocks(3); + + let txid = rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(1, 0, 0), (2, 0, 0), (3, 0, 0)], + witness: envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]), + ..Default::default() + }); + rpc_server.mine_blocks(1); + + let server = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]); + + let response = server.json_request(format!("/output/{}:0", txid)); + assert_eq!(response.status(), StatusCode::OK); + + let output_json: OutputJson = serde_json::from_str(&response.text().unwrap()).unwrap(); + + pretty_assert_eq!( + output_json, + OutputJson { + value: 3 * 50 * COIN_VALUE, + script_pubkey: "".to_string(), + address: None, + transaction: txid.to_string(), + sat_ranges: Some(vec![ + (5000000000, 10000000000,), + (10000000000, 15000000000,), + (15000000000, 20000000000,), + ],), + inscriptions: vec![ + InscriptionId { txid, index: 0 }, + InscriptionId { txid, index: 2 }, + InscriptionId { txid, index: 1 } + ] + } + ); +} + #[test] fn json_request_fails_when_not_enabled() { let rpc_server = test_bitcoincore_rpc::spawn(); diff --git a/tests/lib.rs b/tests/lib.rs index 6ad150a58e..037b97110c 100644 --- a/tests/lib.rs +++ b/tests/lib.rs @@ -62,6 +62,24 @@ fn inscribe(rpc_server: &test_bitcoincore_rpc::Handle) -> Inscribe { output } +fn envelope(payload: &[&[u8]]) -> bitcoin::Witness { + let mut builder = bitcoin::script::Builder::new() + .push_opcode(bitcoin::opcodes::OP_FALSE) + .push_opcode(bitcoin::opcodes::all::OP_IF); + + for data in payload { + let mut buf = bitcoin::script::PushBytesBuf::new(); + buf.extend_from_slice(data).unwrap(); + builder = builder.push_slice(buf); + } + + let script = builder + .push_opcode(bitcoin::opcodes::all::OP_ENDIF) + .into_script(); + + bitcoin::Witness::from_slice(&[script.into_bytes(), Vec::new()]) +} + #[derive(Deserialize)] struct Create { mnemonic: Mnemonic, From 621a7470c07b307450e1b0fb6857c70dd8e6270f Mon Sep 17 00:00:00 2001 From: ordinally <11798624+veryordinally@users.noreply.github.com> Date: Thu, 17 Aug 2023 14:54:43 +0200 Subject: [PATCH 09/61] Make retrieving inscriptions in block fast (#2333) --- src/index.rs | 32 ++----- src/index/block_index.rs | 201 +++++++++++++++++++++++++++++++++++++++ src/subcommand/server.rs | 38 ++++++-- tests/json_api.rs | 11 ++- 4 files changed, 249 insertions(+), 33 deletions(-) create mode 100644 src/index/block_index.rs diff --git a/src/index.rs b/src/index.rs index 19f6f7c513..c09a50c31f 100644 --- a/src/index.rs +++ b/src/index.rs @@ -4,6 +4,7 @@ use { BlockHashValue, Entry, InscriptionEntry, InscriptionEntryValue, InscriptionIdValue, OutPointValue, SatPointValue, SatRange, }, + index::block_index::BlockIndex, reorg::*, updater::Updater, }, @@ -13,7 +14,6 @@ use { bitcoincore_rpc::{json::GetBlockHeaderResult, Client}, chrono::SubsecRound, indicatif::{ProgressBar, ProgressStyle}, - itertools::Itertools, log::log_enabled, redb::{ Database, MultimapTable, MultimapTableDefinition, ReadableMultimapTable, ReadableTable, Table, @@ -23,6 +23,7 @@ use { std::io::{BufWriter, Read, Write}, }; +pub mod block_index; mod entry; mod fetcher; mod reorg; @@ -925,29 +926,12 @@ impl Index { Ok((inscriptions, prev, next, lowest, highest)) } - pub(crate) fn get_inscriptions_in_block(&self, block_height: u64) -> Result> { - // This is a naive approach and will require optimization, but we don't have an index by block - let block_inscriptions = self - .database - .begin_read()? - .open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY)? - .iter()? - .filter_map(|result| match result { - Ok((key, entry_value)) => { - let entry = InscriptionEntry::load(entry_value.value()); - if entry.height == block_height { - Some((InscriptionId::load(*key.value()), entry.number)) - } else { - None - } - } - Err(_) => None, - }) - .sorted_by_key(|&(_id, number)| number) - .map(|(id, _)| id) - .collect(); - - Ok(block_inscriptions) + pub(crate) fn get_inscriptions_in_block( + &self, + block_index: &BlockIndex, + block_height: u64, + ) -> Result> { + block_index.get_inscriptions_in_block(self, block_height) } pub(crate) fn get_feed_inscriptions(&self, n: usize) -> Result> { diff --git a/src/index/block_index.rs b/src/index/block_index.rs new file mode 100644 index 0000000000..c036080f22 --- /dev/null +++ b/src/index/block_index.rs @@ -0,0 +1,201 @@ +use super::*; + +#[derive(Clone)] +pub struct BlockIndex { + first_inscription_height: u64, + lowest_blessed_by_block: Vec, + lowest_cursed_by_block: Vec, + highest_indexed_blessed: i64, + lowest_indexed_cursed: i64, +} + +impl BlockIndex { + pub(crate) fn new(index: &Index) -> Result { + Ok(BlockIndex { + first_inscription_height: index.options.first_inscription_height(), + lowest_blessed_by_block: Vec::new(), + lowest_cursed_by_block: Vec::new(), + highest_indexed_blessed: i64::MIN, + lowest_indexed_cursed: i64::MAX, + }) + } + + pub(crate) fn update(&mut self, index: &Index) -> Result { + let index_height = index.block_count()?; + let inscribed_block_count = index_height.saturating_sub(self.first_inscription_height); + let indexed_up_to: isize = self + .lowest_blessed_by_block + .len() + .try_into() + .unwrap_or(isize::MAX); + + let gap = inscribed_block_count.try_into().unwrap_or(isize::MAX) - indexed_up_to; + if gap <= 0 { + return Ok(()); + } + + log::debug!( + "Updating block index for {} new blocks ({} to {})", + gap, + indexed_up_to, + inscribed_block_count + ); + + self + .lowest_blessed_by_block + .resize(usize::try_from(inscribed_block_count)?, i64::MAX); + + self + .lowest_cursed_by_block + .resize(usize::try_from(inscribed_block_count)?, i64::MAX); + + let rtx = index.database.begin_read()?; + + // Use a more efficient approach for the initial indexing - since we have + // to traverse all inscriptions, it is most efficient to do so using one table. + if indexed_up_to == 0 { + for result in rtx + .open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY)? + .iter()? + { + let (_, entry) = result?; + let entry = InscriptionEntry::load(entry.value()); + let height_index: usize = entry + .height + .try_into() + .unwrap_or(usize::MAX) + .saturating_sub(self.first_inscription_height.try_into().unwrap()); + + if entry.number < 0 { + self.lowest_cursed_by_block[height_index] = + cmp::min(self.lowest_cursed_by_block[height_index], entry.number); + self.lowest_indexed_cursed = cmp::min(self.lowest_indexed_cursed, entry.number); + } else { + self.lowest_blessed_by_block[height_index] = + cmp::min(self.lowest_blessed_by_block[height_index], entry.number); + self.highest_indexed_blessed = cmp::max(self.highest_indexed_blessed, entry.number); + } + } + } else { + // Use default approach where we iterate in order of inscription number + // so we can easily skip over already indexed inscriptions. + let mut prev_block_height = usize::MAX; + + for result in rtx + .open_table(INSCRIPTION_NUMBER_TO_INSCRIPTION_ID)? + .iter()? + { + let (number, id) = result?; + + if number.value() >= self.lowest_indexed_cursed + && number.value() <= self.highest_indexed_blessed + { + continue; + } + + let inscription_id = InscriptionId::load(*id.value()); + + if let Some(entry) = index.get_inscription_entry(inscription_id)? { + let current_height = entry.height.try_into().unwrap_or(usize::MAX); + + if prev_block_height != current_height { + prev_block_height = current_height; + + if number.value() < 0 { + self.lowest_cursed_by_block[prev_block_height + .saturating_sub(usize::try_from(self.first_inscription_height)?)] = number.value(); + self.lowest_indexed_cursed = cmp::min(self.lowest_indexed_cursed, number.value()); + } else { + self.lowest_blessed_by_block[prev_block_height + .saturating_sub(usize::try_from(self.first_inscription_height)?)] = number.value(); + self.highest_indexed_blessed = cmp::max(self.highest_indexed_blessed, number.value()); + } + } + } + } + } + + log::debug!( + "Updated block index for {} new blocks ({} to {})", + gap, + indexed_up_to, + inscribed_block_count + ); + + Ok(()) + } + + // Return all consecutively numbered inscriptions in the block at the given height, starting from the given number + fn get_inscriptions_in_block_from( + &self, + index: &Index, + block_height: u64, + from_number: i64, + cursed: bool, + ) -> Result> { + let mut block_inscriptions = Vec::new(); + + let rtx = index.database.begin_read()?; + let inscription_id_by_number = rtx.open_table(INSCRIPTION_NUMBER_TO_INSCRIPTION_ID)?; + + let highest = if cursed { + -1 + } else { + match inscription_id_by_number.iter()?.next_back() { + Some(Ok((number, _id))) => number.value(), + Some(Err(err)) => return Err(err.into()), + None => i64::MIN, + } + }; + + for number in from_number..=highest { + match inscription_id_by_number.get(number)? { + Some(inscription_id) => { + let inscription_id = InscriptionId::load(*inscription_id.value()); + if let Some(entry) = index.get_inscription_entry(inscription_id)? { + if entry.height != block_height { + break; + } + block_inscriptions.push(inscription_id); + } + } + None => break, + } + } + + Ok(block_inscriptions) + } + + pub(crate) fn get_inscriptions_in_block( + &self, + index: &Index, + block_height: u64, + ) -> Result> { + if block_height >= index.block_count()? || block_height < self.first_inscription_height { + return Ok(Vec::new()); + } + let lowest_cursed = self.lowest_cursed_by_block + [usize::try_from(block_height.saturating_sub(self.first_inscription_height))?]; + let lowest_blessed = self.lowest_blessed_by_block + [usize::try_from(block_height.saturating_sub(self.first_inscription_height))?]; + + let mut inscriptions = + self.get_inscriptions_in_block_from(index, block_height, lowest_cursed, true)?; + inscriptions.extend(self.get_inscriptions_in_block_from( + index, + block_height, + lowest_blessed, + false, + )?); + + log::debug!( + "Got {} inscriptions in block {} ({} - {})", + inscriptions.len(), + block_height, + lowest_cursed, + lowest_blessed + ); + + Ok(inscriptions) + } +} diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 8cc8f672a1..46b9fac786 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -5,6 +5,7 @@ use { error::{OptionExt, ServerError, ServerResult}, }, super::*, + crate::index::block_index::BlockIndex, crate::page_config::PageConfig, crate::templates::{ BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionJson, InscriptionsHtml, @@ -29,7 +30,7 @@ use { caches::DirCache, AcmeConfig, }, - std::{cmp::Ordering, str, sync::Arc}, + std::{cmp::Ordering, str, sync::Arc, sync::RwLock}, tokio_stream::StreamExt, tower_http::{ compression::CompressionLayer, @@ -46,6 +47,10 @@ pub struct ServerConfig { pub is_json_api_enabled: bool, } +struct BlockIndexState { + block_index: RwLock, +} + enum BlockQuery { Height(u64), Hash(BlockHash), @@ -134,18 +139,38 @@ pub(crate) struct Server { impl Server { pub(crate) fn run(self, options: Options, index: Arc, handle: Handle) -> Result { Runtime::new()?.block_on(async { + let block_index_state = BlockIndexState { + block_index: RwLock::new(BlockIndex::new(&index)?), + }; + + let block_index_state = Arc::new(block_index_state); + let index_clone = index.clone(); + let block_index_clone = block_index_state.clone(); + let index_thread = thread::spawn(move || loop { if SHUTTING_DOWN.load(atomic::Ordering::Relaxed) { break; } if let Err(error) = index_clone.update() { - log::warn!("{error}"); + log::warn!("Updating index: {error}"); + } + if let Err(error) = block_index_clone + .block_index + .write() + .unwrap() + .update(&index_clone) + { + log::warn!("Updating block index: {error}"); } thread::sleep(Duration::from_millis(5000)); }); INDEXER.lock().unwrap().replace(index_thread); + let server_config = Arc::new(ServerConfig { + is_json_api_enabled: index.is_json_api_enabled(), + }); + let config = options.load_config()?; let acme_domains = self.acme_domains()?; @@ -154,10 +179,6 @@ impl Server { domain: acme_domains.first().cloned(), }); - let server_config = Arc::new(ServerConfig { - is_json_api_enabled: index.is_json_api_enabled(), - }); - let router = Router::new() .route("/", get(Self::home)) .route("/block/:query", get(Self::block)) @@ -193,6 +214,7 @@ impl Server { .layer(Extension(index)) .layer(Extension(page_config)) .layer(Extension(Arc::new(config))) + .layer(Extension(block_index_state)) .layer(SetResponseHeaderLayer::if_not_present( header::CONTENT_SECURITY_POLICY, HeaderValue::from_static("default-src 'self'"), @@ -1007,10 +1029,12 @@ impl Server { async fn inscriptions_in_block( Extension(page_config): Extension>, Extension(index): Extension>, + Extension(block_index_state): Extension>, Path(block_height): Path, accept_json: AcceptJson, ) -> ServerResult { - let inscriptions = index.get_inscriptions_in_block(block_height)?; + let inscriptions = index + .get_inscriptions_in_block(&block_index_state.block_index.read().unwrap(), block_height)?; Ok(if accept_json.0 { Json(InscriptionsJson::new(inscriptions, None, None, None, None)).into_response() } else { diff --git a/tests/json_api.rs b/tests/json_api.rs index 893ede9327..893d7c7673 100644 --- a/tests/json_api.rs +++ b/tests/json_api.rs @@ -321,7 +321,15 @@ fn get_inscriptions_in_block() { } rpc_server.mine_blocks(1); - let server = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]); + let server = TestServer::spawn_with_args( + &rpc_server, + &[ + "--index-sats", + "--enable-json-api", + "--first-inscription-height", + "0", + ], + ); // get all inscriptions from block 11 let response = server.json_request(format!("/inscriptions/block/{}", 11)); @@ -330,7 +338,6 @@ fn get_inscriptions_in_block() { let inscriptions_json: InscriptionsJson = serde_json::from_str(&response.text().unwrap()).unwrap(); - assert_eq!(inscriptions_json.inscriptions.len(), 3); pretty_assert_eq!( inscriptions_json.inscriptions, vec![ From 8f15f6bcb12ef530f6feabe7baea93e683c92153 Mon Sep 17 00:00:00 2001 From: raph Date: Thu, 17 Aug 2023 15:17:49 +0200 Subject: [PATCH 10/61] Allow setting custom postage (#2331) --- src/subcommand/preview.rs | 1 + src/subcommand/wallet.rs | 2 +- src/subcommand/wallet/inscribe.rs | 27 +- src/subcommand/wallet/send.rs | 19 +- src/subcommand/wallet/transaction_builder.rs | 331 ++++++++++--------- tests/wallet/inscribe.rs | 21 ++ tests/wallet/send.rs | 33 ++ 7 files changed, 277 insertions(+), 157 deletions(-) diff --git a/src/subcommand/preview.rs b/src/subcommand/preview.rs index 462f762952..915fb136f2 100644 --- a/src/subcommand/preview.rs +++ b/src/subcommand/preview.rs @@ -87,6 +87,7 @@ impl Preview { dry_run: false, no_limit: false, destination: None, + postage: Some(TransactionBuilder::TARGET_POSTAGE), }, )), } diff --git a/src/subcommand/wallet.rs b/src/subcommand/wallet.rs index 00c1d8da97..875ed5200f 100644 --- a/src/subcommand/wallet.rs +++ b/src/subcommand/wallet.rs @@ -24,7 +24,7 @@ pub mod receive; mod restore; pub mod sats; pub mod send; -pub(crate) mod transaction_builder; +pub mod transaction_builder; pub mod transactions; #[derive(Debug, Parser)] diff --git a/src/subcommand/wallet/inscribe.rs b/src/subcommand/wallet/inscribe.rs index c3b8aac51b..f712890656 100644 --- a/src/subcommand/wallet/inscribe.rs +++ b/src/subcommand/wallet/inscribe.rs @@ -1,6 +1,6 @@ use { super::*, - crate::wallet::Wallet, + crate::{subcommand::wallet::transaction_builder::Target, wallet::Wallet}, bitcoin::{ blockdata::{opcodes, script}, key::PrivateKey, @@ -51,6 +51,11 @@ pub(crate) struct Inscribe { pub(crate) dry_run: bool, #[clap(long, help = "Send inscription to .")] pub(crate) destination: Option>, + #[clap( + long, + help = "Amount of postage to include in the inscription. Default `10000sat`" + )] + pub(crate) postage: Option, } impl Inscribe { @@ -88,6 +93,10 @@ impl Inscribe { self.commit_fee_rate.unwrap_or(self.fee_rate), self.fee_rate, self.no_limit, + match self.postage { + Some(postage) => postage, + _ => TransactionBuilder::TARGET_POSTAGE, + }, )?; utxos.insert( @@ -155,6 +164,7 @@ impl Inscribe { commit_fee_rate: FeeRate, reveal_fee_rate: FeeRate, no_limit: bool, + postage: Amount, ) -> Result<(Transaction, Transaction, TweakedKeyPair)> { let satpoint = if let Some(satpoint) = satpoint { satpoint @@ -220,15 +230,16 @@ impl Inscribe { &reveal_script, ); - let unsigned_commit_tx = TransactionBuilder::build_transaction_with_value( + let unsigned_commit_tx = TransactionBuilder::new( satpoint, inscriptions, utxos, commit_tx_address.clone(), change, commit_fee_rate, - reveal_fee + TransactionBuilder::TARGET_POSTAGE, - )?; + Target::Value(reveal_fee + postage), + ) + .build_transaction()?; let (vout, output) = unsigned_commit_tx .output @@ -393,6 +404,7 @@ mod tests { FeeRate::try_from(1.0).unwrap(), FeeRate::try_from(1.0).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap(); @@ -424,6 +436,7 @@ mod tests { FeeRate::try_from(1.0).unwrap(), FeeRate::try_from(1.0).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap(); @@ -459,6 +472,7 @@ mod tests { FeeRate::try_from(1.0).unwrap(), FeeRate::try_from(1.0).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap_err() .to_string(); @@ -501,6 +515,7 @@ mod tests { FeeRate::try_from(1.0).unwrap(), FeeRate::try_from(1.0).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .is_ok()) } @@ -537,6 +552,7 @@ mod tests { FeeRate::try_from(fee_rate).unwrap(), FeeRate::try_from(fee_rate).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap(); @@ -599,6 +615,7 @@ mod tests { FeeRate::try_from(commit_fee_rate).unwrap(), FeeRate::try_from(fee_rate).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap(); @@ -648,6 +665,7 @@ mod tests { FeeRate::try_from(1.0).unwrap(), FeeRate::try_from(1.0).unwrap(), false, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap_err() .to_string(); @@ -679,6 +697,7 @@ mod tests { FeeRate::try_from(1.0).unwrap(), FeeRate::try_from(1.0).unwrap(), true, + TransactionBuilder::TARGET_POSTAGE, ) .unwrap(); diff --git a/src/subcommand/wallet/send.rs b/src/subcommand/wallet/send.rs index f55c4601e5..2c6c9fd574 100644 --- a/src/subcommand/wallet/send.rs +++ b/src/subcommand/wallet/send.rs @@ -1,4 +1,4 @@ -use {super::*, crate::wallet::Wallet}; +use {super::*, crate::subcommand::wallet::transaction_builder::Target, crate::wallet::Wallet}; #[derive(Debug, Parser)] pub(crate) struct Send { @@ -6,6 +6,11 @@ pub(crate) struct Send { outgoing: Outgoing, #[clap(long, help = "Use fee rate of sats/vB")] fee_rate: FeeRate, + #[clap( + long, + help = "Target amount of postage to include with sent inscriptions. Default `10000sat`" + )] + pub(crate) postage: Option, } #[derive(Serialize, Deserialize)] @@ -67,14 +72,22 @@ impl Send { get_change_address(&client, &options)?, ]; - let unsigned_transaction = TransactionBuilder::build_transaction_with_postage( + let postage = if let Some(postage) = self.postage { + Target::ExactPostage(postage) + } else { + Target::Postage + }; + + let unsigned_transaction = TransactionBuilder::new( satpoint, inscriptions, unspent_outputs, address, change, self.fee_rate, - )?; + postage, + ) + .build_transaction()?; let signed_tx = client .sign_raw_transaction_with_wallet(&unsigned_transaction, None, None)? diff --git a/src/subcommand/wallet/transaction_builder.rs b/src/subcommand/wallet/transaction_builder.rs index 5738e0c8ee..77003bc53a 100644 --- a/src/subcommand/wallet/transaction_builder.rs +++ b/src/subcommand/wallet/transaction_builder.rs @@ -63,9 +63,10 @@ pub enum Error { } #[derive(Debug, PartialEq)] -enum Target { +pub enum Target { Value(Amount), Postage, + ExactPostage(Amount), } impl fmt::Display for Error { @@ -97,7 +98,7 @@ impl fmt::Display for Error { impl std::error::Error for Error {} -#[derive(Debug)] +#[derive(Debug, PartialEq)] pub struct TransactionBuilder { amounts: BTreeMap, change_addresses: BTreeSet
, @@ -117,61 +118,59 @@ type Result = std::result::Result; impl TransactionBuilder { const ADDITIONAL_INPUT_VBYTES: usize = 58; const ADDITIONAL_OUTPUT_VBYTES: usize = 43; - const MAX_POSTAGE: Amount = Amount::from_sat(2 * 10_000); const SCHNORR_SIGNATURE_SIZE: usize = 64; pub(crate) const TARGET_POSTAGE: Amount = Amount::from_sat(10_000); + pub(crate) const MAX_POSTAGE: Amount = Amount::from_sat(2 * 10_000); - pub fn build_transaction_with_postage( + pub(crate) fn new( outgoing: SatPoint, inscriptions: BTreeMap, amounts: BTreeMap, recipient: Address, change: [Address; 2], fee_rate: FeeRate, - ) -> Result { - Self::new( - outgoing, - inscriptions, + target: Target, + ) -> Self { + Self { + utxos: amounts.keys().cloned().collect(), amounts, - recipient, - change, + change_addresses: change.iter().cloned().collect(), fee_rate, - Target::Postage, - )? - .build_transaction() + inputs: Vec::new(), + inscriptions, + outgoing, + outputs: Vec::new(), + recipient, + unused_change_addresses: change.to_vec(), + target, + } } - pub fn build_transaction_with_value( - outgoing: SatPoint, - inscriptions: BTreeMap, - amounts: BTreeMap, - recipient: Address, - change: [Address; 2], - fee_rate: FeeRate, - output_value: Amount, - ) -> Result { - let dust_value = recipient.script_pubkey().dust_value(); + pub(crate) fn build_transaction(self) -> Result { + if self.change_addresses.len() < 2 { + return Err(Error::DuplicateAddress( + self.change_addresses.first().unwrap().clone(), + )); + } - if output_value < dust_value { - return Err(Error::Dust { - output_value, - dust_value, - }); + if self.change_addresses.contains(&self.recipient) { + return Err(Error::DuplicateAddress(self.recipient)); } - Self::new( - outgoing, - inscriptions, - amounts, - recipient, - change, - fee_rate, - Target::Value(output_value), - )? - .build_transaction() - } + match self.target { + Target::Value(output_value) | Target::ExactPostage(output_value) => { + let dust_value = self.recipient.script_pubkey().dust_value(); + + if output_value < dust_value { + return Err(Error::Dust { + output_value, + dust_value, + }); + } + } + _ => (), + } - fn build_transaction(self) -> Result { self .select_outgoing()? .align_outgoing() @@ -182,38 +181,6 @@ impl TransactionBuilder { .build() } - fn new( - outgoing: SatPoint, - inscriptions: BTreeMap, - amounts: BTreeMap, - recipient: Address, - change: [Address; 2], - fee_rate: FeeRate, - target: Target, - ) -> Result { - if change.contains(&recipient) { - return Err(Error::DuplicateAddress(recipient)); - } - - if change[0] == change[1] { - return Err(Error::DuplicateAddress(change[0].clone())); - } - - Ok(Self { - utxos: amounts.keys().cloned().collect(), - amounts, - change_addresses: change.iter().cloned().collect(), - fee_rate, - inputs: Vec::new(), - inscriptions, - outgoing, - outputs: Vec::new(), - recipient, - unused_change_addresses: change.to_vec(), - target, - }) - } - fn select_outgoing(mut self) -> Result { for (inscribed_satpoint, inscription_id) in &self.inscriptions { if self.outgoing.outpoint == inscribed_satpoint.outpoint @@ -315,7 +282,7 @@ impl TransactionBuilder { let min_value = match self.target { Target::Postage => self.outputs.last().unwrap().0.script_pubkey().dust_value(), - Target::Value(value) => value, + Target::Value(value) | Target::ExactPostage(value) => value, }; let total = min_value @@ -372,6 +339,7 @@ impl TransactionBuilder { if let Some(excess) = value.checked_sub(self.fee_rate.fee(self.estimate_vbytes())) { let (max, target) = match self.target { + Target::ExactPostage(postage) => (postage, postage), Target::Postage => (Self::MAX_POSTAGE, Self::TARGET_POSTAGE), Target::Value(value) => (value, value), }; @@ -461,7 +429,7 @@ impl TransactionBuilder { previous_output: OutPoint::null(), script_sig: ScriptBuf::new(), sequence: Sequence::ENABLE_RBF_NO_LOCKTIME, - witness: Witness::from_slice(&[&[0; TransactionBuilder::SCHNORR_SIGNATURE_SIZE]]), + witness: Witness::from_slice(&[&[0; Self::SCHNORR_SIGNATURE_SIZE]]), }) .collect(), output: outputs @@ -588,6 +556,12 @@ impl TransactionBuilder { "invariant: excess postage is stripped" ); } + Target::ExactPostage(postage) => { + assert!( + Amount::from_sat(output.value) <= postage + slop, + "invariant: excess postage is stripped" + ); + } Target::Value(value) => { assert!( Amount::from_sat(output.value).checked_sub(value).unwrap() @@ -747,7 +721,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap(); @@ -810,14 +783,16 @@ mod tests { fn transactions_are_rbf() { let utxos = vec![(outpoint(1), Amount::from_sat(5_000))]; - assert!(TransactionBuilder::build_transaction_with_postage( + assert!(TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), + Target::Postage, ) + .build_transaction() .unwrap() .is_explicitly_rbf()) } @@ -827,14 +802,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(5_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -858,7 +835,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap() .align_outgoing() @@ -874,14 +850,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 4_950), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -896,14 +874,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(5_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 4_950), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Err(Error::NotEnoughCardinalUtxos), ) } @@ -916,14 +896,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 4_950), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage + ) + .build_transaction(), Err(Error::NotEnoughCardinalUtxos), ) } @@ -936,14 +918,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 4_950), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -971,7 +955,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .build() .unwrap(); } @@ -990,7 +973,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .build() .unwrap(); } @@ -1009,7 +991,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .build() .unwrap(); } @@ -1028,7 +1009,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap(); @@ -1054,7 +1034,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap(); @@ -1068,14 +1047,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(1_000_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1102,7 +1083,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap() .build() @@ -1114,14 +1094,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(10_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 3_333), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1139,14 +1121,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 1), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1170,7 +1154,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap() .align_outgoing() @@ -1198,7 +1181,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap() .align_outgoing() @@ -1224,7 +1206,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap() .strip_value() @@ -1247,7 +1228,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Postage, ) - .unwrap() .select_outgoing() .unwrap() .strip_value() @@ -1321,14 +1301,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::from([(satpoint(2, 10 * COIN_VALUE), inscription_id(1))]), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Err(Error::NotEnoughCardinalUtxos) ) } @@ -1338,14 +1320,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(1_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::from([(satpoint(1, 500), inscription_id(1))]), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Err(Error::UtxoContainsAdditionalInscription { outgoing_satpoint: satpoint(1, 0), inscribed_satpoint: satpoint(1, 500), @@ -1360,14 +1344,16 @@ mod tests { let fee_rate = FeeRate::try_from(17.3).unwrap(); - let transaction = TransactionBuilder::build_transaction_with_postage( + let transaction = TransactionBuilder::new( satpoint(1, 0), BTreeMap::from([(satpoint(1, 0), inscription_id(1))]), utxos.into_iter().collect(), recipient(), [change(0), change(1)], fee_rate, + Target::Postage, ) + .build_transaction() .unwrap(); let fee = @@ -1389,15 +1375,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(5_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1415,15 +1402,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - Amount::from_sat(1500) - ), + Target::Value(Amount::from_sat(1500)) + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1438,15 +1426,16 @@ mod tests { let utxos = vec![(outpoint(1), Amount::from_sat(1_000))]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::from([(satpoint(1, 500), inscription_id(1))]), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - Amount::from_sat(1) - ), + Target::Value(Amount::from_sat(1)) + ) + .build_transaction(), Err(Error::Dust { output_value: Amount::from_sat(1), dust_value: Amount::from_sat(294) @@ -1462,15 +1451,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Err(Error::NotEnoughCardinalUtxos), ) } @@ -1483,15 +1473,16 @@ mod tests { ]; pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), utxos.into_iter().collect(), recipient(), [change(0), change(1)], FeeRate::try_from(4.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Err(Error::NotEnoughCardinalUtxos), ) } @@ -1521,7 +1512,7 @@ mod tests { #[test] fn do_not_strip_excess_value_if_it_would_create_dust() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(1_000))] @@ -1530,8 +1521,9 @@ mod tests { recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - Amount::from_sat(707) - ), + Target::Value(Amount::from_sat(707)) + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1544,7 +1536,7 @@ mod tests { #[test] fn possible_to_create_output_of_exactly_max_postage() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(20_099))] @@ -1553,7 +1545,9 @@ mod tests { recipient(), [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1566,7 +1560,7 @@ mod tests { #[test] fn do_not_strip_excess_value_if_additional_output_cannot_pay_fee() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(1_500))] @@ -1575,8 +1569,9 @@ mod tests { recipient(), [change(0), change(1)], FeeRate::try_from(5.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1589,7 +1584,7 @@ mod tests { #[test] fn correct_error_is_returned_when_fee_cannot_be_paid() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(1_500))] @@ -1598,8 +1593,9 @@ mod tests { recipient(), [change(0), change(1)], FeeRate::try_from(6.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Err(Error::NotEnoughCardinalUtxos) ); } @@ -1607,7 +1603,7 @@ mod tests { #[test] fn recipient_address_must_be_unique() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(1000))] @@ -1616,8 +1612,9 @@ mod tests { recipient(), [recipient(), change(1)], FeeRate::try_from(0.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Err(Error::DuplicateAddress(recipient())) ); } @@ -1625,7 +1622,7 @@ mod tests { #[test] fn change_addresses_must_be_unique() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(1000))] @@ -1634,8 +1631,9 @@ mod tests { recipient(), [change(0), change(0)], FeeRate::try_from(0.0).unwrap(), - Amount::from_sat(1000) - ), + Target::Value(Amount::from_sat(1000)) + ) + .build_transaction(), Err(Error::DuplicateAddress(change(0))) ); } @@ -1643,7 +1641,7 @@ mod tests { #[test] fn output_over_value_because_fees_prevent_excess_value_stripping() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_value( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(2000))] @@ -1652,8 +1650,9 @@ mod tests { recipient(), [change(0), change(1)], FeeRate::try_from(2.0).unwrap(), - Amount::from_sat(1500) - ), + Target::Value(Amount::from_sat(1500)) + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1666,7 +1665,7 @@ mod tests { #[test] fn output_over_max_postage_because_fees_prevent_excess_value_stripping() { pretty_assert_eq!( - TransactionBuilder::build_transaction_with_postage( + TransactionBuilder::new( satpoint(1, 0), BTreeMap::new(), vec![(outpoint(1), Amount::from_sat(45000))] @@ -1675,7 +1674,9 @@ mod tests { recipient(), [change(0), change(1)], FeeRate::try_from(250.0).unwrap(), - ), + Target::Postage, + ) + .build_transaction(), Ok(Transaction { version: 1, lock_time: LockTime::ZERO, @@ -1705,7 +1706,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Value(Amount::from_sat(10_000)), ) - .unwrap() .select_outgoing() .unwrap() .add_value() @@ -1750,7 +1750,6 @@ mod tests { FeeRate::try_from(1.0).unwrap(), Target::Value(Amount::from_sat(10_000)), ) - .unwrap() .select_outgoing() .unwrap() .align_outgoing() @@ -1801,8 +1800,7 @@ mod tests { [change(0), change(1)], FeeRate::try_from(1.0).unwrap(), Target::Value(Amount::from_sat(10_000)), - ) - .unwrap(); + ); assert_eq!( tx_builder @@ -1841,4 +1839,39 @@ mod tests { Amount::from_sat(20_000), ); } + + #[test] + fn build_transaction_with_custom_postage() { + let utxos = vec![(outpoint(1), Amount::from_sat(1_000_000))]; + + let fee_rate = FeeRate::try_from(17.3).unwrap(); + + let transaction = TransactionBuilder::new( + satpoint(1, 0), + BTreeMap::from([(satpoint(1, 0), inscription_id(1))]), + utxos.into_iter().collect(), + recipient(), + [change(0), change(1)], + fee_rate, + Target::ExactPostage(Amount::from_sat(66_000)), + ) + .build_transaction() + .unwrap(); + + let fee = + fee_rate.fee(transaction.vsize() + TransactionBuilder::SCHNORR_SIGNATURE_SIZE / 4 + 1); + + pretty_assert_eq!( + transaction, + Transaction { + version: 1, + lock_time: LockTime::ZERO, + input: vec![tx_in(outpoint(1))], + output: vec![ + tx_out(66_000, recipient()), + tx_out(1_000_000 - 66_000 - fee.to_sat(), change(1)) + ], + } + ) + } } diff --git a/tests/wallet/inscribe.rs b/tests/wallet/inscribe.rs index 03a20ee03a..47527a8284 100644 --- a/tests/wallet/inscribe.rs +++ b/tests/wallet/inscribe.rs @@ -417,3 +417,24 @@ fn inscribe_with_no_limit() { .write("degenerate.png", four_megger) .rpc_server(&rpc_server); } + +#[test] +fn inscribe_works_with_postage() { + let rpc_server = test_bitcoincore_rpc::spawn(); + create_wallet(&rpc_server); + rpc_server.mine_blocks(1); + + CommandBuilder::new("wallet inscribe foo.txt --postage 5btc --fee-rate 10".to_string()) + .write("foo.txt", [0; 350]) + .rpc_server(&rpc_server) + .run_and_check_output::(); + + rpc_server.mine_blocks(1); + + let inscriptions = CommandBuilder::new("wallet inscriptions".to_string()) + .write("foo.txt", [0; 350]) + .rpc_server(&rpc_server) + .run_and_check_output::>(); + + pretty_assert_eq!(inscriptions[0].postage, 5 * COIN_VALUE); +} diff --git a/tests/wallet/send.rs b/tests/wallet/send.rs index 5ecf6122c8..0e26079877 100644 --- a/tests/wallet/send.rs +++ b/tests/wallet/send.rs @@ -346,3 +346,36 @@ fn user_must_provide_fee_rate_to_send() { ) .run_and_extract_stdout(); } + +#[test] +fn wallet_send_with_fee_rate_and_target_postage() { + let rpc_server = test_bitcoincore_rpc::spawn(); + create_wallet(&rpc_server); + rpc_server.mine_blocks(1); + + let Inscribe { inscription, .. } = inscribe(&rpc_server); + + CommandBuilder::new(format!( + "wallet send bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {inscription} --fee-rate 2.0 --postage 77000sat" + )) + .rpc_server(&rpc_server) + .stdout_regex("[[:xdigit:]]{64}\n") + .run_and_extract_stdout(); + + let tx = &rpc_server.mempool()[0]; + let mut fee = 0; + for input in &tx.input { + fee += rpc_server + .get_utxo_amount(&input.previous_output) + .unwrap() + .to_sat(); + } + for output in &tx.output { + fee -= output.value; + } + + let fee_rate = fee as f64 / tx.vsize() as f64; + + pretty_assert_eq!(fee_rate, 2.0); + pretty_assert_eq!(tx.output[0].value, 77_000); +} From 462f38bb8b13b0794c515356337b7b972ce4abc5 Mon Sep 17 00:00:00 2001 From: gmart7t2 <49558347+gmart7t2@users.noreply.github.com> Date: Thu, 17 Aug 2023 12:11:02 -0300 Subject: [PATCH 11/61] Use `--fee-rate` when sending an amount (#1922) Co-authored-by: raphjaph --- src/fee_rate.rs | 4 ++ src/subcommand/wallet/send.rs | 68 +++++++++++++++++++++--------- test-bitcoincore-rpc/src/api.rs | 3 ++ test-bitcoincore-rpc/src/server.rs | 40 ++++++++++++++++++ tests/wallet/send.rs | 29 ++++++++++--- 5 files changed, 118 insertions(+), 26 deletions(-) diff --git a/src/fee_rate.rs b/src/fee_rate.rs index 72d30834f5..9a0290040d 100644 --- a/src/fee_rate.rs +++ b/src/fee_rate.rs @@ -28,6 +28,10 @@ impl FeeRate { #[allow(clippy::cast_sign_loss)] Amount::from_sat((self.0 * vsize as f64).round() as u64) } + + pub(crate) fn n(&self) -> f64 { + self.0 + } } #[cfg(test)] diff --git a/src/subcommand/wallet/send.rs b/src/subcommand/wallet/send.rs index 2c6c9fd574..ac8faf52c4 100644 --- a/src/subcommand/wallet/send.rs +++ b/src/subcommand/wallet/send.rs @@ -20,7 +20,10 @@ pub struct Output { impl Send { pub(crate) fn run(self, options: Options) -> Result { - let address = self.address.require_network(options.chain().network())?; + let address = self + .address + .clone() + .require_network(options.chain().network())?; let index = Index::open(&options)?; index.update()?; @@ -44,25 +47,9 @@ impl Send { .get_inscription_satpoint_by_id(id)? .ok_or_else(|| anyhow!("Inscription {id} not found"))?, Outgoing::Amount(amount) => { - let all_inscription_outputs = inscriptions - .keys() - .map(|satpoint| satpoint.outpoint) - .collect::>(); - - let wallet_inscription_outputs = unspent_outputs - .keys() - .filter(|utxo| all_inscription_outputs.contains(utxo)) - .cloned() - .collect::>(); - - if !client.lock_unspent(&wallet_inscription_outputs)? { - bail!("failed to lock ordinal UTXOs"); - } - - let txid = client.send_to_address(&address, amount, None, None, None, None, None, None)?; - + Self::lock_inscriptions(&client, inscriptions, unspent_outputs)?; + let txid = Self::send_amount(&client, amount, address, self.fee_rate.n())?; print_json(Output { transaction: txid })?; - return Ok(()); } }; @@ -82,7 +69,7 @@ impl Send { satpoint, inscriptions, unspent_outputs, - address, + address.clone(), change, self.fee_rate, postage, @@ -99,4 +86,45 @@ impl Send { Ok(()) } + + fn lock_inscriptions( + client: &Client, + inscriptions: BTreeMap, + unspent_outputs: BTreeMap, + ) -> Result { + let all_inscription_outputs = inscriptions + .keys() + .map(|satpoint| satpoint.outpoint) + .collect::>(); + + let wallet_inscription_outputs = unspent_outputs + .keys() + .filter(|utxo| all_inscription_outputs.contains(utxo)) + .cloned() + .collect::>(); + + if !client.lock_unspent(&wallet_inscription_outputs)? { + bail!("failed to lock ordinal UTXOs"); + } + + Ok(()) + } + + fn send_amount(client: &Client, amount: Amount, address: Address, fee_rate: f64) -> Result { + Ok(client.call( + "sendtoaddress", + &[ + address.to_string().into(), // 1. address + amount.to_btc().into(), // 2. amount + serde_json::Value::Null, // 3. comment + serde_json::Value::Null, // 4. comment_to + serde_json::Value::Null, // 5. subtractfeefromamount + serde_json::Value::Null, // 6. replaceable + serde_json::Value::Null, // 7. conf_target + serde_json::Value::Null, // 8. estimate_mode + serde_json::Value::Null, // 9. avoid_reuse + fee_rate.into(), // 10. fee_rate + ], + )?) + } } diff --git a/test-bitcoincore-rpc/src/api.rs b/test-bitcoincore-rpc/src/api.rs index fc88b0e121..8c1226ebcc 100644 --- a/test-bitcoincore-rpc/src/api.rs +++ b/test-bitcoincore-rpc/src/api.rs @@ -72,6 +72,9 @@ pub trait Api { replaceable: Option, confirmation_target: Option, estimate_mode: Option, + avoid_reuse: Option, + fee_rate: Option, + verbose: Option, ) -> Result; #[rpc(name = "gettransaction")] diff --git a/test-bitcoincore-rpc/src/server.rs b/test-bitcoincore-rpc/src/server.rs index 58c185e07d..5cdcbb0037 100644 --- a/test-bitcoincore-rpc/src/server.rs +++ b/test-bitcoincore-rpc/src/server.rs @@ -279,6 +279,9 @@ impl Api for Server { replaceable: Option, confirmation_target: Option, estimate_mode: Option, + avoid_reuse: Option, + fee_rate: Option, + verbose: Option, ) -> Result { assert_eq!(comment, None); assert_eq!(comment_to, None); @@ -286,10 +289,47 @@ impl Api for Server { assert_eq!(replaceable, None); assert_eq!(confirmation_target, None); assert_eq!(estimate_mode, None); + assert_eq!(avoid_reuse, None); + assert_eq!(verbose, None); let mut state = self.state.lock().unwrap(); let locked = state.locked.iter().cloned().collect(); + let value = Amount::from_btc(amount).expect("error converting amount to sat"); + + let utxo = state.utxos.first_entry().expect("failed to get a utxo"); + let outpoint = utxo.key(); + let utxo_amount = utxo.get(); + + let mut transaction = Transaction { + version: 1, + lock_time: LockTime::ZERO, + input: vec![TxIn { + previous_output: *outpoint, + script_sig: ScriptBuf::new(), + sequence: Sequence::ENABLE_RBF_NO_LOCKTIME, + witness: Witness::new(), + }], + output: vec![ + TxOut { + value: value.to_sat(), + script_pubkey: address.payload.script_pubkey(), + }, + TxOut { + value: (*utxo_amount - value).to_sat(), + script_pubkey: address.payload.script_pubkey(), + }, + ], + }; + + #[allow(clippy::cast_possible_truncation)] + #[allow(clippy::cast_sign_loss)] + let fee = (fee_rate.unwrap_or(1.0) * transaction.vsize() as f64).round() as u64; + + transaction.output[1].value -= fee; + + state.mempool.push(transaction); + state.sent.push(Sent { address: address.assume_checked(), amount, diff --git a/tests/wallet/send.rs b/tests/wallet/send.rs index 0e26079877..996be6e737 100644 --- a/tests/wallet/send.rs +++ b/tests/wallet/send.rs @@ -211,16 +211,33 @@ fn inscriptions_cannot_be_sent_by_satpoint() { } #[test] -fn send_btc() { +fn send_btc_with_fee_rate() { let rpc_server = test_bitcoincore_rpc::spawn(); create_wallet(&rpc_server); rpc_server.mine_blocks(1); - let output = - CommandBuilder::new("wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 1btc") - .rpc_server(&rpc_server) - .run_and_check_output::(); + let output = CommandBuilder::new( + "wallet send --fee-rate 13.3 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 1btc", + ) + .rpc_server(&rpc_server) + .run_and_check_output::(); + + let tx = &rpc_server.mempool()[0]; + let mut fee = 0; + for input in &tx.input { + fee += rpc_server + .get_utxo_amount(&input.previous_output) + .unwrap() + .to_sat(); + } + for output in &tx.output { + fee -= output.value; + } + + let fee_rate = fee as f64 / tx.vsize() as f64; + + assert!(f64::abs(fee_rate - 13.3) < 0.1); assert_eq!( output.transaction, @@ -239,7 +256,7 @@ fn send_btc() { .assume_checked(), locked: Vec::new(), }] - ) + ); } #[test] From f6a2bac89b5019efccb492daefc22f8a90ab432e Mon Sep 17 00:00:00 2001 From: raph Date: Thu, 17 Aug 2023 17:30:08 +0200 Subject: [PATCH 12/61] Release 0.8.2 (#2334) --- CHANGELOG.md | 29 +++++++++++++++++++++++++++++ Cargo.lock | 2 +- Cargo.toml | 2 +- 3 files changed, 31 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3694c217d5..a0e2b86b17 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,35 @@ Changelog ========= +[0.8.2](https://github.com/ordinals/ord/releases/tag/0.8.2) - 2023-08-17 +--------------------------------------------------------------------- + +### Added + +- Allow setting custom postage (#2331) +- Make retrieving inscriptions in block fast (#2333) +- JSON API for `/inscription`, `/inscriptions` and `/output` (#2323) +- Ignore invalid content type header values (#2326) +- Add reorg resistance (#2320) +- Add JSON API endpoint `/sat/` (#2250) +- Add `amount` field to `wallet inscriptions` output. (#1928) + + +### Changed + +- Only fetch inscriptions that are owned by the ord wallet (#2310) +- Inform user when redb starts in recovery mode (#2304) +- Select multiple utxos (#2303) + + +### Fixed + +- Use `--fee-rate` when sending an amount (#1922) +- Fix typos in documentation (#2328) +- Fix dust limit for padding in `TransactionBuilder` (#1929) +- Fix remote RPC wallet commands (#1766) + + [0.8.1](https://github.com/ordinals/ord/releases/tag/0.8.1) - 2023-07-23 --------------------------------------------------------------------- diff --git a/Cargo.lock b/Cargo.lock index 30f559314a..32492ffb3b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2009,7 +2009,7 @@ checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" [[package]] name = "ord" -version = "0.8.1" +version = "0.8.2" dependencies = [ "anyhow", "async-trait", diff --git a/Cargo.toml b/Cargo.toml index e852c4aa04..1fdf664a24 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,7 +1,7 @@ [package] name = "ord" description = "◉ Ordinal wallet and block explorer" -version = "0.8.1" +version = "0.8.2" license = "CC0-1.0" edition = "2021" autotests = false From 3a3e9f55f00223ea1d48c870f6c1bb9ad0ba72c8 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Mon, 21 Aug 2023 12:57:56 -0700 Subject: [PATCH 13/61] Homogenize design (#2346) --- src/templates.rs | 2 +- static/index.css | 28 ++++++---------------------- templates/page.html | 2 +- 3 files changed, 8 insertions(+), 24 deletions(-) diff --git a/src/templates.rs b/src/templates.rs index 259ad50370..145e4f679d 100644 --- a/src/templates.rs +++ b/src/templates.rs @@ -137,7 +137,7 @@ mod tests { rare.txt
- +
diff --git a/static/index.css b/static/index.css index 36a1551793..a8faeb6616 100644 --- a/static/index.css +++ b/static/index.css @@ -4,8 +4,8 @@ --dark-fg: #98a3ad; --epic: darkorchid; --legendary: gold; - --light-bg: #000000; - --light-fg: #ffffff; + --light-bg: #292c2f; + --light-fg: #a1adb8; --link: #4169e1; --mythic: #f2a900; --rare: cornflowerblue; @@ -48,6 +48,10 @@ a:hover { text-decoration: underline; } +a:visited { + color: var(--link); +} + dt { font-weight: bold; margin-top: 0.5rem; @@ -66,15 +70,6 @@ nav > :first-child { font-weight: bold; } -nav a:hover { - color: var(--link); - text-decoration: none; -} - -nav a { - color: var(--light-fg); -} - form { display: flex; flex-grow: 1; @@ -92,17 +87,6 @@ input[type=text] { min-width: 0; } -input[type=submit] { - background: none; - border: none; - color: var(--light-fg); - cursor: pointer; - font-weight: bold; - font: inherit; - outline: inherit; - padding: 0; -} - dl { overflow-wrap: break-word; } diff --git a/templates/page.html b/templates/page.html index 3d939bbfc5..34d3e04484 100644 --- a/templates/page.html +++ b/templates/page.html @@ -25,7 +25,7 @@ %% }
- +
From 4b9a530bbb4df6776baaa0d743a382af130c867c Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Mon, 21 Aug 2023 14:01:12 -0700 Subject: [PATCH 14/61] Reduce index durability when testing (#2347) --- src/index.rs | 35 ++++++++++++++++++++++++++++------- src/index/reorg.rs | 8 ++++++++ src/subcommand/server.rs | 6 +++--- 3 files changed, 39 insertions(+), 10 deletions(-) diff --git a/src/index.rs b/src/index.rs index c09a50c31f..018f43c7eb 100644 --- a/src/index.rs +++ b/src/index.rs @@ -137,12 +137,13 @@ impl BitcoinCoreRpcResultExt for Result { pub(crate) struct Index { client: Client, database: Database, - path: PathBuf, + durability: redb::Durability, first_inscription_height: u64, genesis_block_coinbase_transaction: Transaction, genesis_block_coinbase_txid: Txid, height_limit: Option, options: Options, + path: PathBuf, unrecoverably_reorged: AtomicBool, } @@ -188,6 +189,12 @@ impl Index { log::info!("Setting DB cache size to {} bytes", db_cache_size); + let durability = if cfg!(test) { + redb::Durability::None + } else { + redb::Durability::Immediate + }; + let database = match Database::builder() .set_cache_size(db_cache_size) .open(&path) @@ -224,7 +231,7 @@ impl Index { let mut tx = database.begin_write()?; - tx.set_durability(redb::Durability::Immediate); + tx.set_durability(durability); tx.open_table(HEIGHT_TO_BLOCK_HASH)?; tx.open_table(INSCRIPTION_ID_TO_INSCRIPTION_ENTRY)?; @@ -258,15 +265,21 @@ impl Index { genesis_block_coinbase_txid: genesis_block_coinbase_transaction.txid(), client, database, - path, + durability, first_inscription_height: options.first_inscription_height(), genesis_block_coinbase_transaction, height_limit: options.height_limit, options: options.clone(), + path, unrecoverably_reorged: AtomicBool::new(false), }) } + #[cfg(test)] + fn set_durability(&mut self, durability: redb::Durability) { + self.durability = durability; + } + pub(crate) fn get_unspent_outputs(&self, _wallet: Wallet) -> Result> { let mut utxos = BTreeMap::new(); utxos.extend( @@ -500,7 +513,9 @@ impl Index { } fn begin_write(&self) -> Result { - Ok(self.database.begin_write()?) + let mut tx = self.database.begin_write()?; + tx.set_durability(self.durability); + Ok(tx) } fn increment_statistic(wtx: &WriteTransaction, statistic: Statistic, n: u64) -> Result { @@ -3298,7 +3313,9 @@ mod tests { #[test] fn recover_from_reorg() { - for context in Context::configurations() { + for mut context in Context::configurations() { + context.index.set_durability(redb::Durability::Immediate); + context.mine_blocks(1); let txid = context.rpc_server.broadcast_tx(TransactionTemplate { @@ -3348,7 +3365,9 @@ mod tests { #[test] fn recover_from_3_block_deep_and_consecutive_reorg() { - for context in Context::configurations() { + for mut context in Context::configurations() { + context.index.set_durability(redb::Durability::Immediate); + context.mine_blocks(1); let txid = context.rpc_server.broadcast_tx(TransactionTemplate { @@ -3401,7 +3420,9 @@ mod tests { #[test] fn recover_from_very_unlikely_7_block_deep_reorg() { - for context in Context::configurations() { + for mut context in Context::configurations() { + context.index.set_durability(redb::Durability::Immediate); + context.mine_blocks(1); let txid = context.rpc_server.broadcast_tx(TransactionTemplate { diff --git a/src/index/reorg.rs b/src/index/reorg.rs index 8de5da973c..a1cb3b79f1 100644 --- a/src/index/reorg.rs +++ b/src/index/reorg.rs @@ -56,6 +56,10 @@ impl Reorg { pub(crate) fn handle_reorg(index: &Index, height: u64, depth: u64) -> Result { log::info!("rolling back database after reorg of depth {depth} at height {height}"); + if let redb::Durability::None = index.durability { + panic!("set index durability to `Durability::Immediate` to test reorg handling"); + } + let mut wtx = index.begin_write()?; let oldest_savepoint = @@ -75,6 +79,10 @@ impl Reorg { } pub(crate) fn update_savepoints(index: &Index, height: u64) -> Result { + if let redb::Durability::None = index.durability { + return Ok(()); + } + if (height < SAVEPOINT_INTERVAL || height % SAVEPOINT_INTERVAL == 0) && index .client diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 46b9fac786..185b461c53 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -2147,7 +2147,7 @@ mod tests { let server = TestServer::new(); thread::sleep(Duration::from_millis(100)); - assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 3); + assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 1); let info = server.index.info().unwrap(); assert_eq!(info.transactions.len(), 1); @@ -2155,7 +2155,7 @@ mod tests { server.index.update().unwrap(); - assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 3); + assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 1); let info = server.index.info().unwrap(); assert_eq!(info.transactions.len(), 1); @@ -2166,7 +2166,7 @@ mod tests { thread::sleep(Duration::from_millis(10)); server.index.update().unwrap(); - assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 6); + assert_eq!(server.index.statistic(crate::index::Statistic::Commits), 2); let info = server.index.info().unwrap(); assert_eq!(info.transactions.len(), 2); From 6b846dc036d02c8f1ee1e40a382f975043ef4e65 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Mon, 21 Aug 2023 14:13:45 -0700 Subject: [PATCH 15/61] Add hint about maximum number of open files for testing (#2348) --- README.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/README.md b/README.md index be2203823b..8f8db824db 100644 --- a/README.md +++ b/README.md @@ -151,6 +151,10 @@ just doc just watch ltest --all ``` +If the tests are failing or hanging, you might need to increase the maximum +number of open files by running `ulimit -n 1024` in your shell before you run +the tests, or in your shell configuration. + We also try to follow a TDD (Test-Driven-Development) approach, which means we use tests as a way to get visibility into the code. Tests have to run fast for that reason so that the feedback loop between making a change, running the test and From 8d146ca9a8ab3ca71d76fea20e56e0ed5fe37df6 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Mon, 21 Aug 2023 14:21:45 -0700 Subject: [PATCH 16/61] Recognize media types without explicit charset (#2349) --- src/media.rs | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/media.rs b/src/media.rs index 2ec4e63dbe..c5ea8ac488 100644 --- a/src/media.rs +++ b/src/media.rs @@ -34,10 +34,13 @@ impl Media { ("model/gltf-binary", Media::Unknown, &["glb"]), ("model/stl", Media::Unknown, &["stl"]), ("text/css", Media::Text, &["css"]), + ("text/html", Media::Iframe, &[]), ("text/html;charset=utf-8", Media::Iframe, &["html"]), ("text/javascript", Media::Text, &["js"]), - ("text/plain;charset=utf-8", Media::Text, &["txt"]), + ("text/markdown", Media::Text, &[]), ("text/markdown;charset=utf-8", Media::Text, &["md"]), + ("text/plain", Media::Text, &[]), + ("text/plain;charset=utf-8", Media::Text, &["txt"]), ("video/mp4", Media::Video, &["mp4"]), ("video/webm", Media::Video, &["webm"]), ]; From fb0457c044c24fa9a0c06312f71b51296dbaa165 Mon Sep 17 00:00:00 2001 From: Eloc <42568538+elocremarc@users.noreply.github.com> Date: Tue, 22 Aug 2023 04:53:02 -0700 Subject: [PATCH 17/61] Update explorer.md (#2215) --- docs/src/guides/explorer.md | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/docs/src/guides/explorer.md b/docs/src/guides/explorer.md index 311390aac0..94799a71e7 100644 --- a/docs/src/guides/explorer.md +++ b/docs/src/guides/explorer.md @@ -5,6 +5,19 @@ The `ord` binary includes a block explorer. We host a instance of the block explorer on mainnet at [ordinals.com](https://ordinals.com), and on signet at [signet.ordinals.com](https://signet.ordinals.com). +### Running The Explorer +The server can be run locally with: + +`ord server` + +To specify a port add the `--http-port` flag: + +`ord server --http-port 8080` + +To test how your inscriptions will look you can run: + +`ord preview ...` + Search ------ From 47d2affcb75bcf428c4a70e50aa185ab1754d38c Mon Sep 17 00:00:00 2001 From: raph Date: Wed, 23 Aug 2023 00:02:36 +0200 Subject: [PATCH 18/61] Add tags and inscription id documentation (#2351) --- docs/src/SUMMARY.md | 1 + docs/src/inscriptions.md | 71 +++++++++++++++++++----------- docs/src/inscriptions/recursion.md | 27 ++++++++++++ 3 files changed, 73 insertions(+), 26 deletions(-) create mode 100644 docs/src/inscriptions/recursion.md diff --git a/docs/src/SUMMARY.md b/docs/src/SUMMARY.md index 46b956fd39..519a01c043 100644 --- a/docs/src/SUMMARY.md +++ b/docs/src/SUMMARY.md @@ -3,6 +3,7 @@ - [Overview](overview.md) - [Digital Artifacts](digital-artifacts.md) - [Inscriptions](inscriptions.md) + - [Recursion](inscriptions/recursion.md) - [FAQ](faq.md) - [Contributing](contributing.md) - [Donate](donate.md) diff --git a/docs/src/inscriptions.md b/docs/src/inscriptions.md index 27ba6971f0..056918dd2c 100644 --- a/docs/src/inscriptions.md +++ b/docs/src/inscriptions.md @@ -69,40 +69,59 @@ Content The data model of inscriptions is that of a HTTP response, allowing inscription content to be served by a web server and viewed in a web browser. -Sandboxing ----------- +Fields +------ -HTML and SVG inscriptions are sandboxed in order to prevent references to -off-chain content, thus keeping inscriptions immutable and self-contained. +Inscriptions may include fields before an optional body. Each field consists of +two data pushes, a tag and a value. -This is accomplished by loading HTML and SVG inscriptions inside `iframes` with -the `sandbox` attribute, as well as serving inscription content with -`Content-Security-Policy` headers. +Currently, the only defined field is `content-type`, with a tag of `1`, whose +value is the MIME type of the body. + +The beginning of the body and end of fields is indicated with an empty data +push. + +Unrecognized tags are interpreted differently depending on whether they are +even or odd, following the "it's okay to be odd" rule used by the Lightning +Network. -Recursion ---------- +Even tags are used for fields which may affect creation, initial assignment, or +transfer of an inscription. Thus, inscriptions with unrecognized even fields +must be displayed as "unbound", that is, without a location. -An important exception to sandboxing is recursion: access to `ord`'s `/content` -endpoint is permitted, allowing inscriptions to access the content of other -inscriptions by requesting `/content/`. +Odd tags are used for fields which do not affect creation, initial assignment, +or transfer, such as additional metadata, and thus are safe to ignore. -This has a number of interesting use-cases: +Inscription IDs +--------------- -- Remixing the content of existing inscriptions. +The inscriptions are contained within the inputs of a reveal transaction. In +order to uniquely identify them they are assigned an ID of the form: -- Publishing snippets of code, images, audio, or stylesheets as shared public - resources. +`521f8eccffa4c41a3a7728dd012ea5a4a02feed81f41159231251ecf1e5c79dai0` -- Generative art collections where an algorithm is inscribed as JavaScript, - and instantiated from multiple inscriptions with unique seeds. +The part in front of the `i` is the transaction ID (`txid`) of the reveal +transaction. The number after the `i` defines the index (starting at 0) of new inscriptions +being inscribed in the reveal transaction. -- Generative profile picture collections where accessories and attributes are - inscribed as individual images, or in a shared texture atlas, and then - combined, collage-style, in unique combinations in multiple inscriptions. +Inscriptions can either be located in different inputs, within the same input or +a combination of both. In any case the ordering is clear, since a parser would +go through the inputs consecutively and look for all inscription `envelopes`. -A couple other endpoints that inscriptions may access are the following: +| Input | Inscription Count | Indices | +|:-----:|:-----------------:|:----------:| +| 0 | 2 | i0, i1 | +| 1 | 1 | i2 | +| 2 | 3 | i3, i4, i5 | +| 3 | 0 | | +| 4 | 1 | i6 | -- `/blockheight`: latest block height. -- `/blockhash`: latest block hash. -- `/blockhash/`: block hash at given block height. -- `/blocktime`: UNIX time stamp of latest block. +Sandboxing +---------- + +HTML and SVG inscriptions are sandboxed in order to prevent references to +off-chain content, thus keeping inscriptions immutable and self-contained. + +This is accomplished by loading HTML and SVG inscriptions inside `iframes` with +the `sandbox` attribute, as well as serving inscription content with +`Content-Security-Policy` headers. diff --git a/docs/src/inscriptions/recursion.md b/docs/src/inscriptions/recursion.md new file mode 100644 index 0000000000..43a2eabdb7 --- /dev/null +++ b/docs/src/inscriptions/recursion.md @@ -0,0 +1,27 @@ +Recursion +========= + +An important exception to [sandboxing](../inscriptions.md#sandboxing) is recursion: access to `ord`'s `/content` +endpoint is permitted, allowing inscriptions to access the content of other +inscriptions by requesting `/content/`. + +This has a number of interesting use-cases: + +- Remixing the content of existing inscriptions. + +- Publishing snippets of code, images, audio, or stylesheets as shared public + resources. + +- Generative art collections where an algorithm is inscribed as JavaScript, + and instantiated from multiple inscriptions with unique seeds. + +- Generative profile picture collections where accessories and attributes are + inscribed as individual images, or in a shared texture atlas, and then + combined, collage-style, in unique combinations in multiple inscriptions. + +A few other endpoints that inscriptions may access are the following: + +- `/blockheight`: latest block height. +- `/blockhash`: latest block hash. +- `/blockhash/`: block hash at given block height. +- `/blocktime`: UNIX time stamp of latest block. From 8a95a0fd47ba9b82b007f27fb3aa963af6f8cd53 Mon Sep 17 00:00:00 2001 From: gmart7t2 <49558347+gmart7t2@users.noreply.github.com> Date: Tue, 22 Aug 2023 19:34:05 -0300 Subject: [PATCH 19/61] Use correct height and depth in reorg log (#2352) --- src/index.rs | 2 +- src/index/reorg.rs | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/index.rs b/src/index.rs index 018f43c7eb..75e3f0ace5 100644 --- a/src/index.rs +++ b/src/index.rs @@ -417,7 +417,7 @@ impl Index { log::info!("{}", err.to_string()); match err.downcast_ref() { - Some(&ReorgError::Recoverable((height, depth))) => { + Some(&ReorgError::Recoverable { height, depth }) => { Reorg::handle_reorg(self, height, depth)?; updater = Updater::new(self)?; diff --git a/src/index/reorg.rs b/src/index/reorg.rs index a1cb3b79f1..fc6164282c 100644 --- a/src/index/reorg.rs +++ b/src/index/reorg.rs @@ -2,14 +2,14 @@ use {super::*, updater::BlockData}; #[derive(Debug, PartialEq)] pub(crate) enum ReorgError { - Recoverable((u64, u64)), + Recoverable { height: u64, depth: u64 }, Unrecoverable, } impl fmt::Display for ReorgError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match self { - ReorgError::Recoverable((height, depth)) => { + ReorgError::Recoverable { height, depth } => { write!(f, "{depth} block deep reorg detected at height {height}") } ReorgError::Unrecoverable => write!(f, "unrecoverable reorg detected"), @@ -43,7 +43,7 @@ impl Reorg { .into_option()?; if index_block_hash == bitcoind_block_hash { - return Err(anyhow!(ReorgError::Recoverable((depth, height)))); + return Err(anyhow!(ReorgError::Recoverable { height, depth })); } } From 68a41f72e83b26729967a3ca873fd93779c43260 Mon Sep 17 00:00:00 2001 From: gmart7t2 <49558347+gmart7t2@users.noreply.github.com> Date: Tue, 22 Aug 2023 20:02:28 -0300 Subject: [PATCH 20/61] Allow splitting merged inscriptions (#1927) --- src/subcommand/wallet/transaction_builder.rs | 11 +- tests/json_api.rs | 7 +- tests/lib.rs | 9 ++ tests/wallet/send.rs | 126 ++++++++++++++++++- 4 files changed, 142 insertions(+), 11 deletions(-) diff --git a/src/subcommand/wallet/transaction_builder.rs b/src/subcommand/wallet/transaction_builder.rs index 77003bc53a..87a12e624a 100644 --- a/src/subcommand/wallet/transaction_builder.rs +++ b/src/subcommand/wallet/transaction_builder.rs @@ -182,9 +182,18 @@ impl TransactionBuilder { } fn select_outgoing(mut self) -> Result { - for (inscribed_satpoint, inscription_id) in &self.inscriptions { + let dust_limit = self + .unused_change_addresses + .last() + .unwrap() + .script_pubkey() + .dust_value() + .to_sat(); + + for (inscribed_satpoint, inscription_id) in self.inscriptions.iter().rev() { if self.outgoing.outpoint == inscribed_satpoint.outpoint && self.outgoing.offset != inscribed_satpoint.offset + && self.outgoing.offset < inscribed_satpoint.offset + dust_limit { return Err(Error::UtxoContainsAdditionalInscription { outgoing_satpoint: self.outgoing, diff --git a/tests/json_api.rs b/tests/json_api.rs index 893d7c7673..94eca276bb 100644 --- a/tests/json_api.rs +++ b/tests/json_api.rs @@ -1,9 +1,4 @@ -use { - super::*, ord::inscription_id::InscriptionId, ord::rarity::Rarity, - ord::templates::inscription::InscriptionJson, ord::templates::inscriptions::InscriptionsJson, - ord::templates::output::OutputJson, ord::templates::sat::SatJson, ord::SatPoint, - test_bitcoincore_rpc::TransactionTemplate, -}; +use super::*; #[test] fn get_sat_without_sat_index() { diff --git a/tests/lib.rs b/tests/lib.rs index 037b97110c..2beade9cc3 100644 --- a/tests/lib.rs +++ b/tests/lib.rs @@ -9,6 +9,14 @@ use { Network, OutPoint, Txid, }, executable_path::executable_path, + ord::inscription_id::InscriptionId, + ord::rarity::Rarity, + ord::subcommand::wallet::send::Output, + ord::templates::inscription::InscriptionJson, + ord::templates::inscriptions::InscriptionsJson, + ord::templates::output::OutputJson, + ord::templates::sat::SatJson, + ord::SatPoint, pretty_assertions::assert_eq as pretty_assert_eq, regex::Regex, reqwest::{StatusCode, Url}, @@ -24,6 +32,7 @@ use { }, tempfile::TempDir, test_bitcoincore_rpc::Sent, + test_bitcoincore_rpc::TransactionTemplate, }; macro_rules! assert_regex_match { diff --git a/tests/wallet/send.rs b/tests/wallet/send.rs index 996be6e737..3d22c6c70f 100644 --- a/tests/wallet/send.rs +++ b/tests/wallet/send.rs @@ -1,4 +1,4 @@ -use {super::*, ord::subcommand::wallet::send::Output}; +use super::*; #[test] fn inscriptions_can_be_sent() { @@ -164,7 +164,7 @@ fn send_does_not_use_inscribed_sats_as_cardinal_utxos() { } #[test] -fn do_not_accidentally_send_an_inscription() { +fn do_not_send_within_dust_limit_of_an_inscription() { let rpc_server = test_bitcoincore_rpc::spawn(); create_wallet(&rpc_server); @@ -182,16 +182,134 @@ fn do_not_accidentally_send_an_inscription() { }; CommandBuilder::new(format!( - "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {output}:55" + "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {output}:329" )) .rpc_server(&rpc_server) .expected_exit_code(1) .expected_stderr(format!( - "error: cannot send {output}:55 without also sending inscription {inscription} at {output}:0\n" + "error: cannot send {output}:329 without also sending inscription {inscription} at {output}:0\n" )) .run_and_extract_stdout(); } +#[test] +fn can_send_after_dust_limit_from_an_inscription() { + let rpc_server = test_bitcoincore_rpc::spawn(); + create_wallet(&rpc_server); + + let Inscribe { reveal, .. } = inscribe(&rpc_server); + + rpc_server.mine_blocks(1); + + let output = OutPoint { + txid: reveal, + vout: 0, + }; + + CommandBuilder::new(format!( + "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {output}:330" + )) + .rpc_server(&rpc_server) + .stdout_regex("[[:xdigit:]]{64}\n") + .run_and_extract_stdout(); +} + +#[test] +fn splitting_merged_inscriptions_is_possible() { + let rpc_server = test_bitcoincore_rpc::spawn(); + create_wallet(&rpc_server); + rpc_server.mine_blocks(3); + + // merging 3 inscriptions into one utxo + let reveal_txid = rpc_server.broadcast_tx(TransactionTemplate { + inputs: &[(1, 0, 0), (2, 0, 0), (3, 0, 0)], + witness: envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]), + outputs: 1, + ..Default::default() + }); + + rpc_server.mine_blocks(1); + + let server = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]); + + let response = server.json_request(format!("/output/{}:0", reveal_txid)); + assert_eq!(response.status(), StatusCode::OK); + + let output_json: OutputJson = serde_json::from_str(&response.text().unwrap()).unwrap(); + + pretty_assert_eq!( + output_json, + OutputJson { + value: 3 * 50 * COIN_VALUE, + script_pubkey: "".to_string(), + address: None, + transaction: reveal_txid.to_string(), + sat_ranges: Some(vec![ + (5000000000, 10000000000,), + (10000000000, 15000000000,), + (15000000000, 20000000000,), + ],), + inscriptions: vec![ + InscriptionId { + txid: reveal_txid, + index: 0 + }, + InscriptionId { + txid: reveal_txid, + index: 2 + }, + InscriptionId { + txid: reveal_txid, + index: 1 + } + ] + } + ); + + // try and fail to send first + CommandBuilder::new(format!( + "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {}i0", + reveal_txid, + )) + .rpc_server(&rpc_server) + .expected_exit_code(1) + .expected_stderr(format!( + "error: cannot send {reveal_txid}:0:0 without also sending inscription {reveal_txid}i2 at {reveal_txid}:0:{}\n", 100 * COIN_VALUE + )) + .run_and_extract_stdout(); + + // splitting out last + CommandBuilder::new(format!( + "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {}i2", + reveal_txid, + )) + .rpc_server(&rpc_server) + .stdout_regex("[[:xdigit:]]{64}\n") + .run_and_extract_stdout(); + + rpc_server.mine_blocks(1); + + // splitting second to last + CommandBuilder::new(format!( + "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {}i1", + reveal_txid, + )) + .rpc_server(&rpc_server) + .stdout_regex("[[:xdigit:]]{64}\n") + .run_and_extract_stdout(); + + rpc_server.mine_blocks(1); + + // splitting send first + CommandBuilder::new(format!( + "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {}i0", + reveal_txid, + )) + .rpc_server(&rpc_server) + .stdout_regex("[[:xdigit:]]{64}\n") + .run_and_extract_stdout(); +} + #[test] fn inscriptions_cannot_be_sent_by_satpoint() { let rpc_server = test_bitcoincore_rpc::spawn(); From 5c09dd6c38136a95370eb5274d23a38b59306bb8 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Wed, 23 Aug 2023 12:44:59 -0700 Subject: [PATCH 21/61] Return JSON from all commands (#2355) --- src/arguments.rs | 2 +- src/lib.rs | 33 ++++++++-------- src/subcommand.rs | 27 ++++++++++---- src/subcommand/epochs.rs | 6 +-- src/subcommand/find.rs | 7 +--- src/subcommand/index.rs | 10 ++--- src/subcommand/info.rs | 8 ++-- src/subcommand/list.rs | 10 ++--- src/subcommand/parse.rs | 7 ++-- src/subcommand/preview.rs | 6 +-- src/subcommand/server.rs | 4 +- src/subcommand/subsidy.rs | 8 ++-- src/subcommand/supply.rs | 8 ++-- src/subcommand/traits.rs | 8 ++-- src/subcommand/wallet.rs | 4 +- src/subcommand/wallet/balance.rs | 6 +-- src/subcommand/wallet/cardinals.rs | 12 +++--- src/subcommand/wallet/create.rs | 16 ++++---- src/subcommand/wallet/inscribe.rs | 26 ++++++------- src/subcommand/wallet/inscriptions.rs | 6 +-- src/subcommand/wallet/outputs.rs | 6 +-- src/subcommand/wallet/receive.rs | 6 +-- src/subcommand/wallet/restore.rs | 5 +-- src/subcommand/wallet/sats.rs | 8 ++-- src/subcommand/wallet/send.rs | 9 ++--- src/subcommand/wallet/transactions.rs | 6 +-- tests/command_builder.rs | 2 +- tests/epochs.rs | 2 +- tests/find.rs | 2 +- tests/index.rs | 13 ++++--- tests/info.rs | 6 +-- tests/json_api.rs | 4 +- tests/lib.rs | 43 ++++++++------------- tests/list.rs | 2 +- tests/parse.rs | 4 +- tests/server.rs | 9 ++--- tests/subsidy.rs | 8 ++-- tests/supply.rs | 2 +- tests/traits.rs | 4 +- tests/wallet/balance.rs | 8 ++-- tests/wallet/cardinals.rs | 6 +-- tests/wallet/create.rs | 16 ++++---- tests/wallet/inscribe.rs | 30 +++++++-------- tests/wallet/inscriptions.rs | 29 +++++++------- tests/wallet/outputs.rs | 4 +- tests/wallet/receive.rs | 2 +- tests/wallet/restore.rs | 17 +++++---- tests/wallet/sats.rs | 4 +- tests/wallet/send.rs | 54 +++++++++++---------------- tests/wallet/transactions.rs | 10 ++--- 50 files changed, 244 insertions(+), 291 deletions(-) diff --git a/src/arguments.rs b/src/arguments.rs index c3af6814d6..17c6050ebb 100644 --- a/src/arguments.rs +++ b/src/arguments.rs @@ -10,7 +10,7 @@ pub(crate) struct Arguments { } impl Arguments { - pub(crate) fn run(self) -> Result { + pub(crate) fn run(self) -> SubcommandResult { self.subcommand.run(self.options) } } diff --git a/src/lib.rs b/src/lib.rs index 1ef8d220ab..1287aa0a4d 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -27,7 +27,7 @@ use { options::Options, outgoing::Outgoing, representation::Representation, - subcommand::Subcommand, + subcommand::{Subcommand, SubcommandResult}, tally::Tally, }, anyhow::{anyhow, bail, Context, Error}, @@ -180,22 +180,25 @@ pub fn main() { }) .expect("Error setting handler"); - if let Err(err) = Arguments::parse().run() { - eprintln!("error: {err}"); - err - .chain() - .skip(1) - .for_each(|cause| eprintln!("because: {cause}")); - if env::var_os("RUST_BACKTRACE") - .map(|val| val == "1") - .unwrap_or_default() - { - eprintln!("{}", err.backtrace()); - } + match Arguments::parse().run() { + Err(err) => { + eprintln!("error: {err}"); + err + .chain() + .skip(1) + .for_each(|cause| eprintln!("because: {cause}")); + if env::var_os("RUST_BACKTRACE") + .map(|val| val == "1") + .unwrap_or_default() + { + eprintln!("{}", err.backtrace()); + } - gracefully_shutdown_indexer(); + gracefully_shutdown_indexer(); - process::exit(1); + process::exit(1); + } + Ok(output) => output.print_json(), } gracefully_shutdown_indexer(); diff --git a/src/subcommand.rs b/src/subcommand.rs index a1e97b017e..eea4a3a9aa 100644 --- a/src/subcommand.rs +++ b/src/subcommand.rs @@ -13,12 +13,6 @@ pub mod supply; pub mod traits; pub mod wallet; -fn print_json(output: impl Serialize) -> Result { - serde_json::to_writer_pretty(io::stdout(), &output)?; - println!(); - Ok(()) -} - #[derive(Debug, Parser)] pub(crate) enum Subcommand { #[clap(about = "List the first satoshis of each reward epoch")] @@ -48,7 +42,7 @@ pub(crate) enum Subcommand { } impl Subcommand { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { match self { Self::Epochs => epochs::run(), Self::Preview(preview) => preview.run(), @@ -70,3 +64,22 @@ impl Subcommand { } } } + +#[derive(Serialize, Deserialize)] +pub struct Empty {} + +pub(crate) trait Output: Send { + fn print_json(&self); +} + +impl Output for T +where + T: Serialize + Send, +{ + fn print_json(&self) { + serde_json::to_writer_pretty(io::stdout(), self).ok(); + println!(); + } +} + +pub(crate) type SubcommandResult = Result>; diff --git a/src/subcommand/epochs.rs b/src/subcommand/epochs.rs index 53293cb025..39534a65ca 100644 --- a/src/subcommand/epochs.rs +++ b/src/subcommand/epochs.rs @@ -5,13 +5,11 @@ pub struct Output { pub starting_sats: Vec, } -pub(crate) fn run() -> Result { +pub(crate) fn run() -> SubcommandResult { let mut starting_sats = Vec::new(); for sat in Epoch::STARTING_SATS { starting_sats.push(sat); } - print_json(Output { starting_sats })?; - - Ok(()) + Ok(Box::new(Output { starting_sats })) } diff --git a/src/subcommand/find.rs b/src/subcommand/find.rs index fdf314cc48..fb81e19b33 100644 --- a/src/subcommand/find.rs +++ b/src/subcommand/find.rs @@ -12,16 +12,13 @@ pub struct Output { } impl Find { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; match index.find(self.sat.0)? { - Some(satpoint) => { - print_json(Output { satpoint })?; - Ok(()) - } + Some(satpoint) => Ok(Box::new(Output { satpoint })), None => Err(anyhow!("sat has not been mined as of index height")), } } diff --git a/src/subcommand/index.rs b/src/subcommand/index.rs index 0193011636..aea938a8a3 100644 --- a/src/subcommand/index.rs +++ b/src/subcommand/index.rs @@ -9,7 +9,7 @@ pub(crate) enum IndexSubcommand { } impl IndexSubcommand { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { match self { Self::Export(export) => export.run(options), Self::Run => index::run(options), @@ -30,20 +30,20 @@ pub(crate) struct Export { } impl Export { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; index.export(&self.tsv, self.include_addresses)?; - Ok(()) + Ok(Box::new(Empty {})) } } -pub(crate) fn run(options: Options) -> Result { +pub(crate) fn run(options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; - Ok(()) + Ok(Box::new(Empty {})) } diff --git a/src/subcommand/info.rs b/src/subcommand/info.rs index aa96de00b4..737cdd2228 100644 --- a/src/subcommand/info.rs +++ b/src/subcommand/info.rs @@ -15,7 +15,7 @@ pub struct TransactionsOutput { } impl Info { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; let info = index.info()?; @@ -32,11 +32,9 @@ impl Info { elapsed: (end.starting_timestamp - start.starting_timestamp) as f64 / 1000.0 / 60.0, }); } - print_json(output)?; + Ok(Box::new(output)) } else { - print_json(info)?; + Ok(Box::new(info)) } - - Ok(()) } } diff --git a/src/subcommand/list.rs b/src/subcommand/list.rs index 4ec93b14b0..dfad38efa6 100644 --- a/src/subcommand/list.rs +++ b/src/subcommand/list.rs @@ -18,7 +18,7 @@ pub struct Output { } impl List { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; @@ -47,9 +47,7 @@ impl List { }); } - print_json(outputs)?; - - Ok(()) + Ok(Box::new(outputs)) } Some(crate::index::List::Spent) => Err(anyhow!("output spent.")), None => Err(anyhow!("output not found")), @@ -92,8 +90,8 @@ mod tests { offset: u64, rarity: Rarity, name: String, - ) -> Output { - Output { + ) -> super::Output { + super::Output { output, start, end, diff --git a/src/subcommand/parse.rs b/src/subcommand/parse.rs index 73c815d3de..40854b9b73 100644 --- a/src/subcommand/parse.rs +++ b/src/subcommand/parse.rs @@ -12,10 +12,9 @@ pub struct Output { } impl Parse { - pub(crate) fn run(self) -> Result { - print_json(Output { + pub(crate) fn run(self) -> SubcommandResult { + Ok(Box::new(Output { object: self.object, - })?; - Ok(()) + })) } } diff --git a/src/subcommand/preview.rs b/src/subcommand/preview.rs index 915fb136f2..6926e2e631 100644 --- a/src/subcommand/preview.rs +++ b/src/subcommand/preview.rs @@ -16,7 +16,7 @@ impl Drop for KillOnDrop { } impl Preview { - pub(crate) fn run(self) -> Result { + pub(crate) fn run(self) -> SubcommandResult { let tmpdir = TempDir::new()?; let rpc_port = TcpListener::bind("127.0.0.1:0")?.local_addr()?.port(); @@ -102,8 +102,6 @@ impl Preview { options, subcommand: Subcommand::Server(self.server), } - .run()?; - - Ok(()) + .run() } } diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 185b461c53..1f4f93453d 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -137,7 +137,7 @@ pub(crate) struct Server { } impl Server { - pub(crate) fn run(self, options: Options, index: Arc, handle: Handle) -> Result { + pub(crate) fn run(self, options: Options, index: Arc, handle: Handle) -> SubcommandResult { Runtime::new()?.block_on(async { let block_index_state = BlockIndexState { block_index: RwLock::new(BlockIndex::new(&index)?), @@ -272,7 +272,7 @@ impl Server { (None, None) => unreachable!(), } - Ok(()) + Ok(Box::new(Empty {}) as Box) }) } diff --git a/src/subcommand/subsidy.rs b/src/subcommand/subsidy.rs index e293d6d7c6..0a7fbb54ce 100644 --- a/src/subcommand/subsidy.rs +++ b/src/subcommand/subsidy.rs @@ -14,7 +14,7 @@ pub struct Output { } impl Subsidy { - pub(crate) fn run(self) -> Result { + pub(crate) fn run(self) -> SubcommandResult { let first = self.height.starting_sat(); let subsidy = self.height.subsidy(); @@ -23,12 +23,10 @@ impl Subsidy { bail!("block {} has no subsidy", self.height); } - print_json(Output { + Ok(Box::new(Output { first: first.0, subsidy, name: first.name(), - })?; - - Ok(()) + })) } } diff --git a/src/subcommand/supply.rs b/src/subcommand/supply.rs index 4a4b468cca..dcf86d3ed5 100644 --- a/src/subcommand/supply.rs +++ b/src/subcommand/supply.rs @@ -8,7 +8,7 @@ pub struct Output { pub last_mined_in_block: u64, } -pub(crate) fn run() -> Result { +pub(crate) fn run() -> SubcommandResult { let mut last = 0; loop { @@ -18,12 +18,10 @@ pub(crate) fn run() -> Result { last += 1; } - print_json(Output { + Ok(Box::new(Output { supply: Sat::SUPPLY, first: 0, last: Sat::SUPPLY - 1, last_mined_in_block: last, - })?; - - Ok(()) + })) } diff --git a/src/subcommand/traits.rs b/src/subcommand/traits.rs index bfb4999be6..9ef2726655 100644 --- a/src/subcommand/traits.rs +++ b/src/subcommand/traits.rs @@ -21,8 +21,8 @@ pub struct Output { } impl Traits { - pub(crate) fn run(self) -> Result { - print_json(Output { + pub(crate) fn run(self) -> SubcommandResult { + Ok(Box::new(Output { number: self.sat.n(), decimal: self.sat.decimal().to_string(), degree: self.sat.degree().to_string(), @@ -33,8 +33,6 @@ impl Traits { period: self.sat.period(), offset: self.sat.third(), rarity: self.sat.rarity(), - })?; - - Ok(()) + })) } } diff --git a/src/subcommand/wallet.rs b/src/subcommand/wallet.rs index 875ed5200f..4bfd6ef401 100644 --- a/src/subcommand/wallet.rs +++ b/src/subcommand/wallet.rs @@ -17,7 +17,7 @@ use { pub mod balance; pub mod cardinals; pub mod create; -pub(crate) mod inscribe; +pub mod inscribe; pub mod inscriptions; pub mod outputs; pub mod receive; @@ -54,7 +54,7 @@ pub(crate) enum Wallet { } impl Wallet { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { match self { Self::Balance => balance::run(options), Self::Create(create) => create.run(options), diff --git a/src/subcommand/wallet/balance.rs b/src/subcommand/wallet/balance.rs index 9813634118..be08751479 100644 --- a/src/subcommand/wallet/balance.rs +++ b/src/subcommand/wallet/balance.rs @@ -5,7 +5,7 @@ pub struct Output { pub cardinal: u64, } -pub(crate) fn run(options: Options) -> Result { +pub(crate) fn run(options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; @@ -24,7 +24,5 @@ pub(crate) fn run(options: Options) -> Result { } } - print_json(Output { cardinal: balance })?; - - Ok(()) + Ok(Box::new(Output { cardinal: balance })) } diff --git a/src/subcommand/wallet/cardinals.rs b/src/subcommand/wallet/cardinals.rs index 4df85204fd..ccbb187fcb 100644 --- a/src/subcommand/wallet/cardinals.rs +++ b/src/subcommand/wallet/cardinals.rs @@ -1,12 +1,12 @@ use {super::*, crate::wallet::Wallet, std::collections::BTreeSet}; #[derive(Serialize, Deserialize)] -pub struct Cardinal { +pub struct CardinalUtxo { pub output: OutPoint, pub amount: u64, } -pub(crate) fn run(options: Options) -> Result { +pub(crate) fn run(options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; @@ -24,15 +24,13 @@ pub(crate) fn run(options: Options) -> Result { if inscribed_utxos.contains(output) { None } else { - Some(Cardinal { + Some(CardinalUtxo { output: *output, amount: amount.to_sat(), }) } }) - .collect::>(); + .collect::>(); - print_json(cardinal_utxos)?; - - Ok(()) + Ok(Box::new(cardinal_utxos)) } diff --git a/src/subcommand/wallet/create.rs b/src/subcommand/wallet/create.rs index a236f8ee80..4efb9579a4 100644 --- a/src/subcommand/wallet/create.rs +++ b/src/subcommand/wallet/create.rs @@ -1,9 +1,9 @@ use super::*; -#[derive(Serialize)] -struct Output { - mnemonic: Mnemonic, - passphrase: Option, +#[derive(Serialize, Deserialize)] +pub struct Output { + pub mnemonic: Mnemonic, + pub passphrase: Option, } #[derive(Debug, Parser)] @@ -17,7 +17,7 @@ pub(crate) struct Create { } impl Create { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let mut entropy = [0; 16]; rand::thread_rng().fill_bytes(&mut entropy); @@ -25,11 +25,9 @@ impl Create { initialize_wallet(&options, mnemonic.to_seed(self.passphrase.clone()))?; - print_json(Output { + Ok(Box::new(Output { mnemonic, passphrase: Some(self.passphrase), - })?; - - Ok(()) + })) } } diff --git a/src/subcommand/wallet/inscribe.rs b/src/subcommand/wallet/inscribe.rs index f712890656..98c2328c2b 100644 --- a/src/subcommand/wallet/inscribe.rs +++ b/src/subcommand/wallet/inscribe.rs @@ -19,12 +19,12 @@ use { std::collections::BTreeSet, }; -#[derive(Serialize)] -struct Output { - commit: Txid, - inscription: InscriptionId, - reveal: Txid, - fees: u64, +#[derive(Serialize, Deserialize)] +pub struct Output { + pub commit: Txid, + pub inscription: InscriptionId, + pub reveal: Txid, + pub fees: u64, } #[derive(Debug, Parser)] @@ -59,7 +59,7 @@ pub(crate) struct Inscribe { } impl Inscribe { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let inscription = Inscription::from_file(options.chain(), &self.file)?; let index = Index::open(&options)?; @@ -110,12 +110,12 @@ impl Inscribe { Self::calculate_fee(&unsigned_commit_tx, &utxos) + Self::calculate_fee(&reveal_tx, &utxos); if self.dry_run { - print_json(Output { + Ok(Box::new(Output { commit: unsigned_commit_tx.txid(), reveal: reveal_tx.txid(), inscription: reveal_tx.txid().into(), fees, - })?; + })) } else { if !self.no_backup { Inscribe::backup_recovery_key(&client, recovery_key_pair, options.chain().network())?; @@ -133,15 +133,13 @@ impl Inscribe { .send_raw_transaction(&reveal_tx) .context("Failed to send reveal transaction")?; - print_json(Output { + Ok(Box::new(Output { commit, reveal, inscription: reveal.into(), fees, - })?; - }; - - Ok(()) + })) + } } fn calculate_fee(tx: &Transaction, utxos: &BTreeMap) -> u64 { diff --git a/src/subcommand/wallet/inscriptions.rs b/src/subcommand/wallet/inscriptions.rs index 2b9395e378..bb8397089f 100644 --- a/src/subcommand/wallet/inscriptions.rs +++ b/src/subcommand/wallet/inscriptions.rs @@ -8,7 +8,7 @@ pub struct Output { pub postage: u64, } -pub(crate) fn run(options: Options) -> Result { +pub(crate) fn run(options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; @@ -35,7 +35,5 @@ pub(crate) fn run(options: Options) -> Result { } } - print_json(&output)?; - - Ok(()) + Ok(Box::new(output)) } diff --git a/src/subcommand/wallet/outputs.rs b/src/subcommand/wallet/outputs.rs index d064cd312f..dee75fe70e 100644 --- a/src/subcommand/wallet/outputs.rs +++ b/src/subcommand/wallet/outputs.rs @@ -6,7 +6,7 @@ pub struct Output { pub amount: u64, } -pub(crate) fn run(options: Options) -> Result { +pub(crate) fn run(options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; @@ -18,7 +18,5 @@ pub(crate) fn run(options: Options) -> Result { }); } - print_json(outputs)?; - - Ok(()) + Ok(Box::new(outputs)) } diff --git a/src/subcommand/wallet/receive.rs b/src/subcommand/wallet/receive.rs index 94228ff364..1eb35d417c 100644 --- a/src/subcommand/wallet/receive.rs +++ b/src/subcommand/wallet/receive.rs @@ -5,12 +5,10 @@ pub struct Output { pub address: Address, } -pub(crate) fn run(options: Options) -> Result { +pub(crate) fn run(options: Options) -> SubcommandResult { let address = options .bitcoin_rpc_client_for_wallet_command(false)? .get_new_address(None, Some(bitcoincore_rpc::json::AddressType::Bech32m))?; - print_json(Output { address })?; - - Ok(()) + Ok(Box::new(Output { address })) } diff --git a/src/subcommand/wallet/restore.rs b/src/subcommand/wallet/restore.rs index 3e54bfc016..f2afbcf5d1 100644 --- a/src/subcommand/wallet/restore.rs +++ b/src/subcommand/wallet/restore.rs @@ -13,9 +13,8 @@ pub(crate) struct Restore { } impl Restore { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { initialize_wallet(&options, self.mnemonic.to_seed(self.passphrase))?; - - Ok(()) + Ok(Box::new(Empty {})) } } diff --git a/src/subcommand/wallet/sats.rs b/src/subcommand/wallet/sats.rs index 8006e7477b..79cd2f510d 100644 --- a/src/subcommand/wallet/sats.rs +++ b/src/subcommand/wallet/sats.rs @@ -24,7 +24,7 @@ pub struct OutputRare { } impl Sats { - pub(crate) fn run(&self, options: Options) -> Result { + pub(crate) fn run(&self, options: Options) -> SubcommandResult { let index = Index::open(&options)?; index.update()?; @@ -42,7 +42,7 @@ impl Sats { output: outpoint, }); } - print_json(output)?; + Ok(Box::new(output)) } else { let mut output = Vec::new(); for (outpoint, sat, offset, rarity) in rare_sats(utxos) { @@ -53,10 +53,8 @@ impl Sats { rarity, }); } - print_json(output)?; + Ok(Box::new(output)) } - - Ok(()) } } diff --git a/src/subcommand/wallet/send.rs b/src/subcommand/wallet/send.rs index ac8faf52c4..8aa015a297 100644 --- a/src/subcommand/wallet/send.rs +++ b/src/subcommand/wallet/send.rs @@ -19,7 +19,7 @@ pub struct Output { } impl Send { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let address = self .address .clone() @@ -49,8 +49,7 @@ impl Send { Outgoing::Amount(amount) => { Self::lock_inscriptions(&client, inscriptions, unspent_outputs)?; let txid = Self::send_amount(&client, amount, address, self.fee_rate.n())?; - print_json(Output { transaction: txid })?; - return Ok(()); + return Ok(Box::new(Output { transaction: txid })); } }; @@ -82,9 +81,7 @@ impl Send { let txid = client.send_raw_transaction(&signed_tx)?; - println!("{txid}"); - - Ok(()) + Ok(Box::new(Output { transaction: txid })) } fn lock_inscriptions( diff --git a/src/subcommand/wallet/transactions.rs b/src/subcommand/wallet/transactions.rs index 9e5961f911..9f63779652 100644 --- a/src/subcommand/wallet/transactions.rs +++ b/src/subcommand/wallet/transactions.rs @@ -13,7 +13,7 @@ pub struct Output { } impl Transactions { - pub(crate) fn run(self, options: Options) -> Result { + pub(crate) fn run(self, options: Options) -> SubcommandResult { let mut output = Vec::new(); for tx in options .bitcoin_rpc_client_for_wallet_command(false)? @@ -30,8 +30,6 @@ impl Transactions { }); } - print_json(output)?; - - Ok(()) + Ok(Box::new(output)) } } diff --git a/tests/command_builder.rs b/tests/command_builder.rs index 84c14f168a..c2e9dadd53 100644 --- a/tests/command_builder.rs +++ b/tests/command_builder.rs @@ -154,7 +154,7 @@ impl CommandBuilder { stdout.into() } - pub(crate) fn run_and_check_output(self) -> T { + pub(crate) fn run_and_deserialize_output(self) -> T { let stdout = self.stdout_regex(".*").run_and_extract_stdout(); serde_json::from_str(&stdout) .unwrap_or_else(|err| panic!("Failed to deserialize JSON: {err}\n{stdout}")) diff --git a/tests/epochs.rs b/tests/epochs.rs index 648e9e1739..ef49c266d0 100644 --- a/tests/epochs.rs +++ b/tests/epochs.rs @@ -3,7 +3,7 @@ use {super::*, ord::subcommand::epochs::Output, ord::Sat}; #[test] fn empty() { assert_eq!( - CommandBuilder::new("epochs").run_and_check_output::(), + CommandBuilder::new("epochs").run_and_deserialize_output::(), Output { starting_sats: vec![ Sat(0), diff --git a/tests/find.rs b/tests/find.rs index 2efbcea850..63d55497f0 100644 --- a/tests/find.rs +++ b/tests/find.rs @@ -6,7 +6,7 @@ fn find_command_returns_satpoint_for_sat() { assert_eq!( CommandBuilder::new("--index-sats find 0") .rpc_server(&rpc_server) - .run_and_check_output::(), + .run_and_deserialize_output::(), Output { satpoint: "4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b:0:0" .parse() diff --git a/tests/index.rs b/tests/index.rs index b232d8ef4f..091fe3229b 100644 --- a/tests/index.rs +++ b/tests/index.rs @@ -1,4 +1,4 @@ -use {super::*, crate::command_builder::ToArgs}; +use {super::*, crate::command_builder::ToArgs, ord::subcommand::Empty}; #[test] fn custom_index_path() { @@ -11,7 +11,7 @@ fn custom_index_path() { CommandBuilder::new(format!("--index {} index run", index_path.display())) .rpc_server(&rpc_server) - .run_and_extract_stdout(); + .run_and_deserialize_output::(); assert!(index_path.is_file()) } @@ -27,13 +27,13 @@ fn re_opening_database_does_not_trigger_schema_check() { CommandBuilder::new(format!("--index {} index run", index_path.display())) .rpc_server(&rpc_server) - .run_and_extract_stdout(); + .run_and_deserialize_output::(); assert!(index_path.is_file()); CommandBuilder::new(format!("--index {} index run", index_path.display())) .rpc_server(&rpc_server) - .run_and_extract_stdout(); + .run_and_deserialize_output::(); } #[test] @@ -80,6 +80,7 @@ fn export_inscription_number_to_id_tsv() { let tsv = CommandBuilder::new("index export --tsv foo.tsv") .rpc_server(&rpc_server) .temp_dir(temp_dir) + .stdout_regex(r"\{\}\n") .run_and_extract_file("foo.tsv"); let entries: std::collections::BTreeMap = tsv @@ -96,6 +97,6 @@ fn export_inscription_number_to_id_tsv() { assert_eq!( entries.get(&2).unwrap(), - &ord::Object::from_str(&inscription).unwrap() - ) + &ord::Object::InscriptionId(inscription), + ); } diff --git a/tests/info.rs b/tests/info.rs index 1d8d3675e4..414651b175 100644 --- a/tests/info.rs +++ b/tests/info.rs @@ -77,7 +77,7 @@ fn transactions() { index_path.display() )) .rpc_server(&rpc_server) - .run_and_check_output::>() + .run_and_deserialize_output::>() .is_empty()); rpc_server.mine_blocks(10); @@ -87,7 +87,7 @@ fn transactions() { index_path.display() )) .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].start, 0); assert_eq!(output[0].end, 1); @@ -100,7 +100,7 @@ fn transactions() { index_path.display() )) .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[1].start, 1); assert_eq!(output[1].end, 11); diff --git a/tests/json_api.rs b/tests/json_api.rs index 94eca276bb..51586c568b 100644 --- a/tests/json_api.rs +++ b/tests/json_api.rs @@ -88,7 +88,7 @@ fn get_sat_with_inscription_on_common_sat_and_more_inscriptions() { )) .write("foo.txt", "FOO") .rpc_server(&rpc_server) - .run_and_check_output(); + .run_and_deserialize_output(); rpc_server.mine_blocks(1); let inscription_id = InscriptionId::from(reveal); @@ -192,7 +192,7 @@ fn create_210_inscriptions( let Inscribe { reveal, .. } = CommandBuilder::new("wallet inscribe --fee-rate 1 foo.txt") .write("foo.txt", "FOO") .rpc_server(rpc_server) - .run_and_check_output(); + .run_and_deserialize_output(); rpc_server.mine_blocks(1); blessed_inscriptions.push(InscriptionId::from(reveal)); } diff --git a/tests/lib.rs b/tests/lib.rs index 2beade9cc3..3e2eda203f 100644 --- a/tests/lib.rs +++ b/tests/lib.rs @@ -2,25 +2,25 @@ use { self::{command_builder::CommandBuilder, expected::Expected, test_server::TestServer}, - bip39::Mnemonic, bitcoin::{ address::{Address, NetworkUnchecked}, blockdata::constants::COIN_VALUE, - Network, OutPoint, Txid, + Network, OutPoint, }, executable_path::executable_path, - ord::inscription_id::InscriptionId, - ord::rarity::Rarity, - ord::subcommand::wallet::send::Output, - ord::templates::inscription::InscriptionJson, - ord::templates::inscriptions::InscriptionsJson, - ord::templates::output::OutputJson, - ord::templates::sat::SatJson, - ord::SatPoint, + ord::{ + inscription_id::InscriptionId, + rarity::Rarity, + templates::{ + inscription::InscriptionJson, inscriptions::InscriptionsJson, output::OutputJson, + sat::SatJson, + }, + SatPoint, + }, pretty_assertions::assert_eq as pretty_assert_eq, regex::Regex, reqwest::{StatusCode, Url}, - serde::{de::DeserializeOwned, Deserialize}, + serde::de::DeserializeOwned, std::{ fs, net::TcpListener, @@ -31,8 +31,7 @@ use { time::Duration, }, tempfile::TempDir, - test_bitcoincore_rpc::Sent, - test_bitcoincore_rpc::TransactionTemplate, + test_bitcoincore_rpc::{Sent, TransactionTemplate}, }; macro_rules! assert_regex_match { @@ -49,14 +48,7 @@ macro_rules! assert_regex_match { }; } -#[derive(Deserialize, Debug)] -struct Inscribe { - #[allow(dead_code)] - commit: Txid, - inscription: String, - reveal: Txid, - fees: u64, -} +type Inscribe = ord::subcommand::wallet::inscribe::Output; fn inscribe(rpc_server: &test_bitcoincore_rpc::Handle) -> Inscribe { rpc_server.mine_blocks(1); @@ -64,7 +56,7 @@ fn inscribe(rpc_server: &test_bitcoincore_rpc::Handle) -> Inscribe { let output = CommandBuilder::new("wallet inscribe --fee-rate 1 foo.txt") .write("foo.txt", "FOO") .rpc_server(rpc_server) - .run_and_check_output(); + .run_and_deserialize_output(); rpc_server.mine_blocks(1); @@ -89,15 +81,10 @@ fn envelope(payload: &[&[u8]]) -> bitcoin::Witness { bitcoin::Witness::from_slice(&[script.into_bytes(), Vec::new()]) } -#[derive(Deserialize)] -struct Create { - mnemonic: Mnemonic, -} - fn create_wallet(rpc_server: &test_bitcoincore_rpc::Handle) { CommandBuilder::new(format!("--chain {} wallet create", rpc_server.network())) .rpc_server(rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); } mod command_builder; diff --git a/tests/list.rs b/tests/list.rs index 8b82267d3b..d66fd47caa 100644 --- a/tests/list.rs +++ b/tests/list.rs @@ -7,7 +7,7 @@ fn output_found() { "--index-sats list 4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b:0", ) .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!( output, diff --git a/tests/parse.rs b/tests/parse.rs index 241ae0467c..1877c25ec7 100644 --- a/tests/parse.rs +++ b/tests/parse.rs @@ -3,7 +3,7 @@ use {super::*, ord::subcommand::parse::Output, ord::Object}; #[test] fn name() { assert_eq!( - CommandBuilder::new("parse a").run_and_check_output::(), + CommandBuilder::new("parse a").run_and_deserialize_output::(), Output { object: Object::Integer(2099999997689999), } @@ -14,7 +14,7 @@ fn name() { fn hash() { assert_eq!( CommandBuilder::new("parse 0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef") - .run_and_check_output::(), + .run_and_deserialize_output::(), Output { object: "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" .parse::() diff --git a/tests/server.rs b/tests/server.rs index 4ded70df60..c893cc901b 100644 --- a/tests/server.rs +++ b/tests/server.rs @@ -1,4 +1,4 @@ -use {super::*, crate::command_builder::ToArgs}; +use {super::*, crate::command_builder::ToArgs, ord::subcommand::wallet::send::Output}; #[test] fn run() { @@ -148,17 +148,16 @@ fn inscription_page_after_send() { )) .rpc_server(&rpc_server) .stdout_regex(".*") - .run_and_extract_stdout(); + .run_and_deserialize_output::() + .transaction; rpc_server.mine_blocks(1); - let send = txid.trim(); - let ord_server = TestServer::spawn_with_args(&rpc_server, &[]); ord_server.assert_response_regex( format!("/inscription/{inscription}"), format!( - r".*

Inscription 0

.*
address
\s*
bc1qcqgs2pps4u4yedfyl5pysdjjncs8et5utseepv
.*
location
\s*
{send}:0:0
.*", + r".*

Inscription 0

.*
address
\s*
bc1qcqgs2pps4u4yedfyl5pysdjjncs8et5utseepv
.*
location
\s*
{txid}:0:0
.*", ), ) } diff --git a/tests/subsidy.rs b/tests/subsidy.rs index 4ee2a7bf84..624f3fc1d1 100644 --- a/tests/subsidy.rs +++ b/tests/subsidy.rs @@ -3,7 +3,7 @@ use {super::*, ord::subcommand::subsidy::Output}; #[test] fn genesis() { assert_eq!( - CommandBuilder::new("subsidy 0").run_and_check_output::(), + CommandBuilder::new("subsidy 0").run_and_deserialize_output::(), Output { first: 0, subsidy: 5000000000, @@ -15,7 +15,7 @@ fn genesis() { #[test] fn second_block() { assert_eq!( - CommandBuilder::new("subsidy 1").run_and_check_output::(), + CommandBuilder::new("subsidy 1").run_and_deserialize_output::(), Output { first: 5000000000, subsidy: 5000000000, @@ -27,7 +27,7 @@ fn second_block() { #[test] fn second_to_last_block_with_subsidy() { assert_eq!( - CommandBuilder::new("subsidy 6929998").run_and_check_output::(), + CommandBuilder::new("subsidy 6929998").run_and_deserialize_output::(), Output { first: 2099999997689998, subsidy: 1, @@ -39,7 +39,7 @@ fn second_to_last_block_with_subsidy() { #[test] fn last_block_with_subsidy() { assert_eq!( - CommandBuilder::new("subsidy 6929999").run_and_check_output::(), + CommandBuilder::new("subsidy 6929999").run_and_deserialize_output::(), Output { first: 2099999997689999, subsidy: 1, diff --git a/tests/supply.rs b/tests/supply.rs index 581224e474..9e5f396af2 100644 --- a/tests/supply.rs +++ b/tests/supply.rs @@ -3,7 +3,7 @@ use {super::*, ord::subcommand::supply::Output}; #[test] fn genesis() { assert_eq!( - CommandBuilder::new("supply").run_and_check_output::(), + CommandBuilder::new("supply").run_and_deserialize_output::(), Output { supply: 2099999997690000, first: 0, diff --git a/tests/traits.rs b/tests/traits.rs index da2d1232e8..3eb4aa6fa1 100644 --- a/tests/traits.rs +++ b/tests/traits.rs @@ -3,7 +3,7 @@ use {super::*, ord::subcommand::traits::Output, ord::Rarity}; #[test] fn traits_command_prints_sat_traits() { assert_eq!( - CommandBuilder::new("traits 0").run_and_check_output::(), + CommandBuilder::new("traits 0").run_and_deserialize_output::(), Output { number: 0, decimal: "0.0".into(), @@ -21,7 +21,7 @@ fn traits_command_prints_sat_traits() { #[test] fn traits_command_for_last_sat() { assert_eq!( - CommandBuilder::new("traits 2099999997689999").run_and_check_output::(), + CommandBuilder::new("traits 2099999997689999").run_and_deserialize_output::(), Output { number: 2099999997689999, decimal: "6929999.0".into(), diff --git a/tests/wallet/balance.rs b/tests/wallet/balance.rs index bd686b99a5..79557aa562 100644 --- a/tests/wallet/balance.rs +++ b/tests/wallet/balance.rs @@ -8,7 +8,7 @@ fn wallet_balance() { assert_eq!( CommandBuilder::new("wallet balance") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .cardinal, 0 ); @@ -18,7 +18,7 @@ fn wallet_balance() { assert_eq!( CommandBuilder::new("wallet balance") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .cardinal, 50 * COIN_VALUE ); @@ -32,7 +32,7 @@ fn wallet_balance_only_counts_cardinal_utxos() { assert_eq!( CommandBuilder::new("wallet balance") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .cardinal, 0 ); @@ -42,7 +42,7 @@ fn wallet_balance_only_counts_cardinal_utxos() { assert_eq!( CommandBuilder::new("wallet balance") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .cardinal, 100 * COIN_VALUE - 10_000 ); diff --git a/tests/wallet/cardinals.rs b/tests/wallet/cardinals.rs index c52a3bdbb9..db966c39a8 100644 --- a/tests/wallet/cardinals.rs +++ b/tests/wallet/cardinals.rs @@ -1,6 +1,6 @@ use { super::*, - ord::subcommand::wallet::{cardinals::Cardinal, outputs::Output}, + ord::subcommand::wallet::{cardinals::CardinalUtxo, outputs::Output}, }; #[test] @@ -12,11 +12,11 @@ fn cardinals() { let all_outputs = CommandBuilder::new("wallet outputs") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); let cardinal_outputs = CommandBuilder::new("wallet cardinals") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(all_outputs.len() - cardinal_outputs.len(), 1); } diff --git a/tests/wallet/create.rs b/tests/wallet/create.rs index cc8d5ec35c..c5f267e418 100644 --- a/tests/wallet/create.rs +++ b/tests/wallet/create.rs @@ -1,4 +1,4 @@ -use super::*; +use {super::*, ord::subcommand::wallet::create::Output}; #[test] fn create() { @@ -8,16 +8,16 @@ fn create() { CommandBuilder::new("wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert!(rpc_server.wallets().contains("ord")); } #[test] fn seed_phrases_are_twelve_words_long() { - let Create { mnemonic } = CommandBuilder::new("wallet create") + let Output { mnemonic, .. } = CommandBuilder::new("wallet create") .rpc_server(&test_bitcoincore_rpc::spawn()) - .run_and_check_output::(); + .run_and_deserialize_output(); assert_eq!(mnemonic.word_count(), 12); } @@ -28,7 +28,7 @@ fn wallet_creates_correct_mainnet_taproot_descriptor() { CommandBuilder::new("wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert_eq!(rpc_server.descriptors().len(), 2); assert_regex_match!( @@ -49,7 +49,7 @@ fn wallet_creates_correct_test_network_taproot_descriptor() { CommandBuilder::new("--chain signet wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert_eq!(rpc_server.descriptors().len(), 2); assert_regex_match!( @@ -68,7 +68,7 @@ fn detect_wrong_descriptors() { CommandBuilder::new("wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); rpc_server.import_descriptor("wpkh([aslfjk])#a23ad2l".to_string()); @@ -89,7 +89,7 @@ fn create_with_different_name() { CommandBuilder::new("--wallet inscription-wallet wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert!(rpc_server.wallets().contains("inscription-wallet")); } diff --git a/tests/wallet/inscribe.rs b/tests/wallet/inscribe.rs index 47527a8284..4af22348e1 100644 --- a/tests/wallet/inscribe.rs +++ b/tests/wallet/inscribe.rs @@ -35,7 +35,7 @@ fn inscribe_works_with_huge_expensive_inscriptions() { )) .write("foo.txt", [0; 350_000]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); } #[test] @@ -61,7 +61,7 @@ fn inscribe_no_backup() { CommandBuilder::new("wallet inscribe hello.txt --no-backup --fee-rate 1") .write("hello.txt", "HELLOWORLD") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert_eq!(rpc_server.descriptors().len(), 2); } @@ -205,7 +205,7 @@ fn inscribe_with_optional_satpoint_arg() { )) .write("foo.txt", "FOO") .rpc_server(&rpc_server) - .run_and_check_output(); + .run_and_deserialize_output(); rpc_server.mine_blocks(1); @@ -227,7 +227,7 @@ fn inscribe_with_fee_rate() { CommandBuilder::new("--index-sats wallet inscribe degenerate.png --fee-rate 2.0") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); let tx1 = &rpc_server.mempool()[0]; let mut fee = 0; @@ -270,7 +270,7 @@ fn inscribe_with_commit_fee_rate() { ) .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); let tx1 = &rpc_server.mempool()[0]; let mut fee = 0; @@ -308,14 +308,14 @@ fn inscribe_with_wallet_named_foo() { CommandBuilder::new("--wallet foo wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); rpc_server.mine_blocks(1); CommandBuilder::new("--wallet foo wallet inscribe degenerate.png --fee-rate 1") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); } #[test] @@ -327,14 +327,14 @@ fn inscribe_with_dry_run_flag() { CommandBuilder::new("wallet inscribe --dry-run degenerate.png --fee-rate 1") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert!(rpc_server.mempool().is_empty()); CommandBuilder::new("wallet inscribe degenerate.png --fee-rate 1") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert_eq!(rpc_server.mempool().len(), 2); } @@ -349,14 +349,14 @@ fn inscribe_with_dry_run_flag_fees_inscrease() { CommandBuilder::new("wallet inscribe --dry-run degenerate.png --fee-rate 1") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .fees; let total_fee_normal = CommandBuilder::new("wallet inscribe --dry-run degenerate.png --fee-rate 1.1") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .fees; assert!(total_fee_dry_run < total_fee_normal); @@ -370,7 +370,7 @@ fn inscribe_to_specific_destination() { let destination = CommandBuilder::new("wallet receive") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .address; let txid = CommandBuilder::new(format!( @@ -379,7 +379,7 @@ fn inscribe_to_specific_destination() { )) .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .reveal; let reveal_tx = &rpc_server.mempool()[1]; // item 0 is the commit, item 1 is the reveal. @@ -427,14 +427,14 @@ fn inscribe_works_with_postage() { CommandBuilder::new("wallet inscribe foo.txt --postage 5btc --fee-rate 10".to_string()) .write("foo.txt", [0; 350]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); rpc_server.mine_blocks(1); let inscriptions = CommandBuilder::new("wallet inscriptions".to_string()) .write("foo.txt", [0; 350]) .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); pretty_assert_eq!(inscriptions[0].postage, 5 * COIN_VALUE); } diff --git a/tests/wallet/inscriptions.rs b/tests/wallet/inscriptions.rs index 3951e8f072..20ac3283f0 100644 --- a/tests/wallet/inscriptions.rs +++ b/tests/wallet/inscriptions.rs @@ -1,6 +1,6 @@ use { super::*, - ord::subcommand::wallet::{inscriptions::Output, receive}, + ord::subcommand::wallet::{inscriptions, receive, send}, }; #[test] @@ -17,10 +17,10 @@ fn inscriptions() { let output = CommandBuilder::new("wallet inscriptions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output.len(), 1); - assert_eq!(output[0].inscription, inscription.parse().unwrap()); + assert_eq!(output[0].inscription, inscription); assert_eq!(output[0].location, format!("{reveal}:0:0").parse().unwrap()); assert_eq!( output[0].explorer, @@ -29,28 +29,27 @@ fn inscriptions() { let address = CommandBuilder::new("wallet receive") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .address; - let stdout = CommandBuilder::new(format!( + let txid = CommandBuilder::new(format!( "wallet send --fee-rate 1 {} {inscription}", address.assume_checked() )) .rpc_server(&rpc_server) .expected_exit_code(0) .stdout_regex(".*") - .run_and_extract_stdout(); + .run_and_deserialize_output::() + .transaction; rpc_server.mine_blocks(1); - let txid = Txid::from_str(stdout.trim()).unwrap(); - let output = CommandBuilder::new("wallet inscriptions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output.len(), 1); - assert_eq!(output[0].inscription, inscription.parse().unwrap()); + assert_eq!(output[0].inscription, inscription); assert_eq!(output[0].location, format!("{txid}:0:0").parse().unwrap()); } @@ -76,10 +75,10 @@ fn inscriptions_includes_locked_utxos() { let output = CommandBuilder::new("wallet inscriptions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output.len(), 1); - assert_eq!(output[0].inscription, inscription.parse().unwrap()); + assert_eq!(output[0].inscription, inscription); assert_eq!(output[0].location, format!("{reveal}:0:0").parse().unwrap()); } @@ -93,13 +92,13 @@ fn inscriptions_with_postage() { let output = CommandBuilder::new("wallet inscriptions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].postage, 10000); let address = CommandBuilder::new("wallet receive") .rpc_server(&rpc_server) - .run_and_check_output::() + .run_and_deserialize_output::() .address; CommandBuilder::new(format!( @@ -115,7 +114,7 @@ fn inscriptions_with_postage() { let output = CommandBuilder::new("wallet inscriptions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].postage, 9889); } diff --git a/tests/wallet/outputs.rs b/tests/wallet/outputs.rs index de75585fc4..fcef95d39c 100644 --- a/tests/wallet/outputs.rs +++ b/tests/wallet/outputs.rs @@ -11,7 +11,7 @@ fn outputs() { let output = CommandBuilder::new("wallet outputs") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].output, outpoint); assert_eq!(output[0].amount, amount); @@ -30,7 +30,7 @@ fn outputs_includes_locked_outputs() { let output = CommandBuilder::new("wallet outputs") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].output, outpoint); assert_eq!(output[0].amount, amount); diff --git a/tests/wallet/receive.rs b/tests/wallet/receive.rs index 15aba71879..3f5f9120d4 100644 --- a/tests/wallet/receive.rs +++ b/tests/wallet/receive.rs @@ -7,7 +7,7 @@ fn receive() { let output = CommandBuilder::new("wallet receive") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert!(output.address.is_valid_for_network(Network::Bitcoin)); } diff --git a/tests/wallet/restore.rs b/tests/wallet/restore.rs index 6ed175eada..5106c5eb33 100644 --- a/tests/wallet/restore.rs +++ b/tests/wallet/restore.rs @@ -1,13 +1,13 @@ -use super::*; +use {super::*, ord::subcommand::wallet::create, ord::subcommand::Empty}; #[test] fn restore_generates_same_descriptors() { let (mnemonic, descriptors) = { let rpc_server = test_bitcoincore_rpc::spawn(); - let Create { mnemonic } = CommandBuilder::new("wallet create") + let create::Output { mnemonic, .. } = CommandBuilder::new("wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output(); (mnemonic, rpc_server.descriptors()) }; @@ -16,7 +16,7 @@ fn restore_generates_same_descriptors() { CommandBuilder::new(["wallet", "restore", &mnemonic.to_string()]) .rpc_server(&rpc_server) - .run_and_extract_stdout(); + .run_and_deserialize_output::(); assert_eq!(rpc_server.descriptors(), descriptors); } @@ -27,9 +27,10 @@ fn restore_generates_same_descriptors_with_passphrase() { let (mnemonic, descriptors) = { let rpc_server = test_bitcoincore_rpc::spawn(); - let Create { mnemonic } = CommandBuilder::new(["wallet", "create", "--passphrase", passphrase]) - .rpc_server(&rpc_server) - .run_and_check_output::(); + let create::Output { mnemonic, .. } = + CommandBuilder::new(["wallet", "create", "--passphrase", passphrase]) + .rpc_server(&rpc_server) + .run_and_deserialize_output(); (mnemonic, rpc_server.descriptors()) }; @@ -44,7 +45,7 @@ fn restore_generates_same_descriptors_with_passphrase() { &mnemonic.to_string(), ]) .rpc_server(&rpc_server) - .run_and_extract_stdout(); + .run_and_deserialize_output::(); assert_eq!(rpc_server.descriptors(), descriptors); } diff --git a/tests/wallet/sats.rs b/tests/wallet/sats.rs index e835833a2c..0d6ec2657b 100644 --- a/tests/wallet/sats.rs +++ b/tests/wallet/sats.rs @@ -11,7 +11,7 @@ fn sats() { let output = CommandBuilder::new("--index-sats wallet sats") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].sat, 50 * COIN_VALUE); assert_eq!(output[0].output.to_string(), format!("{second_coinbase}:0")); @@ -26,7 +26,7 @@ fn sats_from_tsv_success() { let output = CommandBuilder::new("--index-sats wallet sats --tsv foo.tsv") .write("foo.tsv", "nvtcsezkbtg") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(output[0].sat, "nvtcsezkbtg"); assert_eq!(output[0].output.to_string(), format!("{second_coinbase}:0")); diff --git a/tests/wallet/send.rs b/tests/wallet/send.rs index 3d22c6c70f..79db26942a 100644 --- a/tests/wallet/send.rs +++ b/tests/wallet/send.rs @@ -1,4 +1,4 @@ -use super::*; +use {super::*, ord::subcommand::wallet::send::Output}; #[test] fn inscriptions_can_be_sent() { @@ -10,19 +10,19 @@ fn inscriptions_can_be_sent() { rpc_server.mine_blocks(1); - let stdout = CommandBuilder::new(format!( + let output = CommandBuilder::new(format!( "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {inscription}", )) .rpc_server(&rpc_server) .stdout_regex(r".*") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); let txid = rpc_server.mempool()[0].txid(); - assert_eq!(format!("{txid}\n"), stdout); + assert_eq!(txid, output.transaction); rpc_server.mine_blocks(1); - let send_txid = stdout.trim(); + let send_txid = output.transaction; let ord_server = TestServer::spawn_with_args(&rpc_server, &[]); ord_server.assert_response_regex( @@ -69,16 +69,15 @@ fn send_inscribed_sat() { rpc_server.mine_blocks(1); - let stdout = CommandBuilder::new(format!( + let output = CommandBuilder::new(format!( "wallet send --fee-rate 1 bc1qcqgs2pps4u4yedfyl5pysdjjncs8et5utseepv {inscription}", )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); rpc_server.mine_blocks(1); - let send_txid = stdout.trim(); + let send_txid = output.transaction; let ord_server = TestServer::spawn_with_args(&rpc_server, &[]); ord_server.assert_response_regex( @@ -96,14 +95,13 @@ fn send_on_mainnnet_works_with_wallet_named_foo() { CommandBuilder::new("--wallet foo wallet create") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); CommandBuilder::new(format!( "--wallet foo wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {txid}:0:0" )) .rpc_server(&rpc_server) - .stdout_regex(r"[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); } #[test] @@ -129,15 +127,13 @@ fn send_on_mainnnet_works_with_wallet_named_ord() { let txid = rpc_server.mine_blocks_with_subsidy(1, 1_000_000)[0].txdata[0].txid(); create_wallet(&rpc_server); - let stdout = CommandBuilder::new(format!( + let output = CommandBuilder::new(format!( "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {txid}:0:0" )) .rpc_server(&rpc_server) - .stdout_regex(r".*") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); - let txid = rpc_server.mempool()[0].txid(); - assert_eq!(format!("{txid}\n"), stdout); + assert_eq!(rpc_server.mempool()[0].txid(), output.transaction); } #[test] @@ -151,7 +147,7 @@ fn send_does_not_use_inscribed_sats_as_cardinal_utxos() { )) .write("degenerate.png", [1; 100]) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); let txid = rpc_server.mine_blocks_with_subsidy(1, 100)[0].txdata[0].txid(); CommandBuilder::new(format!( @@ -210,8 +206,7 @@ fn can_send_after_dust_limit_from_an_inscription() { "wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {output}:330" )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); } #[test] @@ -284,8 +279,7 @@ fn splitting_merged_inscriptions_is_possible() { reveal_txid, )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); rpc_server.mine_blocks(1); @@ -295,8 +289,7 @@ fn splitting_merged_inscriptions_is_possible() { reveal_txid, )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); rpc_server.mine_blocks(1); @@ -306,8 +299,7 @@ fn splitting_merged_inscriptions_is_possible() { reveal_txid, )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); } #[test] @@ -339,7 +331,7 @@ fn send_btc_with_fee_rate() { "wallet send --fee-rate 13.3 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 1btc", ) .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); let tx = &rpc_server.mempool()[0]; let mut fee = 0; @@ -389,7 +381,7 @@ fn send_btc_locks_inscriptions() { let output = CommandBuilder::new("wallet send --fee-rate 1 bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 1btc") .rpc_server(&rpc_server) - .run_and_check_output::(); + .run_and_deserialize_output::(); assert_eq!( output.transaction, @@ -442,8 +434,7 @@ fn wallet_send_with_fee_rate() { "wallet send bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {inscription} --fee-rate 2.0" )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); let tx = &rpc_server.mempool()[0]; let mut fee = 0; @@ -494,8 +485,7 @@ fn wallet_send_with_fee_rate_and_target_postage() { "wallet send bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4 {inscription} --fee-rate 2.0 --postage 77000sat" )) .rpc_server(&rpc_server) - .stdout_regex("[[:xdigit:]]{64}\n") - .run_and_extract_stdout(); + .run_and_deserialize_output::(); let tx = &rpc_server.mempool()[0]; let mut fee = 0; diff --git a/tests/wallet/transactions.rs b/tests/wallet/transactions.rs index d3f275711e..8f3fd4ffc9 100644 --- a/tests/wallet/transactions.rs +++ b/tests/wallet/transactions.rs @@ -9,7 +9,7 @@ fn transactions() { CommandBuilder::new("wallet transactions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_eq!(rpc_server.loaded_wallets().len(), 1); assert_eq!(rpc_server.loaded_wallets().first().unwrap(), "ord"); @@ -18,7 +18,7 @@ fn transactions() { let output = CommandBuilder::new("wallet transactions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_regex_match!(output[0].transaction.to_string(), "[[:xdigit:]]{64}"); assert_eq!(output[0].confirmations, 1); @@ -38,7 +38,7 @@ fn transactions_with_limit() { let output = CommandBuilder::new("wallet transactions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_regex_match!(output[0].transaction.to_string(), "[[:xdigit:]]{64}"); assert_eq!(output[0].confirmations, 1); @@ -47,14 +47,14 @@ fn transactions_with_limit() { let output = CommandBuilder::new("wallet transactions") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_regex_match!(output[1].transaction.to_string(), "[[:xdigit:]]{64}"); assert_eq!(output[1].confirmations, 2); let output = CommandBuilder::new("wallet transactions --limit 1") .rpc_server(&rpc_server) - .run_and_check_output::>(); + .run_and_deserialize_output::>(); assert_regex_match!(output[0].transaction.to_string(), "[[:xdigit:]]{64}"); assert_eq!(output[0].confirmations, 1); From c910079fa90a3786d3034b78018c26a7c60f3ff3 Mon Sep 17 00:00:00 2001 From: raph Date: Fri, 25 Aug 2023 21:20:59 +0200 Subject: [PATCH 22/61] Render GLB/GLTF models in preview (#2369) --- src/index/updater.rs | 7 +++++-- src/inscription.rs | 4 +++- src/media.rs | 4 +++- src/subcommand/server.rs | 14 ++++++++++++-- src/templates.rs | 4 ++-- src/templates/preview.rs | 5 +++++ templates/preview-model.html | 17 +++++++++++++++++ tests/parse.rs | 2 +- 8 files changed, 48 insertions(+), 9 deletions(-) create mode 100644 templates/preview-model.html diff --git a/src/index/updater.rs b/src/index/updater.rs index 0828f156d3..d053e1bd8e 100644 --- a/src/index/updater.rs +++ b/src/index/updater.rs @@ -273,7 +273,7 @@ impl<'index> Updater<'_> { // There's no try_iter on tokio::sync::mpsc::Receiver like std::sync::mpsc::Receiver. // So we just loop until BATCH_SIZE doing try_recv until it returns None. let mut outpoints = vec![outpoint]; - for _ in 0..BATCH_SIZE-1 { + for _ in 0..BATCH_SIZE - 1 { let Ok(outpoint) = outpoint_receiver.try_recv() else { break; }; @@ -296,7 +296,10 @@ impl<'index> Updater<'_> { }; // Send all tx output values back in order for (i, tx) in txs.iter().flatten().enumerate() { - let Ok(_) = value_sender.send(tx.output[usize::try_from(outpoints[i].vout).unwrap()].value).await else { + let Ok(_) = value_sender + .send(tx.output[usize::try_from(outpoints[i].vout).unwrap()].value) + .await + else { log::error!("Value channel closed unexpectedly"); return; }; diff --git a/src/inscription.rs b/src/inscription.rs index 5506c5f361..a63d4f0d4f 100644 --- a/src/inscription.rs +++ b/src/inscription.rs @@ -44,7 +44,9 @@ impl Inscription { pub(crate) fn from_transaction(tx: &Transaction) -> Vec { let mut result = Vec::new(); for (index, tx_in) in tx.input.iter().enumerate() { - let Ok(inscriptions) = InscriptionParser::parse(&tx_in.witness) else { continue }; + let Ok(inscriptions) = InscriptionParser::parse(&tx_in.witness) else { + continue; + }; result.extend( inscriptions diff --git a/src/media.rs b/src/media.rs index c5ea8ac488..ff43ba40a2 100644 --- a/src/media.rs +++ b/src/media.rs @@ -9,6 +9,7 @@ pub(crate) enum Media { Audio, Iframe, Image, + Model, Pdf, Text, Unknown, @@ -31,7 +32,8 @@ impl Media { ("image/png", Media::Image, &["png"]), ("image/svg+xml", Media::Iframe, &["svg"]), ("image/webp", Media::Image, &["webp"]), - ("model/gltf-binary", Media::Unknown, &["glb"]), + ("model/gltf+json", Media::Model, &["gltf"]), + ("model/gltf-binary", Media::Model, &["glb"]), ("model/stl", Media::Unknown, &["stl"]), ("text/css", Media::Text, &["css"]), ("text/html", Media::Iframe, &[]), diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 1f4f93453d..dd8d2189b7 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -10,8 +10,8 @@ use { crate::templates::{ BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionJson, InscriptionsHtml, InscriptionsJson, OutputHtml, OutputJson, PageContent, PageHtml, PreviewAudioHtml, - PreviewImageHtml, PreviewPdfHtml, PreviewTextHtml, PreviewUnknownHtml, PreviewVideoHtml, - RangeHtml, RareTxt, SatHtml, SatJson, TransactionHtml, + PreviewImageHtml, PreviewModelHtml, PreviewPdfHtml, PreviewTextHtml, PreviewUnknownHtml, + PreviewVideoHtml, RangeHtml, RareTxt, SatHtml, SatJson, TransactionHtml, }, axum::{ body, @@ -919,6 +919,16 @@ impl Server { ) .into_response(), ), + Media::Model => Ok( + ( + [( + header::CONTENT_SECURITY_POLICY, + "script-src-elem 'self' https://ajax.googleapis.com", + )], + PreviewModelHtml { inscription_id }, + ) + .into_response(), + ), Media::Pdf => Ok( ( [( diff --git a/src/templates.rs b/src/templates.rs index 145e4f679d..ab21051e3b 100644 --- a/src/templates.rs +++ b/src/templates.rs @@ -11,8 +11,8 @@ pub(crate) use { output::{OutputHtml, OutputJson}, page_config::PageConfig, preview::{ - PreviewAudioHtml, PreviewImageHtml, PreviewPdfHtml, PreviewTextHtml, PreviewUnknownHtml, - PreviewVideoHtml, + PreviewAudioHtml, PreviewImageHtml, PreviewModelHtml, PreviewPdfHtml, PreviewTextHtml, + PreviewUnknownHtml, PreviewVideoHtml, }, range::RangeHtml, rare::RareTxt, diff --git a/src/templates/preview.rs b/src/templates/preview.rs index ed9ee4a415..74cb0261dd 100644 --- a/src/templates/preview.rs +++ b/src/templates/preview.rs @@ -10,6 +10,11 @@ pub(crate) struct PreviewImageHtml { pub(crate) inscription_id: InscriptionId, } +#[derive(boilerplate::Boilerplate)] +pub(crate) struct PreviewModelHtml { + pub(crate) inscription_id: InscriptionId, +} + #[derive(boilerplate::Boilerplate)] pub(crate) struct PreviewPdfHtml { pub(crate) inscription_id: InscriptionId, diff --git a/templates/preview-model.html b/templates/preview-model.html new file mode 100644 index 0000000000..5386001130 --- /dev/null +++ b/templates/preview-model.html @@ -0,0 +1,17 @@ + + + + + + + + + + + diff --git a/tests/parse.rs b/tests/parse.rs index 1877c25ec7..a8f7cc381b 100644 --- a/tests/parse.rs +++ b/tests/parse.rs @@ -26,7 +26,7 @@ fn hash() { #[test] fn unrecognized_object() { CommandBuilder::new("parse A") - .stderr_regex(r#"error: .*: unrecognized object\n.*"#) + .stderr_regex(r"error: .*: unrecognized object\n.*") .expected_exit_code(2) .run_and_extract_stdout(); } From 34bb1c6b9ec0b15aa38258e6b5b45c5ce075c8f3 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Fri, 25 Aug 2023 14:06:33 -0700 Subject: [PATCH 23/61] Remove transaction ID to inscription ID conversion (#2370) --- src/index.rs | 78 ++++++++++++++++++++----------- src/inscription_id.rs | 6 --- src/subcommand/server.rs | 38 +++++++-------- src/subcommand/wallet/inscribe.rs | 10 +++- tests/json_api.rs | 20 ++++++-- 5 files changed, 94 insertions(+), 58 deletions(-) diff --git a/src/index.rs b/src/index.rs index 75e3f0ace5..6f520cc5e8 100644 --- a/src/index.rs +++ b/src/index.rs @@ -1298,7 +1298,7 @@ mod tests { let context = Context::builder().build(); context.mine_blocks(1); let txid = context.rpc_server.broadcast_tx(template.clone()); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); assert_eq!( @@ -1324,7 +1324,7 @@ mod tests { .build(); context.mine_blocks(1); let txid = context.rpc_server.broadcast_tx(template); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); assert_eq!( @@ -1639,7 +1639,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1673,7 +1673,7 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1702,7 +1702,7 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1727,7 +1727,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1772,14 +1772,20 @@ mod tests { ..Default::default() }); - let first_inscription_id = InscriptionId::from(first_txid); + let first_inscription_id = InscriptionId { + txid: first_txid, + index: 0, + }; let second_txid = context.rpc_server.broadcast_tx(TransactionTemplate { inputs: &[(2, 0, 0)], witness: inscription("text/png", [1; 100]).to_witness(), ..Default::default() }); - let second_inscription_id = InscriptionId::from(second_txid); + let second_inscription_id = InscriptionId { + txid: second_txid, + index: 0, + }; context.mine_blocks(1); @@ -1826,7 +1832,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1875,7 +1881,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1919,7 +1925,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1955,7 +1961,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -1992,7 +1998,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; let coinbase_tx = context.mine_blocks(1)[0].txdata[0].txid(); @@ -2021,7 +2027,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks_with_subsidy(1, 0); @@ -2047,7 +2053,10 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let first_inscription_id = InscriptionId::from(first_txid); + let first_inscription_id = InscriptionId { + txid: first_txid, + index: 0, + }; context.mine_blocks_with_subsidy(1, 0); context.mine_blocks(1); @@ -2058,7 +2067,10 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let second_inscription_id = InscriptionId::from(second_txid); + let second_inscription_id = InscriptionId { + txid: second_txid, + index: 0, + }; context.mine_blocks_with_subsidy(1, 0); @@ -2172,7 +2184,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); context.rpc_server.broadcast_tx(TransactionTemplate { @@ -2205,7 +2217,7 @@ mod tests { output_values: &[0, 50 * COIN_VALUE], ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); context.index.assert_inscription_location( @@ -2230,7 +2242,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks_with_subsidy(1, 0); context.index.assert_inscription_location( @@ -2340,7 +2352,7 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; assert_eq!( context @@ -2401,7 +2413,10 @@ mod tests { context.mine_blocks(1); - let inscription_id = InscriptionId::from(first); + let inscription_id = InscriptionId { + txid: first, + index: 0, + }; assert_eq!( context @@ -2432,7 +2447,10 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(second); + let inscription_id = InscriptionId { + txid: second, + index: 0, + }; context.mine_blocks(1); @@ -2450,13 +2468,19 @@ mod tests { assert!(context .index - .get_inscription_entry(second.into()) + .get_inscription_by_id(InscriptionId { + txid: second, + index: 0 + }) .unwrap() .is_some()); assert!(context .index - .get_inscription_by_id(second.into()) + .get_inscription_by_id(InscriptionId { + txid: second, + index: 0 + }) .unwrap() .is_some()); } @@ -2472,7 +2496,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); @@ -2499,7 +2523,7 @@ mod tests { witness: inscription("text/plain", "hello").to_witness(), ..Default::default() }); - ids.push(InscriptionId::from(txid)); + ids.push(InscriptionId { txid, index: 0 }); context.mine_blocks(1); } @@ -2572,7 +2596,7 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; context.mine_blocks(1); diff --git a/src/inscription_id.rs b/src/inscription_id.rs index 998faeb3c3..cb70b4baa1 100644 --- a/src/inscription_id.rs +++ b/src/inscription_id.rs @@ -85,12 +85,6 @@ impl FromStr for InscriptionId { } } -impl From for InscriptionId { - fn from(txid: Txid) -> Self { - Self { txid, index: 0 } - } -} - #[cfg(test)] mod tests { use super::*; diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index dd8d2189b7..adcceb0bf6 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -600,7 +600,7 @@ impl Server { Extension(index): Extension>, Path(txid): Path, ) -> ServerResult> { - let inscription = index.get_inscription_by_id(txid.into())?; + let inscription = index.get_inscription_by_id(InscriptionId { txid, index: 0 })?; let blockhash = index.get_transaction_blockhash(txid)?; @@ -610,7 +610,7 @@ impl Server { .get_transaction(txid)? .ok_or_not_found(|| format!("transaction {txid}"))?, blockhash, - inscription.map(|_| txid.into()), + inscription.map(|_| InscriptionId { txid, index: 0 }), page_config.chain, ) .page(page_config, index.has_sat_index()?), @@ -1882,7 +1882,7 @@ mod tests { server.mine_blocks(1); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.assert_response_regex( format!("/inscription/{}", inscription_id), @@ -2349,7 +2349,7 @@ mod tests { server.mine_blocks(1); server.assert_response_csp( - format!("/preview/{}", InscriptionId::from(txid)), + format!("/preview/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, "default-src 'self'", ".*
hello
.*", @@ -2370,7 +2370,7 @@ mod tests { server.mine_blocks(1); server.assert_response( - format!("/preview/{}", InscriptionId::from(txid)), + format!("/preview/{}", InscriptionId { txid, index: 0 }), StatusCode::INTERNAL_SERVER_ERROR, "Internal Server Error", ); @@ -2394,7 +2394,7 @@ mod tests { server.mine_blocks(1); server.assert_response_csp( - format!("/preview/{}", InscriptionId::from(txid)), + format!("/preview/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, "default-src 'self'", r".*
<script>alert\('hello'\);</script>
.*", @@ -2411,7 +2411,7 @@ mod tests { witness: inscription("audio/flac", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.mine_blocks(1); @@ -2432,7 +2432,7 @@ mod tests { witness: inscription("application/pdf", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.mine_blocks(1); @@ -2453,7 +2453,7 @@ mod tests { witness: inscription("image/png", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.mine_blocks(1); @@ -2479,7 +2479,7 @@ mod tests { server.mine_blocks(1); server.assert_response_csp( - format!("/preview/{}", InscriptionId::from(txid)), + format!("/preview/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, "default-src 'self' 'unsafe-eval' 'unsafe-inline' data: blob:", "hello", @@ -2500,7 +2500,7 @@ mod tests { server.mine_blocks(1); server.assert_response_csp( - format!("/preview/{}", InscriptionId::from(txid)), + format!("/preview/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, "default-src 'self'", fs::read_to_string("templates/preview-unknown.html").unwrap(), @@ -2517,7 +2517,7 @@ mod tests { witness: inscription("video/webm", "hello").to_witness(), ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.mine_blocks(1); @@ -2542,7 +2542,7 @@ mod tests { server.mine_blocks(1); server.assert_response_regex( - format!("/inscription/{}", InscriptionId::from(txid)), + format!("/inscription/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, ".*Inscription 0.*", ); @@ -2562,7 +2562,7 @@ mod tests { server.mine_blocks(1); server.assert_response_regex( - format!("/inscription/{}", InscriptionId::from(txid)), + format!("/inscription/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, r".*
sat
\s*
5000000000
\s*
preview
.*", ); @@ -2582,7 +2582,7 @@ mod tests { server.mine_blocks(1); server.assert_response_regex( - format!("/inscription/{}", InscriptionId::from(txid)), + format!("/inscription/{}", InscriptionId { txid, index: 0 }), StatusCode::OK, r".*
output value
\s*
5000000000
\s*
preview
.*", ); @@ -2631,7 +2631,7 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.mine_blocks(1); @@ -2653,7 +2653,7 @@ mod tests { ..Default::default() }); - let inscription_id = InscriptionId::from(txid); + let inscription_id = InscriptionId { txid, index: 0 }; server.mine_blocks(1); @@ -2677,7 +2677,7 @@ mod tests { server.mine_blocks(1); - let response = server.get(format!("/content/{}", InscriptionId::from(txid))); + let response = server.get(format!("/content/{}", InscriptionId { txid, index: 0 })); assert_eq!(response.status(), StatusCode::OK); assert_eq!( @@ -2792,7 +2792,7 @@ mod tests { witness: inscription("text/plain;charset=utf-8", "hello").to_witness(), ..Default::default() }); - let inscription = InscriptionId::from(txid); + let inscription = InscriptionId { txid, index: 0 }; bitcoin_rpc_server.mine_blocks(1); let server = TestServer::new_with_bitcoin_rpc_server_and_config( diff --git a/src/subcommand/wallet/inscribe.rs b/src/subcommand/wallet/inscribe.rs index 98c2328c2b..5312284147 100644 --- a/src/subcommand/wallet/inscribe.rs +++ b/src/subcommand/wallet/inscribe.rs @@ -113,7 +113,10 @@ impl Inscribe { Ok(Box::new(Output { commit: unsigned_commit_tx.txid(), reveal: reveal_tx.txid(), - inscription: reveal_tx.txid().into(), + inscription: InscriptionId { + txid: reveal_tx.txid(), + index: 0, + }, fees, })) } else { @@ -136,7 +139,10 @@ impl Inscribe { Ok(Box::new(Output { commit, reveal, - inscription: reveal.into(), + inscription: InscriptionId { + txid: reveal, + index: 0, + }, fees, })) } diff --git a/tests/json_api.rs b/tests/json_api.rs index 51586c568b..b554ccc6bd 100644 --- a/tests/json_api.rs +++ b/tests/json_api.rs @@ -42,7 +42,10 @@ fn get_sat_with_inscription_and_sat_index() { create_wallet(&rpc_server); let Inscribe { reveal, .. } = inscribe(&rpc_server); - let inscription_id = InscriptionId::from(reveal); + let inscription_id = InscriptionId { + txid: reveal, + index: 0, + }; let response = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]) .json_request(format!("/sat/{}", 50 * COIN_VALUE)); @@ -91,7 +94,10 @@ fn get_sat_with_inscription_on_common_sat_and_more_inscriptions() { .run_and_deserialize_output(); rpc_server.mine_blocks(1); - let inscription_id = InscriptionId::from(reveal); + let inscription_id = InscriptionId { + txid: reveal, + index: 0, + }; let response = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]) .json_request(format!("/sat/{}", 3 * 50 * COIN_VALUE + 1)); @@ -128,7 +134,10 @@ fn get_inscription() { create_wallet(&rpc_server); let Inscribe { reveal, .. } = inscribe(&rpc_server); - let inscription_id = InscriptionId::from(reveal); + let inscription_id = InscriptionId { + txid: reveal, + index: 0, + }; let response = TestServer::spawn_with_args(&rpc_server, &["--index-sats", "--enable-json-api"]) .json_request(format!("/inscription/{}", inscription_id)); @@ -194,7 +203,10 @@ fn create_210_inscriptions( .rpc_server(rpc_server) .run_and_deserialize_output(); rpc_server.mine_blocks(1); - blessed_inscriptions.push(InscriptionId::from(reveal)); + blessed_inscriptions.push(InscriptionId { + txid: reveal, + index: 0, + }); } rpc_server.mine_blocks(1); From 898c2cd0e9d6055da094e81a9c440bd0d88ae006 Mon Sep 17 00:00:00 2001 From: ordinally <11798624+veryordinally@users.noreply.github.com> Date: Sun, 27 Aug 2023 15:12:50 +0200 Subject: [PATCH 24/61] Add proper block inscriptions HTML (#2337) --- src/index/block_index.rs | 5 ++ src/subcommand/server.rs | 68 +++++++++++++---- src/templates.rs | 2 + src/templates/block.rs | 43 +++++++++-- src/templates/inscriptions_block.rs | 113 ++++++++++++++++++++++++++++ templates/block.html | 9 ++- templates/inscriptions-block.html | 26 +++++++ 7 files changed, 245 insertions(+), 21 deletions(-) create mode 100644 src/templates/inscriptions_block.rs create mode 100644 templates/inscriptions-block.html diff --git a/src/index/block_index.rs b/src/index/block_index.rs index c036080f22..0b4cd3381a 100644 --- a/src/index/block_index.rs +++ b/src/index/block_index.rs @@ -174,6 +174,11 @@ impl BlockIndex { if block_height >= index.block_count()? || block_height < self.first_inscription_height { return Ok(Vec::new()); } + + if self.lowest_blessed_by_block.is_empty() { + return Err(anyhow!("Block index not yet initialized")); + } + let lowest_cursed = self.lowest_cursed_by_block [usize::try_from(block_height.saturating_sub(self.first_inscription_height))?]; let lowest_blessed = self.lowest_blessed_by_block diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index adcceb0bf6..291e302721 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -8,10 +8,11 @@ use { crate::index::block_index::BlockIndex, crate::page_config::PageConfig, crate::templates::{ - BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionJson, InscriptionsHtml, - InscriptionsJson, OutputHtml, OutputJson, PageContent, PageHtml, PreviewAudioHtml, - PreviewImageHtml, PreviewModelHtml, PreviewPdfHtml, PreviewTextHtml, PreviewUnknownHtml, - PreviewVideoHtml, RangeHtml, RareTxt, SatHtml, SatJson, TransactionHtml, + BlockHtml, ClockSvg, HomeHtml, InputHtml, InscriptionHtml, InscriptionJson, + InscriptionsBlockHtml, InscriptionsHtml, InscriptionsJson, OutputHtml, OutputJson, PageContent, + PageHtml, PreviewAudioHtml, PreviewImageHtml, PreviewModelHtml, PreviewPdfHtml, + PreviewTextHtml, PreviewUnknownHtml, PreviewVideoHtml, RangeHtml, RareTxt, SatHtml, SatJson, + TransactionHtml, }, axum::{ body, @@ -196,7 +197,14 @@ impl Server { .route("/input/:block/:transaction/:input", get(Self::input)) .route("/inscription/:inscription_id", get(Self::inscription)) .route("/inscriptions", get(Self::inscriptions)) - .route("/inscriptions/block/:n", get(Self::inscriptions_in_block)) + .route( + "/inscriptions/block/:height", + get(Self::inscriptions_in_block), + ) + .route( + "/inscriptions/block/:height/:page_index", + get(Self::inscriptions_in_block_from_page), + ) .route("/inscriptions/:from", get(Self::inscriptions_from)) .route("/inscriptions/:from/:n", get(Self::inscriptions_from_n)) .route("/install.sh", get(Self::install_script)) @@ -566,6 +574,7 @@ impl Server { async fn block( Extension(page_config): Extension>, Extension(index): Extension>, + Extension(block_index_state): Extension>, Path(DeserializeFromStr(query)): Path>, ) -> ServerResult> { let (block, height) = match query { @@ -589,9 +598,17 @@ impl Server { } }; + let inscriptions = + index.get_inscriptions_in_block(&block_index_state.block_index.read().unwrap(), height)?; + Ok( - BlockHtml::new(block, Height(height), Self::index_height(&index)?) - .page(page_config, index.has_sat_index()?), + BlockHtml::new( + block, + Height(height), + Self::index_height(&index)?, + inscriptions, + ) + .page(page_config, index.has_sat_index()?), ) } @@ -1042,19 +1059,40 @@ impl Server { Extension(block_index_state): Extension>, Path(block_height): Path, accept_json: AcceptJson, + ) -> ServerResult { + Self::inscriptions_in_block_from_page( + Extension(page_config), + Extension(index), + Extension(block_index_state), + Path((block_height, 0)), + accept_json, + ) + .await + } + + async fn inscriptions_in_block_from_page( + Extension(page_config): Extension>, + Extension(index): Extension>, + Extension(block_index_state): Extension>, + Path((block_height, page_index)): Path<(u64, usize)>, + accept_json: AcceptJson, ) -> ServerResult { let inscriptions = index - .get_inscriptions_in_block(&block_index_state.block_index.read().unwrap(), block_height)?; + .get_inscriptions_in_block(&block_index_state.block_index.read().unwrap(), block_height) + .map_err(|e| ServerError::NotFound(format!("Failed to get inscriptions in block: {}", e)))?; + Ok(if accept_json.0 { Json(InscriptionsJson::new(inscriptions, None, None, None, None)).into_response() } else { - InscriptionsHtml { - inscriptions, - prev: None, - next: None, - } - .page(page_config, index.has_sat_index()?) - .into_response() + InscriptionsBlockHtml::new(block_height, index.block_count()?, inscriptions, page_index) + .map_err(|e| { + ServerError::NotFound(format!( + "Failed to get inscriptions in inscriptions block page: {}", + e + )) + })? + .page(page_config, index.has_sat_index()?) + .into_response() }) } diff --git a/src/templates.rs b/src/templates.rs index ab21051e3b..7df7f4442a 100644 --- a/src/templates.rs +++ b/src/templates.rs @@ -8,6 +8,7 @@ pub(crate) use { input::InputHtml, inscription::{InscriptionHtml, InscriptionJson}, inscriptions::{InscriptionsHtml, InscriptionsJson}, + inscriptions_block::InscriptionsBlockHtml, output::{OutputHtml, OutputJson}, page_config::PageConfig, preview::{ @@ -27,6 +28,7 @@ mod iframe; mod input; pub mod inscription; pub mod inscriptions; +mod inscriptions_block; pub mod output; mod preview; mod range; diff --git a/src/templates/block.rs b/src/templates/block.rs index d05988ff91..949529e9ad 100644 --- a/src/templates/block.rs +++ b/src/templates/block.rs @@ -7,16 +7,34 @@ pub(crate) struct BlockHtml { best_height: Height, block: Block, height: Height, + num_inscriptions: usize, + txid_to_inscription_ids: BTreeMap>, } impl BlockHtml { - pub(crate) fn new(block: Block, height: Height, best_height: Height) -> Self { + pub(crate) fn new( + block: Block, + height: Height, + best_height: Height, + inscriptions: Vec, + ) -> Self { Self { hash: block.header.block_hash(), target: BlockHash::from_raw_hash(Hash::from_byte_array(block.header.target().to_be_bytes())), block, height, best_height, + num_inscriptions: inscriptions.len(), + txid_to_inscription_ids: inscriptions.into_iter().fold( + BTreeMap::>::new(), + |mut acc, inscription_id| { + acc + .entry(inscription_id.txid) + .and_modify(|inscription_ids| inscription_ids.push(inscription_id)) + .or_insert(vec![inscription_id]); + acc + }, + ), } } } @@ -34,7 +52,12 @@ mod tests { #[test] fn html() { assert_regex_match!( - BlockHtml::new(Chain::Mainnet.genesis_block(), Height(0), Height(0)), + BlockHtml::new( + Chain::Mainnet.genesis_block(), + Height(0), + Height(0), + Vec::new() + ), "

Block 0

@@ -48,7 +71,7 @@ mod tests { prev next .* -

1 Transaction

+

1 Transaction and 0 Inscriptions

@@ -60,7 +83,12 @@ mod tests { #[test] fn next_active_when_not_last() { assert_regex_match!( - BlockHtml::new(Chain::Mainnet.genesis_block(), Height(0), Height(1)), + BlockHtml::new( + Chain::Mainnet.genesis_block(), + Height(0), + Height(1), + Vec::new() + ), r"

Block 0

.*prev\s*.*" ); } @@ -68,7 +96,12 @@ mod tests { #[test] fn prev_active_when_not_first() { assert_regex_match!( - BlockHtml::new(Chain::Mainnet.genesis_block(), Height(1), Height(1)), + BlockHtml::new( + Chain::Mainnet.genesis_block(), + Height(1), + Height(1), + Vec::new() + ), r"

Block 1

.*\s*next.*", ); } diff --git a/src/templates/inscriptions_block.rs b/src/templates/inscriptions_block.rs new file mode 100644 index 0000000000..a3fc6a2a51 --- /dev/null +++ b/src/templates/inscriptions_block.rs @@ -0,0 +1,113 @@ +use super::*; + +#[derive(Boilerplate)] +pub(crate) struct InscriptionsBlockHtml { + pub(crate) block: u64, + pub(crate) inscriptions: Vec, + pub(crate) prev_block: Option, + pub(crate) next_block: Option, + pub(crate) prev_page: Option, + pub(crate) next_page: Option, +} + +impl InscriptionsBlockHtml { + pub(crate) fn new( + block: u64, + current_blockheight: u64, + inscriptions: Vec, + page_index: usize, + ) -> Result { + let start = page_index * 100; + let end = start + 100; + let num_inscriptions = inscriptions.len(); + let inscriptions = inscriptions[start..end].to_vec(); + + Ok(Self { + block, + inscriptions, + prev_block: if block == 0 { None } else { Some(block - 1) }, + next_block: if block >= current_blockheight { + None + } else { + Some(block + 1) + }, + prev_page: if page_index > 0 { + Some(page_index - 1) + } else { + None + }, + next_page: if page_index * 100 <= num_inscriptions { + Some(page_index + 1) + } else { + None + }, + }) + } +} + +impl PageContent for InscriptionsBlockHtml { + fn title(&self) -> String { + format!("Inscriptions in Block {0}", self.block) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn without_prev_and_next() { + assert_regex_match!( + InscriptionsBlockHtml { + block: 21, + inscriptions: vec![inscription_id(1), inscription_id(2)], + prev_block: None, + next_block: None, + prev_page: None, + next_page: None, + }, + " +

Inscriptions in Block 21

+
+ + +
+ .* + prev + next + .* + " + .unindent() + ); + } + + #[test] + fn with_prev_and_next() { + assert_regex_match!( + InscriptionsBlockHtml { + block: 21, + inscriptions: vec![inscription_id(1), inscription_id(2)], + prev_block: Some(20), + next_block: Some(22), + next_page: Some(3), + prev_page: Some(1), + }, + " +

Inscriptions in Block 21

+
+ + +
+ .* + + • + + + • + + .* + " + .unindent() + ); + } +} diff --git a/templates/block.html b/templates/block.html index e5047f842d..352101aba8 100644 --- a/templates/block.html +++ b/templates/block.html @@ -21,10 +21,17 @@

Block {{ self.height }}

next %% } -

{{"Transaction".tally(self.block.txdata.len())}}

+

{{"Transaction".tally(self.block.txdata.len())}} and {{"Inscription".tally(self.num_inscriptions)}}

    %% for tx in &self.block.txdata { %% let txid = tx.txid();
  • {{txid}}
  • +%% if let Some(inscription_ids) = &self.txid_to_inscription_ids.get(&txid) { + +%% } %% }
diff --git a/templates/inscriptions-block.html b/templates/inscriptions-block.html new file mode 100644 index 0000000000..467ce3eba7 --- /dev/null +++ b/templates/inscriptions-block.html @@ -0,0 +1,26 @@ +

Inscriptions in Block {{ &self.block }}

+
+%% for id in &self.inscriptions { + {{ Iframe::thumbnail(*id) }} +%% } +
+
+%% if let Some(prev_block) = &self.prev_block { + +%% } +• +%% if let Some(prev_page) = &self.prev_page { + +%% } else { +prev +%% } +%% if let Some(next_page) = &self.next_page { + +%% } else { +next +%% } +• +%% if let Some(next_block) = &self.next_block { + +%% } +
From 9eb09aa5dfb6614e994980972894d319a484840c Mon Sep 17 00:00:00 2001 From: raph Date: Sun, 27 Aug 2023 19:20:34 +0200 Subject: [PATCH 25/61] Make homepage more interesting (#2374) --- src/index.rs | 14 -------- src/index/block_index.rs | 25 ++++++++++++++ src/subcommand/server.rs | 45 ++++++++++++++++-------- src/templates/home.rs | 53 ++++++++++++++++++++--------- src/templates/inscriptions_block.rs | 4 +-- static/index.css | 23 +++++++++++-- templates/home.html | 23 +++++++------ templates/inscriptions-block.html | 2 +- tests/server.rs | 43 ----------------------- 9 files changed, 128 insertions(+), 104 deletions(-) diff --git a/src/index.rs b/src/index.rs index 6f520cc5e8..799a8d2169 100644 --- a/src/index.rs +++ b/src/index.rs @@ -876,20 +876,6 @@ impl Index { Ok(result) } - pub(crate) fn get_homepage_inscriptions(&self) -> Result> { - Ok( - self - .database - .begin_read()? - .open_table(INSCRIPTION_NUMBER_TO_INSCRIPTION_ID)? - .iter()? - .rev() - .take(8) - .flat_map(|result| result.map(|(_number, id)| Entry::load(*id.value()))) - .collect(), - ) - } - pub(crate) fn get_latest_inscriptions_with_prev_and_next( &self, n: usize, diff --git a/src/index/block_index.rs b/src/index/block_index.rs index 0b4cd3381a..8ab3e033cb 100644 --- a/src/index/block_index.rs +++ b/src/index/block_index.rs @@ -203,4 +203,29 @@ impl BlockIndex { Ok(inscriptions) } + + pub(crate) fn get_highest_paying_inscriptions_in_block( + &self, + index: &Index, + block_height: u64, + n: usize, + ) -> Result> { + let inscription_ids = self.get_inscriptions_in_block(index, block_height)?; + + let mut inscription_to_fee: Vec<(InscriptionId, u64)> = Vec::new(); + for id in inscription_ids { + inscription_to_fee.push((id, index.get_inscription_entry(id)?.unwrap().fee)); + } + + inscription_to_fee.sort_by_key(|(_, fee)| *fee); + + Ok( + inscription_to_fee + .iter() + .map(|(id, _)| *id) + .rev() + .take(n) + .collect(), + ) + } } diff --git a/src/subcommand/server.rs b/src/subcommand/server.rs index 291e302721..90f2ef1963 100644 --- a/src/subcommand/server.rs +++ b/src/subcommand/server.rs @@ -560,15 +560,24 @@ impl Server { async fn home( Extension(page_config): Extension>, Extension(index): Extension>, + Extension(block_index_state): Extension>, ) -> ServerResult> { - Ok( - HomeHtml::new(index.blocks(100)?, index.get_homepage_inscriptions()?) - .page(page_config, index.has_sat_index()?), - ) + let blocks = index.blocks(100)?; + let mut featured_blocks = BTreeMap::new(); + for (height, hash) in blocks.iter().take(5) { + let inscriptions = block_index_state + .block_index + .read() + .unwrap() + .get_highest_paying_inscriptions_in_block(&index, *height, 8)?; + featured_blocks.insert(*hash, inscriptions); + } + + Ok(HomeHtml::new(blocks, featured_blocks).page(page_config, index.has_sat_index()?)) } async fn install_script() -> Redirect { - Redirect::to("https://raw.githubusercontent.com/casey/ord/master/install.sh") + Redirect::to("https://raw.githubusercontent.com/ordinals/ord/master/install.sh") } async fn block( @@ -1545,7 +1554,7 @@ mod tests { fn install_sh_redirects_to_github() { TestServer::new().assert_redirect( "/install.sh", - "https://raw.githubusercontent.com/casey/ord/master/install.sh", + "https://raw.githubusercontent.com/ordinals/ord/master/install.sh", ); } @@ -1960,15 +1969,21 @@ mod tests { test_server.mine_blocks(1); test_server.assert_response_regex( - "/", - StatusCode::OK, - ".*Ordinals.* -

Latest Blocks

-
    -
  1. [[:xdigit:]]{64}
  2. -
  3. 000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f
  4. + "/", + StatusCode::OK, + ".*Ordinals.* +
    +

    Block 1

    +
    +
    +
    +
    +

    Block 0

    +
    +
    +
.*", - ); + ); } #[test] @@ -1989,7 +2004,7 @@ mod tests { test_server.assert_response_regex( "/", StatusCode::OK, - ".*
    \n(
  1. [[:xdigit:]]{64}
  2. \n){100}
.*" + ".*
    \n(
  1. [[:xdigit:]]{64}
  2. \n){95}
.*" ); } diff --git a/src/templates/home.rs b/src/templates/home.rs index 8cfa012b0b..137415cde5 100644 --- a/src/templates/home.rs +++ b/src/templates/home.rs @@ -4,11 +4,14 @@ use super::*; pub(crate) struct HomeHtml { last: u64, blocks: Vec, - inscriptions: Vec, + featured_blocks: BTreeMap>, } impl HomeHtml { - pub(crate) fn new(blocks: Vec<(u64, BlockHash)>, inscriptions: Vec) -> Self { + pub(crate) fn new( + blocks: Vec<(u64, BlockHash)>, + featured_blocks: BTreeMap>, + ) -> Self { Self { last: blocks .get(0) @@ -16,7 +19,7 @@ impl HomeHtml { .cloned() .unwrap_or(0), blocks: blocks.into_iter().map(|(_, hash)| hash).collect(), - inscriptions, + featured_blocks, } } } @@ -33,9 +36,23 @@ mod tests { #[test] fn html() { + let mut feature_blocks = BTreeMap::new(); + feature_blocks.insert( + "2222222222222222222222222222222222222222222222222222222222222222" + .parse() + .unwrap(), + vec![inscription_id(1), inscription_id(2)], + ); + assert_regex_match!( &HomeHtml::new( vec![ + ( + 1260002, + "2222222222222222222222222222222222222222222222222222222222222222" + .parse() + .unwrap() + ), ( 1260001, "1111111111111111111111111111111111111111111111111111111111111111" @@ -49,21 +66,23 @@ mod tests { .unwrap() ) ], - vec![inscription_id(1), inscription_id(2)], + feature_blocks, ) - .to_string(), - "

Latest Inscriptions

-
- - -
- -

Latest Blocks

-
    -
  1. 1{64}
  2. -
  3. 0{64}
  4. -
-", + .to_string() + .unindent(), + "
+

Block 1260002

+
+ + +
+
+
    +
  1. 1{64}
  2. +
  3. 0{64}
  4. +
+ " + .unindent(), ); } } diff --git a/src/templates/inscriptions_block.rs b/src/templates/inscriptions_block.rs index a3fc6a2a51..9619934535 100644 --- a/src/templates/inscriptions_block.rs +++ b/src/templates/inscriptions_block.rs @@ -67,7 +67,7 @@ mod tests { next_page: None, }, " -

Inscriptions in Block 21

+

Inscriptions in Block 21

@@ -93,7 +93,7 @@ mod tests { prev_page: Some(1), }, " -

Inscriptions in Block 21

+

Inscriptions in Block 21

diff --git a/static/index.css b/static/index.css index a8faeb6616..d80f739a8e 100644 --- a/static/index.css +++ b/static/index.css @@ -101,13 +101,32 @@ ol, ul { } } -.blocks { +.block-list { font-family: monospace, monospace; list-style-position: inside; - padding-inline-start: 1rem; white-space: nowrap; } +.block { + background-color: var(--light-bg); + box-shadow: rgba(0, 0, 0, 0.16) 0px 1px 4px, var(--light-bg) 0px 0px 0px 3px; + margin: 3%; + padding: 0.5%; + text-align: center; +} + +.block a { + color: var(--light-fg); + margin-top: 1%; +} + +.block h2 { + margin: 1%; +} + +.light-fg a { + color: var(--light-fg); +} .center { text-align: center; } diff --git a/templates/home.html b/templates/home.html index 24a503a6d7..328e2ffa14 100644 --- a/templates/home.html +++ b/templates/home.html @@ -1,15 +1,18 @@ -%% if !&self.inscriptions.is_empty() { -

Latest Inscriptions

-
-%% for id in &self.inscriptions { - {{Iframe::thumbnail(*id)}} +%% for (i, hash) in self.blocks.iter().enumerate() { +%% if let Some(inscription_ids) = &self.featured_blocks.get(hash) { +
+

Block {{ self.last - i as u64 }}

+
+%% for id in *inscription_ids { + {{ Iframe::thumbnail(*id) }} %% } +
- +%% } else { +%% if i == self.featured_blocks.len() { +
    +%% } +
  1. {{ hash }}
  2. %% } -

    Latest Blocks

    -
      -%% for hash in &self.blocks { -
    1. {{hash}}
    2. %% }
    diff --git a/templates/inscriptions-block.html b/templates/inscriptions-block.html index 467ce3eba7..d087d147b0 100644 --- a/templates/inscriptions-block.html +++ b/templates/inscriptions-block.html @@ -1,4 +1,4 @@ -

    Inscriptions in Block {{ &self.block }}

    +

    Inscriptions in Block {{ &self.block }}

    %% for id in &self.inscriptions { {{ Iframe::thumbnail(*id) }} diff --git a/tests/server.rs b/tests/server.rs index c893cc901b..ff69555cae 100644 --- a/tests/server.rs +++ b/tests/server.rs @@ -195,49 +195,6 @@ fn inscription_content() { assert_eq!(response.bytes().unwrap(), "FOO"); } -#[test] -fn home_page_includes_latest_inscriptions() { - let rpc_server = test_bitcoincore_rpc::spawn(); - create_wallet(&rpc_server); - - let Inscribe { inscription, .. } = inscribe(&rpc_server); - - TestServer::spawn_with_args(&rpc_server, &[]).assert_response_regex( - "/", - format!( - ".*

    Latest Inscriptions

    -
    - +
    +
    +
    +
    id
    +
    1{64}i1
    +
    parent
    +
    2{64}i2
    +
    preview
    +
    link
    +
    content
    +
    link
    +
    content length
    +
    10 bytes
    +
    content type
    +
    text/plain;charset=utf-8
    +
    timestamp
    +
    +
    genesis height
    +
    0
    +
    genesis fee
    +
    1
    +
    genesis transaction
    +
    1{64}
    +
    location
    +
    1{64}:1:0
    +
    output
    +
    1{64}:1
    +
    offset
    +
    0
    +
    + " + .unindent() + ); + } + + #[test] + fn with_children() { + assert_regex_match!( + InscriptionHtml { + children: vec![inscription_id(2), inscription_id(3)], + parent: None, + chain: Chain::Mainnet, + genesis_fee: 1, + genesis_height: 0, + inscription: inscription("text/plain;charset=utf-8", "HELLOWORLD"), + inscription_id: inscription_id(1), + next: None, + number: 1, + output: None, + previous: None, + sat: None, + satpoint: satpoint(1, 0), + timestamp: timestamp(0), + }, + " +

    Inscription 1

    +
    +
    + +
    +
    +
    +
    id
    +
    1{64}i1
    +
    preview
    +
    link
    +
    content
    +
    link
    +
    content length
    +
    10 bytes
    +
    content type
    +
    text/plain;charset=utf-8
    +
    timestamp
    +
    +
    genesis height
    +
    0
    +
    genesis fee
    +
    1
    +
    genesis transaction
    +
    1{64}
    +
    location
    +
    1{64}:1:0
    +
    output
    +
    1{64}:1
    +
    offset
    +
    0
    +
    children
    +
    +
    + + +
    +
    +
    + " + .unindent() + ); + } } diff --git a/templates/inscription.html b/templates/inscription.html index 11b7542ac5..1b272941c5 100644 --- a/templates/inscription.html +++ b/templates/inscription.html @@ -19,6 +19,10 @@

    Inscription {{ self.number }} (unstable)

    id
    {{ self.inscription_id }}
    +%% if let Some(parent) = &self.parent { +
    parent
    +
    {{ parent }}
    +%% } %% if let Some(output) = &self.output { %% if let Ok(address) = self.chain.address_from_script(&output.script_pubkey ) {
    address
    @@ -63,4 +67,14 @@

    Inscription {{ self.number }} (unstable)

    %% }
    offset
    {{ self.satpoint.offset }}
    +%% if !self.children.is_empty() { +
    children
    +
    +
    +%% for id in &self.children { + {{Iframe::thumbnail(*id)}} +%% } +
    +
    +%% }
    diff --git a/test-bitcoincore-rpc/src/lib.rs b/test-bitcoincore-rpc/src/lib.rs index 7a7e314a17..ab4f35d3ff 100644 --- a/test-bitcoincore-rpc/src/lib.rs +++ b/test-bitcoincore-rpc/src/lib.rs @@ -119,10 +119,9 @@ pub fn spawn() -> Handle { #[derive(Clone)] pub struct TransactionTemplate<'a> { pub fee: u64, - pub inputs: &'a [(usize, usize, usize)], + pub inputs: &'a [(usize, usize, usize, Witness)], pub output_values: &'a [u64], pub outputs: usize, - pub witness: Witness, } #[derive(Clone, Debug, PartialEq)] @@ -154,7 +153,6 @@ impl<'a> Default for TransactionTemplate<'a> { inputs: &[], output_values: &[], outputs: 1, - witness: Witness::default(), } } } diff --git a/test-bitcoincore-rpc/src/state.rs b/test-bitcoincore-rpc/src/state.rs index 0a05039aa5..9d1a6c8d4d 100644 --- a/test-bitcoincore-rpc/src/state.rs +++ b/test-bitcoincore-rpc/src/state.rs @@ -131,14 +131,14 @@ impl State { pub(crate) fn broadcast_tx(&mut self, template: TransactionTemplate) -> Txid { let mut total_value = 0; let mut input = Vec::new(); - for (height, tx, vout) in template.inputs.iter() { + for (height, tx, vout, witness) in template.inputs.iter() { let tx = &self.blocks.get(&self.hashes[*height]).unwrap().txdata[*tx]; total_value += tx.output[*vout].value; input.push(TxIn { previous_output: OutPoint::new(tx.txid(), *vout as u32), script_sig: ScriptBuf::new(), sequence: Sequence::MAX, - witness: template.witness.clone(), + witness: witness.clone(), }); } diff --git a/tests/decode.rs b/tests/decode.rs index 1d4ac1c0da..a11adb751f 100644 --- a/tests/decode.rs +++ b/tests/decode.rs @@ -53,7 +53,8 @@ fn from_file() { inscriptions: vec![Inscription { body: Some(vec![0, 1, 2, 3]), content_type: Some(b"text/plain;charset=utf-8".to_vec()), - unrecognized_even_field: false + unrecognized_even_field: false, + parent: None, }], } ); @@ -69,7 +70,8 @@ fn from_stdin() { inscriptions: vec![Inscription { body: Some(vec![0, 1, 2, 3]), content_type: Some(b"text/plain;charset=utf-8".to_vec()), - unrecognized_even_field: false + unrecognized_even_field: false, + parent: None, }], } ); diff --git a/tests/json_api.rs b/tests/json_api.rs index b554ccc6bd..9fab0d6381 100644 --- a/tests/json_api.rs +++ b/tests/json_api.rs @@ -152,6 +152,8 @@ fn get_inscription() { pretty_assert_eq!( inscription_json, InscriptionJson { + parent: None, + children: Vec::new(), inscription_id, number: 0, genesis_height: 2, @@ -184,8 +186,11 @@ fn create_210_inscriptions( rpc_server.mine_blocks(1); let txid = rpc_server.broadcast_tx(TransactionTemplate { - inputs: &[(i * 3 + 1, 0, 0), (i * 3 + 2, 0, 0), (i * 3 + 3, 0, 0)], - witness: witness.clone(), + inputs: &[ + (i * 3 + 1, 0, 0, witness.clone()), + (i * 3 + 2, 0, 0, witness.clone()), + (i * 3 + 3, 0, 0, witness.clone()), + ], ..Default::default() }); @@ -316,9 +321,14 @@ fn get_inscriptions_in_block() { create_wallet(&rpc_server); rpc_server.mine_blocks(10); + let envelope = envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]); + let txid = rpc_server.broadcast_tx(TransactionTemplate { - inputs: &[(1, 0, 0), (2, 0, 0), (3, 0, 0)], - witness: envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]), + inputs: &[ + (1, 0, 0, envelope.clone()), + (2, 0, 0, envelope.clone()), + (3, 0, 0, envelope.clone()), + ], ..Default::default() }); rpc_server.mine_blocks(1); @@ -362,9 +372,14 @@ fn get_output() { create_wallet(&rpc_server); rpc_server.mine_blocks(3); + let envelope = envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]); + let txid = rpc_server.broadcast_tx(TransactionTemplate { - inputs: &[(1, 0, 0), (2, 0, 0), (3, 0, 0)], - witness: envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]), + inputs: &[ + (1, 0, 0, envelope.clone()), + (2, 0, 0, envelope.clone()), + (3, 0, 0, envelope.clone()), + ], ..Default::default() }); rpc_server.mine_blocks(1); diff --git a/tests/wallet/send.rs b/tests/wallet/send.rs index 66b2d1b426..e485dd3a3c 100644 --- a/tests/wallet/send.rs +++ b/tests/wallet/send.rs @@ -215,10 +215,15 @@ fn splitting_merged_inscriptions_is_possible() { create_wallet(&rpc_server); rpc_server.mine_blocks(3); + let inscription = envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]); + // merging 3 inscriptions into one utxo let reveal_txid = rpc_server.broadcast_tx(TransactionTemplate { - inputs: &[(1, 0, 0), (2, 0, 0), (3, 0, 0)], - witness: envelope(&[b"ord", &[1], b"text/plain;charset=utf-8", &[], b"bar"]), + inputs: &[ + (1, 0, 0, inscription.clone()), + (2, 0, 0, inscription.clone()), + (3, 0, 0, inscription.clone()), + ], outputs: 1, ..Default::default() }); From e6ceb9709240cd02cb48bdc1b8c025ae53136403 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Mon, 4 Sep 2023 11:31:54 -0700 Subject: [PATCH 50/61] Add provenance spec (#2278) --- docs/src/inscriptions.md | 13 ++--- docs/src/inscriptions/provenance.md | 82 +++++++++++++++++++++++++++++ docs/src/inscriptions/recursion.md | 7 +-- 3 files changed, 93 insertions(+), 9 deletions(-) create mode 100644 docs/src/inscriptions/provenance.md diff --git a/docs/src/inscriptions.md b/docs/src/inscriptions.md index 056918dd2c..dad8216aa9 100644 --- a/docs/src/inscriptions.md +++ b/docs/src/inscriptions.md @@ -53,14 +53,15 @@ OP_ENDIF First the string `ord` is pushed, to disambiguate inscriptions from other uses of envelopes. -`OP_PUSH 1` indicates that the next push contains the content type, and `OP_PUSH -0` indicates that subsequent data pushes contain the content itself. Multiple data -pushes must be used for large inscriptions, as one of taproot's few -restrictions is that individual data pushes may not be larger than 520 bytes. +`OP_PUSH 1` indicates that the next push contains the content type, and +`OP_PUSH 0`indicates that subsequent data pushes contain the content itself. +Multiple data pushes must be used for large inscriptions, as one of taproot's +few restrictions is that individual data pushes may not be larger than 520 +bytes. The inscription content is contained within the input of a reveal transaction, -and the inscription is made on the first sat of its input. This sat can -then be tracked using the familiar rules of ordinal theory, allowing it to be +and the inscription is made on the first sat of its input. This sat can then be +tracked using the familiar rules of ordinal theory, allowing it to be transferred, bought, sold, lost to fees, and recovered. Content diff --git a/docs/src/inscriptions/provenance.md b/docs/src/inscriptions/provenance.md new file mode 100644 index 0000000000..22d3567c19 --- /dev/null +++ b/docs/src/inscriptions/provenance.md @@ -0,0 +1,82 @@ +Provenance +========== + +The owner of an inscription can create child inscriptions, trustlessly +establishing the provenance of those children on-chain as having been created +by the owner of the parent inscription. This can be used for collections, with +the children of a parent inscription being members of the same collection. + +Children can themselves have children, allowing for complex hierarchies. For +example, an artist might create an inscription representing themselves, with +sub inscriptions representing collections that they create, with the children +of those sub inscriptions being items in those collections. + +### Specification + +To create a child inscription C with parent inscription P: + +- Create an inscribe transaction T as usual for C. +- Spend the parent P in one of the inputs of T. +- Include tag `3`, i.e. `OP_PUSH 3`, in C, with the value of the serialized + binary inscription ID of P, serialized as the 32-byte `TXID`, followed by the + four-byte little-endian `INDEX`, with trailing zeroes omitted. + +_NB_ The bytes of a bitcoin transaction ID are reversed in their text +representation, so the serialized transaction ID will be in the opposite order. + +### Example + +An example of a child inscription of +`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi0`: + +``` +OP_FALSE +OP_IF + OP_PUSH "ord" + OP_PUSH 1 + OP_PUSH "text/plain;charset=utf-8" + OP_PUSH 3 + OP_PUSH 0x1f1e1d1c1b1a191817161514131211100f0e0d0c0b0a09080706050403020100 + OP_PUSH 0 + OP_PUSH "Hello, world!" +OP_ENDIF +``` + +Note that the value of tag `3` is binary, not hex, and that for the child +inscription to be recognized as a child, +`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi0` must be +spent as one of the inputs of the inscribe transaction. + +Example encoding of inscription ID +`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi255`: + +``` +OP_FALSE +OP_IF + … + OP_PUSH 3 + OP_PUSH 0x1f1e1d1c1b1a191817161514131211100f0e0d0c0b0a09080706050403020100ff + … +OP_ENDIF +``` + +And of inscription ID `000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi256`: + +``` +OP_FALSE +OP_IF + … + OP_PUSH 3 + OP_PUSH 0x1f1e1d1c1b1a191817161514131211100f0e0d0c0b0a090807060504030201000001 + … +OP_ENDIF +``` + +### Notes + +The tag `3` is used because it is the first available odd tag. Unrecognized odd +tags do not make an inscription unbound, so child inscriptions would be +recognized and tracked by old versions of `ord`. + +A collection can be closed by burning the collection's parent inscription, +which guarantees that no more items in the collection can be issued. diff --git a/docs/src/inscriptions/recursion.md b/docs/src/inscriptions/recursion.md index 43a2eabdb7..d11e1bef99 100644 --- a/docs/src/inscriptions/recursion.md +++ b/docs/src/inscriptions/recursion.md @@ -1,9 +1,10 @@ Recursion ========= -An important exception to [sandboxing](../inscriptions.md#sandboxing) is recursion: access to `ord`'s `/content` -endpoint is permitted, allowing inscriptions to access the content of other -inscriptions by requesting `/content/`. +An important exception to [sandboxing](../inscriptions.md#sandboxing) is +recursion: access to `ord`'s `/content` endpoint is permitted, allowing +inscriptions to access the content of other inscriptions by requesting +`/content/`. This has a number of interesting use-cases: From 393848fad1b102286a13a1ff096c29de92e74c6f Mon Sep 17 00:00:00 2001 From: "Dr.JingLee" <95461562+DrJingLee@users.noreply.github.com> Date: Tue, 5 Sep 2023 02:40:47 +0800 Subject: [PATCH 51/61] Fix Chinese translation typos and format errors (#2419) --- docs/po/zh.po | 47 +++++++++++++++++++++++------------------------ 1 file changed, 23 insertions(+), 24 deletions(-) diff --git a/docs/po/zh.po b/docs/po/zh.po index 92f437b25f..291142fb3f 100644 --- a/docs/po/zh.po +++ b/docs/po/zh.po @@ -1,9 +1,9 @@ msgid "" msgstr "" -"Project-Id-Version: Ordinals Theory Handbook\n" +"Project-Id-Version: 序数理论手册\n" "POT-Creation-Date: \n" "PO-Revision-Date: 2023-08-30 10:03+0800\n" -"Last-Translator: Michael \n" +"Last-Translator: Dr.JingLee \n" "Language-Team: Chinese\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" @@ -123,7 +123,7 @@ msgstr "" msgid "" "Ordinal theory imbues satoshis with numismatic value, allowing them to be " "collected and traded as curios." -msgstr "序数理论赋予 聪 以收藏价值,使它们可以作为古玩被收藏和交易。" +msgstr "序数理论赋予聪以收藏价值,使它们可以作为古玩被收藏和交易。" #: src/introduction.md:17 msgid "" @@ -143,8 +143,8 @@ msgid "" "DNS. For now though, such use-cases are speculative, and exist only in the " "minds of fringe ordinal theorists." msgstr "" -"其他非常规的应用也是可能的:链下彩色硬币off-chain colored-coins,、具有密钥轮换的公钥基础设施" -"DNS 的去中心化替代品 decentralized replacement for the DNS。 " +"其他非常规的应用也是可能的:链下染色硬币,具有密钥轮换的公钥基础设施" +"DNS 的去中心化替代品等等。 " "不过就目前而言,这样的应用是推测性的,只存在于非主流的序数理论家的脑海中。" #: src/introduction.md:27 @@ -153,7 +153,7 @@ msgstr "有关序数理论的更多详细信息,请参阅 [概述](overview.md #: src/introduction.md:29 msgid "For more details on inscriptions, see [inscriptions](inscriptions.md)." -msgstr "有关铭文的更多详细信息,请参阅 (inscriptions.md))." +msgstr "有关铭文的更多详细信息,请参阅[铭文](inscriptions.md)." #: src/introduction.md:31 msgid "" @@ -213,7 +213,7 @@ msgid "" "[Ordinals Workshop with Rodarmor](https://www.youtube.com/watch?" "v=MC_haVa6N3I)" msgstr "" -"[Rodarmor的序数工作坊 ](https://www.youtube.com/watch?" +"[CaseyRodarmor的序数理论工作坊 ](https://www.youtube.com/watch?" "v=MC_haVa6N3I)" #: src/introduction.md:51 @@ -241,8 +241,8 @@ msgid "" msgstr "" "序数是一种比特币的编号方案,允许跟踪和转移单个聪。这些数字被称作[序号](https://ordinals.com)。" "比特币是按照它们被挖掘的顺序编号的,并从交易输入转移到交易输出(遵循先进先出原则)。" -"编号方案和传输方案都依赖于*顺序*,编号方案依赖于比特币被挖掘的*顺序*,而传输方案依赖于交易输入和输出的*顺序*。" -"因此得名,*序数(Ordinals)*。" +"编号方案和传输方案都依赖于_顺序_,编号方案依赖于比特币被挖掘的_顺序_,而传输方案依赖于交易输入和输出的_顺序_。" +"因此得名,_序数(Ordinals_。" #: src/overview.md:13 msgid "" @@ -351,8 +351,8 @@ msgid "" "Bitcoin has periodic events, some frequent, some more uncommon, and these " "naturally lend themselves to a system of rarity. These periodic events are:" msgstr "" -“比特币有周期性的事件,有些频繁,有些不常见,这些事件自然而然地形成了一个稀有系统。“ -”这些周期性事件是:” +"比特币有周期性的事件,有些频繁,有些不常见,这些事件自然而然地形成了一个稀有度系统。" +"这些周期性事件是:" #: src/overview.md:62 msgid "" @@ -391,7 +391,7 @@ msgstr "" #: src/overview.md:77 msgid "This gives us the following rarity levels:" msgstr "" -“这给了我们以下稀缺度等级:” +"这给了我们以下稀缺度等级:" #: src/overview.md:79 msgid "`common`: Any sat that is not the first sat of its block" @@ -423,7 +423,7 @@ msgid "" "ordinal number in a way that makes the rarity of a satoshi easy to see at a " "glance:" msgstr "" -"这给我们带来了度数表示法,它以一种使 satoshi 的稀有性一目了然的方式明确地表示一个序数: " +"这给我们带来了度数表示法,它以一种使聪的稀有性一目了然的方式明确地表示一个序数: " #: src/overview.md:89 msgid "" @@ -693,8 +693,8 @@ msgid "" "less is out there, or will be out there, someday." msgstr "" msgid "" -"举个例子, 1905530482684727°'的名字是 \"iaiufjszmoba\".最后一个被挖掘的聪的名字会是\"a\" " -" 每个 10 个或更少字符的组合都存在,或者总有一天会存在。" +"举个例子, 1905530482684727°'的名字是 \"iaiufjszmoba\".最后一个被挖掘的聪的名字会是\"a\"。10个字母" +"或更少字符的组合都会存在,或者总有一天会存在。" #: src/overview.md:208 msgid "Exotics" @@ -751,7 +751,7 @@ msgid "" "N29oF4iwCgX3lacrvaG9Kjko)" msgstr "" "致力于编目和收集早期 NFT 的活跃考古学家社区如雨后春笋般涌现 " -" [Chainleft对历史NFT的精彩总结] (https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-N29oF4iwCgX3lacrvaG9Kjko)" +"[Chainleft对历史NFT的精彩总结](https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-N29oF4iwCgX3lacrvaG9Kjko)" #: src/overview.md:238 msgid "" @@ -759,8 +759,8 @@ msgid "" "first ERC-721 contract, [SU SQUARES](https://tenthousandsu.com/), was " "deployed on Ethereum." msgstr "" -"普遍接受的古老NFT 的截止日期是 2018 年 3 月 19 日,即 " -"第一个 ERC-721 合约, [SU SQUARES](https://tenthousandsu.com/), 被部署在以太坊上的时间 " +"普遍接受的古老NFT 的截止日期是 2018年3月19日,即 " +"第一个 ERC-721 合约,[SU SQUARES](https://tenthousandsu.com/), 被部署在以太坊上的时间 " #: src/overview.md:242 msgid "" @@ -988,7 +988,7 @@ msgid "" "they are included, and can be combined with any other locking script." msgstr "" "铭文内容使用未执行条件中的数据推送进行序列化,称为“信封”。" -" 信封由 OP_FALSE OP_IF … OP_ENDIF 组成,包装任意数量的数据推送。" +"信封由 OP_FALSE OP_IF … OP_ENDIF 组成,包装任意数量的数据推送。" "因为信封实际上是空操作,所以它们不会改变包含它们的脚本的语义," "并且可以与任何其他锁定脚本结合使用。" @@ -1142,7 +1142,6 @@ msgstr "" "铭文可以位于同一输入中的不同输入中,可以是同一个输入或两者的组合。" "在任何情况下,顺序都是明确的,因为解析器将连续检查输入并查找所有铭文`信封`" - #: src/inscriptions.md:111 msgid "Input" msgstr "" @@ -1193,7 +1192,7 @@ msgstr "" #: src/inscriptions.md:119 msgid "Sandboxing" -msgstr "沙盒" +msgstr "沙盒化" #: src/inscriptions.md:122 msgid "" @@ -1218,7 +1217,7 @@ msgid "" "inscriptions to access the content of other inscriptions by requesting `/" "content/`." msgstr "" -"[沙盒](../inscriptions.md#sandboxing)的一个重要例外是递归:" +"[沙盒化](../inscriptions.md#sandboxing)的一个重要例外是递归:" "访问“ord”的“/content”允许端点,允许铭文访问" "其他端点的内容通过请求 `/content/` 来获取铭文。" @@ -1549,7 +1548,7 @@ msgid "" "Ethereum NFTs _can_ be immutable, but many are not, and can be changed or " "deleted by the NFT contract owner." msgstr "" -"以太坊NFTs _可以_ 是不可更改的,但很多都不是,且是可以由 NFT 合约所有者更改或删除。" +"以太坊NFTs_可以_是不可更改的,但很多都不是,且是可以由 NFT 合约所有者更改或删除。" #: src/faq.md:121 msgid "" @@ -3698,7 +3697,7 @@ msgid "" "not always) the inscription will be inscribed on the first satoshi in the " "UTXO." msgstr "" -"如我们之前所述,铭文是刻在sats上的,sats存储在UTXO中。" +"如我们之前所述,铭文是刻在聪上的,sats存储在UTXO中。" "UTXO是具有某个特定数量的satoshi(输出值)的satoshi集合。通常(但不总是)铭文会被刻在UTXO中" "的第一个satoshi上。" From 44f8b2a14605fdee6dd55da44ce28d6400b28fe2 Mon Sep 17 00:00:00 2001 From: Eloc <42568538+elocremarc@users.noreply.github.com> Date: Wed, 6 Sep 2023 11:58:16 -0700 Subject: [PATCH 52/61] Prevent search when query field is empty (#2425) --- static/index.js | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/static/index.js b/static/index.js index 2c6632f9d4..e3153179b7 100644 --- a/static/index.js +++ b/static/index.js @@ -23,3 +23,12 @@ window.addEventListener('keydown', e => { return; } }); + +const search = document.querySelector('form[action="/search"]'); +const query = search.querySelector('input[name="query"]'); + +search.addEventListener('submit', (e) => { + if (!query.value) { + e.preventDefault(); + } +}); From 04bc261cfc605da72d82632a2831cceb1bad01dc Mon Sep 17 00:00:00 2001 From: raph Date: Wed, 6 Sep 2023 22:49:39 +0200 Subject: [PATCH 53/61] Inscribe with parent (#2388) --- Cargo.lock | 8 +- Cargo.toml | 2 +- src/fee_rate.rs | 2 +- src/inscription.rs | 10 +- src/inscription_id.rs | 1 - src/subcommand/preview.rs | 1 + src/subcommand/wallet/inscribe.rs | 458 ++++++++++++++----- src/subcommand/wallet/transaction_builder.rs | 15 +- src/test.rs | 1 + test-bitcoincore-rpc/src/api.rs | 2 +- test-bitcoincore-rpc/src/lib.rs | 3 +- test-bitcoincore-rpc/src/server.rs | 7 +- tests/wallet/inscribe.rs | 100 +++- 13 files changed, 459 insertions(+), 151 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 2147614369..4d6ed8ea87 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2138,9 +2138,9 @@ dependencies = [ [[package]] name = "ord-bitcoincore-rpc" -version = "0.17.0" +version = "0.17.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ece3c1158d3391127ddd7615868792a602745cf4f9fbea259c4449c9ed89814a" +checksum = "3d57a4297d466506cde088e020b33819f9a496d50272b14d7890e6fde5595a3e" dependencies = [ "bitcoin-private", "jsonrpc", @@ -2152,9 +2152,9 @@ dependencies = [ [[package]] name = "ord-bitcoincore-rpc-json" -version = "0.17.0" +version = "0.17.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "45f46eb509cc95759461d3cb483ba51b2adbea0e31747cb1e3f388e71e13a321" +checksum = "5bb35088f918c775bc27fa79e372d034892b216fb7900aeedd6e06654879ad33" dependencies = [ "bitcoin", "bitcoin-private", diff --git a/Cargo.toml b/Cargo.toml index c08b3258d3..f2fd86a6e0 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -41,7 +41,7 @@ mime = "0.3.16" mime_guess = "2.0.4" miniscript = "10.0.0" mp4 = "0.14.0" -ord-bitcoincore-rpc = "0.17.0" +ord-bitcoincore-rpc = "0.17.1" redb = "1.0.5" regex = "1.6.0" rss = "2.0.1" diff --git a/src/fee_rate.rs b/src/fee_rate.rs index 9a0290040d..9f85ac5133 100644 --- a/src/fee_rate.rs +++ b/src/fee_rate.rs @@ -23,7 +23,7 @@ impl TryFrom for FeeRate { } impl FeeRate { - pub(crate) fn fee(&self, vsize: usize) -> Amount { + pub fn fee(&self, vsize: usize) -> Amount { #[allow(clippy::cast_possible_truncation)] #[allow(clippy::cast_sign_loss)] Amount::from_sat((self.0 * vsize as f64).round() as u64) diff --git a/src/inscription.rs b/src/inscription.rs index 80252c9294..79147ac7cd 100644 --- a/src/inscription.rs +++ b/src/inscription.rs @@ -73,7 +73,11 @@ impl Inscription { result } - pub(crate) fn from_file(chain: Chain, path: impl AsRef) -> Result { + pub(crate) fn from_file( + chain: Chain, + path: impl AsRef, + parent: Option, + ) -> Result { let path = path.as_ref(); let body = fs::read(path).with_context(|| format!("io error reading {}", path.display()))?; @@ -90,7 +94,7 @@ impl Inscription { Ok(Self { body: Some(body), content_type: Some(content_type.into()), - parent: None, + parent: parent.map(|id| id.parent_value()), unrecognized_even_field: false, }) } @@ -830,8 +834,8 @@ mod tests { Ok(vec![Inscription { content_type: None, body: None, - unrecognized_even_field: true, parent: None, + unrecognized_even_field: true, }]), ); } diff --git a/src/inscription_id.rs b/src/inscription_id.rs index 29a87b2945..7ea91939a8 100644 --- a/src/inscription_id.rs +++ b/src/inscription_id.rs @@ -7,7 +7,6 @@ pub struct InscriptionId { } impl InscriptionId { - #[cfg(test)] pub(crate) fn parent_value(self) -> Vec { let index = self.index.to_le_bytes(); let mut index_slice = index.as_slice(); diff --git a/src/subcommand/preview.rs b/src/subcommand/preview.rs index e97859b34b..f645a979d9 100644 --- a/src/subcommand/preview.rs +++ b/src/subcommand/preview.rs @@ -87,6 +87,7 @@ impl Preview { dry_run: false, no_limit: false, destination: None, + parent: None, postage: Some(TransactionBuilder::TARGET_POSTAGE), }, )), diff --git a/src/subcommand/wallet/inscribe.rs b/src/subcommand/wallet/inscribe.rs index 823b7106a0..1bb4c125ad 100644 --- a/src/subcommand/wallet/inscribe.rs +++ b/src/subcommand/wallet/inscribe.rs @@ -7,14 +7,13 @@ use { key::{TapTweak, TweakedKeyPair, TweakedPublicKey, UntweakedKeyPair}, locktime::absolute::LockTime, policy::MAX_STANDARD_TX_WEIGHT, - secp256k1::{ - self, constants::SCHNORR_SIGNATURE_SIZE, rand, schnorr::Signature, Secp256k1, XOnlyPublicKey, - }, + secp256k1::{self, constants::SCHNORR_SIGNATURE_SIZE, rand, Secp256k1, XOnlyPublicKey}, sighash::{Prevouts, SighashCache, TapSighashType}, + taproot::Signature, taproot::{ControlBlock, LeafVersion, TapLeafHash, TaprootBuilder}, ScriptBuf, Witness, }, - bitcoincore_rpc::bitcoincore_rpc_json::{ImportDescriptors, Timestamp}, + bitcoincore_rpc::bitcoincore_rpc_json::{ImportDescriptors, SignRawTransactionInput, Timestamp}, bitcoincore_rpc::Client, std::collections::BTreeSet, }; @@ -23,22 +22,30 @@ use { pub struct Output { pub commit: Txid, pub inscription: InscriptionId, + pub parent: Option, pub reveal: Txid, - pub fees: u64, + pub total_fees: u64, +} + +#[derive(Clone)] +struct ParentInfo { + destination: Address, + location: SatPoint, + tx_out: TxOut, } #[derive(Debug, Parser)] pub(crate) struct Inscribe { - #[arg(long, help = "Inscribe ")] + #[arg(long, help = "Inscribe .")] pub(crate) satpoint: Option, - #[arg(long, help = "Use fee rate of sats/vB")] + #[arg(long, help = "Use fee rate of sats/vB.")] pub(crate) fee_rate: FeeRate, #[arg( long, help = "Use sats/vbyte for commit transaction.\nDefaults to if unset." )] pub(crate) commit_fee_rate: Option, - #[arg(help = "Inscribe sat with contents of ")] + #[arg(help = "Inscribe sat with contents of .")] pub(crate) file: PathBuf, #[arg(long, help = "Do not back up recovery key.")] pub(crate) no_backup: bool, @@ -53,21 +60,23 @@ pub(crate) struct Inscribe { pub(crate) destination: Option>, #[arg( long, - help = "Amount of postage to include in the inscription. Default `10000sat`" + help = "Amount of postage to include in the inscription. Default `10000sat`." )] pub(crate) postage: Option, + #[clap(long, help = "Make inscription a child of .")] + pub(crate) parent: Option, } impl Inscribe { pub(crate) fn run(self, options: Options) -> SubcommandResult { - let inscription = Inscription::from_file(options.chain(), &self.file)?; + let inscription = Inscription::from_file(options.chain(), &self.file, self.parent)?; let index = Index::open(&options)?; index.update()?; let client = options.bitcoin_rpc_client_for_wallet_command(false)?; - let mut utxos = index.get_unspent_outputs(Wallet::load(&options)?)?; + let utxos = index.get_unspent_outputs(Wallet::load(&options)?)?; let inscriptions = index.get_inscriptions(utxos.clone())?; @@ -81,13 +90,38 @@ impl Inscribe { None => get_change_address(&client, &options)?, }; - let (unsigned_commit_tx, reveal_tx, recovery_key_pair) = + let parent_info = if let Some(parent_id) = self.parent { + if let Some(satpoint) = index.get_inscription_satpoint_by_id(parent_id)? { + if !utxos.contains_key(&satpoint.outpoint) { + return Err(anyhow!(format!("parent {parent_id} not in wallet"))); + } + + Some(ParentInfo { + destination: get_change_address(&client, &options)?, + location: satpoint, + tx_out: index + .get_transaction(satpoint.outpoint.txid)? + .expect("parent transaction not found in index") + .output + .into_iter() + .nth(satpoint.outpoint.vout.try_into().unwrap()) + .expect("current transaction output"), + }) + } else { + return Err(anyhow!(format!("parent {parent_id} does not exist"))); + } + } else { + None + }; + + let (commit_tx, reveal_tx, recovery_key_pair, total_fees) = Inscribe::create_inscription_transactions( self.satpoint, + parent_info, inscription, inscriptions, options.chain().network(), - utxos.clone(), + utxos, commit_tx_change, reveal_tx_destination, self.commit_fee_rate.unwrap_or(self.fee_rate), @@ -99,77 +133,84 @@ impl Inscribe { }, )?; - utxos.insert( - reveal_tx.input[0].previous_output, - Amount::from_sat( - unsigned_commit_tx.output[reveal_tx.input[0].previous_output.vout as usize].value, - ), - ); - - let fees = - Self::calculate_fee(&unsigned_commit_tx, &utxos) + Self::calculate_fee(&reveal_tx, &utxos); - if self.dry_run { - Ok(Box::new(Output { - commit: unsigned_commit_tx.txid(), + return Ok(Box::new(Output { + commit: commit_tx.txid(), reveal: reveal_tx.txid(), inscription: InscriptionId { txid: reveal_tx.txid(), index: 0, }, - fees, - })) - } else { - if !self.no_backup { - Inscribe::backup_recovery_key(&client, recovery_key_pair, options.chain().network())?; - } + parent: self.parent, + total_fees, + })); + } - let signed_raw_commit_tx = client - .sign_raw_transaction_with_wallet(&unsigned_commit_tx, None, None)? - .hex; + let signed_commit_tx = client + .sign_raw_transaction_with_wallet(&commit_tx, None, None)? + .hex; + + let signed_reveal_tx = if self.parent.is_some() { + client + .sign_raw_transaction_with_wallet( + &reveal_tx, + Some( + &commit_tx + .output + .iter() + .enumerate() + .map(|(vout, output)| SignRawTransactionInput { + txid: commit_tx.txid(), + vout: vout.try_into().unwrap(), + script_pub_key: output.script_pubkey.clone(), + redeem_script: None, + amount: Some(Amount::from_sat(output.value)), + }) + .collect::>(), + ), + None, + )? + .hex + } else { + bitcoin::consensus::encode::serialize(&reveal_tx) + }; - let commit = client - .send_raw_transaction(&signed_raw_commit_tx) - .context("Failed to send commit transaction")?; + if !self.no_backup { + Inscribe::backup_recovery_key(&client, recovery_key_pair, options.chain().network())?; + } - let reveal = client - .send_raw_transaction(&reveal_tx) - .context("Failed to send reveal transaction")?; + let commit = client.send_raw_transaction(&signed_commit_tx)?; - Ok(Box::new(Output { - commit, - reveal, - inscription: InscriptionId { - txid: reveal, - index: 0, - }, - fees, - })) - } - } + let reveal = client + .send_raw_transaction(&signed_reveal_tx) + .context("Failed to send reveal transaction")?; - fn calculate_fee(tx: &Transaction, utxos: &BTreeMap) -> u64 { - tx.input - .iter() - .map(|txin| utxos.get(&txin.previous_output).unwrap().to_sat()) - .sum::() - .checked_sub(tx.output.iter().map(|txout| txout.value).sum::()) - .unwrap() + Ok(Box::new(Output { + commit, + reveal, + parent: self.parent, + inscription: InscriptionId { + txid: reveal, + index: 0, + }, + total_fees, + })) } fn create_inscription_transactions( satpoint: Option, + parent_info: Option, inscription: Inscription, inscriptions: BTreeMap, network: Network, - utxos: BTreeMap, + mut utxos: BTreeMap, change: [Address; 2], destination: Address, commit_fee_rate: FeeRate, reveal_fee_rate: FeeRate, no_limit: bool, postage: Amount, - ) -> Result<(Transaction, Transaction, TweakedKeyPair)> { + ) -> Result<(Transaction, Transaction, TweakedKeyPair, u64)> { let satpoint = if let Some(satpoint) = satpoint { satpoint } else { @@ -223,21 +264,43 @@ impl Inscribe { let commit_tx_address = Address::p2tr_tweaked(taproot_spend_info.output_key(), network); + let mut inputs = vec![OutPoint::null()]; + let mut outputs = vec![TxOut { + script_pubkey: destination.script_pubkey(), + value: 0, + }]; + + if let Some(ParentInfo { + location, + destination, + tx_out, + }) = parent_info.clone() + { + inputs.insert(0, location.outpoint); + outputs.insert( + 0, + TxOut { + script_pubkey: destination.script_pubkey(), + value: tx_out.value, + }, + ); + } + + let commit_input = if parent_info.is_some() { 1 } else { 0 }; + let (_, reveal_fee) = Self::build_reveal_transaction( &control_block, reveal_fee_rate, - OutPoint::null(), - TxOut { - script_pubkey: destination.script_pubkey(), - value: 0, - }, + inputs.clone(), + commit_input, + outputs.clone(), &reveal_script, ); let unsigned_commit_tx = TransactionBuilder::new( satpoint, inscriptions, - utxos, + utxos.clone(), commit_tx_address.clone(), change, commit_fee_rate, @@ -252,50 +315,74 @@ impl Inscribe { .find(|(_vout, output)| output.script_pubkey == commit_tx_address.script_pubkey()) .expect("should find sat commit/inscription output"); + inputs[commit_input] = OutPoint { + txid: unsigned_commit_tx.txid(), + vout: vout.try_into().unwrap(), + }; + + outputs[commit_input] = TxOut { + script_pubkey: destination.script_pubkey(), + value: output.value, + }; + let (mut reveal_tx, fee) = Self::build_reveal_transaction( &control_block, reveal_fee_rate, - OutPoint { - txid: unsigned_commit_tx.txid(), - vout: vout.try_into().unwrap(), - }, - TxOut { - script_pubkey: destination.script_pubkey(), - value: output.value, - }, + inputs, + commit_input, + outputs.clone(), &reveal_script, ); - reveal_tx.output[0].value = reveal_tx.output[0] + reveal_tx.output[commit_input].value = reveal_tx.output[commit_input] .value .checked_sub(fee.to_sat()) .context("commit transaction output value insufficient to pay transaction fee")?; - if reveal_tx.output[0].value < reveal_tx.output[0].script_pubkey.dust_value().to_sat() { + if reveal_tx.output[commit_input].value + < reveal_tx.output[commit_input] + .script_pubkey + .dust_value() + .to_sat() + { bail!("commit transaction output would be dust"); } + let mut prevouts = vec![unsigned_commit_tx.output[0].clone()]; + + if let Some(parent_info) = parent_info { + prevouts.insert(0, parent_info.tx_out); + } + let mut sighash_cache = SighashCache::new(&mut reveal_tx); - let signature_hash = sighash_cache + let sighash = sighash_cache .taproot_script_spend_signature_hash( - 0, - &Prevouts::All(&[output]), + commit_input, + &Prevouts::All(&prevouts), TapLeafHash::from_script(&reveal_script, LeafVersion::TapScript), TapSighashType::Default, ) .expect("signature hash should compute"); - let signature = secp256k1.sign_schnorr( - &secp256k1::Message::from_slice(signature_hash.as_ref()) + let sig = secp256k1.sign_schnorr( + &secp256k1::Message::from_slice(sighash.as_ref()) .expect("should be cryptographically secure hash"), &key_pair, ); let witness = sighash_cache - .witness_mut(0) + .witness_mut(commit_input) .expect("getting mutable witness reference should work"); - witness.push(signature.as_ref()); + + witness.push( + Signature { + sig, + hash_ty: TapSighashType::Default, + } + .to_vec(), + ); + witness.push(reveal_script); witness.push(&control_block.serialize()); @@ -318,7 +405,74 @@ impl Inscribe { ); } - Ok((unsigned_commit_tx, reveal_tx, recovery_key_pair)) + utxos.insert( + reveal_tx.input[commit_input].previous_output, + Amount::from_sat( + unsigned_commit_tx.output[reveal_tx.input[commit_input].previous_output.vout as usize] + .value, + ), + ); + + let total_fees = + Self::calculate_fee(&unsigned_commit_tx, &utxos) + Self::calculate_fee(&reveal_tx, &utxos); + + Ok((unsigned_commit_tx, reveal_tx, recovery_key_pair, total_fees)) + } + + fn build_reveal_transaction( + control_block: &ControlBlock, + fee_rate: FeeRate, + inputs: Vec, + commit_input_index: usize, + outputs: Vec, + script: &Script, + ) -> (Transaction, Amount) { + let reveal_tx = Transaction { + input: inputs + .iter() + .map(|outpoint| TxIn { + previous_output: *outpoint, + script_sig: script::Builder::new().into_script(), + witness: Witness::new(), + sequence: Sequence::ENABLE_RBF_NO_LOCKTIME, + }) + .collect(), + output: outputs, + lock_time: LockTime::ZERO, + version: 1, + }; + + let fee = { + let mut reveal_tx = reveal_tx.clone(); + + for (current_index, txin) in reveal_tx.input.iter_mut().enumerate() { + // add dummy inscription witness for reveal input/commit output + if current_index == commit_input_index { + txin.witness.push( + Signature::from_slice(&[0; SCHNORR_SIGNATURE_SIZE]) + .unwrap() + .to_vec(), + ); + txin.witness.push(script); + txin.witness.push(&control_block.serialize()); + } else { + txin.witness = Witness::from_slice(&[&[0; SCHNORR_SIGNATURE_SIZE]]); + } + } + + fee_rate.fee(reveal_tx.vsize()) + }; + + (reveal_tx, fee) + } + + fn calculate_fee(tx: &Transaction, utxos: &BTreeMap) -> u64 { + tx.input + .iter() + .map(|txin| utxos.get(&txin.previous_output).unwrap().to_sat()) + .sum::() + .checked_sub(tx.output.iter().map(|txout| txout.value).sum::()) + .unwrap() } fn backup_recovery_key( @@ -348,42 +502,6 @@ impl Inscribe { Ok(()) } - - fn build_reveal_transaction( - control_block: &ControlBlock, - fee_rate: FeeRate, - input: OutPoint, - output: TxOut, - script: &Script, - ) -> (Transaction, Amount) { - let reveal_tx = Transaction { - input: vec![TxIn { - previous_output: input, - script_sig: script::Builder::new().into_script(), - witness: Witness::new(), - sequence: Sequence::ENABLE_RBF_NO_LOCKTIME, - }], - output: vec![output], - lock_time: LockTime::ZERO, - version: 1, - }; - - let fee = { - let mut reveal_tx = reveal_tx.clone(); - - reveal_tx.input[0].witness.push( - Signature::from_slice(&[0; SCHNORR_SIGNATURE_SIZE]) - .unwrap() - .as_ref(), - ); - reveal_tx.input[0].witness.push(script); - reveal_tx.input[0].witness.push(&control_block.serialize()); - - fee_rate.fee(reveal_tx.vsize()) - }; - - (reveal_tx, fee) - } } #[cfg(test)] @@ -397,8 +515,9 @@ mod tests { let commit_address = change(0); let reveal_address = recipient(); - let (commit_tx, reveal_tx, _private_key) = Inscribe::create_inscription_transactions( + let (commit_tx, reveal_tx, _private_key, _) = Inscribe::create_inscription_transactions( Some(satpoint(1, 0)), + None, inscription, BTreeMap::new(), Network::Bitcoin, @@ -429,8 +548,9 @@ mod tests { let commit_address = change(0); let reveal_address = recipient(); - let (commit_tx, reveal_tx, _) = Inscribe::create_inscription_transactions( + let (commit_tx, reveal_tx, _, _) = Inscribe::create_inscription_transactions( Some(satpoint(1, 0)), + None, inscription, BTreeMap::new(), Network::Bitcoin, @@ -467,6 +587,7 @@ mod tests { let error = Inscribe::create_inscription_transactions( satpoint, + None, inscription, inscriptions, Network::Bitcoin, @@ -510,6 +631,7 @@ mod tests { assert!(Inscribe::create_inscription_transactions( satpoint, + None, inscription, inscriptions, Network::Bitcoin, @@ -545,8 +667,9 @@ mod tests { let reveal_address = recipient(); let fee_rate = 3.3; - let (commit_tx, reveal_tx, _private_key) = Inscribe::create_inscription_transactions( + let (commit_tx, reveal_tx, _private_key, _) = Inscribe::create_inscription_transactions( satpoint, + None, inscription, inscriptions, bitcoin::Network::Signet, @@ -586,6 +709,88 @@ mod tests { ); } + #[test] + fn inscribe_with_parent() { + let utxos = vec![ + (outpoint(1), Amount::from_sat(10_000)), + (outpoint(2), Amount::from_sat(20_000)), + ]; + + let mut inscriptions = BTreeMap::new(); + let parent_inscription = inscription_id(1); + let parent_info = ParentInfo { + destination: change(3), + location: SatPoint { + outpoint: outpoint(1), + offset: 0, + }, + tx_out: TxOut { + script_pubkey: change(0).script_pubkey(), + value: 10000, + }, + }; + + inscriptions.insert(parent_info.location, parent_inscription); + + let child_inscription = inscription("text/plain", [b'O'; 100]); + + let commit_address = change(1); + let reveal_address = recipient(); + let fee_rate = 4.0; + + let (commit_tx, reveal_tx, _private_key, _) = Inscribe::create_inscription_transactions( + None, + Some(parent_info.clone()), + child_inscription, + inscriptions, + bitcoin::Network::Signet, + utxos.into_iter().collect(), + [commit_address, change(2)], + reveal_address, + FeeRate::try_from(fee_rate).unwrap(), + FeeRate::try_from(fee_rate).unwrap(), + false, + TransactionBuilder::TARGET_POSTAGE, + ) + .unwrap(); + + let sig_vbytes = 17; + let fee = FeeRate::try_from(fee_rate) + .unwrap() + .fee(commit_tx.vsize() + sig_vbytes) + .to_sat(); + + let reveal_value = commit_tx + .output + .iter() + .map(|o| o.value) + .reduce(|acc, i| acc + i) + .unwrap(); + + assert_eq!(reveal_value, 20_000 - fee); + + let sig_vbytes = 16; + let fee = FeeRate::try_from(fee_rate) + .unwrap() + .fee(reveal_tx.vsize() + sig_vbytes) + .to_sat(); + + assert_eq!(fee, commit_tx.output[0].value - reveal_tx.output[1].value,); + assert_eq!( + reveal_tx.output[0].script_pubkey, + parent_info.destination.script_pubkey() + ); + assert_eq!(reveal_tx.output[0].value, parent_info.tx_out.value); + pretty_assert_eq!( + reveal_tx.input[0], + TxIn { + previous_output: parent_info.location.outpoint, + sequence: Sequence::ENABLE_RBF_NO_LOCKTIME, + ..Default::default() + } + ); + } + #[test] fn inscribe_with_commit_fee_rate() { let utxos = vec![ @@ -608,8 +813,9 @@ mod tests { let commit_fee_rate = 3.3; let fee_rate = 1.0; - let (commit_tx, reveal_tx, _private_key) = Inscribe::create_inscription_transactions( + let (commit_tx, reveal_tx, _private_key, _) = Inscribe::create_inscription_transactions( satpoint, + None, inscription, inscriptions, bitcoin::Network::Signet, @@ -660,6 +866,7 @@ mod tests { let error = Inscribe::create_inscription_transactions( satpoint, + None, inscription, BTreeMap::new(), Network::Bitcoin, @@ -690,8 +897,9 @@ mod tests { let commit_address = change(0); let reveal_address = recipient(); - let (_commit_tx, reveal_tx, _private_key) = Inscribe::create_inscription_transactions( + let (_commit_tx, reveal_tx, _private_key, _) = Inscribe::create_inscription_transactions( satpoint, + None, inscription, BTreeMap::new(), Network::Bitcoin, diff --git a/src/subcommand/wallet/transaction_builder.rs b/src/subcommand/wallet/transaction_builder.rs index 87a12e624a..4e7c549640 100644 --- a/src/subcommand/wallet/transaction_builder.rs +++ b/src/subcommand/wallet/transaction_builder.rs @@ -8,18 +8,17 @@ //! constructing ordinal-aware transactions that take these additional //! conditions into account. //! -//! The external interfaces are -//! `TransactionBuilder::build_transaction_with_postage`, and -//! `TransactionBuilder::build_transaction_with_value`. Both return a -//! constructed transaction given the arguments, which include the outgoing sat +//! The external interface is +//! `TransactionBuilder::new`, which returns a +//! constructed transaction given the `Target`, which include the outgoing sat //! to send, the wallets current UTXOs and their sat ranges, and the -//! recipient's address. +//! recipient's address. To build the transaction call `Transaction::build_transaction`. //! -//! `TransactionBuilder::build_transaction_with_postage` ensures that the +//! `Target::Postage` ensures that the //! outgoing value is at most 20,000 sats, reducing it to 10,000 sats if coin //! selection requires adding excess value. //! -//! `TransactionBuilder::build_transaction_with_value` ensures that the +//! `Target::Value(Amount)` ensures that the //! outgoing value is exactly the requested amount, //! //! Internally, `TransactionBuilder` calls multiple methods that implement @@ -28,7 +27,7 @@ //! //! This module is heavily tested. For all features of transaction //! construction, there should be a positive test that checks that the feature -//! is implemented correctly, an assertion in the final `Transaction::build` +//! is implemented correctly, an assertion in the final `Transaction::build_transaction` //! method that the built transaction is correct with respect to the feature, //! and a test that the assertion fires as expected. diff --git a/src/test.rs b/src/test.rs index 1ea1634daf..cdc70671ce 100644 --- a/src/test.rs +++ b/src/test.rs @@ -88,6 +88,7 @@ pub(crate) fn change(n: u64) -> Address { 0 => "tb1qjsv26lap3ffssj6hfy8mzn0lg5vte6a42j75ww", 1 => "tb1qakxxzv9n7706kc3xdcycrtfv8cqv62hnwexc0l", 2 => "tb1qxz9yk0td0yye009gt6ayn7jthz5p07a75luryg", + 3 => "tb1qe62s57n77pfhlw2vtqlhm87dwj75l6fguavjjq", _ => panic!(), } .parse::>() diff --git a/test-bitcoincore-rpc/src/api.rs b/test-bitcoincore-rpc/src/api.rs index 8c1226ebcc..3477faa0e7 100644 --- a/test-bitcoincore-rpc/src/api.rs +++ b/test-bitcoincore-rpc/src/api.rs @@ -54,7 +54,7 @@ pub trait Api { fn sign_raw_transaction_with_wallet( &self, tx: String, - utxos: Option<()>, + utxos: Option>, sighash_type: Option<()>, ) -> Result; diff --git a/test-bitcoincore-rpc/src/lib.rs b/test-bitcoincore-rpc/src/lib.rs index ab4f35d3ff..a0c4c660ef 100644 --- a/test-bitcoincore-rpc/src/lib.rs +++ b/test-bitcoincore-rpc/src/lib.rs @@ -22,7 +22,8 @@ use { GetNetworkInfoResult, GetRawTransactionResult, GetTransactionResult, GetTransactionResultDetail, GetTransactionResultDetailCategory, GetWalletInfoResult, ImportDescriptors, ImportMultiResult, ListDescriptorsResult, ListTransactionResult, - ListUnspentResultEntry, LoadWalletResult, SignRawTransactionResult, Timestamp, WalletTxInfo, + ListUnspentResultEntry, LoadWalletResult, SignRawTransactionInput, SignRawTransactionResult, + Timestamp, WalletTxInfo, }, jsonrpc_core::{IoHandler, Value}, jsonrpc_http_server::{CloseHandle, ServerBuilder}, diff --git a/test-bitcoincore-rpc/src/server.rs b/test-bitcoincore-rpc/src/server.rs index 1c783769c8..4bbabf74a6 100644 --- a/test-bitcoincore-rpc/src/server.rs +++ b/test-bitcoincore-rpc/src/server.rs @@ -241,15 +241,16 @@ impl Api for Server { fn sign_raw_transaction_with_wallet( &self, tx: String, - utxos: Option<()>, + _utxos: Option>, sighash_type: Option<()>, ) -> Result { - assert_eq!(utxos, None, "utxos param not supported"); assert_eq!(sighash_type, None, "sighash_type param not supported"); let mut transaction: Transaction = deserialize(&hex::decode(tx).unwrap()).unwrap(); for input in &mut transaction.input { - input.witness = Witness::from_slice(&[&[0; 64]]); + if input.witness.is_empty() { + input.witness = Witness::from_slice(&[&[0; 64]]); + } } Ok( diff --git a/tests/wallet/inscribe.rs b/tests/wallet/inscribe.rs index 4af22348e1..4e6a831640 100644 --- a/tests/wallet/inscribe.rs +++ b/tests/wallet/inscribe.rs @@ -224,7 +224,7 @@ fn inscribe_with_fee_rate() { create_wallet(&rpc_server); rpc_server.mine_blocks(1); - CommandBuilder::new("--index-sats wallet inscribe degenerate.png --fee-rate 2.0") + let output = CommandBuilder::new("--index-sats wallet inscribe degenerate.png --fee-rate 2.0") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) .run_and_deserialize_output::(); @@ -257,6 +257,13 @@ fn inscribe_with_fee_rate() { let fee_rate = fee as f64 / tx2.vsize() as f64; pretty_assert_eq!(fee_rate, 2.0); + assert_eq!( + ord::FeeRate::try_from(2.0) + .unwrap() + .fee(tx1.vsize() + tx2.vsize()) + .to_sat(), + output.total_fees + ); } #[test] @@ -350,14 +357,14 @@ fn inscribe_with_dry_run_flag_fees_inscrease() { .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) .run_and_deserialize_output::() - .fees; + .total_fees; let total_fee_normal = CommandBuilder::new("wallet inscribe --dry-run degenerate.png --fee-rate 1.1") .write("degenerate.png", [1; 520]) .rpc_server(&rpc_server) .run_and_deserialize_output::() - .fees; + .total_fees; assert!(total_fee_dry_run < total_fee_normal); } @@ -438,3 +445,90 @@ fn inscribe_works_with_postage() { pretty_assert_eq!(inscriptions[0].postage, 5 * COIN_VALUE); } + +#[test] +fn inscribe_with_non_existent_parent_inscription() { + let rpc_server = test_bitcoincore_rpc::spawn(); + create_wallet(&rpc_server); + rpc_server.mine_blocks(1); + + let parent_id = "0000000000000000000000000000000000000000000000000000000000000000i0"; + + CommandBuilder::new(format!( + "wallet inscribe --fee-rate 1.0 --parent {parent_id} child.png" + )) + .write("child.png", [1; 520]) + .rpc_server(&rpc_server) + .expected_stderr(format!("error: parent {parent_id} does not exist\n")) + .expected_exit_code(1) + .run_and_extract_stdout(); +} + +#[test] +fn inscribe_with_parent_inscription_and_fee_rate() { + let rpc_server = test_bitcoincore_rpc::spawn(); + create_wallet(&rpc_server); + rpc_server.mine_blocks(1); + + let parent_output = CommandBuilder::new("wallet inscribe --fee-rate 5.0 parent.png") + .write("parent.png", [1; 520]) + .rpc_server(&rpc_server) + .run_and_deserialize_output::(); + + assert_eq!(rpc_server.descriptors().len(), 3); + let parent_id = parent_output.inscription; + + let commit_tx = &rpc_server.mempool()[0]; + let reveal_tx = &rpc_server.mempool()[1]; + + assert_eq!( + ord::FeeRate::try_from(5.0) + .unwrap() + .fee(commit_tx.vsize() + reveal_tx.vsize()) + .to_sat(), + parent_output.total_fees + ); + + rpc_server.mine_blocks(1); + + let child_output = CommandBuilder::new(format!( + "wallet inscribe --fee-rate 7.3 --parent {parent_id} child.png" + )) + .write("child.png", [1; 520]) + .rpc_server(&rpc_server) + .run_and_deserialize_output::(); + + assert_eq!(rpc_server.descriptors().len(), 4); + assert_eq!(parent_id, child_output.parent.unwrap()); + + let commit_tx = &rpc_server.mempool()[0]; + let reveal_tx = &rpc_server.mempool()[1]; + + assert_eq!( + ord::FeeRate::try_from(7.3) + .unwrap() + .fee(commit_tx.vsize() + reveal_tx.vsize()) + .to_sat(), + child_output.total_fees + ); + + rpc_server.mine_blocks(1); + + let ord_server = TestServer::spawn_with_args(&rpc_server, &[]); + + ord_server.assert_response_regex( + format!("/inscription/{}", child_output.parent.unwrap()), + format!( + ".*
    children
    .*.*", + child_output.inscription + ), + ); + + ord_server.assert_response_regex( + format!("/inscription/{}", child_output.inscription), + format!( + ".*
    parent
    .*
    .*", + child_output.parent.unwrap() + ), + ); +} From 4b1709d87d05047d72cb2e94606fb756fd1e4cb7 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Wed, 6 Sep 2023 14:07:45 -0700 Subject: [PATCH 54/61] Add provenance to docs summary (#2427) --- docs/src/SUMMARY.md | 1 + 1 file changed, 1 insertion(+) diff --git a/docs/src/SUMMARY.md b/docs/src/SUMMARY.md index 7a0258fd26..2ed5bace41 100644 --- a/docs/src/SUMMARY.md +++ b/docs/src/SUMMARY.md @@ -3,6 +3,7 @@ - [Overview](overview.md) - [Digital Artifacts](digital-artifacts.md) - [Inscriptions](inscriptions.md) + - [Provenance](inscriptions/provenance.md) - [Recursion](inscriptions/recursion.md) - [FAQ](faq.md) - [Contributing](contributing.md) From ca30715aae5868bf7fb9956923c6e8e212b33ecd Mon Sep 17 00:00:00 2001 From: "Dr.JingLee" <95461562+DrJingLee@users.noreply.github.com> Date: Sun, 10 Sep 2023 01:02:06 +0800 Subject: [PATCH 55/61] Add Japanese version of handbook (#2426) --- .github/workflows/ci.yaml | 2 +- docs/po/ja.po | 4675 +++++++++++++++++++++++++++++++++++++ docs/po/zh.po | 1403 ++++++----- docs/theme/index.hbs | 3 + justfile | 2 +- 5 files changed, 5475 insertions(+), 610 deletions(-) create mode 100644 docs/po/ja.po diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 3b7849dbdc..410497f52a 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -14,7 +14,7 @@ defaults: env: RUSTFLAGS: --deny warnings - LANGUAGES: zh + LANGUAGES: zh ja jobs: docs: diff --git a/docs/po/ja.po b/docs/po/ja.po new file mode 100644 index 0000000000..c1e1b20baf --- /dev/null +++ b/docs/po/ja.po @@ -0,0 +1,4675 @@ +msgid "" +msgstr "" +"Project-Id-Version: 序数理論マニュアル\n" +"POT-Creation-Date: \n" +"PO-Revision-Date: 2023-09-06 10:03+0800\n" +"Last-Translator: Dr.JingLee @ordjingle \n" +"Language-Team: Japanese\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ja\n" +"Plural-Forms: nplurals=1; plural=0;\n" + +#: src/SUMMARY.md:2 src/introduction.md:1 +msgid "Introduction" +msgstr "紹介" + +#: src/SUMMARY.md:3 +msgid "Overview" +msgstr "概要" + +#: src/SUMMARY.md:4 src/digital-artifacts.md:1 +msgid "Digital Artifacts" +msgstr "数字文物" + +#: src/SUMMARY.md:5 src/SUMMARY.md:12 src/overview.md:221 src/inscriptions.md:1 +msgid "Inscriptions" +msgstr "銘文" + +#: src/SUMMARY.md:6 src/inscriptions/recursion.md:1 +msgid "Recursion" +msgstr "再帰" + +#: src/SUMMARY.md:7 +msgid "FAQ" +msgstr "一般的問題" + +#: src/SUMMARY.md:8 +msgid "Contributing" +msgstr "貢献" + +#: src/SUMMARY.md:9 src/donate.md:1 +msgid "Donate" +msgstr "寄付" + +#: src/SUMMARY.md:10 +msgid "Guides" +msgstr "ガイド" + +#: src/SUMMARY.md:11 +msgid "Explorer" +msgstr "ブラウザ" + +#: src/SUMMARY.md:13 src/guides/sat-hunting.md:1 +msgid "Sat Hunting" +msgstr "聡狩り" + +#: src/SUMMARY.md:14 src/guides/collecting.md:1 +msgid "Collecting" +msgstr "コレクション" + +#: src/SUMMARY.md:15 src/guides/sat-hunting.md:239 +msgid "Sparrow Wallet" +msgstr "雀のウォレット" + +#: src/SUMMARY.md:16 src/guides/testing.md:1 +msgid "Testing" +msgstr "テスティング" + +#: src/SUMMARY.md:17 src/guides/moderation.md:1 +msgid "Moderation" +msgstr "調節" + +#: src/SUMMARY.md:18 src/guides/reindexing.md:1 +msgid "Reindexing" +msgstr "再インデックス" + +#: src/SUMMARY.md:19 +msgid "Bounties" +msgstr "賞金" + +#: src/SUMMARY.md:20 +msgid "Bounty 0: 100,000 sats Claimed!" +msgstr "ミッション 0: 100,000 sats 完成!" + +#: src/SUMMARY.md:21 +msgid "Bounty 1: 200,000 sats Claimed!" +msgstr "ミッション 1: 200,000 sats 完成!" + +#: src/SUMMARY.md:22 +msgid "Bounty 2: 300,000 sats Claimed!" +msgstr "ミッション 2: 300,000 sats 完成!" + +#: src/SUMMARY.md:23 +msgid "Bounty 3: 400,000 sats" +msgstr "ミッション 3: 400,000 sats" + +#: src/introduction.md:4 +msgid "" +"This handbook is a guide to ordinal theory. Ordinal theory concerns itself " +"with satoshis, giving them individual identities and allowing them to be " +"tracked, transferred, and imbued with meaning." +msgstr "このマニュアルは序数理論の(Ordinals Theory)ガイドです。 序数理論は聡を注目され(Satoshi)、彼らに独立な身分を付与され、追跡され、移行され、意義を付与されます。" + +#: src/introduction.md:8 +msgid "" +"Satoshis, not bitcoin, are the atomic, native currency of the Bitcoin " +"network. One bitcoin can be sub-divided into 100,000,000 satoshis, but no " +"further." +msgstr "" +"聡は(Satoshi),ビットコインではなく、ビットコインのインタネットのネイティブ貨幣と最小の単位です。" +"一つのビットコインは100,000,000聡を細分されますが、これ以上細分できません。" + +#: src/introduction.md:11 +msgid "" +"Ordinal theory does not require a sidechain or token aside from Bitcoin, and " +"can be used without any changes to the Bitcoin network. It works right now." +msgstr "" +"序数理論はビットコインブロックチェーン以外のサイドチェーンまたはトークンが必要ではなく、ビットコインインターネットに何も変更しない場合にも使うことができます。" +"すぐに使うことができます。" + +#: src/introduction.md:14 +msgid "" +"Ordinal theory imbues satoshis with numismatic value, allowing them to be " +"collected and traded as curios." +msgstr "序数理論は聡にコレックション価値に与えられ、骨董品としてコレックションし、取引をすることができます。" + +#: src/introduction.md:17 +msgid "" +"Individual satoshis can be inscribed with arbitrary content, creating unique " +"Bitcoin-native digital artifacts that can be held in Bitcoin wallets and " +"transferred using Bitcoin transactions. Inscriptions are as durable, " +"immutable, secure, and decentralized as Bitcoin itself." +msgstr "" +"一つの聡はどんな内容でも刻むことができます。独特なビットコインネイティブ数字文物(Digital Artifact)を作り上げ、" +"ビットコインウォレットの中に保存して、ビットコインの取引で送信することができます。" +"銘文(Inscription)はビットコインのように一样長持ちして、永久、安全と分散化されます。" + +#: src/introduction.md:22 +msgid "" +"Other, more unusual use-cases are possible: off-chain colored-coins, public " +"key infrastructure with key rotation, a decentralized replacement for the " +"DNS. For now though, such use-cases are speculative, and exist only in the " +"minds of fringe ordinal theorists." +msgstr "" +"他の非正規の応用でも可能となり、チェーンの下の染色コインはシークレットキー交替の公開キーのインフラがあります。" +"DNSの分散化の代替品などです。 " +"今のところこういう応用は推測的で、非主流の序数の理論家の頭だけに存在しています" + +#: src/introduction.md:27 +msgid "For more details on ordinal theory, see the [overview](overview.md)." +msgstr "序数理論のより多くの詳細な情報について, [概述](overview.md)を閲見してください。" + +#: src/introduction.md:29 +msgid "For more details on inscriptions, see [inscriptions](inscriptions.md)." +msgstr "銘文のより多くの詳細な情報について、 [銘文](inscriptions.md)を閲見してください。" + +#: src/introduction.md:31 +msgid "" +"When you're ready to get your hands dirty, a good place to start is with " +"[inscriptions](guides/inscriptions.md), a curious species of digital " +"artifact enabled by ordinal theory." +msgstr "" +"準備ができて、自分でする時に、銘文は良いなスタートです。[銘文](guides/inscriptions.md)" +"序数理論で支持した独特な数字文物です。" + +#: src/introduction.md:35 +msgid "Links" +msgstr "リンク" + +#: src/introduction.md:38 +msgid "[GitHub](https://github.com/ordinals/ord/)" +msgstr "[GitHub倉庫](https://github.com/ordinals/ord/)" + +#: src/introduction.md:39 +msgid "[BIP](https://github.com/ordinals/ord/blob/master/bip.mediawiki)" +msgstr "[BIP](https://github.com/ordinals/ord/blob/master/bip.mediawiki)" + +#: src/introduction.md:40 +msgid "[Discord](https://discord.gg/ordinals)" +msgstr "[Discord](https://discord.gg/ordinals)" + +#: src/introduction.md:41 +msgid "[Open Ordinals Institute Website](https://ordinals.org/)" +msgstr "[Open Ordinals Institute ウェブサイト](https://ordinals.org/)" + +#: src/introduction.md:42 +msgid "[Open Ordinals Institute X](https://x.com/ordinalsorg)" +msgstr "[Open Ordinals Institute Xアカウント](https://x.com/ordinalsorg)" + +#: src/introduction.md:43 +msgid "[Mainnet Block Explorer](https://ordinals.com)" +msgstr "[メインネットブロック](https://ordinals.com)" + +#: src/introduction.md:44 +msgid "[Signet Block Explorer](https://signet.ordinals.com)" +msgstr "[Signetブロックブラウザ](https://signet.ordinals.com)" + +#: src/introduction.md:46 +msgid "Videos" +msgstr "動画" + +#: src/introduction.md:49 +msgid "" +"[Ordinal Theory Explained: Satoshi Serial Numbers and NFTs on Bitcoin]" +"(https://www.youtube.com/watch?v=rSS0O2KQpsI)" +msgstr "" +"[序数理論を解釈して: 聡のシリアル番号とビットコインでのNFT]" +"(https://www.youtube.com/watch?v=rSS0O2KQpsI)" + +#: src/introduction.md:50 +msgid "" +"[Ordinals Workshop with Rodarmor](https://www.youtube.com/watch?" +"v=MC_haVa6N3I)" +msgstr "" +"[CaseyRodarmorの序数理論のワークショップ ](https://www.youtube.com/watch?" +"v=MC_haVa6N3I)" + +#: src/introduction.md:51 +msgid "" +"[Ordinal Art: Mint Your own NFTs on Bitcoin w/ @rodarmor](https://www." +"youtube.com/watch?v=j5V33kV3iqo)" +msgstr "" +"[序数の芸術:ビットコインの上で自分のNFTを鋳造します。 w/ @rodarmor](https://www." +"youtube.com/watch?v=j5V33kV3iqo)" + +#: src/overview.md:1 +msgid "Ordinal Theory Overview" +msgstr "序数理論の概要" + +#: src/overview.md:4 +msgid "" +"Ordinals are a numbering scheme for satoshis that allows tracking and " +"transferring individual sats. These numbers are called [ordinal numbers]" +"(https://ordinals.com). Satoshis are numbered in the order in which they're " +"mined, and transferred from transaction inputs to transaction outputs first-" +"in-first-out. Both the numbering scheme and the transfer scheme rely on " +"_order_, the numbering scheme on the _order_ in which satoshis are mined, " +"and the transfer scheme on the _order_ of transaction inputs and outputs. " +"Thus the name, _ordinals_." +msgstr "" +"序数は一つのビットコインの番号プランであり、一つの聡を追跡し、移転することができます。これらの数字はシリアル番号といいます。(https://ordinals.com)。" +"ビットコインはそれらの発掘した順序により番号を書いて、取引輸入から取引輸出まで移行します(先入れ先出しの原則に従います)。" +"番号プランと転送プランは順序に依存して、番号プランはビットコインが発掘された順序に依存し、転送プランは取引の輸入と輸出に順序に依存します。" +"ここで_序数と呼ばれます(Ordinals_。" + +#: src/overview.md:13 +msgid "" +"Technical details are available in [the BIP](https://github.com/ordinals/ord/" +"blob/master/bip.mediawiki)." +msgstr "" +"技術的な詳細は[the BIP](https://github.com/ordinals/ord/blob/master/bip.mediawiki)で獲得することができます." + +#: src/overview.md:16 +msgid "" +"Ordinal theory does not require a separate token, another blockchain, or any " +"changes to Bitcoin. It works right now." +msgstr "" +"序数理論は独特なトークン、独特なブロックチェーン、またはビットコインにどんな変更でも必要ではなくて、すぐに有効的に運行することができません。" + +#: src/overview.md:19 +msgid "Ordinal numbers have a few different representations:" +msgstr "" +"シリアル番号はいくつか異なる表示方式があります" + +#: src/overview.md:21 +msgid "" +"_Integer notation_: [`2099994106992659`](https://ordinals.com/" +"sat/2099994106992659) The ordinal number, assigned according to the order in " +"which the satoshi was mined." +msgstr "" +"_整数符号_:[`2099994106992659`](https://ordinals.com/sat/2099994106992659) " +"このシリアル番号は発掘された聡の順序に従って分配されます。" + + +#: src/overview.md:26 +msgid "" +"_Decimal notation_: [`3891094.16797`](https://ordinals.com/" +"sat/3891094.16797) The first number is the block height in which the satoshi " +"was mined, the second the offset of the satoshi within the block." +msgstr "" +"_十進法符号n_: [`3891094.16797`](https://ordinals.com/" +"sat/3891094.16797) 最初の数字は聡のブロック高度を発掘され、 " +"二番目の数字はブロック内の聡の定常偏差です。" + +#: src/overview.md:31 +msgid "" +"_Degree notation_: [`3°111094′214″16797‴`](https://ordinals.com/" +"sat/3%C2%B0111094%E2%80%B2214%E2%80%B316797%E2%80%B4). We'll get to that in " +"a moment." +msgstr "" +"_度数符号_: [`3°111094′214″16797‴`](https://ordinals.com/" +"sat/3%C2%B0111094%E2%80%B2214%E2%80%B316797%E2%80%B4). " +"それについてはすぐに説明します。" + +#: src/overview.md:35 +msgid "" +"_Percentile notation_: [`99.99971949060254%`](https://ordinals.com/" +"sat/99.99971949060254%25) . The satoshi's position in Bitcoin's supply, " +"expressed as a percentage." +msgstr "" +"_パーセント_: [`99.99971949060254%`](https://ordinals.com/" +"sat/99.99971949060254%25) . パーセントで聡がビットコイン供給の中の位置を示します" + +#: src/overview.md:39 +msgid "" +"_Name_: [`satoshi`](https://ordinals.com/sat/satoshi). An encoding of the " +"ordinal number using the characters `a` through `z`." +msgstr "" +"_名前_: [`satoshi`](https://ordinals.com/sat/satoshi). " +"一つのアルファベットaからbまでのシリアル番号にコーディングを書く方法です。" + +#: src/overview.md:42 +msgid "" +"Arbitrary assets, such as NFTs, security tokens, accounts, or stablecoins " +"can be attached to satoshis using ordinal numbers as stable identifiers." +msgstr "" +"任意の資産、例えばNFT、セキュリティトークン、アカウントまたは安定した貨幣、 " +"序数を使って、安定した標識符号として、聡に付けます。" + +#: src/overview.md:45 +msgid "" +"Ordinals is an open-source project, developed [on GitHub](https://github.com/" +"ordinals/ord). The project consists of a BIP describing the ordinal scheme, " +"an index that communicates with a Bitcoin Core node to track the location of " +"all satoshis, a wallet that allows making ordinal-aware transactions, a " +"block explorer for interactive exploration of the blockchain, functionality " +"for inscribing satoshis with digital artifacts, and this manual." +msgstr "" +"Ordinalsはオープンソースプロジェクトです、[on GitHub](https://github.com/" +"ordinals/ord)に配置されます. このプロジェクトは序数プランを記述するBIPを含まれます。 " +"ビットコインのコアノードと通信して、すべての聡を追跡した索引です" +"シリアル番号の感知取引を許可するウォレットです。 " +"ブロックチェーンインタラクティブな探索のブロック資源管理プログラム、数字文物で聡の機能及びこのマニュアルを嵌めます。 " + +#: src/overview.md:52 +msgid "Rarity" +msgstr "希少度" + +#: src/overview.md:55 +msgid "" +"Humans are collectors, and since satoshis can now be tracked and " +"transferred, people will naturally want to collect them. Ordinal theorists " +"can decide for themselves which sats are rare and desirable, but there are " +"some hints…" +msgstr "" +"人間はコレクターです。現在、聡は追跡、移行されることができて、人々は聡をコレクションしたいです。 " +"序数理論家は自分でどちらの聡が珍しい聡と合意的な聡であることを決めます。 " +"ヒントがあります…" + +#: src/overview.md:59 +msgid "" +"Bitcoin has periodic events, some frequent, some more uncommon, and these " +"naturally lend themselves to a system of rarity. These periodic events are:" +msgstr "" +"ビットコインは周期的な事件があり、頻繫的なのはあるし、一般的なのはあります。これらの事件は自然にレア度システムが形成されます。" +"この周期的な事件は:" + +#: src/overview.md:62 +msgid "" +"_Blocks_: A new block is mined approximately every 10 minutes, from now " +"until the end of time." +msgstr "" +"_ブロック_: 現在から終わるまで10分ごとに新たなブロックを発掘されます。" + +#: src/overview.md:65 +msgid "" +"_Difficulty adjustments_: Every 2016 blocks, or approximately every two " +"weeks, the Bitcoin network responds to changes in hashrate by adjusting the " +"difficulty target which blocks must meet in order to be accepted." +msgstr "" +"_難易度調整_: 2016個ごとのブロック、または約二週間ごとです " +"ビットコインインターネットはブロックが難易度の目標に満足させることを調整して、ハッシュレートの変化に対応します。 " + + +#: src/overview.md:69 +msgid "" +"_Halvings_: Every 210,000 blocks, or roughly every four years, the amount of " +"new sats created in every block is cut in half." +msgstr "" +"_半減_: 21万のブロックごとに、または約4年ごとに,各ブロックが生じた新たな聡の数が半減になります。 " + +#: src/overview.md:72 +msgid "" +"_Cycles_: Every six halvings, something magical happens: the halving and the " +"difficulty adjustment coincide. This is called a conjunction, and the time " +"period between conjunctions a cycle. A conjunction occurs roughly every 24 " +"years. The first conjunction should happen sometime in 2032." +msgstr "" +"_周期_6回ごとに半減にするたび不思議なことが起こります。半分にすることと難易度の調整は同時に起こります。" +"これは相合と言います。相合の時間周期は一周期です。 " +"ほぼ24年ごとに一回の相合が起こります、最初の相合は2032年のある時に起こるはずです。 " + +#: src/overview.md:77 +msgid "This gives us the following rarity levels:" +msgstr "" +"以下の希少度のレベルを与えられました:" + +#: src/overview.md:79 +msgid "`common`: Any sat that is not the first sat of its block" +msgstr "`普通`: 全てのそのブロック最初の聡ではないの聡" + +#: src/overview.md:80 +msgid "`uncommon`: The first sat of each block" +msgstr "`普通ではない`: 各ブロックの最初の聡" + +#: src/overview.md:81 +msgid "`rare`: The first sat of each difficulty adjustment period" +msgstr "`珍しく`: 各難易度の調整周期の最初の聡" + +#: src/overview.md:82 +msgid "`epic`: The first sat of each halving epoch" +msgstr "`史詩`: 各半減周期の最初の聡" + +#: src/overview.md:83 +msgid "`legendary`: The first sat of each cycle" +msgstr "`伝奇`: 各循環周期の最初の聡" + +#: src/overview.md:84 +msgid "`mythic`: The first sat of the genesis block" +msgstr "`神話`: 創世ブロックの最初の聡" + +#: src/overview.md:86 +msgid "" +"Which brings us to degree notation, which unambiguously represents an " +"ordinal number in a way that makes the rarity of a satoshi easy to see at a " +"glance:" +msgstr "" +"これは私たちに度数表記法をもたらしました。それは聡の希少性を一目で見ることができる方法で序数を明確に表しています " + +#: src/overview.md:89 +msgid "" +"```\n" +"A°B′C″D‴\n" +"│ │ │ ╰─ Index of sat in the block\n" +"│ │ ╰─── Index of block in difficulty adjustment period\n" +"│ ╰───── Index of block in halving epoch\n" +"╰─────── Cycle, numbered starting from 0\n" +"```" +msgstr "" +"```\n" +"A°B′C″D‴\n" +"│ │ │ ╰─ 聡の索引位置\n" +"│ │ ╰─── 難易度調整期のブロックの位置\n" +"│ ╰───── 半減周期ブロックの索引位置\n" +"╰─────── 循環周期、0の数字からスタートします。\n" +"```" + +#: src/overview.md:97 +msgid "" +"Ordinal theorists often use the terms \"hour\", \"minute\", \"second\", and " +"\"third\" for _A_, _B_, _C_, and _D_, respectively." +msgstr "" +"序理論家通常使用 \"時間\",\"分\"、\"秒\"、そして\"3\" などの対応を専用の言叶は、a、b、_c、と_d_。 " + +#: src/overview.md:100 +msgid "Now for some examples. This satoshi is common:" +msgstr "いくつか例をお見せしましょうこれは普通の聡です" + +#: src/overview.md:102 +msgid "" +"```\n" +"1°1′1″1‴\n" +"│ │ │ ╰─ Not first sat in block\n" +"│ │ ╰─── Not first block in difficulty adjustment period\n" +"│ ╰───── Not first block in halving epoch\n" +"╰─────── Second cycle\n" +"```" +msgstr "" +"```\n" +"1°1′1″1‴\n" +"│ │ │ ╰─ ブロック最初の聡ではない\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:111 +msgid "This satoshi is uncommon:" +msgstr "これは一般的な聡ではない" + +#: src/overview.md:113 +msgid "" +"```\n" +"1°1′1″0‴\n" +"│ │ │ ╰─ First sat in block\n" +"│ │ ╰─── Not first block in difficulty adjustment period\n" +"│ ╰───── Not first block in halving epoch\n" +"╰─────── Second cycle\n" +"```" +msgstr "" +"```\n" +"1°1′1″0‴\n" +"│ │ │ ╰─ ブロック最初の聡\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:121 +msgid "This satoshi is rare:" +msgstr "珍しい聡です。" + +#: src/overview.md:123 +msgid "" +"```\n" +"1°1′0″0‴\n" +"│ │ │ ╰─ First sat in block\n" +"│ │ ╰─── First block in difficulty adjustment period\n" +"│ ╰───── Not the first block in halving epoch\n" +"╰─────── Second cycle\n" +"```" +msgstr "" +"```\n" +"1°1′0″0‴\n" +"│ │ │ ╰─ ブロック最初の聡\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:131 +msgid "This satoshi is epic:" +msgstr "史詩レベルの聡です" + +#: src/overview.md:133 +msgid "" +"```\n" +"1°0′1″0‴\n" +"│ │ │ ╰─ First sat in block\n" +"│ │ ╰─── Not first block in difficulty adjustment period\n" +"│ ╰───── First block in halving epoch\n" +"╰─────── Second cycle\n" +"```" +msgstr "" +"```\n" +"1°0′1″0‴\n" +"│ │ │ ╰─ ブロック最初の聡\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:141 +msgid "This satoshi is legendary:" +msgstr "伝奇レベルの聡です。" + +#: src/overview.md:143 +msgid "" +"```\n" +"1°0′0″0‴\n" +"│ │ │ ╰─ First sat in block\n" +"│ │ ╰─── First block in difficulty adjustment period\n" +"│ ╰───── First block in halving epoch\n" +"╰─────── Second cycle\n" +"```" +msgstr "" +"```\n" +"1°0′0″0‴\n" +"│ │ │ ╰─ ブロック最初の聡\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:151 +msgid "And this satoshi is mythic:" +msgstr "神話レベルの聡です:" + +#: src/overview.md:153 +msgid "" +"```\n" +"0°0′0″0‴\n" +"│ │ │ ╰─ First sat in block\n" +"│ │ ╰─── First block in difficulty adjustment period\n" +"│ ╰───── First block in halving epoch\n" +"╰─────── First cycle\n" +"```" +msgstr "" +"```\n" +"0°0′0″0‴\n" +"│ │ │ ╰─ ブロック最初の聡\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:161 +msgid "" +"If the block offset is zero, it may be omitted. This is the uncommon satoshi " +"from above:" +msgstr "" +"ブロック オフセットがゼロの場合は省略できます。こちらは上から見た珍しいサトシです。" +#: src/overview.md:164 +msgid "" +"```\n" +"1°1′1″\n" +"│ │ ╰─ Not first block in difficulty adjustment period\n" +"│ ╰─── Not first block in halving epoch\n" +"╰───── Second cycle\n" +"```" +msgstr "" +"```\n" +"1°1′1″\n" +"│ │ ╰─── 難易度調整周期の最初の聡ではない\n" +"│ ╰───── 半減周期の最初の聡ではない\n" +"╰─────── 二番目の循環周期\n" +"```" + +#: src/overview.md:171 +msgid "Rare Satoshi Supply" +msgstr "珍しい聡の総供給量" + +#: src/overview.md:174 +msgid "Total Supply" +msgstr "総供給量" + +#: src/overview.md:176 +msgid "`common`: 2.1 quadrillion" +msgstr "`普通`: 2千100万亿" + +#: src/overview.md:177 +msgid "`uncommon`: 6,929,999" +msgstr "`普通ではない`: 6,929,999" + +#: src/overview.md:178 +msgid "`rare`: 3437" +msgstr "`珍しく`: 3437" + +#: src/overview.md:179 +msgid "`epic`: 32" +msgstr "`史詩`: 32" + +#: src/overview.md:180 +msgid "`legendary`: 5" +msgstr "`伝奇`: 5" + +#: src/overview.md:181 src/overview.md:190 +msgid "`mythic`: 1" +msgstr "`神話`: 1" + +#: src/overview.md:183 +msgid "Current Supply" +msgstr "現有の供給量" + +#: src/overview.md:185 +msgid "`common`: 1.9 quadrillion" +msgstr "`普通`: 1千900万亿" + +#: src/overview.md:186 +msgid "`uncommon`: 745,855" +msgstr "`普通ではない`: 745,855" + +#: src/overview.md:187 +msgid "`rare`: 369" +msgstr "`珍しく`: 369" + +#: src/overview.md:188 +msgid "`epic`: 3" +msgstr "`史詩`: 3" + +#: src/overview.md:189 +msgid "`legendary`: 0" +msgstr "`伝奇`: 0" + +#: src/overview.md:192 +msgid "" +"At the moment, even uncommon satoshis are quite rare. As of this writing, " +"745,855 uncommon satoshis have been mined - one per 25.6 bitcoin in " +"circulation." +msgstr "" +"今は普通ではないの聡にもとても珍しいです。 この文を書くまで, " +"もう745,855の普通ではないの聡を発掘され、およそ25.6の流通ビットコインのうちに一つがあります。 " + +#: src/overview.md:196 +msgid "Names" +msgstr "名前" + +#: src/overview.md:199 +msgid "" +"Each satoshi has a name, consisting of the letters _A_ through _Z_, that get " +"shorter the further into the future the satoshi was mined. They could start " +"short and get longer, but then all the good, short names would be trapped in " +"the unspendable genesis block." +msgstr "" +"それぞれの聡は名前があり、アルファベットAからZまで構成されます " +"随着聪被开采的时间越长聡の発掘された時間が長いほど、名前が短いです。 短い名前から始めて、" +"ますます長くなると、それでは、すべての良い、短い名前は使用できない創世ブロックに閉じ込められます。 " + +#: src/overview.md:204 +msgid "" +"As an example, 1905530482684727°'s name is \"iaiufjszmoba\". The name of the " +"last satoshi to be mined is \"a\". Every combination of 10 characters or " +"less is out there, or will be out there, someday." +msgstr "" +msgid "" +"例えば、1905530482684727°'の名前は\"iaiufjszmoba\".最後に発掘された聡の名前は\"a\"になります。10文字" +"以下の文字の組み合わせが存在するか、いつか存在します。" + +#: src/overview.md:208 +msgid "Exotics" +msgstr "きけいです。" + +#: src/overview.md:211 +msgid "" +"Satoshis may be prized for reasons other than their name or rarity. This " +"might be due to a quality of the number itself, like having an integer " +"square or cube root. Or it might be due to a connection to a historical " +"event, such as satoshis from block 477,120, the block in which SegWit " +"activated, or 2099999997689999°, the last satoshi that will ever be mined." +msgstr "" +"彼らの名前や希少性以外にも、聡は他の理由で重視されているかもしれない。" +"これは、整数の平方根や立方根を持つような数値自体の性質によるものかもしれません。" +"またはそれは、ブロック477,120(SegWitによって活性化されたブロック)からのような歴史的なイベントに関連しています" +"は2099999997689999°で、これが最後に掘り出された聡です。" + +#: src/overview.md:217 +msgid "" +"Such satoshis are termed \"exotic\". Which satoshis are exotic and what " +"makes them so is subjective. Ordinal theorists are encouraged to seek out " +"exotics based on criteria of their own devising." +msgstr "" +"このビットコインは「奇妙な」と呼ばれています。どの聡が「変わった」のですか?何がそんなに重要視されているのか?」" +"順序理論家は、彼ら自身が設計した基準に基づいて「奇妙な」コングを探すことを奨励されています。" + +#: src/overview.md:224 +msgid "" +"Satoshis can be inscribed with arbitrary content, creating Bitcoin-native " +"digital artifacts. Inscribing is done by sending the satoshi to be inscribed " +"in a transaction that reveals the inscription content on-chain. This content " +"is then inextricably linked to that satoshi, turning it into an immutable " +"digital artifact that can be tracked, transferred, hoarded, bought, sold, " +"lost, and rediscovered." +msgstr "" +"コングは、ビットコインネイティブのデジタルアーティファクト(デジタルアート)を作成するために、任意のコンテンツを刻印することができます。" +"「刻印は、刻印する内容を取引に送ることで行われます。この取引はチェーンに銘文の内容を表示します。」" +"は銘文の内容が聡と不可分なつながりを持っているため、から変えられないデジタル人工製品を創造した。" +"このデジタル遺物は、追跡、移転、保管、購入、販売、紛失、再発見することができます。" + + + +#: src/overview.md:231 +msgid "Archaeology" +msgstr "考古" + +#: src/overview.md:234 +msgid "" +"A lively community of archaeologists devoted to cataloging and collecting " +"early NFTs has sprung up. [Here's a great summary of historical NFTs by " +"Chainleft.](https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-" +"N29oF4iwCgX3lacrvaG9Kjko)" +msgstr "" +"初期のNFTのカタログ作成と収集に取り組む活発な考古学者コミュニティが雨後の筍" +"のように出現しました" +"[Chainleftによる歴史NFTの素晴らしいまとめ](https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-N29oF4iwCgX3lacrvaG9Kjko)" + +#: src/overview.md:238 +msgid "" +"A commonly accepted cut-off for early NFTs is March 19th, 2018, the date the " +"first ERC-721 contract, [SU SQUARES](https://tenthousandsu.com/), was " +"deployed on Ethereum." +msgstr "" +"一般的に受け入れられている古いNFTの締め切りは2018年3月19日、つまり" +"最初のERC-721契約、[SU SQUARES](https://tenthousandsu.com/),がイーサリアムに配備された時間です。" + +#: src/overview.md:242 +msgid "" +"Whether or not ordinals are of interest to NFT archaeologists is an open " +"question! In one sense, ordinals were created in early 2022, when the " +"Ordinals specification was finalized. In this sense, they are not of " +"historical interest." +msgstr "" +"NFT考古学者が序数に興味を持っているかどうかは未解決の問題です!" +"ある意味で、序数は2022年初頭に作成され、当時序数規範がに制定されています。" + +#: src/overview.md:247 +msgid "" +"In another sense though, ordinals were in fact created by Satoshi Nakamoto " +"in 2009 when he mined the Bitcoin genesis block. In this sense, ordinals, " +"and especially early ordinals, are certainly of historical interest." +msgstr "" +"そういう意味では歴史的な意味がありません" +"しかし、別の意味では、序数は実際には中本聡が2009年にビットコイン創世ブロックを採掘した時に作られました。" +"この意味で、序数、特に初期の序数は、もちろん歴史的な意味を持っています。" + +#: src/overview.md:251 +msgid "" +"Many ordinal theorists favor the latter view. This is not least because the " +"ordinals were independently discovered on at least two separate occasions, " +"long before the era of modern NFTs began." +msgstr "" +"多くの順序理論家は後者の見方に賛成しています。これは単に序数が少なくとも2つの異なる場面で独立して発見されたからではありません。" +"は現代のNFT時代よりはるかに早く始まった。" + +#: src/overview.md:255 +msgid "" +"On August 21st, 2012, Charlie Lee [posted a proposal to add proof-of-stake " +"to Bitcoin to the Bitcoin Talk forum](https://bitcointalk.org/index.php?" +"topic=102355.0). This wasn't an asset scheme, but did use the ordinal " +"algorithm, and was implemented but never deployed." +msgstr "" +"2012年8月21日、Charlie LeeでCharlie Lee[Bitcoin Talkフォーラムに投稿する」" +"ビットコイン権益証明Pro of-of-stake追加の提案](https://bitcointalk.org/index.php?" +"topic=102355.0).これは資産シナリオではありませんが、序数アルゴリズムが使用されており、実装されていますが、導入されたことはありません。" + +#: src/overview.md:261 +msgid "" +"On October 8th, 2012, jl2012 [posted a scheme to the same forum](https://" +"bitcointalk.org/index.php?topic=117224.0) which uses decimal notation and " +"has all the important properties of ordinals. The scheme was discussed but " +"never implemented." +msgstr "" +"2012年10月8日、jl 2012は[同じフォーラムで案を発表しました](https://" +"bitcointalk.org/index.php?topic=117224.0)このスキームは、10進表記法を使用し、" +"序数を持つすべての重要な属性。その計画は議論されたが、決して実施されませんでした" + +#: src/overview.md:266 +msgid "" +"These independent inventions of ordinals indicate in some way that ordinals " +"were discovered, or rediscovered, and not invented. The ordinals are an " +"inevitability of the mathematics of Bitcoin, stemming not from their modern " +"documentation, but from their ancient genesis. They are the culmination of a " +"sequence of events set in motion with the mining of the first block, so many " +"years ago." +msgstr "" +"これらの序数の独立した発明は、序数が発見されたことをある程度示しています。" +"または再発見されたもので、発明されたものではありません。順序数はビットコイン数学の必然性であります" +"はそれらの現代の文書からではなく、それらの古い起源から来ています。" +"彼らは何年も前に最初のブロックの採掘とともに始まった一連の出来事のクライマックスです" + +#: src/digital-artifacts.md:4 +msgid "" +"Imagine a physical artifact. A rare coin, say, held safe for untold years in " +"the dark, secret clutch of a Viking hoard, now dug from the earth by your " +"grasping hands. It…" +msgstr "" +"実体のある人工物を想像してください。例えば、珍しいコインの場合、" +"バイキングの宝庫の闇の中で何年も秘密に保管されています" +"今、あなたの手で地下から掘り出されました。それ…" + +#: src/digital-artifacts.md:8 +msgid "" +"…has an owner. You. As long as you keep it safe, nobody can take it from you." +msgstr "" +"……一人の主人がいます。それはあなたです。あなたが大切に保管していれば、誰もあなたからそれを奪うことはできません。" + +#: src/digital-artifacts.md:10 +msgid "…is complete. It has no missing parts." +msgstr "…完全です。漏れている部分はありません。" + +#: src/digital-artifacts.md:12 +msgid "" +"…can only be changed by you. If you were a trader, and you made your way to " +"18th century China, none but you could stamp it with your chop-mark." +msgstr "" +"…あなたしか変えることができません。もしあなたがビジネスマンで、あなたが来たなら" +"の18世紀の中国では、あなた以外に印鑑を押す人はいません。" + +#: src/digital-artifacts.md:15 +msgid "" +"…can only be disposed of by you. The sale, trade, or gift is yours to make, " +"to whomever you wish." +msgstr "" +"……あなたしか処分するすることができません。販売、取引、または贈り物はあなたの決定であり、誰にあげたいかは誰にでもあげます。 " + +#: src/digital-artifacts.md:18 +msgid "" +"What are digital artifacts? Simply put, they are the digital equivalent of " +"physical artifacts." +msgstr "" +"デジタル文化財(デジタル工作物、デジタル人工物)とは?" +"は簡単に言えば、物理的人工物のデジタル等価物であります。" + +#: src/digital-artifacts.md:21 +msgid "" +"For a digital thing to be a digital artifact, it must be like that coin of " +"yours:" +msgstr "デジタル化されたものをデジタルアーティファクトにするには、あなたのコインのようでなければなりません。" + +#: src/digital-artifacts.md:24 +msgid "" +"Digital artifacts can have owners. A number is not a digital artifact, " +"because nobody can own it." +msgstr "" +"デジタル文化財には所有者がいることができるので、デジタル文化財とは異なります。誰も数字を持つことができないからです。" + +#: src/digital-artifacts.md:27 +msgid "" +"Digital artifacts are complete. An NFT that points to off-chain content on " +"IPFS or Arweave is incomplete, and thus not a digital artifact." +msgstr "" +"デジタル文化財は完全であり、IP FSまたはArweave上のチェーンの下のコンテンツを指すNFTは不完全であるため、デジタル文化財ではありません。" + +#: src/digital-artifacts.md:30 +msgid "" +"Digital artifacts are permissionless. An NFT which cannot be sold without " +"paying a royalty is not permissionless, and thus not a digital artifact." +msgstr "" +"デジタル文化財は許可が必要なく、印税を支払わなければ販売できないNFTは許可が必要ないわけではないので、デジタル文化財ではありません。" + +#: src/digital-artifacts.md:33 +msgid "" +"Digital artifacts are uncensorable. Perhaps you can change a database entry " +"on a centralized ledger today, but maybe not tomorrow, and thus one cannot " +"be a digital artifact." +msgstr "" +"デジタル文化財は審査不可能です。今日は集中元帳のデータベースエントリを変更できるかもしれませんが、明日はできないかもしれません。" +"そのため、デジタル文化財ではありません。" + +#: src/digital-artifacts.md:37 +msgid "" +"Digital artifacts are immutable. An NFT with an upgrade key is not a digital " +"artifact." +msgstr "" +"デジタルコンテンツは改ざんできません。アップグレードキーを持つNFTはデジタルコンテンツではありません。" + +#: src/digital-artifacts.md:40 +msgid "" +"The definition of a digital artifact is intended to reflect what NFTs " +"_should_ be, sometimes are, and what inscriptions _always_ are, by their " +"very nature." +msgstr "" +"デジタル文化財の定義は、その特定の性質からNFTを反映することを目的としています" +"_べき_は何か、時には何か、そして銘文_終始_は何ですか " + +#: src/inscriptions.md:4 +msgid "" +"Inscriptions inscribe sats with arbitrary content, creating bitcoin-native " +"digital artifacts, more commonly known as NFTs. Inscriptions do not require " +"a sidechain or separate token." +msgstr "" +"銘文には任意の内容が刻まれ、ビットコインネイティブのデジタル人工製品が作られました。通常NFTと呼ばれています。" +"の銘文には側鎖や個別のトークンは必要ありません。 " + +#: src/inscriptions.md:8 +msgid "" +"These inscribed sats can then be transferred using bitcoin transactions, " +"sent to bitcoin addresses, and held in bitcoin UTXOs. These transactions, " +"addresses, and UTXOs are normal bitcoin transactions, addresses, and UTXOS " +"in all respects, with the exception that in order to send individual sats, " +"transactions must control the order and value of inputs and outputs " +"according to ordinal theory." +msgstr "" +"これらの刻まれたコングは、ビットコイン取引転送を使用してビットコインアドレスに送信され、ビットコインUTXOに保存されます。" +"これらの取引、アドレス、UTXOはすべての点で通常のビットコイン取引、アドレス、UTXOです。" +"は単一の聡を送るために加えて、取引は序数理論に基づいて入力と出力の順序と値を制御しなければなりません" + +#: src/inscriptions.md:15 +msgid "" +"The inscription content model is that of the web. An inscription consists of " +"a content type, also known as a MIME type, and the content itself, which is " +"a byte string. This allows inscription content to be returned from a web " +"server, and for creating HTML inscriptions that use and remix the content of " +"other inscriptions." +msgstr "" +"銘文の内容はワールドワイドウェブ基準に基づいている。銘文はコンテンツタイプ(MI MEタイプとも呼ばれる)とコンテンツ自体(バイト列)で構成されています。" +"これにより、Webサーバーから銘文コンテンツを返すことができ、HTML銘文を作成して使用し、他の銘文コンテンツを再混合するために使用されます。" + +#: src/inscriptions.md:21 +msgid "" +"Inscription content is entirely on-chain, stored in taproot script-path " +"spend scripts. Taproot scripts have very few restrictions on their content, " +"and additionally receive the witness discount, making inscription content " +"storage relatively economical." +msgstr "" +"銘文の内容は完全にチェーン上にあり、保存されているtaproot script-path spendスクリプト内。" +"のTaprootスクリプトはその内容に対する制限が少なく、証人割引を追加で受けることで、銘文内容の保存が比較的経済的になります。" + +#: src/inscriptions.md:26 +msgid "" +"Since taproot script spends can only be made from existing taproot outputs, " +"inscriptions are made using a two-phase commit/reveal procedure. First, in " +"the commit transaction, a taproot output committing to a script containing " +"the inscription content is created. Second, in the reveal transaction, the " +"output created by the commit transaction is spent, revealing the inscription " +"content on-chain." +msgstr "" +"taproot script-path spendスクリプトは既存のtaproot出力からしか生成できないため、" +"そのため、は2段階のcommit/revealプロセスを使用して刻印されます。まず、commitでは、" +"銘文の内容を含むスクリプトに提出するtaproot出力を作成します。次に、reveal取引では、" +"commit取引による出力を使用して、チェーン上の銘文の内容を表示します。" + +#: src/inscriptions.md:33 +msgid "" +"Inscription content is serialized using data pushes within unexecuted " +"conditionals, called \"envelopes\". Envelopes consist of an `OP_FALSE OP_IF " +"… OP_ENDIF` wrapping any number of data pushes. Because envelopes are " +"effectively no-ops, they do not change the semantics of the script in which " +"they are included, and can be combined with any other locking script." +msgstr "" +"銘文の内容は、実行されていない条件のデータプッシュを使用してシリアル化され \"エンベロープ\"と呼ばれます。" +"封筒はOP_FALSE OP_IF…OP_ENDIF任意の数のデータプッシュを構成し、ラップします。" +"エンベロープは実際には空の操作なので、それらを含むスクリプトのセマンティクスは変更されません。" +"となり、他のロックスクリプトと組み合わせて使用できます。" + +#: src/inscriptions.md:39 +msgid "" +"A text inscription containing the string \"Hello, world!\" is serialized as " +"follows:" +msgstr "" +"文字列「Hello,world!」を含むのテキスト銘文を以下のようにシリアル化されています:" + + +#: src/inscriptions.md:42 +msgid "" +"```\n" +"OP_FALSE\n" +"OP_IF\n" +" OP_PUSH \"ord\"\n" +" OP_PUSH 1\n" +" OP_PUSH \"text/plain;charset=utf-8\"\n" +" OP_PUSH 0\n" +" OP_PUSH \"Hello, world!\"\n" +"OP_ENDIF\n" +"```" +msgstr "" + +#: src/inscriptions.md:53 +msgid "" +"First the string `ord` is pushed, to disambiguate inscriptions from other " +"uses of envelopes." +msgstr "" +"まず文字列'ord'がプッシュされ、銘文と封筒の他の用途との曖昧さを解消します。" + +#: src/inscriptions.md:56 +msgid "" +"`OP_PUSH 1` indicates that the next push contains the content type, and " +"`OP_PUSH 0` indicates that subsequent data pushes contain the content " +"itself. Multiple data pushes must be used for large inscriptions, as one of " +"taproot's few restrictions is that individual data pushes may not be larger " +"than 520 bytes." +msgstr "" +"`OP_PUSH 1'は、次のプッシュにコンテンツタイプが含まれていることを示します。" +"'OP_PUSH 0'は、後続のデータプッシュがコンテンツ自体を含むことを示す。" +"の大規模な銘文は複数回のデータプッシュを使用しなければならない。taprootの少数の制限の一つは、単一のデータプッシュが520バイトを超えないことであります。" + + +#: src/inscriptions.md:61 +msgid "" +"The inscription content is contained within the input of a reveal " +"transaction, and the inscription is made on the first sat of its input. This " +"sat can then be tracked using the familiar rules of ordinal theory, allowing " +"it to be transferred, bought, sold, lost to fees, and recovered." +msgstr "" +"銘文の内容はreveal取引の入力に含まれ、銘文はその最初の出力の最初の聡(Satoshi)に刻まれている。" +"私たちはよく知っている序数理論のルールを使ってこの聡satを追跡することができます。移転、購入、販売、紛失、回復を許可します。" + +#: src/inscriptions.md:66 +msgid "Content" +msgstr "内容" + +#: src/inscriptions.md:69 +msgid "" +"The data model of inscriptions is that of a HTTP response, allowing " +"inscription content to be served by a web server and viewed in a web browser." +msgstr "" +"銘文のデータモデルはHTTP応答のデータモデルであり、銘文がウェブサーバによってサービスされ、ウェブブラウザで閲覧することを可能にします。" + +#: src/inscriptions.md:72 +msgid "Fields" +msgstr "フィールド" + +#: src/inscriptions.md:75 +msgid "" +"Inscriptions may include fields before an optional body. Each field consists " +"of two data pushes, a tag and a value." +msgstr "" +"銘文には、オプションの本文の前にフィールドを含めることができます。各フィールドには" +"の2つのデータプッシュ、1つのラベルと1つの値。" + +#: src/inscriptions.md:78 +msgid "" +"Currently, the only defined field is `content-type`, with a tag of `1`, " +"whose value is the MIME type of the body." +msgstr "" +"現在、定義されている唯一のフィールドは『content-type』です。タグは「1」、" +"その値は本文のMI MEタイプです。" + +#: src/inscriptions.md:81 +msgid "" +"The beginning of the body and end of fields is indicated with an empty data " +"push." +msgstr "" +"本文の先頭とフィールドの末尾には \"空のデータ\"でプッシュが表示されます。" + +#: src/inscriptions.md:84 +msgid "" +"Unrecognized tags are interpreted differently depending on whether they are " +"even or odd, following the \"it's okay to be odd\" rule used by the " +"Lightning Network." +msgstr "" +"認識できないラベルの解釈は、偶数か奇数かによって異なり、稲妻ネットワーク\"は奇数\"にすることができるというルールに従います。" + +#: src/inscriptions.md:88 +msgid "" +"Even tags are used for fields which may affect creation, initial assignment, " +"or transfer of an inscription. Thus, inscriptions with unrecognized even " +"fields must be displayed as \"unbound\", that is, without a location." +msgstr "" +"ラベルさえも、作成、初期割り当て、または銘文の転送に影響を及ぼす可能性のあるフィールドのために使用されます。だから、識別できない銘文でも、" +"フィールドも\"未バインド\"として表示されなければなりません。つまり、場所がありません。" + + +#: src/inscriptions.md:92 +msgid "" +"Odd tags are used for fields which do not affect creation, initial " +"assignment, or transfer, such as additional metadata, and thus are safe to " +"ignore." +msgstr "" +"奇数タグは、追加のメタデータなど、作成、初期フィールド、割り当て、または転送に影響を与えないために使用され、したがって、無視することが安全であることを選択します。" + +#: src/inscriptions.md:95 +msgid "Inscription IDs" +msgstr "銘文の身分ID" + +#: src/inscriptions.md:98 +msgid "" +"The inscriptions are contained within the inputs of a reveal transaction. In " +"order to uniquely identify them they are assigned an ID of the form:" +msgstr "" +"銘文は取引を明らかにする入力に含まれている。彼らを一意に識別するために、彼らには以下の形式のIDが割り当てられています:" + +#: src/inscriptions.md:101 +msgid "`521f8eccffa4c41a3a7728dd012ea5a4a02feed81f41159231251ecf1e5c79dai0`" +msgstr "" + +#: src/inscriptions.md:103 +msgid "" +"The part in front of the `i` is the transaction ID (`txid`) of the reveal " +"transaction. The number after the `i` defines the index (starting at 0) of " +"new inscriptions being inscribed in the reveal transaction." +msgstr "" +" ’i’の先頭部分は、トランザクションID(’txid’)である。`i`の後の数字はを定義します。" +"新しい銘文は、取引で常に銘記されたインデックスの位置(0から開始)" + +#: src/inscriptions.md:107 +msgid "" +"Inscriptions can either be located in different inputs, within the same " +"input or a combination of both. In any case the ordering is clear, since a " +"parser would go through the inputs consecutively and look for all " +"inscription `envelopes`." +msgstr "" +"銘文は、入力の中の異なる入力の中にあってもよく、同じ入力であってもよく、あるいは両者の組み合わせであってもよい。" +"いずれにしても、パーサーは入力を連続的にチェックし、すべての銘文『封筒を検索するため、順序は明確です" + +#: src/inscriptions.md:111 +msgid "Input" +msgstr "" + +#: src/inscriptions.md:111 +msgid "Inscription Count" +msgstr "" + +#: src/inscriptions.md:111 +msgid "Indices" +msgstr "" + +#: src/inscriptions.md:113 src/inscriptions.md:116 +msgid "0" +msgstr "" + +#: src/inscriptions.md:113 src/inscriptions.md:115 +msgid "2" +msgstr "" + +#: src/inscriptions.md:113 +msgid "i0, i1" +msgstr "" + +#: src/inscriptions.md:114 src/inscriptions.md:117 +msgid "1" +msgstr "" + +#: src/inscriptions.md:114 +msgid "i2" +msgstr "" + +#: src/inscriptions.md:115 src/inscriptions.md:116 +msgid "3" +msgstr "" + +#: src/inscriptions.md:115 +msgid "i3, i4, i5" +msgstr "" + +#: src/inscriptions.md:117 +msgid "4" +msgstr "" + +#: src/inscriptions.md:117 +msgid "i6" +msgstr "" + +#: src/inscriptions.md:119 +msgid "Sandboxing" +msgstr "サンドボックス化" + +#: src/inscriptions.md:122 +msgid "" +"HTML and SVG inscriptions are sandboxed in order to prevent references to " +"off-chain content, thus keeping inscriptions immutable and self-contained." +msgstr "" +"HTML と SVG 銘文はサンドボックス化され、チェーンの下の内容を引用しないようにして、銘文の不変性と独立性を保ちます。" + +#: src/inscriptions.md:125 +msgid "" +"This is accomplished by loading HTML and SVG inscriptions inside `iframes` " +"with the `sandbox` attribute, as well as serving inscription content with " +"`Content-Security-Policy` headers." +msgstr "" +"これは、HTMLとSVGの銘文を `iframes` にロードし、銘文コンテンツを提供することによって行われる'sandbox'属性です。" +"Content-Security-Policy”ヘッダ。" + +#: src/inscriptions/recursion.md:4 +msgid "" +"An important exception to [sandboxing](../inscriptions.md#sandboxing) is " +"recursion: access to `ord`'s `/content` endpoint is permitted, allowing " +"inscriptions to access the content of other inscriptions by requesting `/" +"content/`." +msgstr "" +"[サンドボックス化](../inscriptions.md#sandboxing)の重要な例外は再帰です:" +"`ord`の /content`エンドポイントへのアクセスが許可され、碑文が`/content/`" +"をリクエストすることで他の碑文のコンテンツにアクセスできるようになります。" + +#: src/inscriptions/recursion.md:8 +msgid "This has a number of interesting use-cases:" +msgstr "ここは面白い用例がとても多いです:" + +#: src/inscriptions/recursion.md:10 +msgid "Remixing the content of existing inscriptions." +msgstr "既存の銘文の内容を再混合します。" + +#: src/inscriptions/recursion.md:12 +msgid "" +"Publishing snippets of code, images, audio, or stylesheets as shared public " +"resources." +msgstr "コード、イメージ、オーディオ、またはスタイルシートフラグメントを共通の共有アセットとしてパブリッシュします。" + +#: src/inscriptions/recursion.md:15 +msgid "" +"Generative art collections where an algorithm is inscribed as JavaScript, " +"and instantiated from multiple inscriptions with unique seeds." +msgstr "アルゴリズムがJava Scriptを使用して刻印され、ユニークなシードを持つ複数の銘文からインスタンス化されたアートコレクションを生成します。" + +#: src/inscriptions/recursion.md:18 +msgid "" +"Generative profile picture collections where accessories and attributes are " +"inscribed as individual images, or in a shared texture atlas, and then " +"combined, collage-style, in unique combinations in multiple inscriptions." +msgstr "" +"アクセサリーやプロパティを含むプロファイル画像セットを生成し、個別の画像として書き込むか、共有テクスチャマップセットに書き込み、" +"の組み合わせ、コラージュスタイル、複数の銘文の中で独特の組み合わせで。" + +#: src/inscriptions/recursion.md:22 +msgid "A few other endpoints that inscriptions may access are the following:" +msgstr "銘文がアクセスできる他のいくつかのエンドポイントは以下の通りです:" + +#: src/inscriptions/recursion.md:24 +msgid "`/blockheight`: latest block height." +msgstr "`/blockheight`:最新のブロック高度。" + +#: src/inscriptions/recursion.md:25 +msgid "`/blockhash`: latest block hash." +msgstr "`/blockhash`:最新のブロックハッシュ。" + +#: src/inscriptions/recursion.md:26 +msgid "`/blockhash/`: block hash at given block height." +msgstr "`/blockhash/`:指定されたブロック高さのブロックハッシュ。" + +#: src/inscriptions/recursion.md:27 +msgid "`/blocktime`: UNIX time stamp of latest block." +msgstr "`/blocktime`:最新のブロックのUNIXタイムスタンプ。" + +#: src/faq.md:1 +msgid "Ordinal Theory FAQ" +msgstr "序数理論のよくある質問" + +#: src/faq.md:4 +msgid "What is ordinal theory?" +msgstr "序数理論はなんですか" + +#: src/faq.md:7 +msgid "" +"Ordinal theory is a protocol for assigning serial numbers to satoshis, the " +"smallest subdivision of a bitcoin, and tracking those satoshis as they are " +"spent by transactions." +msgstr "" +"序数理論は、聡(satoshi、以下は「聡」と書きます。ビットコインの最小単位)シリアル番号を割り当てるプロトコル、" +"と取引中にこれらの聡を追跡します。" + +#: src/faq.md:11 +msgid "" +"These serial numbers are large numbers, like this 804766073970493. Every " +"satoshi, which is ¹⁄₁₀₀₀₀₀₀₀₀ of a bitcoin, has an ordinal number." +msgstr "" +"これらの番号はすべて大きな数字で、例えば、804766073970493.すべての聡satoshi、" +"もビットコインの¹⁄₁₀₀₀₀₀₀₀₀はすべて序数番号を持っています" + +#: src/faq.md:14 +msgid "" +"Does ordinal theory require a side chain, a separate token, or changes to " +"Bitcoin?" +msgstr "" +"序数理論はサイドチェーン、個別のトークン、またはビットコインに変更を加える必要がありますか?" + +#: src/faq.md:17 +msgid "" +"Nope! Ordinal theory works right now, without a side chain, and the only " +"token needed is bitcoin itself." +msgstr "" +"全然いりません!序数理論は現在有効に利用可能で、サイドチェーンがなく、唯一必要なトークンはビットコイン自体であります。" + +#: src/faq.md:20 +msgid "What is ordinal theory good for?" +msgstr "" +"序数理論はどんな用途がありますか?" + +#: src/faq.md:23 +msgid "" +"Collecting, trading, and scheming. Ordinal theory assigns identities to " +"individual satoshis, allowing them to be individually tracked and traded, as " +"curios and for numismatic value." +msgstr "" +"収集、取引、企画。序数理論は身分を個々の聡に割り当て、骨董品や貨幣価値として個別に追跡され取引されることを可能にします。" + +#: src/faq.md:27 +msgid "" +"Ordinal theory also enables inscriptions, a protocol for attaching arbitrary " +"content to individual satoshis, turning them into bitcoin-native digital " +"artifacts." +msgstr "" +"序数理論はまた、任意の内容を単一のコングに付加し、ビットコインのネイティブデジタルアーティファクトにするプロトコルである銘文を与えます。" + +#: src/faq.md:31 +msgid "How does ordinal theory work?" +msgstr "" +"序数理論はどのように機能しますか?" + +#: src/faq.md:34 +msgid "" +"Ordinal numbers are assigned to satoshis in the order in which they are " +"mined. The first satoshi in the first block has ordinal number 0, the second " +"has ordinal number 1, and the last satoshi of the first block has ordinal " +"number 4,999,999,999." +msgstr "" +"序数は採掘の順序に従って聡に割り当てられた。最初のブロックの最初のリスニングの序数は0で、2番目のリスニングの序数は1です。" +"最初のブロックの最後のセグメントの序数は4,999,999,999です。" + +#: src/faq.md:39 +msgid "" +"Satoshis live in outputs, but transactions destroy outputs and create new " +"ones, so ordinal theory uses an algorithm to determine how satoshis hop from " +"the inputs of a transaction to its outputs." +msgstr "" +"コングは出力に存在しますが、トランザクションは出力を破壊し、新しい出力を作成します。" +"したがって、序数理論では、取引の入力から出力にどのようにジャンプするかを決定するアルゴリズ" +"ムを使用します。" + +#: src/faq.md:43 +msgid "Fortunately, that algorithm is very simple." +msgstr "" +"幸いなことに、このアルゴリズムはとても簡単です。" + +#: src/faq.md:45 +msgid "" +"Satoshis transfer in first-in-first-out order. Think of the inputs to a " +"transaction as being a list of satoshis, and the outputs as a list of slots, " +"waiting to receive a satoshi. To assign input satoshis to slots, go through " +"each satoshi in the inputs in order, and assign each to the first available " +"slot in the outputs." +msgstr "" +"聡は先入れ先出しの順に振り込みを行う。トランザクションの入力をスマートリストとして、出力をスロットスロットリストとして、" +"聡の受信を待つ。入力セグメントをスロットに割り当てるには、入力セグメントの各セグメントを順番にチェックします。" +"と表示され、出力の最初の使用可能なスロットに各スマートが割り当てられます。。" + +#: src/faq.md:51 +msgid "" +"Let's imagine a transaction with three inputs and two outputs. The inputs " +"are on the left of the arrow and the outputs are on the right, all labeled " +"with their values:" +msgstr "" +"3つの入力と2つの出力を持つ取引を想像してみましょう。入力は矢印の左側にあり、出力は右側にあり、値のラベルが付いています:" + +#: src/faq.md:55 +msgid "" +"```\n" +"[2] [1] [3] → [4] [2]\n" +"```" +msgstr "" + +#: src/faq.md:57 +msgid "" +"Now let's label the same transaction with the ordinal numbers of the " +"satoshis that each input contains, and question marks for each output slot. " +"Ordinal numbers are large, so let's use letters to represent them:" +msgstr "" +"今は、各入力に含まれる小数点以下の番号で同じトランザクションをマークし、各出力スロットに疑問符をマークします。" +"序数は大きいので、それらをアルファベットで表します:" + +#: src/faq.md:61 +msgid "" +"```\n" +"[a b] [c] [d e f] → [? ? ? ?] [? ?]\n" +"```" +msgstr "" + +#: src/faq.md:63 +msgid "" +"To figure out which satoshi goes to which output, go through the input " +"satoshis in order and assign each to a question mark:" +msgstr "" +"どの出力からどの出力への出力かを確認するには、入力の出力を順番にチェックし、各出力に疑問符を割り当てます:" + +#: src/faq.md:66 +msgid "" +"```\n" +"[a b] [c] [d e f] → [a b c d] [e f]\n" +"```" +msgstr "" + +#: src/faq.md:68 +msgid "" +"What about fees, you might ask? Good question! Let's imagine the same " +"transaction, this time with a two satoshi fee. Transactions with fees send " +"more satoshis in the inputs than are received by the outputs, so to make our " +"transaction into one that pays fees, we'll remove the second output:" +msgstr "" +"取引費用を聞くかもしれませんか?いい質問だ!同じ取引を想像してみましょう。今回は2人の聡の費用です。" +"有料取引は、で送信された音声が出力で受信された音声より多いので、私たちの取引を料金を支払う取引にするために、" +"2番目の出力を削除します:" + +#: src/faq.md:73 +msgid "" +"```\n" +"[2] [1] [3] → [4]\n" +"```" +msgstr "" + +#: src/faq.md:75 +msgid "The satoshis " +msgstr "聡" + +#: src/faq.md:75 +msgid "e" +msgstr "" + +#: src/faq.md:75 +msgid " and " +msgstr " と " + +#: src/faq.md:75 +msgid "f" +msgstr "" + +#: src/faq.md:75 +msgid " now have nowhere to go in the outputs:" +msgstr "今は輸出しているので、どこへも行けません" + +#: src/faq.md:78 +msgid "" +"```\n" +"[a b] [c] [d e f] → [a b c d]\n" +"```" +msgstr "" +"```\n" +"[a b] [c] [d e f] → [a b c d]\n" +"```" + +#: src/faq.md:80 +msgid "" +"So they go to the miner who mined the block as fees. [The BIP](https://" +"github.com/ordinals/ord/blob/master/bip.mediawiki) has the details, but in " +"short, fees paid by transactions are treated as extra inputs to the coinbase " +"transaction, and are ordered how their corresponding transactions are " +"ordered in the block. The coinbase transaction of the block might look like " +"this:" +msgstr "" +"だから彼らは「費用」としてこのブロックを掘っている鉱山労働者のところに行きます。[The BIP]" +"(https://github.com/ordinals/ord/blob/master/bip.mediawiki)にはもっと詳しい説明があります。" +"しかし、簡単に言えば、取引によって支払われる料金は、Coinbase取引への追加入力とみなされ、その対応する取引に応じてブロック内にあります" +"の順にソートされます。ブロックのCoinbaseトランザクションは次のようになる可能性があります:" + +#: src/faq.md:87 +msgid "" +"```\n" +"[SUBSIDY] [e f] → [SUBSIDY e f]\n" +"```" +msgstr "" + +#: src/faq.md:89 +msgid "Where can I find the nitty-gritty details?" +msgstr "これらの詳細な情報はどこで見つけることができますか" + +#: src/faq.md:92 +msgid "[The BIP!](https://github.com/ordinals/ord/blob/master/bip.mediawiki)" +msgstr "" + +#: src/faq.md:94 +msgid "" +"Why are sat inscriptions called \"digital artifacts\" instead of \"NFTs\"?" +msgstr "なぜ聡の銘文は「NFT」ではなく「デジタル文化財」と呼ばれていますか?" + +#: src/faq.md:97 +msgid "" +"An inscription is an NFT, but the term \"digital artifact\" is used instead, " +"because it's simple, suggestive, and familiar." +msgstr "銘文もNFTの一種です。しかし、その代わりに「デジタル文化財」という用語が使われています。それは簡単で、啓発的で、なじみがあるからです" + +#: src/faq.md:100 +msgid "" +"The phrase \"digital artifact\" is highly suggestive, even to someone who " +"has never heard the term before. In comparison, NFT is an acronym, and " +"doesn't provide any indication of what it means if you haven't heard the " +"term before." +msgstr "" +"\"デジタル文化財\"(デジタル工作物、デジタル人工物)これらの言葉は強い暗示性を持っている。" +"「これまでこの言葉を聞いたことがない人にとってもそうだ」" +"に比べて、NFTは頭文字の略語で、以前この用語を聞いたことがなければ、意味を説明できません。" + +#: src/faq.md:104 +msgid "" +"Additionally, \"NFT\" feels like financial terminology, and the both word " +"\"fungible\" and sense of the word \"token\" as used in \"NFT\" is uncommon " +"outside of financial contexts." +msgstr "" +"さらに、\"NFT\"は金融用語のような気がします。\"NFT\"で使われている\"同質化\"という言葉と" +"\"トークン\"という言葉の意味は金融の文脈の外では一般的ではありません。" + +#: src/faq.md:108 +msgid "How do sat inscriptions compare to…" +msgstr "聡は他のルーンと比較します" + +#: src/faq.md:111 +msgid "Ethereum NFTs?" +msgstr "エセリウムNFT" + +#: src/faq.md:113 +msgid "_Inscriptions are always immutable._" +msgstr "_銘文は永遠に変わりません_" + +#: src/faq.md:115 +msgid "" +"There is simply no way to for the creator of an inscription, or the owner of " +"an inscription, to modify it after it has been created." +msgstr "" +"銘文の作成者や所有者は、銘文を作成した後に修正することはできません。" + +#: src/faq.md:118 +msgid "" +"Ethereum NFTs _can_ be immutable, but many are not, and can be changed or " +"deleted by the NFT contract owner." +msgstr "" +"エセリウムNFTs_は変更不可能であることができますが、多くはそうではなく、NFT契約者によって変更" +"または削除することができます。。" + +#: src/faq.md:121 +msgid "" +"In order to make sure that a particular Ethereum NFT is immutable, the " +"contract code must be audited, which requires detailed knowledge of the EVM " +"and Solidity semantics." +msgstr "" +"特定のエセリウムNFTが不変であることを確実にするためには、契約コードを監査しなければなりません。" +"これにはEVMとSolidityセマンティクスの詳細な理解が必要です。" + +#: src/faq.md:125 +msgid "" +"It is very hard for a non-technical user to determine whether or not a given " +"Ethereum NFT is mutable or immutable, and Ethereum NFT platforms make no " +"effort to distinguish whether an NFT is mutable or immutable, and whether " +"the contract source code is available and has been audited." +msgstr "" +"技術ユーザーではない人にとって、あるイーサリアムNFTが可変かどうかを判断することは困難であり、" +"イーサリアムNFTプラットフォームもNFTが可変かどうかを区別する努力をしていません。" +"と、契約ソースコードが利用可能で監査されているかどうかを確認します。" + +#: src/faq.md:130 +msgid "_Inscription content is always on-chain._" +msgstr "_銘文の内容は永遠にコンタクトします_" + +#: src/faq.md:132 +msgid "" +"There is no way for an inscription to refer to off-chain content. This makes " +"inscriptions more durable, because content cannot be lost, and scarcer, " +"because inscription creators must pay fees proportional to the size of the " +"content." +msgstr "" +"銘文はチェーンの下の内容を引用できない。内容が失われないので、銘文はより長持ちします。" +"また、コンテンツのサイズに比例した費用を銘文作成者に支払わなければなりません。" + +#: src/faq.md:136 +msgid "" +"Some Ethereum NFT content is on-chain, but much is off-chain, and is stored " +"on platforms like IPFS or Arweave, or on traditional, fully centralized web " +"servers. Content on IPFS is not guaranteed to continue to be available, and " +"some NFT content stored on IPFS has already been lost. Platforms like " +"Arweave rely on weak economic assumptions, and will likely fail " +"catastrophically when these economic assumptions are no longer met. " +"Centralized web servers may disappear at any time." +msgstr "" +"イーサリアムのNFTコンテンツの中にはチェーン上にあるものもありますが、ほとんどのコンテンツはチェーンの下にあり、IP FSやArweaveなどのプラットフォームに保存されています。" +"または従来の完全に中心化されたネットワークサーバー上で。IP FSのコンテンツは引き続き利用できることを保証していません。IP FSに保存されているNFTコンテンツの一部は失われました。" +"Arweaveのようなプラットフォームは弱い経済的仮定に依存しており、これらの経済的仮定が満たされなくなったとき" +"は壊滅的な失敗を起こす可能性が高い。一元化されたネットワークサーバはいつでも消える可能性があ。" + +#: src/faq.md:144 +msgid "" +"It is very hard for a non-technical user to determine where the content of a " +"given Ethereum NFT is stored." +msgstr "" +"技術ユーザーではない人にとって、イーサリアムNFTのコンテンツがどこに保存されているかを特定することは困難です。" + +#: src/faq.md:147 +msgid "_Inscriptions are much simpler._" +msgstr "_めいぶんはよりかんたんであります_" + +#: src/faq.md:149 +msgid "" +"Ethereum NFTs depend on the Ethereum network and virtual machine, which are " +"highly complex, constantly changing, and which introduce changes via " +"backwards-incompatible hard forks." +msgstr "" +"イーサリアムNFTはイーサリアムネットワークと仮想マシンに依存しており、高度に複雑で変化し続け、下位互換性のないハードフォークによって変化をもたらします。" + +#: src/faq.md:153 +msgid "" +"Inscriptions, on the other hand, depend on the Bitcoin blockchain, which is " +"relatively simple and conservative, and which introduces changes via " +"backwards-compatible soft forks." +msgstr "" +"対照的に、銘文はビットコインブロックチェーンに依存しており、比較的単純で保守的であり、後方互換性のあるソフトフォークによって変化を導入しています。" + +#: src/faq.md:157 +msgid "_Inscriptions are more secure._" +msgstr "_銘文はより安全です_" + +#: src/faq.md:159 +msgid "" +"Inscriptions inherit Bitcoin's transaction model, which allow a user to see " +"exactly which inscriptions are being transferred by a transaction before " +"they sign it. Inscriptions can be offered for sale using partially signed " +"transactions, which don't require allowing a third party, such as an " +"exchange or marketplace, to transfer them on the user's behalf." +msgstr "" +"銘文はビットコインの取引モデルを継承しており、ユーザーは署名する前に取引でどの銘文が転送されたかを正確に見ることができる。" +"銘文は署名取引の一部を使用して販売することができ、取引所や市場などの第三者がユーザーの代わりにそれらを譲渡することを許可する必要はありません。" + +#: src/faq.md:165 +msgid "" +"By comparison, Ethereum NFTs are plagued with end-user security " +"vulnerabilities. It is commonplace to blind-sign transactions, grant third-" +"party apps unlimited permissions over a user's NFTs, and interact with " +"complex and unpredictable smart contracts. This creates a minefield of " +"hazards for Ethereum NFT users which are simply not a concern for ordinal " +"theorists." +msgstr "" +"対照的に、イーサリアムNFTはエンドユーザーのセキュリティホールに悩まされています。ブラインドサイン取引、ユーザーにNFTの第三者アプリケーションを付与" +"の無限の権限と、複雑で予測不可能なスマートコントラクトとのやり取りは当たり前のことです。これはイーサリアムNFTユーザーのために作られた" +"危険地雷原をしていますが、これらは番号理論家にとっては心配する必要はありません。" + +#: src/faq.md:171 +msgid "_Inscriptions are scarcer._" +msgstr "_銘文はより珍しいです_" + +#: src/faq.md:173 +msgid "" +"Inscriptions require bitcoin to mint, transfer, and store. This seems like a " +"downside on the surface, but the raison d'etre of digital artifacts is to be " +"scarce and thus valuable." +msgstr "" +"銘文は鋳造、移転、保存するためにビットコインが必要です。表面的には、これは障害のように見えるが、デジタル文化財の存在の価値目的は希少であります。" + +#: src/faq.md:177 +msgid "" +"Ethereum NFTs, on the other hand, can be minted in virtually unlimited " +"qualities with a single transaction, making them inherently less scarce, and " +"thus, potentially less valuable." +msgstr "" +"一方、イーサリアムNFTは1回の取引でほぼ無限の品質で鋳造することができ、本質的にそれほど希少ではないです。" +"そのため、あまり価値がないかもしれません。" + +#: src/faq.md:181 +msgid "_Inscriptions do not pretend to support on-chain royalties._" +msgstr "_銘文はチェーン上の版税を支持するふりをしません_" + +#: src/faq.md:183 +msgid "" +"On-chain royalties are a good idea in theory but not in practice. Royalty " +"payment cannot be enforced on-chain without complex and invasive " +"restrictions. The Ethereum NFT ecosystem is currently grappling with " +"confusion around royalties, and is collectively coming to grips with the " +"reality that on-chain royalties, which were messaged to artists as an " +"advantage of NFTs, are not possible, while platforms race to the bottom and " +"remove royalty support." +msgstr "" +"「チェーン上の版税」は理論的には良い考えですが、実際にはうまくいきません。複雑で侵入的な制限がなければ" +"チェーン上の版税の支払いを強制することはできません。イーサリアムNFTエコシステムは、版税をめぐる問題を解決するために努力しています。" +"そして、は共に一つの現実に直面しています。つまり、NFTチェーン上の版税という利器を芸術家に伝えることは実際には不可能です。" +"同時に、複数のプラットフォームが版税のサポートを削除するために競争しています" + +#: src/faq.md:190 +msgid "" +"Inscriptions avoid this situation entirely by making no false promises of " +"supporting royalties on-chain, thus avoiding the confusion, chaos, and " +"negativity of the Ethereum NFT situation." +msgstr "" +"銘文はこの状況を完全に迴避し、チェーン上のロイヤリティをサポートすることを虚偽で約束していません。" +"はイーサリアムNFTのような混乱と消極的な状況を避けました。" + +#: src/faq.md:194 +msgid "_Inscriptions unlock new markets._" +msgstr "_銘文は新たな市場を開きました_" + +#: src/faq.md:196 +msgid "" +"Bitcoin's market capitalization and liquidity are greater than Ethereum's by " +"a large margin. Much of this liquidity is not available to Ethereum NFTs, " +"since many Bitcoiners prefer not to interact with the Ethereum ecosystem due " +"to concerns related to simplicity, security, and decentralization." +msgstr "" +"ビットコインの市場価格と流動性はイーサリアムを大きく上回っている。イーサリアムNFTはこのような流動性の大部分を得ることができません。" +"多くのビットコインユーザーは、シンプルさ、安全性、分散化の観点から、イーサリアムのエコシステムと対話することを望んでいないからです。" + +#: src/faq.md:201 +msgid "" +"Such Bitcoiners may be more interested in inscriptions than Ethereum NFTs, " +"unlocking new classes of collector." +msgstr "" +"イーサリアムNFTと比較して、そのようなビットコインの支持者は、新しいカテゴリのコレクターをロッ" +"ク解除するために、碑文に興味を持っている可能性があります。" + +#: src/faq.md:204 +msgid "_Inscriptions have a richer data model._" +msgstr "_铭文有更丰富的数据模型_" + +#: src/faq.md:206 +msgid "" +"Inscriptions consist of a content type, also known as a MIME type, and " +"content, which is an arbitrary byte string. This is the same data model used " +"by the web, and allows inscription content to evolve with the web, and come " +"to support any kind of content supported by web browsers, without requiring " +"changes to the underlying protocol." +msgstr "" +"銘文はコンテンツタイプ(MI MEタイプとも呼ばれる)とコンテンツ)任意のバイト文字列)で構成される。これは、Webで使用されるデータモデルと同じです。" +"銘文コンテンツをウェブの発展に合わせて発展させ、基盤となるプロトコルを変更することなく、ウェブブラウザがサポートするあらゆる種類のコンテンツをサポートすることを可能に" + +#: src/faq.md:212 +msgid "RGB and Taro assets?" +msgstr "RGB 和 Taro 資産?" + +#: src/faq.md:214 +msgid "" +"RGB and Taro are both second-layer asset protocols built on Bitcoin. " +"Compared to inscriptions, they are much more complicated, but much more " +"featureful." +msgstr "" +"RGBとTaroはいずれもビットコインの上に構築された二層資産協定である。銘文に比べて、それらはずっと複雑ですが、より特色があります。" + +#: src/faq.md:217 +msgid "" +"Ordinal theory has been designed from the ground up for digital artifacts, " +"whereas the primary use-case of RGB and Taro are fungible tokens, so the " +"user experience for inscriptions is likely to be simpler and more polished " +"than the user experience for RGB and Taro NFTs." +msgstr "" +"シリアル番号理論はデジタルアーティファクト用に設計されていますが、RGBとTaroの主なユースケースは代替トークンです。" +"そのため、銘文のユーザー体験はRGBやTaro NFTのユーザー体験よりも簡単で完璧かもしれません。" + +#: src/faq.md:222 +msgid "" +"RGB and Taro both store content off-chain, which requires additional " +"infrastructure, and which may be lost. By contrast, inscription content is " +"stored on-chain, and cannot be lost." +msgstr "" +"RGBとTaroはどちらもチェーンの下にコンテンツを保存します。これには追加のインフラが必要で、紛失する可能性があります。" +"に比べて、銘文の内容はチェーンに保存され、失われることはありません。" + +#: src/faq.md:226 +msgid "" +"Ordinal theory, RGB, and Taro are all very early, so this is speculation, " +"but ordinal theory's focus may give it the edge in terms of features for " +"digital artifacts, including a better content model, and features like " +"globally unique symbols." +msgstr "" +"序数理論、RGB、Taroは非常に初期のものなので、これは推測にすぎませんが、序数理論の重点はデジタル芸術品のにあるかもしれません" +"の特性には、より良いコンテンツモデルや、世界で唯一のシンボルのような特性などの利点があります。" + + +#: src/faq.md:231 +msgid "Counterparty assets?" +msgstr "Counterparty资产" + +#: src/faq.md:233 +msgid "" +"Counterparty has its own token, XCP, which is required for some " +"functionality, which makes most bitcoiners regard it as an altcoin, and not " +"an extension or second layer for bitcoin." +msgstr "" +"Counterpartyは独自のトークンXCPを持っています。これはいくつかの機能に必要で、ほとんどのビットコイン保有者はそれをパクリ通貨と見なしています。" +"ビットコインの拡張や第二層ではありません" + +#: src/faq.md:237 +msgid "" +"Ordinal theory has been designed from the ground up for digital artifacts, " +"whereas Counterparty was primarily designed for financial token issuance." +msgstr "" +"序数理論はデジタル文化財のために最初から設計されたのに対し、Counterpartyは主に金融トークン発行のために設計されました。" + +#: src/faq.md:240 +msgid "Inscriptions for…" +msgstr "銘文は..." + +#: src/faq.md:243 +msgid "Artists" +msgstr "芸術家になれます" + +#: src/faq.md:245 +msgid "" +"_Inscriptions are on Bitcoin._ Bitcoin is the digital currency with the " +"highest status and greatest chance of long-term survival. If you want to " +"guarantee that your art survives into the future, there is no better way to " +"publish it than as inscriptions." +msgstr "" +"_ビットコインの銘文_ビットコインは現在最も地位が高く、長期生存の機会が最も大きいデジタル通貨であります。" +"はあなたの芸術作品が未来に伝わることを保証したいなら、銘文より良い発表方法はありません。" + +#: src/faq.md:250 +msgid "" +"_Cheaper on-chain storage._ At $20,000 per BTC and the minimum relay fee of " +"1 satoshi per vbyte, publishing inscription content costs $50 per 1 million " +"bytes." +msgstr "" +"_チェーン上のストレージはより安価です_ビットコインあたり2万ドルとvbyteあたり1聡の最低中継料金で計算します。" +"碑文の内容を公開するためのコストは100万バイトあたり50ドルです。" + +#: src/faq.md:254 +msgid "" +"_Inscriptions are early!_ Inscriptions are still in development, and have " +"not yet launched on mainnet. This gives you an opportunity to be an early " +"adopter, and explore the medium as it evolves." +msgstr "" +"_铭文还处于项目早期_ 铭文仍在开发中,尚未在主网上发布(建议更新)。 " +"这使您有机会成为早期采用者,并随着媒体的发展探索它。" + +#: src/faq.md:258 +msgid "" +"_Inscriptions are simple._ Inscriptions do not require writing or " +"understanding smart contracts." +msgstr "" +"_铭文很简单_ 铭文不需要你编写或理解智能合约。" + +#: src/faq.md:261 +msgid "" +"_Inscriptions unlock new liquidity._ Inscriptions are more accessible and " +"appealing to bitcoin holders, unlocking an entirely new class of collector." +msgstr "" +"_銘文ロック解除新しい流動性_ビットコインの所有者にとって、銘文はより簡単に入手でき、より魅力的で、新しいコレクションをもたらします。" + +#: src/faq.md:264 +msgid "" +"_Inscriptions are designed for digital artifacts._ Inscriptions are designed " +"from the ground up to support NFTs, and feature a better data model, and " +"features like globally unique symbols and enhanced provenance." +msgstr "" +"_銘文はデジタル文化財のために設計されています_新しいデザインの銘文はNFTをサポートし、より良いデータモデルを持っています," +"や、世界的にユニークなシンボルや拡張ソースなどの機能です。" + +#: src/faq.md:268 +msgid "" +"_Inscriptions do not support on-chain royalties._ This is negative, but only " +"depending on how you look at it. On-chain royalties have been a boon for " +"creators, but have also created a huge amount of confusion in the Ethereum " +"NFT ecosystem. The ecosystem now grapples with this issue, and is engaged in " +"a race to the bottom, towards a royalties-optional future. Inscriptions have " +"no support for on-chain royalties, because they are technically infeasible. " +"If you choose to create inscriptions, there are many ways you can work " +"around this limitation: withhold a portion of your inscriptions for future " +"sale, to benefit from future appreciation, or perhaps offer perks for users " +"who respect optional royalties." +msgstr "" +"_銘文はチェーン上のロイヤリティを奨励しない_これは良いニュースではないかもしれませんが、それをどう見るかによっても異なります。チェーン上のロイヤリティは常にクリエイターにとっての福音です" +"しかし、はイーサリアムNFT生態系にも大きな混乱をもたらした。イーサリアムは今この問題を解決しようと努力しています。底をついた競争でもあります。" +"は「オプションのロイヤリティ」の未来を実現する。銘文はチェーン上のロイヤリティをサポートしていません。技術的に不可能だからです。」" +"銘文を作成することを選択した場合、この制限を迴避する方法はたくさんあります。銘文の一部を将来の販売のために残して、将来の上昇から利益を得ます。" +"またはオプションのロイヤリティを尊重するユーザーに追加の手当を提供します。" + +#: src/faq.md:279 +msgid "Collectors" +msgstr "コレクター" + +#: src/faq.md:281 +msgid "" +"_Inscriptions are simple, clear, and have no surprises._ They are always " +"immutable and on-chain, with no special due diligence required." +msgstr "" +"_銘文は簡単です_、はっきりしていて事故はありません*それらは常に不変でチェーン上にあり、特別なデューデリジェンスは必要ありません。" + +#: src/faq.md:284 +msgid "" +"_Inscriptions are on Bitcoin._ You can verify the location and properties of " +"inscriptions easily with Bitcoin full node that you control." +msgstr "" +"_ビットコイン上の銘文_あなたが制御するビットコインのフルノードを使用して、銘文の位置と属性を簡単に検証できます。" + +#: src/faq.md:287 +msgid "Bitcoiners" +msgstr "ビットコインの信仰者" + +#: src/faq.md:289 +msgid "" +"Let me begin this section by saying: the most important thing that the " +"Bitcoin network does is decentralize money. All other use-cases are " +"secondary, including ordinal theory. The developers of ordinal theory " +"understand and acknowledge this, and believe that ordinal theory helps, at " +"least in a small way, Bitcoin's primary mission." +msgstr "" +"冒頭で説明しましょう。ビットコインネットワークが行っている最も重要なことは、通貨の中心化です。" +"の他のすべてのユースケースは、序数理論を含む副次的なものです。序数理論の開発者はこれを理解し、認めています。" +"そして、序数理論はビットコインの主要な課題に少なくともわずかに寄与すると信じています。" + +#: src/faq.md:295 +msgid "" +"Unlike many other things in the altcoin space, digital artifacts have merit. " +"There are, of course, a great deal of NFTs that are ugly, stupid, and " +"fraudulent. However, there are many that are fantastically creative, and " +"creating and collecting art has been a part of the human story since its " +"inception, and predates even trade and money, which are also ancient " +"technologies." +msgstr "" +"他のパクリ通貨分野のものとは異なり、デジタル文化財には利点がある。もちろん、多くのNFTは醜い、愚か、詐欺的です。" +"しかし、素晴らしいアイデアはたくさんあります。芸術を創造し、収蔵することはもともと人間の物語の一部です。" +"は貿易やお金といった同じ古い技術よりも早いです。" + +#: src/faq.md:302 +msgid "" +"Bitcoin provides an amazing platform for creating and collecting digital " +"artifacts in a secure, decentralized way, that protects users and artists in " +"the same way that it provides an amazing platform for sending and receiving " +"value, and for all the same reasons." +msgstr "" +"ビットコインは、安全で分散化された方法でデジタルアーティファクトを作成し、収集するための素晴らしいプラットフォームを提供しています。" +"も同様にユーザーとアーティストを保護し、同時に価値を送受信する優れたプラットフォームを提供しています。" + +#: src/faq.md:307 +msgid "" +"Ordinals and inscriptions increase demand for Bitcoin block space, which " +"increase Bitcoin's security budget, which is vital for safeguarding " +"Bitcoin's transition to a fee-dependent security model, as the block subsidy " +"is halved into insignificance." +msgstr "" +"序数と銘文はビットコインのブロックスペースの必要性を高め、ビットコインのセキュリティ予算を増加させた。" +"はビットコインが費用依存型のセキュリティモデルに移行することを保障するために重要であります。ブロック補助金の半減はわずかになったからであります。" + +#: src/faq.md:312 +msgid "" +"Inscription content is stored on-chain, and thus the demand for block space " +"for use in inscriptions is unlimited. This creates a buyer of last resort " +"for _all_ Bitcoin block space. This will help support a robust fee market, " +"which ensures that Bitcoin remains secure." +msgstr "" +"銘文の内容はチェーンに保存されているので、銘文のためのブロック空間の需要は無限であります。" +"これにより、すべてのビットコインのブロックスペースに最後の買い手が生まれます。" +"これは強力な有料市場をサポートし、ビットコインが常に安全であることを保証するのに役立ちます。" + +#: src/faq.md:317 +msgid "" +"Inscriptions also counter the narrative that Bitcoin cannot be extended or " +"used for new use-cases. If you follow projects like DLCs, Fedimint, " +"Lightning, Taro, and RGB, you know that this narrative is false, but " +"inscriptions provide a counter argument which is easy to understand, and " +"which targets a popular and proven use case, NFTs, which makes it highly " +"legible." +msgstr "" +"銘文はまた、ビットコインは拡張できない、または新しいユースケースに使用できないという説に反論した。DLC、Fedimint、Lightning、に注目すると" +"TaroやRGBなどのプロジェクトをすると、この言い方が間違っていることがわかります。碑文は、理解しやすい反論を提供しています。" +"そして、一般的で実証済みのユースケースについて:NFT,これにより、非常に理解しやすくなります。" + +#: src/faq.md:323 +msgid "" +"If inscriptions prove, as the authors hope, to be highly sought after " +"digital artifacts with a rich history, they will serve as a powerful hook " +"for Bitcoin adoption: come for the fun, rich art, stay for the decentralized " +"digital money." +msgstr "" +"作者が望むように、銘文が豊富な歴史を持つデジタル文化財であることが証明され、高く支持されていれば" +"それらはビットコインに採用される強力な魅力になります:楽しさ、豊かな芸術に惹かれて、" +"脱中心化されたデジタル通貨のためにも残りたいです。" + +#: src/faq.md:327 +msgid "" +"Inscriptions are an extremely benign source of demand for block space. " +"Unlike, for example, stablecoins, which potentially give large stablecoin " +"issuers influence over the future of Bitcoin development, or DeFi, which " +"might centralize mining by introducing opportunities for MEV, digital art " +"and collectables on Bitcoin, are unlikely to produce individual entities " +"with enough power to corrupt Bitcoin. Art is decentralized." +msgstr "" +"銘文はブロック空間需要の極めて健全な源であり、安定通貨とは異なり、大手発行者がビットコインの未来をする可能性があります" +"発展に影響を与えて;DeFiとは異なり、ビットコインにMEV、デジタルアート、コレクションを導入する機会を通じて、マイニングを集中化することができます。" +"芸術は脱中心化されており、いかなる実体も権力を使ってビットコインを破壊することはできません。" + +#: src/faq.md:334 +msgid "" +"Inscription users and service providers are incentivized to run Bitcoin full " +"nodes, to publish and track inscriptions, and thus throw their economic " +"weight behind the honest chain." +msgstr "" +"銘文ユーザーとサービスプロバイダーは、ビットコインのフルノードを実行し、追跡銘文を発行して、彼らの経済的重みを正直なチェーンに向けるように動機付けられています。" + +#: src/faq.md:338 +msgid "" +"Ordinal theory and inscriptions do not meaningfully affect Bitcoin's " +"fungibility. Bitcoin users can ignore both and be unaffected." +msgstr "" +"序数理論と銘文はビットコインの代替可能性に大きな影響を与えない。ビットコインユーザーはこの2つを無視しても影響を受けません。" + +#: src/faq.md:341 +msgid "" +"We hope that ordinal theory strengthens and enriches bitcoin, and gives it " +"another dimension of appeal and functionality, enabling it more effectively " +"serve its primary use case as humanity's decentralized store of value." +msgstr "" +"序数理論がビットコインを強化し、豊かにし、別の次元の魅力と機能を与えることを望んでいます。" +"人間の分散化された価値ストレージとしての主要なユースケースにより効率的にサービスを提供できるようにします。" + +#: src/contributing.md:1 +msgid "Contributing to `ord`" +msgstr "どのように`ord`のために貢献しますか" + +#: src/contributing.md:4 +msgid "Suggested Steps" +msgstr "お勧めの手順" + +#: src/contributing.md:7 +msgid "Find an issue you want to work on." +msgstr "解決したい問題を見つけました。" + +#: src/contributing.md:8 +msgid "" +"Figure out what would be a good first step towards resolving the issue. This " +"could be in the form of code, research, a proposal, or suggesting that it be " +"closed, if it's out of date or not a good idea in the first place." +msgstr "" +"この問題を解決するための良い第一歩とは何かを明らかにします。これはコード、研究、提案の形にすることができます。または" +"は、それが古くなった場合、または最初から良いアイデアではない場合は、それを閉じることをお勧めします。" + +#: src/contributing.md:11 +msgid "" +"Comment on the issue with an outline of your suggested first step, and " +"asking for feedback. Of course, you can dive in and start writing code or " +"tests immediately, but this avoids potentially wasted effort, if the issue " +"is out of date, not clearly specified, blocked on something else, or " +"otherwise not ready to implement." +msgstr "" +"提案した最初のステップの概要を説明し、問題についてコメントし、フィードバックを求めます。もちろん、すぐに投入して始めることもできます" +"コードを書いたり、テストしたりする。しかし、問題がすでに時代遅れ、明確に制定されていなく、他の理由で妨げられていて、または準備ができていない場合" +"はよく実施され、このステップは潜在的なエネルギーの浪費を避けることができます。" + +#: src/contributing.md:16 +msgid "" +"If the issue requires a code change or bugfix, open a draft PR with tests, " +"and ask for feedback. This makes sure that everyone is on the same page " +"about what needs to be done, or what the first step in solving the issue " +"should be. Also, since tests are required, writing the tests first makes it " +"easy to confirm that the change can be tested easily." +msgstr "" +"コードの変更やエラーの修正が必要な場合は、テストPRのドラフトを開き、フィードバックを求めてください。これは保証されます" +"誰もが同期して何をすべきか、あるいはこの問題を解決する最初のステップは何かを知っています。同様に、デバッグ" +"は必須なので、まずテストドラフトを作成し、更新が容易にテストできることを確認します。" + +#: src/contributing.md:21 +msgid "" +"Mash the keyboard randomly until the tests pass, and refactor until the code " +"is ready to submit." +msgstr "" +"テストに合格するまでキーボードをランダムにタップし、コードをコミットする準備ができるまでリファクタリングします。" + +#: src/contributing.md:23 +msgid "Mark the PR as ready to review." +msgstr "" +"PRをレビュー準備完了としてマーク。" + +#: src/contributing.md:24 +msgid "Revise the PR as needed." +msgstr "必要であれば PRを直します 。" + +#: src/contributing.md:25 +msgid "And finally, mergies!" +msgstr "最後のの一歩!" + +#: src/contributing.md:27 +msgid "Start small" +msgstr "塵を重ねば、山となる" + +#: src/contributing.md:30 +msgid "" +"Small changes will allow you to make an impact quickly, and if you take the " +"wrong tack, you won't have wasted much time." +msgstr "小さな変化はあなたに迅速に影響力を与えることができ、間違った戦略をとっても、あまり時間を無駄にしません。" + +#: src/contributing.md:33 +msgid "Ideas for small issues:" +msgstr "いくつかの小さな問題の考え:" + +#: src/contributing.md:34 +msgid "Add a new test or test case that increases test coverage" +msgstr "新しいテストまたはテストケースを追加して、テストのカバー率を高めます。" + +#: src/contributing.md:35 +msgid "Add or improve documentation" +msgstr "ドキュメントの追加または改善" + +#: src/contributing.md:36 +msgid "" +"Find an issue that needs more research, and do that research and summarize " +"it in a comment" +msgstr "より多くの研究が必要な問題を見つけ、研究を行い、コメントにまとめます" + +#: src/contributing.md:38 +msgid "Find an out-of-date issue and comment that it can be closed" +msgstr "古い質問を見つけ、コメントして閉じます。" + +#: src/contributing.md:39 +msgid "" +"Find an issue that shouldn't be done, and provide constructive feedback " +"detailing why you think that is the case" +msgstr "本来すべきではない問題を見つけ、建設的なフィードバックを提供して、このような状況が発生すると考えられる理由を詳しく説明します" + +#: src/contributing.md:42 +msgid "Merge early and often" +msgstr "早く合併し、よく合併します" + +#: src/contributing.md:45 +msgid "" +"Break up large tasks into multiple smaller steps that individually make " +"progress. If there's a bug, you can open a PR that adds a failing ignored " +"test. This can be merged, and the next step can be to fix the bug and " +"unignore the test. Do research or testing, and report on your results. Break " +"a feature into small sub-features, and implement them one at a time." +msgstr "" +"大規模なタスクを複数の小さなステップに分け、これらのステップが個別に取ることができる進展。プログラムエラーがある場合は、" +"PRを開くこともできます。失敗した無視テストを追加します。これはマージすることができ、次のステップでエラーを修正することができます" +"とテストを無視します。あなたの研究またはテスト結果を報告します。大きな機能を小さなサブ機能に分解します。" +"そして、それらを一度に1つずつ段階的に実現します。" + +#: src/contributing.md:51 +msgid "" +"Figuring out how to break down a larger PR into smaller PRs where each can " +"be merged is an art form well-worth practicing. The hard part is that each " +"PR must itself be an improvement." +msgstr "" +"大きなPRを小さなPRに分解する方法を明らかにし、すべてのPRが統合できるのは非常に練習価値があります。" +"これもプログラミングの芸術です。難しい部分は、各PR自体が改善でなければならないことです。" + +#: src/contributing.md:55 +msgid "" +"I strive to follow this advice myself, and am always better off when I do." +msgstr "" +"私は自分でこのアドバイスに従うように努力しています。そして、私がそうすると、私はいつももっと上手にできます。" + +#: src/contributing.md:57 +msgid "" +"Small changes are fast to write, review, and merge, which is much more fun " +"than laboring over a single giant PR that takes forever to write, review, " +"and merge. Small changes don't take much time, so if you need to stop " +"working on a small change, you won't have wasted much time as compared to a " +"larger change that represents many hours of work. Getting a PR in quickly " +"improves the project a little bit immediately, instead of having to wait a " +"long time for larger improvement. Small changes are less likely to " +"accumulate merge conflict. As the Athenians said: _The fast commit what they " +"will, the slow merge what they must._" +msgstr "" +"小さな変更は、迅速に作成、レビュー、マージすることができます。これは、永遠に作成、レビュー、マージする必要がある大規模な" +"PRの仕事はもっと面白いです。小さな変更にはあまり時間がかからないので、小さな変更の処理を中止する必要がある場合は、" +"何時間もの仕事を表す大きな変化に比べて、あまり時間を無駄にしません。迅速なPR獲得はプロジェクトをすぐに改善することができます。" +"はより大きな改善を行うために長い時間を待つ必要はありません。小さな変更ではマージの競合が累積する可能性は低くなります。" +"はアテネ人が言ったように、_速い者はその望みを尽くし、遅い者はその必要を併合します。_" + +#: src/contributing.md:67 +msgid "Get help" +msgstr "助けを求めます。" + +#: src/contributing.md:70 +msgid "" +"If you're stuck for more than 15 minutes, ask for help, like a Rust Discord, " +"Stack Exchange, or in a project issue or discussion." +msgstr "" +"15分以上困っている場合は、例えば、助けを求めてくださいRust Discord、Stack Exchange," +"またはプロジェクトの問題や議論で助けを求めます。" + +#: src/contributing.md:73 +msgid "Practice hypothesis-driven debugging" +msgstr "「仮説駆動型」のデバッグを実践します。" + +#: src/contributing.md:76 +msgid "" +"Formulate a hypothesis as to what is causing the problem. Figure out how to " +"test that hypothesis. Perform that tests. If it works, great, you fixed the " +"issue or now you know how to fix the issue. If not, repeat with a new " +"hypothesis." +msgstr "" +"問題の原因について仮説を立てる。この仮説をどのように検証するかを明らかにする。テストを実行します。効果があれば、それは素晴らしいことです。" +"は問題を解決しました。または、問題を解決する方法を知っています。そうでなければ、新しい仮定を繰り返してください。。" + +#: src/contributing.md:81 +msgid "Pay attention to error messages" +msgstr "エラーメッセージを注目します" + +#: src/contributing.md:84 +msgid "Read all error messages and don't tolerate warnings." +msgstr "すべてのエラーメッセージを読み、警告を容認しないでください。" + +#: src/donate.md:4 +msgid "" +"Ordinals is open-source and community funded. The current lead maintainer of " +"`ord` is [raphjaph](https://github.com/raphjaph/). Raph's work on `ord` is " +"entirely funded by donations. If you can, please consider donating!" +msgstr "" +"Ordinals序数はオープンソースで、コミュニティが資金を提供するプロジェクトです。現在の『ord』のチーフメンテナンス担当者は" +"[raphjaph](https://github.com/raphjaph/)." +"「Raphの『ord』でのメンテナンスはすべて寄付金で行われました。よろしければ、寄付を検討してください!!" + +#: src/donate.md:8 +msgid "" +"The donation address for Bitcoin is " +"[bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt](https://" +"mempool.space/address/" +"bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt). The " +"donation address for inscriptions is " +"[bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0](https://" +"mempool.space/address/" +"bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0)." +msgstr "" +"寄与のアドレスは " +"[bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt](https://" +"mempool.space/address/" +"bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt). " +"銘文の寄与のアドレスは " +"[bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0](https://" +"mempool.space/address/" +"bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0)." + +#: src/donate.md:11 +msgid "" +"Both addresses are in a 2 of 4 multisig wallet with keys held by [raphjaph]" +"(https://twitter.com/raphjaph), [erin](https://twitter.com/realizingerin), " +"[rodarmor](https://twitter.com/rodarmor), and [ordinally](https://twitter." +"com/veryordinally)." +msgstr "" +"上記の2つの住所は以下の多署名者(2/4)が保有管理しています。[raphjaph]: [raphjaph]" +"(https://twitter.com/raphjaph), [erin](https://twitter.com/realizingerin), " +"[rodarmor](https://twitter.com/rodarmor), and [ordinally](https://twitter." +"com/veryordinally)." + +#: src/donate.md:17 +msgid "" +"Donations received will go towards funding maintenance and development of " +"`ord`, as well as hosting costs for [ordinals.com](https://ordinals.com)." +msgstr "" +"受け取った寄付金は『ord』の維持と更なる開発に資金を提供するために使われます。" +"は同時に[ordinals.com](https://ordinals.com)のホスト費用を支払います。" + +#: src/donate.md:20 +msgid "Thank you for donating!" +msgstr "ご寄与を感謝いたします!" + +#: src/guides.md:1 +msgid "Ordinal Theory Guides" +msgstr "序数理論の指導" + +#: src/guides.md:4 +msgid "" +"See the table of contents for a list of guides, including a guide to the " +"explorer, a guide for sat hunters, and a guide to inscriptions." +msgstr "" +"ガイドの一覧については、カタログを参照してください。これには、ブロックブラウザガイド、スマートフォンガイド、銘文ガイドなどがあります。" + +#: src/guides/explorer.md:1 +msgid "Ordinal Explorer" +msgstr "序数ブラウザ" + +#: src/guides/explorer.md:4 +msgid "" +"The `ord` binary includes a block explorer. We host a instance of the block " +"explorer on mainnet at [ordinals.com](https://ordinals.com), and on signet " +"at [signet.ordinals.com](https://signet.ordinals.com)." +msgstr "" +"`ord` ファイルにはブロックブラウザが含まれています。当社のメインネットワークブロックチェーンは [ordinals.com](https://ordinals.com), " +"signet[signet.ordinals.com](https://signet.ordinals.com)に配置します。" + +#: src/guides/explorer.md:8 +msgid "Running The Explorer" +msgstr "運行ブラウザ" + +#: src/guides/explorer.md:9 +msgid "The server can be run locally with:" +msgstr "サーバはローカルで運行可能:" + +#: src/guides/explorer.md:11 +msgid "`ord server`" +msgstr "" + +#: src/guides/explorer.md:13 +msgid "To specify a port add the `--http-port` flag:" +msgstr "指定されたポートは、'--http-port'タグを使用します。" + +#: src/guides/explorer.md:15 +msgid "`ord server --http-port 8080`" +msgstr "" + +#: src/guides/explorer.md:17 +msgid "To test how your inscriptions will look you can run:" +msgstr "銘文をテストして、運行できます:" + +#: src/guides/explorer.md:19 +msgid "`ord preview ...`" +msgstr "" + +#: src/guides/explorer.md:21 +msgid "Search" +msgstr "検索" + +#: src/guides/explorer.md:24 +msgid "The search box accepts a variety of object representations." +msgstr "検索ボックスではさまざまなオブジェクトを使用できます:" + +#: src/guides/explorer.md:26 +msgid "Blocks" +msgstr "ブロック" + +#: src/guides/explorer.md:28 +msgid "Blocks can be searched by hash, for example, the genesis block:" +msgstr "ブロックはハッシュで見つけることができます。例えば、創世ブロックです:" + +#: src/guides/explorer.md:30 +msgid "" +"[000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f](https://" +"ordinals.com/" +"search/000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f)" +msgstr "" + +#: src/guides/explorer.md:32 +msgid "Transactions" +msgstr "取引" + +#: src/guides/explorer.md:34 +msgid "" +"Transactions can be searched by hash, for example, the genesis block " +"coinbase transaction:" +msgstr "創世記ブロックのcoinbaseトランザクションなど、ハッシュによってトランザクションを見つけることができます。:" + +#: src/guides/explorer.md:37 +msgid "" +"[4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b](https://" +"ordinals.com/" +"search/4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b)" +msgstr "" + +#: src/guides/explorer.md:39 +msgid "Outputs" +msgstr "輸出" + +#: src/guides/explorer.md:41 +msgid "" +"Transaction outputs can searched by outpoint, for example, the only output " +"of the genesis block coinbase transaction:" +msgstr "" +"アウトポイントを使用して、創世ブロックcoinbaseトランザクションのユニークな出力などのトランザクション出力を検索できます:" + +#: src/guides/explorer.md:44 +msgid "" +"[4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b:0](https://" +"ordinals.com/" +"search/4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b:0)" +msgstr "" + +#: src/guides/explorer.md:46 +msgid "Sats" +msgstr "聡" + +#: src/guides/explorer.md:48 +msgid "" +"Sats can be searched by integer, their position within the entire bitcoin " +"supply:" +msgstr "" +"聡は整数で検索することができ、ビットコインの供給全体の中のそれらの位置:" + +#: src/guides/explorer.md:51 +msgid "[2099994106992659](https://ordinals.com/search/2099994106992659)" +msgstr "" + +#: src/guides/explorer.md:53 +msgid "By decimal, their block and offset within that block:" +msgstr "10進数では、そのブロックとそのブロック内のオフセット:" + +#: src/guides/explorer.md:55 +msgid "[481824.0](https://ordinals.com/search/481824.0)" +msgstr "" + +#: src/guides/explorer.md:57 +msgid "" +"By degree, their cycle, blocks since the last halving, blocks since the last " +"difficulty adjustment, and offset within their block:" +msgstr "度数、それらの週期、前回の半減以来のブロック、前回の難易度調整以来のブロック、そしてブロック内のオフセット:" + +#: src/guides/explorer.md:60 +msgid "[1°0′0″0‴](https://ordinals.com/search/1°0′0″0‴)" +msgstr "" + +#: src/guides/explorer.md:62 +msgid "" +"By name, their base 26 representation using the letters \"a\" through \"z\":" +msgstr "名前に応じて、文字\"a\"~\"z\"の26文字の組み合わせで表されます:" + +#: src/guides/explorer.md:64 +msgid "[ahistorical](https://ordinals.com/search/ahistorical)" +msgstr "" + +#: src/guides/explorer.md:66 +msgid "" +"Or by percentile, the percentage of bitcoin's supply that has been or will " +"have been issued when they are mined:" +msgstr "または百分率で、採掘時にすでに発行された、または発行される予定のビットコインの供給量のパーセント:" + +#: src/guides/explorer.md:69 +msgid "[100%](https://ordinals.com/search/100%)" +msgstr "" + +#: src/guides/inscriptions.md:1 +msgid "Ordinal Inscription Guide" +msgstr "銘文のガイド" + +#: src/guides/inscriptions.md:4 +msgid "" +"Individual sats can be inscribed with arbitrary content, creating Bitcoin-" +"native digital artifacts that can be held in a Bitcoin wallet and " +"transferred using Bitcoin transactions. Inscriptions are as durable, " +"immutable, secure, and decentralized as Bitcoin itself." +msgstr "" +"単一のコングは、ビットコインウォレットに保存され、ビットコイントランザクションを使用して転送" +"されるビットコインネイティブデジタルアーティファクトを作成するために、任意のコンテンツを刻印することができます。" +"の銘文はビットコイン自体と同じくらい持続的で、不変で、安全で、分散化されています。" + +#: src/guides/inscriptions.md:9 +msgid "" +"Working with inscriptions requires a Bitcoin full node, to give you a view " +"of the current state of the Bitcoin blockchain, and a wallet that can create " +"inscriptions and perform sat control when constructing transactions to send " +"inscriptions to another wallet." +msgstr "" +"銘文を使用するには、ビットコインのブロックチェーンの現在の状態を知ることができるビットコイン" +"の完全なノードが必要です。また、銘文を作成し、取引を構築することができます」" +"他のウォレットに銘文を送信するときにスマートコントロールを実行するウォレット。" + +#: src/guides/inscriptions.md:14 +msgid "" +"Bitcoin Core provides both a Bitcoin full node and wallet. However, the " +"Bitcoin Core wallet cannot create inscriptions and does not perform sat " +"control." +msgstr "" +"Bitcoin Core ビットコインのフルノードとウォレットを提供します。しかし,Bitcoin Coreウォ" +"レットは碑文を作成できず、スマートコントロールを実行しません。" + +#: src/guides/inscriptions.md:17 +msgid "" +"This requires [`ord`](https://github.com/ordinals/ord), the ordinal utility. " +"`ord` doesn't implement its own wallet, so `ord wallet` subcommands interact " +"with Bitcoin Core wallets." +msgstr "" +"これには['ord'](https://github.com/ordinals/ord),序数ユーティリティ。" +"'ord'は自分のウォレットを持っていないので、'ord wallet'サブコマンドはBitcoin Coreウォレットと対話します。" + +#: src/guides/inscriptions.md:21 +msgid "This guide covers:" +msgstr "このガイドの含まれ:" + +#: src/guides/inscriptions.md:23 src/guides/inscriptions.md:39 +msgid "Installing Bitcoin Core" +msgstr " Bitcoin Coreを取り付けます。" + +#: src/guides/inscriptions.md:24 +msgid "Syncing the Bitcoin blockchain" +msgstr "ビットコインのブロックを同期します。" + +#: src/guides/inscriptions.md:25 +msgid "Creating a Bitcoin Core wallet" +msgstr "Bitcoin Core ウォレットを作り上げます。" + +#: src/guides/inscriptions.md:26 +msgid "Using `ord wallet receive` to receive sats" +msgstr " `ord wallet receive`を使って聡を取り受けます。" + +#: src/guides/inscriptions.md:27 +msgid "Creating inscriptions with `ord wallet inscribe`" +msgstr "`ord wallet inscribe`を使って、銘文を作り上げます。" + +#: src/guides/inscriptions.md:28 +msgid "Sending inscriptions with `ord wallet send`" +msgstr "`ord wallet send`を使って`銘文を発送します" + +#: src/guides/inscriptions.md:29 +msgid "Receiving inscriptions with `ord wallet receive`" +msgstr "`ord wallet receive`を使って`銘文を受け取ります。" + +#: src/guides/inscriptions.md:31 +msgid "Getting Help" +msgstr "助けを求めます。" + +#: src/guides/inscriptions.md:34 +msgid "" +"If you get stuck, try asking for help on the [Ordinals Discord Server]" +"(https://discord.com/invite/87cjuz4FYg), or checking GitHub for relevant " +"[issues](https://github.com/ordinals/ord/issues) and [discussions](https://" +"github.com/ordinals/ord/discussions)." +msgstr "" +"困ったことがあったら、[Ordinals Discord Server](https://discord.com/invite/87cjuz4FYg)," +"またはGithubの関連内容をチェックする[質問](https://github.com/ordinals/ord/issues)と" +"[ディスカッション](https://github.com/ordinals/ord/discussions)." + +#: src/guides/inscriptions.md:42 +msgid "" +"Bitcoin Core is available from [bitcoincore.org](https://bitcoincore.org/) " +"on the [download page](https://bitcoincore.org/en/download/)." +msgstr "" +"[bitcoincore.org](https://bitcoincore.org/)で」" +"の[ダウンロードページ](https://bitcoincore.org/en/download/)" + +#: src/guides/inscriptions.md:45 +msgid "Making inscriptions requires Bitcoin Core 24 or newer." +msgstr "銘文を作成するにはBitcoin Core 24以降が必要です。" + +#: src/guides/inscriptions.md:47 +msgid "" +"This guide does not cover installing Bitcoin Core in detail. Once Bitcoin " +"Core is installed, you should be able to run `bitcoind -version` " +"successfully from the command line." +msgstr "" +"このガイドには、Bitcoin Coreの詳細なインストール方法は含まれていません。Bitcoin Coreのインストールに成功した後、" +"コマンドラインで'bitcoind-version'コマンドを使うことができます。" + +#: src/guides/inscriptions.md:51 +msgid "Configuring Bitcoin Core" +msgstr " Bitcoin Coreを配置します。" + +#: src/guides/inscriptions.md:54 +msgid "`ord` requires Bitcoin Core's transaction index." +msgstr "`ord` Bitcoin Core の取引索引が必要です。" + +#: src/guides/inscriptions.md:56 +msgid "" +"To configure your Bitcoin Core node to maintain a transaction index, add the " +"following to your `bitcoin.conf`:" +msgstr "" +"トランザクションインデックスを配置するためにBitcoin Coreフェーズを設定します,'bitcoin.conf'に追加する必要があります::" + +#: src/guides/inscriptions.md:59 src/guides/sat-hunting.md:30 +msgid "" +"```\n" +"txindex=1\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:63 +msgid "Or, run `bitcoind` with `-txindex`:" +msgstr "または, `bitcoind` 和 `-txindex`を運行します:" + +#: src/guides/inscriptions.md:65 src/guides/inscriptions.md:74 +msgid "" +"```\n" +"bitcoind -txindex\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:69 +msgid "Syncing the Bitcoin Blockchain" +msgstr "ビットコインのブロックを同期にします。" + +#: src/guides/inscriptions.md:72 +msgid "To sync the chain, run:" +msgstr "ブロックを同期にして、運行します:" + +#: src/guides/inscriptions.md:78 +msgid "…and leave it running until `getblockcount`:" +msgstr "… `getblockcount運行するまで`:" + +#: src/guides/inscriptions.md:80 +msgid "" +"```\n" +"bitcoin-cli getblockcount\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:84 +msgid "" +"agrees with the block count on a block explorer like [the mempool.space " +"block explorer](https://mempool.space/). `ord` interacts with `bitcoind`, so " +"you should leave `bitcoind` running in the background when you're using " +"`ord`." +msgstr "" +"ブロックチェーンブラウザのように[the mempool.space block explorer](https://mempool.space/)のように" +"ブロックを記述する...`ord`は'bitcoind'と相互作用するので、'ord'を使用するときは'bitcoind'をバックグラウンドで実行する必要があります。" + +#: src/guides/inscriptions.md:88 +msgid "Installing `ord`" +msgstr " `ord取り付けます。`" + +#: src/guides/inscriptions.md:91 +msgid "" +"The `ord` utility is written in Rust and can be built from [source](https://" +"github.com/ordinals/ord). Pre-built binaries are available on the [releases " +"page](https://github.com/ordinals/ord/releases)." +msgstr "" +"`ord` プログラムはRust言語で書かれており、[ソースコード](https://github.com/ordinals/ord)インストール." +"事前に作成されたファイルは、[バージョンリリースページ](https://github.com/ordinals/ord/releases))からダウンロードできます。" + +#: src/guides/inscriptions.md:95 +msgid "You can install the latest pre-built binary from the command line with:" +msgstr "コマンドラインで次のコマンドを使用して、最新のファイルをインストールすることもできます。:" + +#: src/guides/inscriptions.md:97 +msgid "" +"```sh\n" +"curl --proto '=https' --tlsv1.2 -fsLS https://ordinals.com/install.sh | bash " +"-s\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:101 +msgid "Once `ord` is installed, you should be able to run:" +msgstr "'ord'が正常にインストールされた後、あなたは実行することができます :" + +#: src/guides/inscriptions.md:103 +msgid "" +"```\n" +"ord --version\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:107 +msgid "Which prints out `ord`'s version number." +msgstr "これにより、「ord」のバージョン番号が出力されます。" + +#: src/guides/inscriptions.md:109 +msgid "Creating a Bitcoin Core Wallet" +msgstr "『ord』という名前のBitcoin Coreウォレットを作成します。" + +#: src/guides/inscriptions.md:112 +msgid "" +"`ord` uses Bitcoin Core to manage private keys, sign transactions, and " +"broadcast transactions to the Bitcoin network." +msgstr "`ord` Bitcoin Coreを使用して秘密鍵を管理し、トランザクションに署名し、ビットコインネットワークにトランザクションをブロードキャストします。。" + +#: src/guides/inscriptions.md:115 +msgid "To create a Bitcoin Core wallet named `ord` for use with `ord`, run:" +msgstr "『ord』という名前のBitcoin Coreウォレットを作成し、運行します:" + +#: src/guides/inscriptions.md:117 +msgid "" +"```\n" +"ord wallet create\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:121 +msgid "Receiving Sats" +msgstr "聡を取り受けます。" + +#: src/guides/inscriptions.md:124 +msgid "" +"Inscriptions are made on individual sats, using normal Bitcoin transactions " +"that pay fees in sats, so your wallet will need some sats." +msgstr "銘文は、単一のコングに作成され、通常のビットコイン取引のためにコングを使用して料金を支払うので、あなたの財布はいくつかのコング(ビットコイン)を必要とします。" + +#: src/guides/inscriptions.md:127 +msgid "Get a new address from your `ord` wallet by running:" +msgstr "'ord'ウォレットの新しいアドレスを作成して、実行します::" + +#: src/guides/inscriptions.md:129 src/guides/inscriptions.md:201 +#: src/guides/inscriptions.md:229 +msgid "" +"```\n" +"ord wallet receive\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:133 +msgid "And send it some funds." +msgstr "上記の住所に資金を発送します。" + +#: src/guides/inscriptions.md:135 +msgid "You can see pending transactions with:" +msgstr "以下のコマンドで取引状況を見ることができます:" + +#: src/guides/inscriptions.md:137 src/guides/inscriptions.md:213 +#: src/guides/inscriptions.md:240 +msgid "" +"```\n" +"ord wallet transactions\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:141 +msgid "" +"Once the transaction confirms, you should be able to see the transactions " +"outputs with `ord wallet outputs`." +msgstr "取引が確認されたら、'ord wallet outputs'を使って取引の輸出を見ることができるはずです;" + +#: src/guides/inscriptions.md:144 +msgid "Creating Inscription Content" +msgstr "銘文の内容を作成します。" + +#: src/guides/inscriptions.md:147 +msgid "" +"Sats can be inscribed with any kind of content, but the `ord` wallet only " +"supports content types that can be displayed by the `ord` block explorer." +msgstr "聡の上ではあらゆる種類のコンテンツを書き込むことができますが、'ord'ウォレットは'ord'ブロックブラウザで表示できる種類のコンテンツしかサポートしていません。" + +#: src/guides/inscriptions.md:150 +msgid "" +"Additionally, inscriptions are included in transactions, so the larger the " +"content, the higher the fee that the inscription transaction must pay." +msgstr "また、銘文は取引に含まれているため、内容が大きいほど銘文取引にかかる費用が高くなります。" + +#: src/guides/inscriptions.md:153 +msgid "" +"Inscription content is included in transaction witnesses, which receive the " +"witness discount. To calculate the approximate fee that an inscribe " +"transaction will pay, divide the content size by four and multiply by the " +"fee rate." +msgstr "" +"銘文の内容は取引証言に含まれ、証言割引を受ける。トランザクションへの書き込みで支払われる費用の概算を計算するには、" +"コンテンツサイズを4で割ってから、レートを掛けてください。" + +#: src/guides/inscriptions.md:157 +msgid "" +"Inscription transactions must be less than 400,000 weight units, or they " +"will not be relayed by Bitcoin Core. One byte of inscription content costs " +"one weight unit. Since an inscription transaction includes not just the " +"inscription content, limit inscription content to less than 400,000 weight " +"units. 390,000 weight units should be safe." +msgstr "" +"銘文取引は400,000重量測定単位未満でなければならない。そうでなければ、Bitcoin Coreに中継されない" +"の1バイトの銘文内容には重み測定単位が必要です。銘文取引は銘文の内容だけではないので、" +"銘文の内容は400、000重量計量単位以内に制限されている。390、000の重み測定単位は安全でなければならなりません。" + +#: src/guides/inscriptions.md:163 +msgid "Creating Inscriptions" +msgstr "銘文を作成します。" + +#: src/guides/inscriptions.md:166 +msgid "To create an inscription with the contents of `FILE`, run:" +msgstr "『FILE』の内容で銘文を作成し、実行する必要があります:" + +#: src/guides/inscriptions.md:168 +msgid "" +"```\n" +"ord wallet inscribe --fee-rate FEE_RATE FILE\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:172 +msgid "" +"Ord will output two transactions IDs, one for the commit transaction, and " +"one for the reveal transaction, and the inscription ID. Inscription IDs are " +"of the form `TXIDiN`, where `TXID` is the transaction ID of the reveal " +"transaction, and `N` is the index of the inscription in the reveal " +"transaction." +msgstr "" +"Ordは2つの取引IDが出力されます。1つはcommit取引、もう1つはreveal取引、そして銘文IDです。" +"銘文IDのフォーマットは、'TXIDiN'であり、'TXID'は、取引を開示する取引IDであり、'N'は、取引" +"における銘文を開示するインデックスであ。" + +#: src/guides/inscriptions.md:177 +msgid "" +"The commit transaction commits to a tapscript containing the content of the " +"inscription, and the reveal transaction spends from that tapscript, " +"revealing the content on chain and inscribing it on the first sat of the " +"input that contains the corresponding tapscript." +msgstr "" +"CommitCommitトランザクションは銘文の内容を含むtapscriptに提出され、revealトランザクションはそのtapscriptから費やされます。" +"チェーン上のコンテンツを表示し、revealトランザクションの最初の出力の最初のsatにそれらをマークします。" + +#: src/guides/inscriptions.md:182 +msgid "" +"Wait for the reveal transaction to be mined. You can check the status of the " +"commit and reveal transactions using [the mempool.space block explorer]" +"(https://mempool.space/)." +msgstr "" +"reveal取引が記録されるのを待っている間、あなたは使用することができます[the mempool.space block explorer]" +"(https://mempool.space/)は取引の状態をチェックします。" + +#: src/guides/inscriptions.md:186 +msgid "" +"Once the reveal transaction has been mined, the inscription ID should be " +"printed when you run:" +msgstr "" +"reveal取引が記帳を完了したら、以下のコマンドを使用して銘文IDを照会できます" + + +#: src/guides/inscriptions.md:189 src/guides/inscriptions.md:220 +#: src/guides/inscriptions.md:246 +msgid "" +"```\n" +"ord wallet inscriptions\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:193 +msgid "" +"And when you visit [the ordinals explorer](https://ordinals.com/) at " +"`ordinals.com/inscription/INSCRIPTION_ID`." +msgstr "" +"[the ordinals explorer](https://ordinals.com/)で以下の形式で銘文にアクセスできます`." + +#: src/guides/inscriptions.md:196 +msgid "Sending Inscriptions" +msgstr "銘文を発送します" + +#: src/guides/inscriptions.md:199 +msgid "Ask the recipient to generate a new address by running:" +msgstr "銘文受信者は次のコマンドを使って住所を生成する" + +#: src/guides/inscriptions.md:205 +msgid "Send the inscription by running:" +msgstr "コマンドを使用して銘文を発送します:" + +#: src/guides/inscriptions.md:207 +msgid "" +"```\n" +"ord wallet send --fee-rate
    \n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:211 src/guides/inscriptions.md:239 +msgid "See the pending transaction with:" +msgstr "できなかった取引状況を検査します:" + +#: src/guides/inscriptions.md:217 +msgid "" +"Once the send transaction confirms, the recipient can confirm receipt by " +"running:" +msgstr "取引が確認されると、受信者は次のコマンドを使用して受信した銘文を見ることができます" + +#: src/guides/inscriptions.md:224 +msgid "Receiving Inscriptions" +msgstr "銘文を受け取ります" + +#: src/guides/inscriptions.md:227 +msgid "Generate a new receive address using:" +msgstr "次のコマンドを使用して新しい受信アドレスを生成する" + +#: src/guides/inscriptions.md:233 +msgid "The sender can transfer the inscription to your address using:" +msgstr "送信者はコマンドを使用してあなたの住所に銘文を送信します" + +#: src/guides/inscriptions.md:235 +msgid "" +"```\n" +"ord wallet send ADDRESS INSCRIPTION_ID\n" +"```" +msgstr "" + +#: src/guides/inscriptions.md:244 +msgid "" +"Once the send transaction confirms, you can can confirm receipt by running:" +msgstr "取引が確認されると、受信者は次のコマンドを使用して受信した銘文を見ることができます" + +#: src/guides/sat-hunting.md:4 +msgid "" +"_This guide is out of date. Since it was written, the `ord` binary was " +"changed to only build the full satoshi index when the `--index-sats` flag is " +"supplied. Additionally, `ord` now has a built-in wallet that wraps a Bitcoin " +"Core wallet. See `ord wallet --help`._" +msgstr "" +"_このガイドは時代遅れです。書かれて以来、'ord'インストールファイルが変更されました。" +"「--index-sats」フラグが指定されている場合にのみ、完全なスマートインデックスが作成されます。" +"さらに「ord」には現在、ビットコインのコアウォレットを含むウォレットが内蔵されていて、ord wallet --help`を参照してください。_" + +#: src/guides/sat-hunting.md:9 +msgid "" +"Ordinal hunting is difficult but rewarding. The feeling of owning a wallet " +"full of UTXOs, redolent with the scent of rare and exotic sats, is beyond " +"compare." +msgstr "" + +#: src/guides/sat-hunting.md:12 +msgid "" +"Ordinals are numbers for satoshis. Every satoshi has an ordinal number and " +"every ordinal number has a satoshi." +msgstr "" + +#: src/guides/sat-hunting.md:15 +msgid "Preparation" +msgstr "" + +#: src/guides/sat-hunting.md:18 +msgid "There are a few things you'll need before you start." +msgstr "" + +#: src/guides/sat-hunting.md:20 +msgid "" +"First, you'll need a synced Bitcoin Core node with a transaction index. To " +"turn on transaction indexing, pass `-txindex` on the command-line:" +msgstr "" + +#: src/guides/sat-hunting.md:23 +msgid "" +"```sh\n" +"bitcoind -txindex\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:27 +msgid "" +"Or put the following in your [Bitcoin configuration file](https://github.com/" +"bitcoin/bitcoin/blob/master/doc/bitcoin-conf.md#configuration-file-path):" +msgstr "" + +#: src/guides/sat-hunting.md:34 +msgid "" +"Launch it and wait for it to catch up to the chain tip, at which point the " +"following command should print out the current block height:" +msgstr "" + +#: src/guides/sat-hunting.md:37 +msgid "" +"```sh\n" +"bitcoin-cli getblockcount\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:41 +msgid "Second, you'll need a synced `ord` index." +msgstr "" + +#: src/guides/sat-hunting.md:43 +msgid "Get a copy of `ord` from [the repo](https://github.com/ordinals/ord/)." +msgstr "" + +#: src/guides/sat-hunting.md:45 +msgid "" +"Run `RUST_LOG=info ord index`. It should connect to your bitcoin core node " +"and start indexing." +msgstr "" + +#: src/guides/sat-hunting.md:48 +msgid "Wait for it to finish indexing." +msgstr "" + +#: src/guides/sat-hunting.md:50 +msgid "Third, you'll need a wallet with UTXOs that you want to search." +msgstr "" + +#: src/guides/sat-hunting.md:52 +msgid "Searching for Rare Ordinals" +msgstr "" + +#: src/guides/sat-hunting.md:55 +msgid "Searching for Rare Ordinals in a Bitcoin Core Wallet" +msgstr "" + +#: src/guides/sat-hunting.md:57 +msgid "" +"The `ord wallet` command is just a wrapper around Bitcoin Core's RPC API, so " +"searching for rare ordinals in a Bitcoin Core wallet is Easy. Assuming your " +"wallet is named `foo`:" +msgstr "" + +#: src/guides/sat-hunting.md:61 +msgid "Load your wallet:" +msgstr "" + +#: src/guides/sat-hunting.md:63 +msgid "" +"```sh\n" +"bitcoin-cli loadwallet foo\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:67 +msgid "Display any rare ordinals wallet `foo`'s UTXOs:" +msgstr "" + +#: src/guides/sat-hunting.md:69 src/guides/sat-hunting.md:132 +#: src/guides/sat-hunting.md:233 +msgid "" +"```sh\n" +"ord wallet sats\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:73 +msgid "Searching for Rare Ordinals in a Non-Bitcoin Core Wallet" +msgstr "" + +#: src/guides/sat-hunting.md:75 +msgid "" +"The `ord wallet` command is just a wrapper around Bitcoin Core's RPC API, so " +"to search for rare ordinals in a non-Bitcoin Core wallet, you'll need to " +"import your wallet's descriptors into Bitcoin Core." +msgstr "" + +#: src/guides/sat-hunting.md:79 +msgid "" +"[Descriptors](https://github.com/bitcoin/bitcoin/blob/master/doc/descriptors." +"md) describe the ways that wallets generate private keys and public keys." +msgstr "" + +#: src/guides/sat-hunting.md:82 +msgid "" +"You should only import descriptors into Bitcoin Core for your wallet's " +"public keys, not its private keys." +msgstr "" + +#: src/guides/sat-hunting.md:85 +msgid "" +"If your wallet's public key descriptor is compromised, an attacker will be " +"able to see your wallet's addresses, but your funds will be safe." +msgstr "" + +#: src/guides/sat-hunting.md:88 +msgid "" +"If your wallet's private key descriptor is compromised, an attacker can " +"drain your wallet of funds." +msgstr "" + +#: src/guides/sat-hunting.md:91 +msgid "" +"Get the wallet descriptor from the wallet whose UTXOs you want to search for " +"rare ordinals. It will look something like this:" +msgstr "" + +#: src/guides/sat-hunting.md:94 +msgid "" +"```\n" +"wpkh([bf1dd55e/84'/0'/0']xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/0/" +"*)#csvefu29\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:98 +msgid "Create a watch-only wallet named `foo-watch-only`:" +msgstr "" + +#: src/guides/sat-hunting.md:100 +msgid "" +"```sh\n" +"bitcoin-cli createwallet foo-watch-only true true\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:104 +msgid "Feel free to give it a better name than `foo-watch-only`!" +msgstr "" + +#: src/guides/sat-hunting.md:106 +msgid "Load the `foo-watch-only` wallet:" +msgstr "" + +#: src/guides/sat-hunting.md:108 src/guides/sat-hunting.md:199 +msgid "" +"```sh\n" +"bitcoin-cli loadwallet foo-watch-only\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:112 +msgid "Import your wallet descriptors into `foo-watch-only`:" +msgstr "" + +#: src/guides/sat-hunting.md:114 +msgid "" +"```sh\n" +"bitcoin-cli importdescriptors \\\n" +" '[{ \"desc\": " +"\"wpkh([bf1dd55e/84h/0h/0h]xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/0/" +"*)#tpnxnxax\", \"timestamp\":0 }]'\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:119 +msgid "" +"If you know the Unix timestamp when your wallet first started receive " +"transactions, you may use it for the value of `\"timestamp\"` instead of " +"`0`. This will reduce the time it takes for Bitcoin Core to search for your " +"wallet's UTXOs." +msgstr "" + +#: src/guides/sat-hunting.md:124 src/guides/sat-hunting.md:225 +msgid "Check that everything worked:" +msgstr "" + +#: src/guides/sat-hunting.md:126 src/guides/sat-hunting.md:227 +msgid "" +"```sh\n" +"bitcoin-cli getwalletinfo\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:130 src/guides/sat-hunting.md:231 +msgid "Display your wallet's rare ordinals:" +msgstr "" + +#: src/guides/sat-hunting.md:136 +msgid "" +"Searching for Rare Ordinals in a Wallet that Exports Multi-path Descriptors" +msgstr "" + +#: src/guides/sat-hunting.md:138 +msgid "" +"Some descriptors describe multiple paths in one descriptor using angle " +"brackets, e.g., `<0;1>`. Multi-path descriptors are not yet supported by " +"Bitcoin Core, so you'll first need to convert them into multiple " +"descriptors, and then import those multiple descriptors into Bitcoin Core." +msgstr "" + +#: src/guides/sat-hunting.md:143 +msgid "" +"First get the multi-path descriptor from your wallet. It will look something " +"like this:" +msgstr "" + +#: src/guides/sat-hunting.md:146 +msgid "" +"```\n" +"wpkh([bf1dd55e/84h/0h/0h]xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/" +"<0;1>/*)#fw76ulgt\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:150 +msgid "Create a descriptor for the receive address path:" +msgstr "" + +#: src/guides/sat-hunting.md:152 +msgid "" +"```\n" +"wpkh([bf1dd55e/84'/0'/0']xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/0/" +"*)\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:156 +msgid "And the change address path:" +msgstr "" + +#: src/guides/sat-hunting.md:158 +msgid "" +"```\n" +"wpkh([bf1dd55e/84'/0'/0']xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/1/" +"*)\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:162 +msgid "" +"Get and note the checksum for the receive address descriptor, in this case " +"`tpnxnxax`:" +msgstr "" + +#: src/guides/sat-hunting.md:165 +msgid "" +"```sh\n" +"bitcoin-cli getdescriptorinfo \\\n" +" 'wpkh([bf1dd55e/84h/0h/0h]xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/0/" +"*)'\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:170 +msgid "" +"```json\n" +"{\n" +" \"descriptor\": " +"\"wpkh([bf1dd55e/84'/0'/0']xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/0/" +"*)#csvefu29\",\n" +" \"checksum\": \"tpnxnxax\",\n" +" \"isrange\": true,\n" +" \"issolvable\": true,\n" +" \"hasprivatekeys\": false\n" +"}\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:180 +msgid "And for the change address descriptor, in this case `64k8wnd7`:" +msgstr "" + +#: src/guides/sat-hunting.md:182 +msgid "" +"```sh\n" +"bitcoin-cli getdescriptorinfo \\\n" +" 'wpkh([bf1dd55e/84h/0h/0h]xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/1/" +"*)'\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:187 +msgid "" +"```json\n" +"{\n" +" \"descriptor\": " +"\"wpkh([bf1dd55e/84'/0'/0']xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/1/" +"*)#fyfc5f6a\",\n" +" \"checksum\": \"64k8wnd7\",\n" +" \"isrange\": true,\n" +" \"issolvable\": true,\n" +" \"hasprivatekeys\": false\n" +"}\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:197 +msgid "Load the wallet you want to import the descriptors into:" +msgstr "" + +#: src/guides/sat-hunting.md:203 +msgid "" +"Now import the descriptors, with the correct checksums, into Bitcoin Core." +msgstr "" + +#: src/guides/sat-hunting.md:205 +msgid "" +"```sh\n" +"bitcoin-cli \\\n" +" importdescriptors \\\n" +" '[\n" +" {\n" +" \"desc\": " +"\"wpkh([bf1dd55e/84h/0h/0h]xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/0/" +"*)#tpnxnxax\"\n" +" \"timestamp\":0\n" +" },\n" +" {\n" +" \"desc\": " +"\"wpkh([bf1dd55e/84h/0h/0h]xpub6CcJtWcvFQaMo39ANFi1MyXkEXM8T8ZhnxMtSjQAdPmVSTHYnc8Hwoc11VpuP8cb8JUTboZB5A7YYGDonYySij4XTawL6iNZvmZwdnSEEep/1/" +"*)#64k8wnd7\",\n" +" \"timestamp\":0\n" +" }\n" +" ]'\n" +"```" +msgstr "" + +#: src/guides/sat-hunting.md:220 +msgid "" +"If you know the Unix timestamp when your wallet first started receive " +"transactions, you may use it for the value of the `\"timestamp\"` fields " +"instead of `0`. This will reduce the time it takes for Bitcoin Core to " +"search for your wallet's UTXOs." +msgstr "" + +#: src/guides/sat-hunting.md:237 +msgid "Exporting Descriptors" +msgstr "" + +#: src/guides/sat-hunting.md:241 +msgid "" +"Navigate to the `Settings` tab, then to `Script Policy`, and press the edit " +"button to display the descriptor." +msgstr "" + +#: src/guides/sat-hunting.md:244 +msgid "Transferring Ordinals" +msgstr "" + +#: src/guides/sat-hunting.md:246 +msgid "" +"The `ord` wallet supports transferring specific satoshis. You can also use " +"`bitcoin-cli` commands `createrawtransaction`, " +"`signrawtransactionwithwallet`, and `sendrawtransaction`, how to do so is " +"complex and outside the scope of this guide." +msgstr "" + +#: src/guides/collecting.md:4 +msgid "" +"Currently, [ord](https://github.com/ordinals/ord/) is the only wallet " +"supporting sat-control and sat-selection, which are required to safely store " +"and send rare sats and inscriptions, hereafter ordinals." +msgstr "" +"現在、[ord](https://github.com/ordinals/ord/)はsat-controlとsat-selectionをサポートする唯一のものです" +"の財布、これは希少なsatsと銘文(以下、序数と略称する)を安全に保存して送るために必要です。" + +#: src/guides/collecting.md:8 +msgid "" +"The recommended way to send, receive, and store ordinals is with `ord`, but " +"if you are careful, it is possible to safely store, and in some cases send, " +"ordinals with other wallets." +msgstr "" +"送信、受信、シーケンス番号の格納方法としては、'ord'を用いることが推奨される。しかし、注意すれば安全に保管できます。" +"場合によっては、別のウォレットを使用してシリアル番号を送信します。" + +#: src/guides/collecting.md:12 +msgid "" +"As a general note, receiving ordinals in an unsupported wallet is not " +"dangerous. Ordinals can be sent to any bitcoin address, and are safe as long " +"as the UTXO that contains them is not spent. However, if that wallet is then " +"used to send bitcoin, it may select the UTXO containing the ordinal as an " +"input, and send the inscription or spend it to fees." +msgstr "" +"一般的な説明として、サポートされていないウォレットでシーケンス番号を受信することは危険ではありません。シリアル番号は任意のビットコインアドレスに送信できます。" +"はそれらを含むUTXOが使われていない限り、安全です。しかし、ウォレットがその後ビットコインを送信するために使用された場合、" +"はシリアル番号を含むUTXOを入力として選択し、銘文を送ったり、費用に使用したりすることがあります。。" + +#: src/guides/collecting.md:18 +msgid "" +"A [guide](./collecting/sparrow-wallet.md) to creating an `ord`\\-compatible " +"wallet with [Sparrow Wallet](https://sparrowwallet.com/), is available in " +"this handbook." +msgstr "" +"このマニュアルでは、[Sparrow Wallet](https://sparrowwallet.com/)作成" +"'ord'と互換性のある財布の[ガイド](./collecting/sparrow-wallet.md).。" + +#: src/guides/collecting.md:21 +msgid "" +"Please note that if you follow this guide, you should not use the wallet you " +"create to send BTC, unless you perform manual coin-selection to avoid " +"sending ordinals." +msgstr "" +"このガイドに従う場合は、シリアル番号の送信を避けるために手動コイン選択を実行しない限り、作成したウォレットを使用してBTCを送信しないでください。。" + +#: src/guides/collecting/sparrow-wallet.md:1 +msgid "Collecting Inscriptions and Ordinals with Sparrow Wallet" +msgstr "スズメのSparrow財布を使って銘文をコレクションします。" + +#: src/guides/collecting/sparrow-wallet.md:4 +msgid "" +"Users who cannot or have not yet set up the [ord](https://github.com/" +"ordinals/ord) wallet can receive inscriptions and ordinals with alternative " +"bitcoin wallets, as long as they are _very_ careful about how they spend " +"from that wallet." +msgstr "" +"それらの生きていられないまだ設定されていない[ord](https://github.com/ordinals/ord)" +"ウォレットのユーザーは、ウォレットを使用する際に細心の注意を払っていれば、他のビットコインウォレットを使用して銘文と序数を受け取ることができます。。" + +#: src/guides/collecting/sparrow-wallet.md:6 +msgid "" +"This guide gives some basic steps on how to create a wallet with [Sparrow " +"Wallet](https://sparrowwallet.com/) which is compatible with `ord` and can " +"be later imported into `ord`" +msgstr "このガイドでは、[Sparrow Wallet](https://sparrowwallet.com/)" +"の使用方法を説明します。'ord'と互換性のあるウォレットを作成し、後で'ord'にインポートできます。`" + +#: src/guides/collecting/sparrow-wallet.md:8 +msgid "⚠️⚠️ Warning!! ⚠️⚠️" +msgstr "⚠️⚠️ 警告!! ⚠️⚠️" + +#: src/guides/collecting/sparrow-wallet.md:9 +msgid "" +"As a general rule if you take this approach, you should use this wallet with " +"the Sparrow software as a receive-only wallet." +msgstr "" +"一般的に、この方法を選ぶなら、この財布をお金を受け取る財布として、Sparrowソフトウェアを使うべきです" + +#: src/guides/collecting/sparrow-wallet.md:11 +msgid "" +"Do not spend any satoshis from this wallet unless you are sure you know what " +"you are doing. You could very easily inadvertently lose access to your " +"ordinals and inscriptions if you don't heed this warning." +msgstr "" +"自分が何をしているか分からない限り、この財布からビットコインを使わないでください。この警告に注意しなければ、" +"うっかり序数や銘文へのアクセスを失いやすいかもしれません" + +#: src/guides/collecting/sparrow-wallet.md:13 +msgid "Wallet Setup & Receiving" +msgstr "ウォレットの設置と取り受け" + +#: src/guides/collecting/sparrow-wallet.md:15 +msgid "" +"Download the Sparrow Wallet from the [releases page](https://sparrowwallet." +"com/download/) for your particular operating system." +msgstr "" +"オペレーティングシステムに応じて[リリースページ]から(https://sparrowwallet.com/download/)ダウンロードSparrowウォレット。" + +#: src/guides/collecting/sparrow-wallet.md:17 +msgid "Select `File -> New Wallet` and create a new wallet called `ord`." +msgstr "" +"『File->New Wallet』を選択し、『ord』という名前の新しいウォレットを作成します。" + +#: src/guides/collecting/sparrow-wallet.md:19 +msgid "![](images/wallet_setup_01.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:21 +msgid "" +"Change the `Script Type` to `Taproot (P2TR)` and select the `New or Imported " +"Software Wallet` option." +msgstr "" +"'Script Type'を'Taproot(P 2 TR)'に変更します。そして、New or Imported Software Wallet'オプションを選択します。" + +#: src/guides/collecting/sparrow-wallet.md:23 +msgid "![](images/wallet_setup_02.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:25 +msgid "" +"Select `Use 12 Words` and then click `Generate New`. Leave the passphrase " +"blank." +msgstr "" +"『Use 12 Words』を選択し、『Generate New』をクリックします。パスフレーズは空白のままにします。。" + +#: src/guides/collecting/sparrow-wallet.md:27 +msgid "![](images/wallet_setup_03.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:29 +msgid "" +"A new 12 word BIP39 seed phrase will be generated for you. Write this down " +"somewhere safe as this is your backup to get access to your wallet. NEVER " +"share or show this seed phrase to anyone else." +msgstr "" +"あなたのために新しい12語BIP 39シードフレーズが生成されます。このフレーズを安全な場所に書いてください。" +"これはウォレットへのアクセス権を取得するためのバックアップです。このシードフレーズを他の人と共有したり表示したりしないでください" + +#: src/guides/collecting/sparrow-wallet.md:31 +msgid "Once you have written down the seed phrase click `Confirm Backup`." +msgstr "シードフレーズを書いたら、`Confirm Backup`をクリックしてください' `Confirm Backup`." + +#: src/guides/collecting/sparrow-wallet.md:33 +msgid "![](images/wallet_setup_04.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:35 +msgid "" +"Re-enter the seed phrase which you wrote down, and then click `Create " +"Keystore`." +msgstr "" +"メモしたシードフレーズを再入力して、'Create Key storeをクリックしてください。`." + +#: src/guides/collecting/sparrow-wallet.md:37 +msgid "![](images/wallet_setup_05.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:39 +msgid "Click `Import Keystore`." +msgstr " `Import Keystoreをクリックします`." + + +#: src/guides/collecting/sparrow-wallet.md:41 +msgid "![](images/wallet_setup_06.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:43 +msgid "Click `Apply`. Add a password for the wallet if you want to." +msgstr "『Apply』をクリックします。もしあなたが望むなら、財布にパスワードを追加してもいいです。。" + +#: src/guides/collecting/sparrow-wallet.md:45 +msgid "![](images/wallet_setup_07.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:47 +msgid "" +"You now have a wallet which is compatible with `ord`, and can be imported " +"into `ord` using the BIP39 Seed Phrase. To receive ordinals or inscriptions, " +"click on the `Receive` tab and copy a new address." +msgstr "" +"今、BIP 39シードフレーズを使って'ord'にインポートできる'ord'互換のウォレットを持っています。序数や銘文を受け取るには、" +"'Receive'タブをクリックし、新しいアドレスをコピーします" + +#: src/guides/collecting/sparrow-wallet.md:49 +msgid "" +"Each time you want to receive you should use a brand-new address, and not re-" +"use existing addresses." +msgstr "受信するたびに、既存のアドレスを再利用するのではなく、新しいアドレスを使用する必要があります。" + +#: src/guides/collecting/sparrow-wallet.md:51 +msgid "" +"Note that bitcoin is different to some other blockchain wallets, in that " +"this wallet can generate an unlimited number of new addresses. You can " +"generate a new address by clicking on the `Get Next Address` button. You can " +"see all of your addresses in the `Addresses` tab of the app." +msgstr "" +"ビットコインは他のいくつかのブロックチェーンウォレットとは異なり、無限の数の新しいアドレスを生成できることに注意してください。" +"は次の住所を取得ボタンをクリックすることで新しい住所を生成できます。あなたはアプリケーションの'Addresses'タブですべてのアドレスを見ることができます" + +#: src/guides/collecting/sparrow-wallet.md:53 +msgid "" +"You can add a label to each address, so you can keep track of what it was " +"used for." +msgstr "各アドレスにタグを付けることができますので、目的を追跡することができます。" + +#: src/guides/collecting/sparrow-wallet.md:55 +msgid "![](images/wallet_setup_08.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:57 +msgid "Validating / Viewing Received Inscriptions" +msgstr "査証/受け取った銘文を見ます" + +#: src/guides/collecting/sparrow-wallet.md:59 +msgid "" +"Once you have received an inscription you will see a new transaction in the " +"`Transactions` tab of Sparrow, as well as a new UTXO in the `UTXOs` tab." +msgstr "" +"銘文を受け取ったら、Sparrowの『Transactions』タブに新しい取引が表示されます。" +"'UTXOs'タブに新しいUTXOが表示されます。" + +#: src/guides/collecting/sparrow-wallet.md:61 +msgid "" +"Initially this transaction may have an \"Unconfirmed\" status, and you will " +"need to wait for it to be mined into a bitcoin block before it is fully " +"received." +msgstr "" +"最初は、この取引には\"未確認\"の状態があるかもしれません。実際に受け取ったのはビットコインブロックに採掘されるのを待つ必要があります。" + +#: src/guides/collecting/sparrow-wallet.md:63 +msgid "![](images/validating_viewing_01.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:65 +msgid "" +"To track the status of your transaction you can right-click on it, select " +"`Copy Transaction ID` and then paste that transaction id into [mempool.space]" +"(https://mempool.space)." +msgstr "" +"取引状況を追跡するには、右クリックして'Copy Transaction ID'を選択することができます," +"そして、この取引idを[mempool.space](https://mempool.space)に貼り付けます。" + +#: src/guides/collecting/sparrow-wallet.md:67 +msgid "![](images/validating_viewing_02.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:69 +msgid "" +"Once the transaction has confirmed, you can validate and view your " +"inscription by heading over to the `UTXOs` tab, finding the UTXO you want to " +"check, right-clicking on the `Output` and selecting `Copy Transaction " +"Output`. This transaction output id can then be pasted into the [ordinals." +"com](https://ordinals.com) search." +msgstr "" +"取引が確認されたら、`UTXOs`タブに行くことによって、チェックしたい`UTXO`を見つけることができます,右クリック`Output`" +"'Copy Transaction Out put'を選択して、あなたの銘文を確認し、確認します。そして、この取引はidを輸出します" +"は[ordinals.com](https://ordinals.com)検索)に貼り付けることができます。。" + +#: src/guides/collecting/sparrow-wallet.md:72 +msgid "Freezing UTXO's" +msgstr " UTXOを凍結します" + +#: src/guides/collecting/sparrow-wallet.md:73 +msgid "" +"As explained above, each of your inscriptions is stored in an Unspent " +"Transaction Output (UTXO). You want to be very careful not to accidentally " +"spend your inscriptions, and one way to make it harder for this to happen is " +"to freeze the UTXO." +msgstr "" +"上記のように、あなたのすべての銘文は未費用の取引出力(UTXO)に保存されています。あなたの銘文を誤って使わないように細心の注意を払う必要があります」" +"がUTXOを凍結することは、このような状況を発生させる難易度を高める一つの方法であります。" + +#: src/guides/collecting/sparrow-wallet.md:75 +msgid "" +"To do this, go to the `UTXOs` tab, find the UTXO you want to freeze, right-" +"click on the `Output` and select `Freeze UTXO`." +msgstr "" +"これを行うには、UTXOsタブに行き、フリーズしたい'UTXOs'を見つけ、'Out put'をクリックして'Freeze UTXO'`を選択します。" + +#: src/guides/collecting/sparrow-wallet.md:77 +msgid "" +"This UTXO (Inscription) is now un-spendable within the Sparrow Wallet until " +"you unfreeze it." +msgstr "" +"このUTXO(銘文)は現在Sparrowウォレットでは、解凍するまで消費できません。。" + +#: src/guides/collecting/sparrow-wallet.md:79 +msgid "Importing into `ord` wallet" +msgstr " `ord` ウォレットを導入します。" + +#: src/guides/collecting/sparrow-wallet.md:81 +msgid "" +"For details on setting up Bitcoin Core and the `ord` wallet check out the " +"[Inscriptions Guide](../inscriptions.md)" +msgstr "" +"ビットコインコアと'ord'ウォレットの設定の詳細については、[銘文ガイド](../inscriptions.md)をご覧ください。" + +#: src/guides/collecting/sparrow-wallet.md:83 +msgid "" +"When setting up `ord`, instead of running `ord wallet create` to create a " +"brand-new wallet, you can import your existing wallet using `ord wallet " +"restore \"BIP39 SEED PHRASE\"` using the seed phrase you generated with " +"Sparrow Wallet." +msgstr "" +"'ord'を設定するときは、'ord wallet restore\"BIP 39 SE ED PHRASE\"'コマンドと" +"Sparrow Walletで生成されたシードフレーズを使用して、'ord wallet create'を実行する代わりに、既存のウォレットをインポートします。" +"は新しい財布を作ります。" + +#: src/guides/collecting/sparrow-wallet.md:85 +msgid "" +"There is currently a [bug](https://github.com/ordinals/ord/issues/1589) " +"which causes an imported wallet to not be automatically rescanned against " +"the blockchain. To work around this you will need to manually trigger a " +"rescan using the bitcoin core cli: `bitcoin-cli -rpcwallet=ord " +"rescanblockchain 767430`" +msgstr "" +"現在、[プログラムエラー]が存在します(https://github.com/ordinals/ord/issues/1589)インポートの原因" +"のウォレットはブロックチェーンを自動的に再スキャンできません。この問題を解決するには、手動で再スキャンをトリガーする必要があります。" +"Bitcoinコアコマンドラインインターフェイスを使います:" + +#: src/guides/collecting/sparrow-wallet.md:88 +msgid "" +"You can then check your wallet's inscriptions using `ord wallet inscriptions`" +msgstr "そして、'ord wallet inscriptions'を使用して、財布の碑文を確認することができます。." + +#: src/guides/collecting/sparrow-wallet.md:90 +msgid "" +"Note that if you have previously created a wallet with `ord`, then you will " +"already have a wallet with the default name, and will need to give your " +"imported wallet a different name. You can use the `--wallet` parameter in " +"all `ord` commands to reference a different wallet, eg:" +msgstr "" +"前にウォレットを'ord'で作成したことがある場合は、デフォルトの名前のウォレットを既に持っていることに注意してください。" +"はあなたが導入した財布に違う名前を付ける必要があります。すべての'ord'コマンドで'--wallet'引数を使用できます" +"異なる財布を引用するために、例えば:" + +#: src/guides/collecting/sparrow-wallet.md:92 +msgid "`ord --wallet ord_from_sparrow wallet restore \"BIP39 SEED PHRASE\"`" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:94 +msgid "`ord --wallet ord_from_sparrow wallet inscriptions`" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:96 +msgid "`bitcoin-cli -rpcwallet=ord_from_sparrow rescanblockchain 767430`" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:98 +msgid "Sending inscriptions with Sparrow Wallet" +msgstr "雀の財布を使って銘文を発送します。" + +#: src/guides/collecting/sparrow-wallet.md:100 +msgid "⚠️⚠️ Warning ⚠️⚠️" +msgstr "⚠️⚠️ 警告 ⚠️⚠️" + +#: src/guides/collecting/sparrow-wallet.md:101 +msgid "" +"While it is highly recommended that you set up a bitcoin core node and run " +"the `ord` software, there are certain limited ways you can send inscriptions " +"out of Sparrow Wallet in a safe way. Please note that this is not " +"recommended, and you should only do this if you fully understand what you " +"are doing." +msgstr "" +"ビットコインコアノードを設定し、'ord'ソフトウェアを実行することを強くお勧めしますが、いくつかの安全な方法で" +"Sparrowウォレットに銘文をお送りします。これはお勧めではないことに注意してください。あなたが何をしているのか完全に理解している場合にのみ、そうすることができます。" + +#: src/guides/collecting/sparrow-wallet.md:103 +msgid "" +"Using the `ord` software will remove much of the complexity we are " +"describing here, as it is able to automatically and safely handle sending " +"inscriptions in an easy way." +msgstr "" +"『ord』ソフトウェアを使用すると、ここで説明した複雑さが大幅に簡素化されます。それは、簡単な方法で自動的かつ安全に碑文の送信を処理できるからです。。"" + +#: src/guides/collecting/sparrow-wallet.md:105 +msgid "⚠️⚠️ Additional Warning ⚠️⚠️" +msgstr "⚠️⚠️ 特別警告 ⚠️⚠️" + +#: src/guides/collecting/sparrow-wallet.md:106 +msgid "" +"Don't use your sparrow inscriptions wallet to do general sends of non-" +"inscription bitcoin. You can setup a separate wallet in sparrow if you need " +"to do normal bitcoin transactions, and keep your inscriptions wallet " +"separate." +msgstr "" +"sparrowスズメの銘文財布を使って非銘文ビットコインを送らないでください。通常のビットコイン取引が必要な場合は、" +"はスズメの中に別の財布を設置して、銘文財布を独立させることができます。" + +#: src/guides/collecting/sparrow-wallet.md:108 +msgid "Bitcoin's UTXO model" +msgstr "ビットコインのUTXO模型" + +#: src/guides/collecting/sparrow-wallet.md:109 +msgid "" +"Before sending any transaction it's important that you have a good mental " +"model for bitcoin's Unspent Transaction Output (UTXO) system. The way " +"Bitcoin works is fundamentally different to many other blockchains such as " +"Ethereum. In Ethereum generally you have a single address in which you store " +"ETH, and you cannot differentiate between any of the ETH - it is just all a " +"single value of the total amount in that address. Bitcoin works very " +"differently in that we generate a new address in the wallet for each " +"receive, and every time you receive sats to an address in your wallet you " +"are creating a new UTXO. Each UTXO can be seen and managed individually. You " +"can select specific UTXO's which you want to spend, and you can choose not " +"to spend certain UTXO's." +msgstr "" +"取引を送信する前に、ビットコインの未消費取引出力(UTXO)システムをよく理解しなければなりません。" +"ビットコインの仕組みはイーサリアムなど他の多くのブロックチェーンと根本的に異なっている。イーサリアムでは、通常、ETHを保存する単一のアドレスを持っています。" +"これらのETHのいずれかを区別することはできません。それらはその住所の合計金額の単一の値にすぎません。" +"ビットコインの仕組みは全く異なります。私たちは受信ごとに新しいアドレスを生成します。ウォレットのアドレスの1つにsatsを受信するたびに" +"あなたは新しいUTXOを作っています。各UTXOは個別に表示および管理できます。" +"特定のUTXOを使用するか、または特定のUTXOを使用しないかを選択できます。" + +#: src/guides/collecting/sparrow-wallet.md:111 +msgid "" +"Some Bitcoin wallets do not expose this level of detail, and they just show " +"you a single summed up value of all the bitcoin in your wallet. However, " +"when sending inscriptions it is important that you use a wallet like Sparrow " +"which allows for UTXO control." +msgstr "" +"ビットコインウォレットの中には、このレベルの詳細情報を表示しないものもあります。ウォレット内のすべてのビットコインの単一の合計値しか表示しません。" +"しかし、銘文を送るときは、スズメのようなUTXO制御を許す財布を使うことが非常に重要であります。" + +#: src/guides/collecting/sparrow-wallet.md:113 +msgid "Inspecting your inscription before sending" +msgstr "在发送之前检查你的铭文" + +#: src/guides/collecting/sparrow-wallet.md:114 +msgid "" +"Like we have previously described inscriptions are inscribed onto sats, and " +"sats are stored within UTXOs. UTXO's are a collection of satoshis with some " +"particular value of the number of satoshis (the output value). Usually (but " +"not always) the inscription will be inscribed on the first satoshi in the " +"UTXO." +msgstr "" +"的第一个satoshi上前にも述べたように、銘文はは聡に刻まれ、satsはUTXOに格納されています。" +"UTXOは、ある特定の数のsatoshi(出力値)を持つsatoshiの集合です。銘文はUTXOの最初の聡に刻まれることが多いが、いつもそうではないけど」。" + +#: src/guides/collecting/sparrow-wallet.md:116 +msgid "" +"When inspecting your inscription before sending the main thing you will want " +"to check is which satoshi in the UTXO your inscription is inscribed on." +msgstr "" +"発送前にあなたの銘文をチェックするとき、主にチェックしなければならないのはあなたの銘文がUTXOのどのsatoshiに刻まれているかです。" + +#: src/guides/collecting/sparrow-wallet.md:118 +msgid "" +"To do this, you can follow the [Validating / Viewing Received Inscriptions]" +"(./sparrow-wallet.md#validating--viewing-received-inscriptions) described " +"above to find the inscription page for your inscription on ordinals.com" +msgstr "" +"そのためには、上記のように[受け取った銘文を検証/見る]することができます(./sparrow-wallet.md#validating--viewing-received-inscriptions)" +"ordinals.comのあなたの銘文の銘文ページを見つけるために来ます。" + +#: src/guides/collecting/sparrow-wallet.md:120 +msgid "" +"There you will find some metadata about your inscription which looks like " +"the following:" +msgstr "そこで、あなたの銘文に関するメタデータを見つけます。以下のようになります:" + +#: src/guides/collecting/sparrow-wallet.md:122 +msgid "![](images/sending_01.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:124 +msgid "There is a few of important things to check here:" +msgstr "以下は、チェックすべき重要事項です:" + +#: src/guides/collecting/sparrow-wallet.md:125 +msgid "" +"The `output` identifier matches the identifier of the UTXO you are going to " +"send" +msgstr "`output` 識別子が送信しようとしているUTXOの識別子と一致します" + +#: src/guides/collecting/sparrow-wallet.md:126 +msgid "" +"The `offset` of the inscription is `0` (this means that the inscription is " +"located on the first sat in the UTXO)" +msgstr "銘文の'off set'は'0'であり、(これは銘文がUTXOの最初のsatにあることを意味する)" + +#: src/guides/collecting/sparrow-wallet.md:127 +msgid "" +"the `output_value` has enough sats to cover the transaction fee (postage) " +"for sending the transaction. The exact amount you will need depends on the " +"fee rate you will select for the transaction" +msgstr "" +"`output_value` 取引を送るための取引料金(郵便料金)を支払うのに十分なsatsがあります。必要な正確な金額は" +"あなたが取引に選択したレート" + +#: src/guides/collecting/sparrow-wallet.md:129 +msgid "" +"If all of the above are true for your inscription, it should be safe for you " +"to send it using the method below." +msgstr "" +"上記の内容がすべてあなたの碑文に正しい場合、あなたは安全に以下の方法でそれを送ることができるはずです。" + +#: src/guides/collecting/sparrow-wallet.md:131 +msgid "" +"⚠️⚠️ Be very careful sending your inscription particularly if the `offset` " +"value is not `0`. It is not recommended to use this method if that is the " +"case, as doing so you could accidentally send your inscription to a bitcoin " +"miner unless you know what you are doing." +msgstr "" +"⚠️⚠️ 銘文を送るときは、特に'off set'値が'0'でない場合は細心の注意を払ってください。この場合は、" +"この方法をお勧めしません。そうしないと、自分が何をしているのか分からない限り、あなたの銘文をビットコイン鉱山労働者に送ることになるかもしれません。" + +#: src/guides/collecting/sparrow-wallet.md:133 +msgid "Sending your inscription" +msgstr "あなたの銘文を発送します。" + +#: src/guides/collecting/sparrow-wallet.md:134 +msgid "" +"To send an inscription navigate to the `UTXOs` tab, and find the UTXO which " +"you previously validated contains your inscription." +msgstr "" +"要銘文を送信するには、'UTXOs'タブに移動し、以前にあなたの彫刻が含まれていることを確認したUTXOを見つけてください。" + +#: src/guides/collecting/sparrow-wallet.md:136 +msgid "" +"If you previously froze the UXTO you will need to right-click on it and " +"unfreeze it." +msgstr "" +"以前にUXTOを凍結した場合、右クリックして解凍する必要があります。" + +#: src/guides/collecting/sparrow-wallet.md:138 +msgid "" +"Select the UTXO you want to send, and ensure that is the _only_ UTXO is " +"selected. You should see `UTXOs 1/1` in the interface. Once you are sure " +"this is the case you can hit `Send Selected`." +msgstr "" +"送信するUTXOを選択します。これが唯一選択されたUTXOであることを確認します。インターフェイスでは、'UTXOs 1/1'が表示されるはずです。" +"これを確認した後、'Send Selected'をクリックすることができます。" + +#: src/guides/collecting/sparrow-wallet.md:140 +msgid "![](images/sending_02.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:142 +msgid "" +"You will then be presented with the transaction construction interface. " +"There is a few things you need to check here to make sure that this is a " +"safe send:" +msgstr "" +"次に、トランザクション構築インターフェイスが表示されます。ここでは、これが安全な送信であることを確認するためにいくつかのことを確認する必要があります:" + +#: src/guides/collecting/sparrow-wallet.md:144 +msgid "" +"The transaction should have only 1 input, and this should be the UTXO with " +"the label you want to send" +msgstr "" +"取引には1つの入力しかないはずです。これはラベル付きのUTXOを送りたいはずです" + +#: src/guides/collecting/sparrow-wallet.md:145 +msgid "" +"The transaction should have only 1 output, which is the address/label where " +"you want to send the inscription" +msgstr "" +"取引には1つの出力しかないはずです。これは銘文を送りたい住所/ラベルです" + +#: src/guides/collecting/sparrow-wallet.md:147 +msgid "" +"If your transaction looks any different, for example you have multiple " +"inputs, or multiple outputs then this may not be a safe transfer of your " +"inscription, and you should abandon sending until you understand more, or " +"can import into the `ord` wallet." +msgstr "" +"取引がこれとは異なるように見える場合、例えば複数の入力や複数の出力がある場合、これは安全な碑文の転送方法ではないかもしれません。" +"あなたがよりよく知っているか、'ord'ウォレットにインポートすることができるまで、送信を断念する必要があります。" + +#: src/guides/collecting/sparrow-wallet.md:149 +msgid "" +"You should set an appropriate transaction fee, Sparrow will usually " +"recommend a reasonable one, but you can also check [mempool.space](https://" +"mempool.space) to see what the recommended fee rate is for sending a " +"transaction." +msgstr "" +"適切な取引手数料を設定する必要があります。Sparrowは通常、合理的な料金をお勧めします。" +"しかし、[mempool.space]を見て、(https://mempool.space)送信トランザクションの推奨レートを見ます。" + +#: src/guides/collecting/sparrow-wallet.md:151 +msgid "" +"You should add a label for the recipient address, a label like `alice " +"address for inscription #123` would be ideal." +msgstr "" +"受信者の住所にラベルを追加する必要があります。『alice address for inscription#123』は理想的です。" + +#: src/guides/collecting/sparrow-wallet.md:153 +msgid "" +"Once you have checked the transaction is a safe transaction using the checks " +"above, and you are confident to send it you can click `Create Transaction`." +msgstr "" +"上記のチェックを使用して、取引が安全な取引であることを確認し、それを送信する自信があることを確認したら、『Create Transaction』をクリックしてください。" + +#: src/guides/collecting/sparrow-wallet.md:155 +msgid "![](images/sending_03.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:157 +msgid "" +"Here again you can double check that your transaction looks safe, and once " +"you are confident you can click `Finalize Transaction for Signing`." +msgstr "" +"ここで、あなたの取引が安全かどうかを再確認できます。確認した後、『Finalize Transaction for Signing'をクリックしてください。" + +#: src/guides/collecting/sparrow-wallet.md:159 +msgid "![](images/sending_04.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:161 +msgid "Here you can triple check everything before hitting `Sign`." +msgstr "ここでは、『Sign』をクリックする前にすべてを再確認できます。" + +#: src/guides/collecting/sparrow-wallet.md:163 +msgid "![](images/sending_05.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:165 +msgid "" +"And then actually you get very very last chance to check everything before " +"hitting `Broadcast Transaction`. Once you broadcast the transaction it is " +"sent to the bitcoin network, and starts being propagated into the mempool." +msgstr "" +"そして、実際には『Broadcast Transaction』をクリックする前に、すべてをチェックする最後の機会があります。" +"は取引を放送すると、ビットコインネットワークに送られ、mempoolで広がり始めます。" + +#: src/guides/collecting/sparrow-wallet.md:167 +msgid "![](images/sending_06.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:169 +msgid "" +"If you want to track the status of your transaction you can copy the " +"`Transaction Id (Txid)` and paste that into [mempool.space](https://mempool." +"space)" +msgstr "" +"取引状態を追跡したいなら、『Transaction Id(Txid)をコピーして" +"[mempool.space](https://mempool.space)'に貼り付けます。" + + +#: src/guides/collecting/sparrow-wallet.md:171 +msgid "" +"Once the transaction has confirmed you can check the inscription page on " +"[ordinals.com](https://ordinals.com) to validate that it has moved to the " +"new output location and address." +msgstr "" +"取引が確認されたら、[ordinals.com](https://ordinals.com)" +"の銘文ページで、新しい出力場所と住所に移動したことを確認します。" + +#: src/guides/collecting/sparrow-wallet.md:173 +msgid "Troubleshooting" +msgstr "故障排除" + +#: src/guides/collecting/sparrow-wallet.md:175 +msgid "" +"Sparrow wallet is not showing a transaction/UTXO, but I can see it on " +"mempool.space!" +msgstr "Sparrowウォレットは取引/UTXOを表示していませんが、mempool.spaceでを見ました" + +#: src/guides/collecting/sparrow-wallet.md:177 +msgid "" +"Make sure that your wallet is connected to a bitcoin node. To validate this, " +"head into the `Preferences`\\-> `Server` settings, and click `Edit Existing " +"Connection`." +msgstr "" +"ウォレットがビットコインノードに接続されていることを確認してください。これを確認するには、'Preferences'\\-->'Server'の設定に移動します。" +"そして、'Edit Existing Connection'をクリックしてください。。" + +#: src/guides/collecting/sparrow-wallet.md:179 +msgid "![](images/troubleshooting_01.png)" +msgstr "" + +#: src/guides/collecting/sparrow-wallet.md:181 +msgid "" +"From there you can select a node and click `Test Connection` to validate " +"that Sparrow is able to connect successfully." +msgstr "そこからノードを選択し、'Test Connection'をクリックしてSparrowが正常に接続できることを確認できます。" + +#: src/guides/collecting/sparrow-wallet.md:183 +msgid "![](images/troubleshooting_02.png)" +msgstr "" + +#: src/guides/testing.md:4 +msgid "" +"Ord can be tested using the following flags to specify the test network. For " +"more information on running Bitcoin Core for testing, see [Bitcoin's " +"developer documentation](https://developer.bitcoin.org/examples/testing." +"html)." +msgstr "" +"次のフラグを使用してテストネットワークを指定すると、フォームをテストできます。テストのためのビットコインコアの実行の詳細については、" +"[ビットコインの開発者ドキュメント]を参照してください。" + +#: src/guides/testing.md:7 +msgid "" +"Most `ord` commands in [inscriptions](inscriptions.md) and [explorer]" +"(explorer.md) can be run with the following network flags:" +msgstr "" +"[銘文](inscriptions.md)および[ブラウザ](explorer.md)内のほとんどの'ord'コマンドは、次のネットワークフラグを使用して実行できます:" + +#: src/guides/testing.md:10 +msgid "Network" +msgstr "" + +#: src/guides/testing.md:10 +msgid "Flag" +msgstr "" + +#: src/guides/testing.md:12 +msgid "Testnet" +msgstr "" + +#: src/guides/testing.md:12 +msgid "`--testnet` or `-t`" +msgstr "" + +#: src/guides/testing.md:13 +msgid "Signet" +msgstr "" + +#: src/guides/testing.md:13 +msgid "`--signet` or `-s`" +msgstr "" + +#: src/guides/testing.md:14 +msgid "Regtest" +msgstr "" + +#: src/guides/testing.md:14 +msgid "`--regtest` or `-r`" +msgstr "" + +#: src/guides/testing.md:16 +msgid "Regtest doesn't require downloading the blockchain or indexing ord." +msgstr "Regtestはブロックチェーンをダウンロードしたり、ordインデックスを作成したりする必要はありません" + +#: src/guides/testing.md:18 src/guides/reindexing.md:15 +msgid "Example" +msgstr "サンプル" + +#: src/guides/testing.md:21 +msgid "Run bitcoind in regtest with:" +msgstr "regtestの中でbitcoindを運行し、使います:" + +#: src/guides/testing.md:22 +msgid "" +"```\n" +"bitcoind -regtest -txindex\n" +"```" +msgstr "" + +#: src/guides/testing.md:25 +msgid "Create a wallet in regtest with:" +msgstr "regtestの中でウォレットを作り上げます。" + +#: src/guides/testing.md:26 +msgid "" +"```\n" +"ord -r wallet create\n" +"```" +msgstr "" + +#: src/guides/testing.md:29 +msgid "Get a regtest receive address with:" +msgstr "regtest受け取りアドレスを作成します" + +#: src/guides/testing.md:30 +msgid "" +"```\n" +"ord -r wallet receive\n" +"```" +msgstr "" + +#: src/guides/testing.md:33 +msgid "Mine 101 blocks (to unlock the coinbase) with:" +msgstr "101個のブロックを掘る(ロック解除coinbase)に使います:" + +#: src/guides/testing.md:34 +msgid "" +"```\n" +"bitcoin-cli generatetoaddress 101 \n" +"```" +msgstr "" + +#: src/guides/testing.md:37 +msgid "Inscribe in regtest with:" +msgstr "regtest上で刻みます。" + +#: src/guides/testing.md:38 +msgid "" +"```\n" +"ord -r wallet inscribe --fee-rate 1 \n" +"```" +msgstr "" + +#: src/guides/testing.md:41 +msgid "Mine the inscription with:" +msgstr "銘文を発掘します。" + +#: src/guides/testing.md:42 +msgid "" +"```\n" +"bitcoin-cli generatetoaddress 1 \n" +"```" +msgstr "" + +#: src/guides/testing.md:45 +msgid "View the inscription in the regtest explorer:" +msgstr "regtestブラウザで銘文を見ます。" + +#: src/guides/testing.md:46 +msgid "" +"```\n" +"ord -r server\n" +"```" +msgstr "" + +#: src/guides/testing.md:50 +msgid "Testing Recursion" +msgstr "再帰のテスト" + +#: src/guides/testing.md:53 +msgid "" +"When testing out [recursion](../inscriptions/recursion.md), inscribe the " +"dependencies first (example with [p5.js](https://p5js.org):" +msgstr "" +" [recursion](../inscriptions/recursion.md) のテスト[の場合、最初に依存関係をメモします。" +"( [p5.js](https://p5js.org) を例とします:" + +#: src/guides/testing.md:55 +msgid "" +"```\n" +"ord -r wallet inscribe --fee-rate 1 p5.js\n" +"```" +msgstr "" + +#: src/guides/testing.md:58 +msgid "" +"This should return a `inscription_id` which you can then reference in your " +"recursive inscription." +msgstr "" +"これは'inscription_id'を返す必要があります。その後、再帰的な銘文でそれを参照することができます。。" + +#: src/guides/testing.md:61 +msgid "" +"ATTENTION: These ids will be different when inscribing on mainnet or signet, " +"so be sure to change those in your recursive inscription for each chain." +msgstr "" +"これは'inscription_id'を返す必要があります。その後、再帰的な銘文でそれを参照することができます。" + +#: src/guides/testing.md:65 +msgid "Then you can inscribe your recursive inscription with:" +msgstr "次のコマンドを使用して、再帰的な碑文を刻むことができます。:" + +#: src/guides/testing.md:66 +msgid "" +"```\n" +"ord -r wallet inscribe --fee-rate 1 recursive-inscription.html\n" +"```" +msgstr "" + +#: src/guides/testing.md:69 +msgid "Finally you will have to mine some blocks and start the server:" +msgstr "最終的には、サーバーを開始するためにいくつかのブロックを掘ることができます:" + +#: src/guides/testing.md:70 +msgid "" +"```\n" +"bitcoin-cli generatetoaddress 6 \n" +"ord -r server\n" +"```" +msgstr "" + +#: src/guides/moderation.md:4 +msgid "" +"`ord` includes a block explorer, which you can run locally with `ord server`." +msgstr "" +"`ord` ブロックブラウザが含まれています,あなたはローカルで'ord server`運行することができます." + +#: src/guides/moderation.md:6 +msgid "" +"The block explorer allows viewing inscriptions. Inscriptions are user-" +"generated content, which may be objectionable or unlawful." +msgstr "" +"ブロックブラウザでは、銘文を見ることができます。銘文はユーザーが生成した内容であるため、不快または違法である可能性があります。" + +#: src/guides/moderation.md:9 +msgid "" +"It is the responsibility of each individual who runs an ordinal block " +"explorer instance to understand their responsibilities with respect to " +"unlawful content, and decide what moderation policy is appropriate for their " +"instance." +msgstr "" +"Wordブロックブラウザのインスタンスを実行するすべてのユーザーは、不正なコンテンツに対する責任を理解し、そのインスタンスに適した監査ポリシーを決定する責任があります。" + +#: src/guides/moderation.md:13 +msgid "" +"In order to prevent particular inscriptions from being displayed on an `ord` " +"instance, they can be included in a YAML config file, which is loaded with " +"the `--config` option." +msgstr "" +"特定の銘文が'ord'インスタンスに表示されないようにするには、YAML設定ファイルに含めることができます。" +'--config'オプションを使用してファイルをロードします。" + +#: src/guides/moderation.md:17 +msgid "" +"To hide inscriptions, first create a config file, with the inscription ID " +"you want to hide:" +msgstr "" +"銘文を隠れると、先ずプロファイルを作成しますその中は隠れる銘文IDが含まれます。" + +#: src/guides/moderation.md:20 +msgid "" +"```yaml\n" +"hidden:\n" +"- 0000000000000000000000000000000000000000000000000000000000000000i0\n" +"```" +msgstr "" + +#: src/guides/moderation.md:25 +msgid "" +"The suggested name for `ord` config files is `ord.yaml`, but any filename " +"can be used." +msgstr "" +"'ord'設定ファイルの推奨名は'ord.yaml'です。ただし、任意のファイル名を使用できます。" + +#: src/guides/moderation.md:28 +msgid "Then pass the file to `--config` when starting the server:" +msgstr "" +"その後、サービスの開始時にファイルを使います--config` :" + +#: src/guides/moderation.md:30 +msgid "`ord --config ord.yaml server`" +msgstr "" + +#: src/guides/moderation.md:32 +msgid "" +"Note that the `--config` option comes after `ord` but before the `server` " +"subcommand." +msgstr "" +"注意してください。 『--config'オプションは'ord'の後で'server'サブコマンドの前にあります。" + +#: src/guides/moderation.md:35 +msgid "`ord` must be restarted in to load changes to the config file." +msgstr "" +"'ord'は、設定ファイルの変更をロードするために再起動する必要があります。" + +#: src/guides/moderation.md:37 +msgid "`ordinals.com`" +msgstr "" + +#: src/guides/moderation.md:40 +msgid "" +"The `ordinals.com` instances use `systemd` to run the `ord server` service, " +"which is called `ord`, with a config file located at `/var/lib/ord/ord.yaml`." +msgstr "" +"ordinals.com'インスタンスは'systemd'を使用して'ord'という名前の'ord server'サービスを実行しています。" +"設定ファイルは『/var/lib/ord/ord.yaml'." + +#: src/guides/moderation.md:43 +msgid "To hide an inscription on `ordinals.com`:" +msgstr " ordinals.com 上で銘文を隠れます:" + +#: src/guides/moderation.md:45 +msgid "SSH into the server" +msgstr "SSHでサーバを登録します。" + +#: src/guides/moderation.md:46 +msgid "Add the inscription ID to `/var/lib/ord/ord.yaml`" +msgstr " `/var/lib/ord/ord.yaml`中で銘文を増やしますID" + +#: src/guides/moderation.md:47 +msgid "Restart the service with `systemctl restart ord`" +msgstr " `systemctl restart ord`で サービスを再起動します。" + +#: src/guides/moderation.md:48 +msgid "Monitor the restart with `journalctl -u ord`" +msgstr " `journalctl -u ord` で再起動します。" + +#: src/guides/moderation.md:50 +msgid "" +"Currently, `ord` is slow to restart, so the site will not come back online " +"immediately." +msgstr "今は、ordの再起動の速度が遅いから、そのため、サイトはすぐにオンラインに戻りません。" + +#: src/guides/reindexing.md:4 +msgid "" +"Sometimes the `ord` database must be reindexed, which means deleting the " +"database and restarting the indexing process with either `ord index run` or " +"`ord server`. Reasons to reindex are:" +msgstr "" +"「ord」データベースを再インデックスする必要がある場合があります。これは、データベースを削除し、「ord index run」または「ord server」を使用することを意味します。" +"を使用してデータベースを再インデックス化します。再インデックスを作成する理由は次のとおりです。:" + +#: src/guides/reindexing.md:8 +msgid "A new major release of ord, which changes the database scheme" +msgstr "ord 发布新的主要版本,更改了数据库架构 新たなメインバージョンをリリースして、データベースのスキーマを変更されました。" + +#: src/guides/reindexing.md:9 +msgid "The database got corrupted somehow" +msgstr "データベースは壊れるかもしれません" + +#: src/guides/reindexing.md:11 +msgid "" +"The database `ord` uses is called [redb](https://github.com/cberner/redb), " +"so we give the index the default file name `index.redb`. By default we store " +"this file in different locations depending on your operating system." +msgstr "" +"`ord`は 使っているデータベースは [redb](https://github.com/cberner/redb)を呼ばれています," +"索引のために‘index.redb’を指定された黙認ファイル名として、黙認する場合は保存します" +"操作システムによって、このファイルは異なる位置にあります" + +#: src/guides/reindexing.md:15 +msgid "Platform" +msgstr "プラットフォーム" + +#: src/guides/reindexing.md:15 +msgid "Value" +msgstr "" + +#: src/guides/reindexing.md:17 +msgid "Linux" +msgstr "" + +#: src/guides/reindexing.md:17 +msgid "`$XDG_DATA_HOME`/ord or `$HOME`/.local/share/ord" +msgstr "" + +#: src/guides/reindexing.md:17 +msgid "/home/alice/.local/share/ord" +msgstr "" + +#: src/guides/reindexing.md:18 +msgid "macOS" +msgstr "" + +#: src/guides/reindexing.md:18 +msgid "`$HOME`/Library/Application Support/ord" +msgstr "" + +#: src/guides/reindexing.md:18 +msgid "/Users/Alice/Library/Application Support/ord" +msgstr "" + +#: src/guides/reindexing.md:19 +msgid "Windows" +msgstr "" + +#: src/guides/reindexing.md:19 +msgid "`{FOLDERID_RoamingAppData}`\\\\ord" +msgstr "" + +#: src/guides/reindexing.md:19 +msgid "C:\\Users\\Alice\\AppData\\Roaming\\ord" +msgstr "" + +#: src/guides/reindexing.md:21 +msgid "" +"So to delete the database and reindex on MacOS you would have to run the " +"following commands in the terminal:" +msgstr "" +"ですので,MacOSの上にデータベースを削除して再索引し、ターミナルでこのコマンドを実行します。" + +#: src/guides/reindexing.md:24 +msgid "" +"```bash\n" +"rm ~/Library/Application Support/ord/index.redb\n" +"ord index run\n" +"```" +msgstr "" + +#: src/guides/reindexing.md:29 +msgid "" +"You can of course also set the location of the data directory yourself with " +"`ord --data-dir index run` or give it a specific filename and path " +"with `ord --index index run`." +msgstr "" +"もちろん自分でデータの目録の位置を設置することができます,`ord --data-dir index run` または" +"指定された特定のファイル名とパスに‘ord --index で索引運行します’。" + +#: src/bounties.md:1 +msgid "Ordinal Bounty Hunting Hints" +msgstr "Ordinals賞金計画の提示" + +#: src/bounties.md:4 +msgid "" +"The `ord` wallet can send and receive specific satoshis. Additionally, " +"ordinal theory is extremely simple. A clever hacker should be able to write " +"code from scratch to manipulate satoshis using ordinal theory in no time." +msgstr "" +"`ord` ウォレットは特定の聡を発送して、受け取れます。その他、序数理論は非常に簡単で、賢いハッカが速く最初から始めることができるはずです" +"コードを書いて、序数理論を使って聡を操作します。" + +#: src/bounties.md:8 +msgid "" +"For more information about ordinal theory, check out the [FAQ](./faq.md) for " +"an overview, the [BIP](https://github.com/ordinals/ord/blob/master/bip." +"mediawiki) for the technical details, and the [ord repo](https://github.com/" +"ordinals/ord) for the `ord` wallet and block explorer." +msgstr "" +"序数理論のより多くの情報に関して,[FAQ](./faq.md) を閲見して概要を獲得します。" +"[BIP](https://github.com/ordinals/ord/blob/master/bip.mediawiki) を閲見して、技術の詳細を獲得します" +"[ord repo](https://github.com/ordinals/ord)を閲見して、`ord`ウォレットとブラウザの情報を獲得します." + +#: src/bounties.md:14 +msgid "" +"Satoshi was the original developer of ordinal theory. However, he knew that " +"others would consider it heretical and dangerous, so he hid his knowledge, " +"and it was lost to the sands of time. This potent theory is only now being " +"rediscovered. You can help by researching rare satoshis." +msgstr "" +"聡は序数理論の原始の開発者であり、しかし、他の人は変なことだと、危険だと思われますが彼は自分の知識を隠れて、" +"時間の砂漠に消えてしまいます。現在、この強い理論は発見されて、珍しい聡を研究することによって私たちを手伝っていただきます。" + +#: src/bounties.md:19 +msgid "Good luck and godspeed!" +msgstr "ご好運、順調に進んでいるように!" + +#: src/bounty/0.md:1 +msgid "Ordinal Bounty 0" +msgstr "賞金任務 0" + +#: src/bounty/0.md:4 src/bounty/1.md:4 src/bounty/2.md:4 src/bounty/3.md:4 +msgid "Criteria" +msgstr "標準" + +#: src/bounty/0.md:7 +msgid "" +"Send a sat whose ordinal number ends with a zero to the submission address:" +msgstr "序数の“0”で終わりの聡を提出アドレスに発送します。" + +#: src/bounty/0.md:9 +msgid "✅: [1857578125803250](https://ordinals.com/ordinal/1857578125803250)" +msgstr "" + +#: src/bounty/0.md:11 +msgid "❌: [1857578125803251](https://ordinals.com/ordinal/1857578125803251)" +msgstr "" + +#: src/bounty/0.md:13 +msgid "The sat must be the first sat of the output you send." +msgstr "聡はあなたが発送した“輸出”の初めての聡でなければなりません。" + +#: src/bounty/0.md:15 src/bounty/1.md:14 src/bounty/2.md:15 src/bounty/3.md:63 +msgid "Reward" +msgstr "奨励" + +#: src/bounty/0.md:18 +msgid "100,000 sats" +msgstr "" + +#: src/bounty/0.md:20 src/bounty/1.md:19 src/bounty/2.md:20 src/bounty/3.md:70 +msgid "Submission Address" +msgstr "アドレスを提出する" + +#: src/bounty/0.md:23 +msgid "" +"[`1PE7u4wbDP2RqfKN6geD1bG57v9Gj9FXm3`](https://mempool.space/" +"address/1PE7u4wbDP2RqfKN6geD1bG57v9Gj9FXm3)" +msgstr "" + +#: src/bounty/0.md:25 src/bounty/1.md:24 src/bounty/2.md:25 src/bounty/3.md:75 +msgid "Status" +msgstr "状態" + +#: src/bounty/0.md:28 +msgid "" +"Claimed by [@count_null](https://twitter.com/rodarmor/" +"status/1560793241473400833)!" +msgstr "" +"[@count_null](https://twitter.com/rodarmor/" +"status/1560793241473400833)! 獲得" + +#: src/bounty/1.md:1 +msgid "Ordinal Bounty 1" +msgstr "賞金任務 1" + +#: src/bounty/1.md:7 +msgid "" +"The transaction that submits a UTXO containing the oldest sat, i.e., that " +"with the lowest number, amongst all submitted UTXOs will be judged the " +"winner." +msgstr "一つの最も古い聡を含まれるUTXOを提出されます。例えば、すべて提出されたUTXOの中に最も小さい数字は優勝者とします" + +#: src/bounty/1.md:10 +msgid "" +"The bounty is open for submissions until block 753984—the first block of " +"difficulty adjustment period 374. Submissions included in block 753984 or " +"later will not be considered." +msgstr "" +"賞金はブロック高度753984 の前に有効であり、ブロック高度753984は初めての難易度の調整期374の後の初めてのブロックです。" +"これらを含まれてまたはブロック高度753984より遅いのは考えされません。" + +#: src/bounty/1.md:17 +msgid "200,000 sats" +msgstr "" + +#: src/bounty/1.md:22 +msgid "" +"[`145Z7PFHyVrwiMWwEcUmDgFbmUbQSU9aap`](https://mempool.space/" +"address/145Z7PFHyVrwiMWwEcUmDgFbmUbQSU9aap)" +msgstr "" + +#: src/bounty/1.md:27 +msgid "" +"Claimed by [@ordinalsindex](https://twitter.com/rodarmor/" +"status/1569883266508853251)!" +msgstr "" +" [@ordinalsindex](https://twitter.com/rodarmor/" +"status/1569883266508853251)は獲得!" + +#: src/bounty/2.md:1 +msgid "Ordinal Bounty 2" +msgstr "賞金任務 2" + +#: src/bounty/2.md:7 +msgid "Send an " +msgstr "一つ発送します" + +#: src/bounty/2.md:7 +msgid "uncommon" +msgstr "普通ではないの" + +#: src/bounty/2.md:7 +msgid " sat to the submission address:" +msgstr "聡は下のアドレスに" + +#: src/bounty/2.md:9 +msgid "✅: [347100000000000](https://ordinals.com/sat/347100000000000)" +msgstr "" + +#: src/bounty/2.md:11 +msgid "❌: [6685000001337](https://ordinals.com/sat/6685000001337)" +msgstr "" + +#: src/bounty/2.md:13 +msgid "" +"Confirm that the submission address has not received transactions before " +"submitting your entry. Only the first successful submission will be rewarded." +msgstr "" +"提出する前に、上述のアドレスがこの前に他の珍しい聡が受け取らなかったことを確認し、最初に提出されたのは奨励を得られます。" + +#: src/bounty/2.md:18 +msgid "300,000 sats" +msgstr "" + +#: src/bounty/2.md:23 +msgid "" +"[`1Hyr94uypwWq5CQffaXHvwUMEyBPp3TUZH`](https://mempool.space/" +"address/1Hyr94uypwWq5CQffaXHvwUMEyBPp3TUZH)" +msgstr "" + +#: src/bounty/2.md:28 +msgid "" +"Claimed by [@utxoset](https://twitter.com/rodarmor/" +"status/1582424455615172608)!" +msgstr "" +"[@utxoset](https://twitter.com/rodarmor/" +"status/1582424455615172608) は獲得!" + +#: src/bounty/3.md:1 +msgid "Ordinal Bounty 3" +msgstr "賞金任務 3" + +#: src/bounty/3.md:7 +msgid "" +"Ordinal bounty 3 has two parts, both of which are based on _ordinal names_. " +"Ordinal names are a modified base-26 encoding of ordinal numbers. To avoid " +"locking short names inside the unspendable genesis block coinbase reward, " +"ordinal names get _shorter_ as the ordinal number gets _longer_. The name of " +"sat 0, the first sat to be mined is `nvtdijuwxlp` and the name of sat " +"2,099,999,997,689,999, the last sat to be mined, is `a`." +msgstr "" +"任务3は二つの部分があり、序数の名前に基づきます_" +"序数の名前は序数の数字を直したbase-26によって行われたコードです。短い名前を時間をかけはいけない創成プレート奨励の中に絞らないように" +"序数の長くなるほど、序数の名前は短くなります。例えば、初めに採掘された0番の聡の名前は`nvtdijuwxlpで`、" +"最後に採掘された2,099,999,997,689,999番の聡の名前は `a`です。" + +#: src/bounty/3.md:14 +msgid "" +"The bounty is open for submissions until block 840000—the first block after " +"the fourth halvening. Submissions included in block 840000 or later will not " +"be considered." +msgstr "" +"賞金計画はブロック高度の840000の-第四回の半分にしたの初めてのブロックまで開放します。ブロック高度の840000及びそれ以上のブロックは考えられません" + +#: src/bounty/3.md:18 +msgid "" +"Both parts use [frequency.tsv](frequency.tsv), a list of words and the " +"number of times they occur in the [Google Books Ngram dataset](http://" +"storage.googleapis.com/books/ngrams/books/datasetsv2.html). filtered to only " +"include the names of sats which will have been mined by the end of the " +"submission period, that appear at least 5000 times in the corpus." +msgstr "" +"二つの部分のミッションが使われ [frequency.tsv](frequency.tsv), 一つの単語のリスト及び" +" [Google Books Ngram dataset](http://storage.googleapis.com/books/ngrams/books/datasetsv2.html)" +"で現れた回数。濾過した後、提出時間が終わるときに発掘された聡の名前だけを含まれ、これらの名称は少なくともコーパスの中で5000回以上現れます。" + +#: src/bounty/3.md:24 +msgid "" +"`frequency.tsv` is a file of tab-separated values. The first column is the " +"word, and the second is the number of times it appears in the corpus. The " +"entries are sorted from least-frequently occurring to most-frequently " +"occurring." +msgstr "" +"`frequency.tsv` タブ区切り値のファイルは第一列が単語で,第二列がコーパスに現れる回数です" +"これらの条目は出現の最も低い頻度から最も高い頻度まで順番に並べています。" + +#: src/bounty/3.md:29 +msgid "" +"`frequency.tsv` was compiled using [this program](https://github.com/casey/" +"onegrams)." +msgstr "" +"`frequency.tsv` [このプログラム]を使って(https://github.com/casey/onegrams)纂訳されます。." + + +#: src/bounty/3.md:32 +msgid "" +"To search an `ord` wallet for sats with a name in `frequency.tsv`, use the " +"following [`ord`](https://github.com/ordinals/ord) command:" +msgstr "" +"`ord`ウォレットの中で`frequency.tsv`の中に含まれた聡の名前を検索して , したのを使います" +"[`ord`](https://github.com/ordinals/ord)コマンド: " + +#: src/bounty/3.md:35 +msgid "" +"```\n" +"ord wallet sats --tsv frequency.tsv\n" +"```" +msgstr "" + +#: src/bounty/3.md:39 +msgid "" +"This command requires the sat index, so `--index-sats` must be passed to ord " +"when first creating the index." +msgstr "" +"このコマンドは聡の索引必要なので,`--index-sats` は初めに索引を作り上げる時に使います。" + +#: src/bounty/3.md:42 +msgid "Part 0" +msgstr "第0部分" + +#: src/bounty/3.md:44 +msgid "_Rare sats pair best with rare words._" +msgstr "_珍しい聡と珍しい名前のベストマッチ" + +#: src/bounty/3.md:46 +msgid "" +"The transaction that submits the UTXO containing the sat whose name appears " +"with the lowest number of occurrences in `frequency.tsv` shall be the winner " +"of part 0." +msgstr "" +"提出したUTXOの中に聡の名前が含まれ、`frequency.tsv`の中に頻度が最も低い人は第0部分の優勝者であります。。" + +#: src/bounty/3.md:50 +msgid "Part 1" +msgstr "第1部分" + +#: src/bounty/3.md:52 +msgid "_Popularity is the font of value._" +msgstr "_人気は価値の源泉です_" + + +#: src/bounty/3.md:54 +msgid "" +"The transaction that submits the UTXO containing the sat whose name appears " +"with the highest number of occurrences in `frequency.tsv` shall be the " +"winner of part 1." +msgstr "" +"提出したUTXOの中に聡の名前が含まれ、`frequency.tsv`の中に頻度が最も高い人は第一部分の優勝者であります。" + +#: src/bounty/3.md:58 +msgid "Tie Breaking" +msgstr "勝負なしの場合" + +#: src/bounty/3.md:60 +msgid "" +"In the case of a tie, where two submissions occur with the same frequency, " +"the earlier submission shall be the winner." +msgstr "勝負なしの場合は二つの提出が同じ頻度でしたら、もっと早く提出した人は優勝者となります" + +#: src/bounty/3.md:66 +msgid "Part 0: 200,000 sats" +msgstr "" + +#: src/bounty/3.md:67 +msgid "Part 1: 200,000 sats" +msgstr "" + +#: src/bounty/3.md:68 +msgid "Total: 400,000 sats" +msgstr "" + +#: src/bounty/3.md:73 +msgid "" +"[`17m5rvMpi78zG8RUpCRd6NWWMJtWmu65kg`](https://mempool.space/" +"address/17m5rvMpi78zG8RUpCRd6NWWMJtWmu65kg)" +msgstr "" + +#: src/bounty/3.md:78 +msgid "Unclaimed!" +msgstr "依然有効です!" diff --git a/docs/po/zh.po b/docs/po/zh.po index 291142fb3f..a87fe3f30f 100644 --- a/docs/po/zh.po +++ b/docs/po/zh.po @@ -5,10 +5,10 @@ msgstr "" "PO-Revision-Date: 2023-08-30 10:03+0800\n" "Last-Translator: Dr.JingLee \n" "Language-Team: Chinese\n" +"Language: zh\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" "Content-Transfer-Encoding: 8bit\n" -"Language: zh\n" "Plural-Forms: nplurals=1; plural=0;\n" #: src/SUMMARY.md:2 src/introduction.md:1 @@ -23,75 +23,79 @@ msgstr "概述" msgid "Digital Artifacts" msgstr "数字文物" -#: src/SUMMARY.md:5 src/SUMMARY.md:12 src/overview.md:221 src/inscriptions.md:1 +#: src/SUMMARY.md:5 src/SUMMARY.md:13 src/overview.md:221 src/inscriptions.md:1 msgid "Inscriptions" msgstr "铭文" -#: src/SUMMARY.md:6 src/inscriptions/recursion.md:1 +#: src/SUMMARY.md:6 src/inscriptions/provenance.md:1 +msgid "Provenance" +msgstr "溯源" + +#: src/SUMMARY.md:7 src/inscriptions/recursion.md:1 msgid "Recursion" msgstr "递归" -#: src/SUMMARY.md:7 +#: src/SUMMARY.md:8 msgid "FAQ" msgstr "常见问题" -#: src/SUMMARY.md:8 +#: src/SUMMARY.md:9 msgid "Contributing" msgstr "贡献" -#: src/SUMMARY.md:9 src/donate.md:1 +#: src/SUMMARY.md:10 src/donate.md:1 msgid "Donate" msgstr "捐赠" -#: src/SUMMARY.md:10 +#: src/SUMMARY.md:11 msgid "Guides" msgstr "指引" -#: src/SUMMARY.md:11 +#: src/SUMMARY.md:12 msgid "Explorer" msgstr "浏览器" -#: src/SUMMARY.md:13 src/guides/sat-hunting.md:1 +#: src/SUMMARY.md:14 src/guides/sat-hunting.md:1 msgid "Sat Hunting" msgstr "猎聪" -#: src/SUMMARY.md:14 src/guides/collecting.md:1 +#: src/SUMMARY.md:15 src/guides/collecting.md:1 msgid "Collecting" msgstr "收藏" -#: src/SUMMARY.md:15 src/guides/sat-hunting.md:239 +#: src/SUMMARY.md:16 src/guides/sat-hunting.md:239 msgid "Sparrow Wallet" msgstr "麻雀钱包" -#: src/SUMMARY.md:16 src/guides/testing.md:1 +#: src/SUMMARY.md:17 src/guides/testing.md:1 msgid "Testing" msgstr "调试" -#: src/SUMMARY.md:17 src/guides/moderation.md:1 +#: src/SUMMARY.md:18 src/guides/moderation.md:1 msgid "Moderation" msgstr "调节" -#: src/SUMMARY.md:18 src/guides/reindexing.md:1 +#: src/SUMMARY.md:19 src/guides/reindexing.md:1 msgid "Reindexing" msgstr "重新索引" -#: src/SUMMARY.md:19 +#: src/SUMMARY.md:20 msgid "Bounties" msgstr "赏金" -#: src/SUMMARY.md:20 +#: src/SUMMARY.md:21 msgid "Bounty 0: 100,000 sats Claimed!" msgstr "任务 0: 100,000 sats 完成!" -#: src/SUMMARY.md:21 +#: src/SUMMARY.md:22 msgid "Bounty 1: 200,000 sats Claimed!" msgstr "任务 1: 200,000 sats 完成!" -#: src/SUMMARY.md:22 +#: src/SUMMARY.md:23 msgid "Bounty 2: 300,000 sats Claimed!" msgstr "任务 2: 300,000 sats 完成!" -#: src/SUMMARY.md:23 +#: src/SUMMARY.md:24 msgid "Bounty 3: 400,000 sats" msgstr "任务 3: 400,000 sats" @@ -100,7 +104,9 @@ msgid "" "This handbook is a guide to ordinal theory. Ordinal theory concerns itself " "with satoshis, giving them individual identities and allowing them to be " "tracked, transferred, and imbued with meaning." -msgstr "这本手册是序数理论(Ordinals Theory)的指南。 序数理论本身关注聪(Satoshi),赋予它们个体身份,并允许它们被追踪、转移并赋予意义。" +msgstr "" +"这本手册是序数理论(Ordinals Theory)的指南。 序数理论本身关注聪(Satoshi)," +"赋予它们个体身份,并允许它们被追踪、转移并赋予意义。" #: src/introduction.md:8 msgid "" @@ -108,16 +114,16 @@ msgid "" "network. One bitcoin can be sub-divided into 100,000,000 satoshis, but no " "further." msgstr "" -"聪(Satoshi),并非比特币,是比特币网络的原生货币和最小单位。" -"一个比特币可以被细分为100,000,000聪,但不能再细分了。" +"聪(Satoshi),并非比特币,是比特币网络的原生货币和最小单位。一个比特币可以被" +"细分为100,000,000聪,但不能再细分了。" #: src/introduction.md:11 msgid "" "Ordinal theory does not require a sidechain or token aside from Bitcoin, and " "can be used without any changes to the Bitcoin network. It works right now." msgstr "" -"序数理论不需要比特币区块链之外的侧链或代币,并且可以在不对比特币网络进行任何更改的情况下使用。" -"它即刻可以有效使用。" +"序数理论不需要比特币区块链之外的侧链或代币,并且可以在不对比特币网络进行任何" +"更改的情况下使用。它即刻可以有效使用。" #: src/introduction.md:14 msgid "" @@ -132,9 +138,9 @@ msgid "" "transferred using Bitcoin transactions. Inscriptions are as durable, " "immutable, secure, and decentralized as Bitcoin itself." msgstr "" -"单个聪可以刻有任意内容,创建独特的比特币原生的数字文物(Digital Artifact)"" -"可以保存在比特币钱包中并使用比特币交易进行传输。" -"铭文(Inscription)与比特币本身一样持久、永恒、安全和去中心化。" +"单个聪可以刻有任意内容,创建独特的比特币原生的数字文物(Digital Artifact)可" +"以保存在比特币钱包中并使用比特币交易进行传输。铭文(Inscription)与比特币本身" +"一样持久、永恒、安全和去中心化。" #: src/introduction.md:22 msgid "" @@ -143,9 +149,9 @@ msgid "" "DNS. For now though, such use-cases are speculative, and exist only in the " "minds of fringe ordinal theorists." msgstr "" -"其他非常规的应用也是可能的:链下染色硬币,具有密钥轮换的公钥基础设施" -"DNS 的去中心化替代品等等。 " -"不过就目前而言,这样的应用是推测性的,只存在于非主流的序数理论家的脑海中。" +"其他非常规的应用也是可能的:链下染色硬币,具有密钥轮换的公钥基础设施DNS 的去中" +"心化替代品等等。 不过就目前而言,这样的应用是推测性的,只存在于非主流的序数理" +"论家的脑海中。" #: src/introduction.md:27 msgid "For more details on ordinal theory, see the [overview](overview.md)." @@ -161,8 +167,8 @@ msgid "" "[inscriptions](guides/inscriptions.md), a curious species of digital " "artifact enabled by ordinal theory." msgstr "" -"当您准备好亲自动手时,一个好的起点是[铭文](guides/inscriptions.md)" -"这是一种由序数理论支持的独特的数字文物。" +"当您准备好亲自动手时,一个好的起点是[铭文](guides/inscriptions.md)这是一种由" +"序数理论支持的独特的数字文物。" #: src/introduction.md:35 msgid "Links" @@ -205,24 +211,23 @@ msgid "" "[Ordinal Theory Explained: Satoshi Serial Numbers and NFTs on Bitcoin]" "(https://www.youtube.com/watch?v=rSS0O2KQpsI)" msgstr "" -"[解释序数理论: 聪的序列号和比特币上的NFT]" -"(https://www.youtube.com/watch?v=rSS0O2KQpsI)" +"[解释序数理论: 聪的序列号和比特币上的NFT](https://www.youtube.com/watch?" +"v=rSS0O2KQpsI)" #: src/introduction.md:50 msgid "" "[Ordinals Workshop with Rodarmor](https://www.youtube.com/watch?" "v=MC_haVa6N3I)" msgstr "" -"[CaseyRodarmor的序数理论工作坊 ](https://www.youtube.com/watch?" -"v=MC_haVa6N3I)" +"[CaseyRodarmor的序数理论工作坊 ](https://www.youtube.com/watch?v=MC_haVa6N3I)" #: src/introduction.md:51 msgid "" "[Ordinal Art: Mint Your own NFTs on Bitcoin w/ @rodarmor](https://www." "youtube.com/watch?v=j5V33kV3iqo)" msgstr "" -"[序数艺术:在比特币上铸造你自己的NFT w/ @rodarmor](https://www." -"youtube.com/watch?v=j5V33kV3iqo)" +"[序数艺术:在比特币上铸造你自己的NFT w/ @rodarmor](https://www.youtube.com/" +"watch?v=j5V33kV3iqo)" #: src/overview.md:1 msgid "Ordinal Theory Overview" @@ -239,29 +244,31 @@ msgid "" "and the transfer scheme on the _order_ of transaction inputs and outputs. " "Thus the name, _ordinals_." msgstr "" -"序数是一种比特币的编号方案,允许跟踪和转移单个聪。这些数字被称作[序号](https://ordinals.com)。" -"比特币是按照它们被挖掘的顺序编号的,并从交易输入转移到交易输出(遵循先进先出原则)。" -"编号方案和传输方案都依赖于_顺序_,编号方案依赖于比特币被挖掘的_顺序_,而传输方案依赖于交易输入和输出的_顺序_。" -"因此得名,_序数(Ordinals_。" +"序数是一种比特币的编号方案,允许跟踪和转移单个聪。这些数字被称作[序号]" +"(https://ordinals.com)。比特币是按照它们被挖掘的顺序编号的,并从交易输入转移" +"到交易输出(遵循先进先出原则)。编号方案和传输方案都依赖于_顺序_,编号方案依" +"赖于比特币被挖掘的_顺序_,而传输方案依赖于交易输入和输出的_顺序_。因此得名,_" +"序数(Ordinals_。" #: src/overview.md:13 msgid "" "Technical details are available in [the BIP](https://github.com/ordinals/ord/" "blob/master/bip.mediawiki)." msgstr "" -"技术细节可以在[the BIP](https://github.com/ordinals/ord/blob/master/bip.mediawiki)获取." +"技术细节可以在[the BIP](https://github.com/ordinals/ord/blob/master/bip." +"mediawiki)获取." #: src/overview.md:16 msgid "" "Ordinal theory does not require a separate token, another blockchain, or any " "changes to Bitcoin. It works right now." msgstr "" -"序数理论不需要一个单独的代币,单独区块链,或者对比特币进行任何更改。它即刻可以有效运转。" +"序数理论不需要一个单独的代币,单独区块链,或者对比特币进行任何更改。它即刻可" +"以有效运转。" #: src/overview.md:19 msgid "Ordinal numbers have a few different representations:" -msgstr "" -"序号有几种不同的表示方式:" +msgstr "序号有几种不同的表示方式:" #: src/overview.md:21 msgid "" @@ -269,9 +276,8 @@ msgid "" "sat/2099994106992659) The ordinal number, assigned according to the order in " "which the satoshi was mined." msgstr "" -"_整数符号_:[`2099994106992659`](https://ordinals.com/sat/2099994106992659) " -"这个序号是根据挖掘聪的顺序分配。" - +"_整数符号_:[`2099994106992659`](https://ordinals.com/sat/2099994106992659) 这" +"个序号是根据挖掘聪的顺序分配。" #: src/overview.md:26 msgid "" @@ -279,8 +285,8 @@ msgid "" "sat/3891094.16797) The first number is the block height in which the satoshi " "was mined, the second the offset of the satoshi within the block." msgstr "" -"_十进制符号_: [`3891094.16797`](https://ordinals.com/" -"sat/3891094.16797) 第一个数字是挖掘聪的区块高度,第二个数字是区块内聪的偏移量。 " +"_十进制符号_: [`3891094.16797`](https://ordinals.com/sat/3891094.16797) 第一" +"个数字是挖掘聪的区块高度,第二个数字是区块内聪的偏移量。 " #: src/overview.md:31 msgid "" @@ -305,16 +311,16 @@ msgid "" "_Name_: [`satoshi`](https://ordinals.com/sat/satoshi). An encoding of the " "ordinal number using the characters `a` through `z`." msgstr "" -"_名字_: [`satoshi`](https://ordinals.com/sat/satoshi). " -"一种使用字母`a` 到 `z`对序号进行编码的方法" +"_名字_: [`satoshi`](https://ordinals.com/sat/satoshi). 一种使用字母`a` 到 `z`" +"对序号进行编码的方法" #: src/overview.md:42 msgid "" "Arbitrary assets, such as NFTs, security tokens, accounts, or stablecoins " "can be attached to satoshis using ordinal numbers as stable identifiers." msgstr "" -"任意资产,如NFT、安全令牌、帐户或稳定币, " -"都可以使用序数作为稳定标识符附加到聪上。" +"任意资产,如NFT、安全令牌、帐户或稳定币, 都可以使用序数作为稳定标识符附加到" +"聪上。" #: src/overview.md:45 msgid "" @@ -325,11 +331,10 @@ msgid "" "block explorer for interactive exploration of the blockchain, functionality " "for inscribing satoshis with digital artifacts, and this manual." msgstr "" -"Ordinals是一个开源项目,部署在[on GitHub](https://github.com/" -"ordinals/ord). 该项目包括一个描述序数方案的BIP、 " -"一个与比特币核心节点通信以跟踪所有聪位置的索引" -"一个允许进行序号感知交易的钱包、 " -"一个用于区块链交互探索的区块资源管理器、用数字文物嵌入聪的功能,以及本手册。 " +"Ordinals是一个开源项目,部署在[on GitHub](https://github.com/ordinals/ord). " +"该项目包括一个描述序数方案的BIP、 一个与比特币核心节点通信以跟踪所有聪位置的" +"索引一个允许进行序号感知交易的钱包、 一个用于区块链交互探索的区块资源管理器、" +"用数字文物嵌入聪的功能,以及本手册。 " #: src/overview.md:52 msgid "Rarity" @@ -342,24 +347,22 @@ msgid "" "can decide for themselves which sats are rare and desirable, but there are " "some hints…" msgstr "" -"人类是收藏者。由于聪现在可以被追踪和转移,人们自然会想要收藏它们。 " -"序数理论家可以自己决定哪些聪是稀有和合意的, " -"这里有一些提示…" +"人类是收藏者。由于聪现在可以被追踪和转移,人们自然会想要收藏它们。 序数理论家" +"可以自己决定哪些聪是稀有和合意的, 这里有一些提示…" #: src/overview.md:59 msgid "" "Bitcoin has periodic events, some frequent, some more uncommon, and these " "naturally lend themselves to a system of rarity. These periodic events are:" msgstr "" -"比特币有周期性的事件,有些频繁,有些不常见,这些事件自然而然地形成了一个稀有度系统。" -"这些周期性事件是:" +"比特币有周期性的事件,有些频繁,有些不常见,这些事件自然而然地形成了一个稀有" +"度系统。这些周期性事件是:" #: src/overview.md:62 msgid "" "_Blocks_: A new block is mined approximately every 10 minutes, from now " "until the end of time." -msgstr "" -"_区块_: 从现在到时间结束,大约每10分钟挖掘一个新区块。" +msgstr "_区块_: 从现在到时间结束,大约每10分钟挖掘一个新区块。" #: src/overview.md:65 msgid "" @@ -367,8 +370,8 @@ msgid "" "weeks, the Bitcoin network responds to changes in hashrate by adjusting the " "difficulty target which blocks must meet in order to be accepted." msgstr "" -"_难度调整_: 每2016个区块,或大约每两周, " -"比特币网络通过调整区块必须满足的难度目标来响应哈希率的变化。 " +"_难度调整_: 每2016个区块,或大约每两周, 比特币网络通过调整区块必须满足的难度" +"目标来响应哈希率的变化。 " #: src/overview.md:69 msgid "" @@ -384,14 +387,13 @@ msgid "" "period between conjunctions a cycle. A conjunction occurs roughly every 24 " "years. The first conjunction should happen sometime in 2032." msgstr "" -"_周期_: 每六次减半就会发生一些神奇的事情:减半和难度调整会同时发生," -"这就是所谓的相合,相合之间的时间周期是一个周期。 " -"大约每24年就会发生一次相合,第一次相合应该会发生在2032年的某个时候。 " +"_周期_: 每六次减半就会发生一些神奇的事情:减半和难度调整会同时发生,这就是所" +"谓的相合,相合之间的时间周期是一个周期。 大约每24年就会发生一次相合,第一次相" +"合应该会发生在2032年的某个时候。 " #: src/overview.md:77 msgid "This gives us the following rarity levels:" -msgstr "" -"这给了我们以下稀缺度等级:" +msgstr "这给了我们以下稀缺度等级:" #: src/overview.md:79 msgid "`common`: Any sat that is not the first sat of its block" @@ -423,7 +425,8 @@ msgid "" "ordinal number in a way that makes the rarity of a satoshi easy to see at a " "glance:" msgstr "" -"这给我们带来了度数表示法,它以一种使聪的稀有性一目了然的方式明确地表示一个序数: " +"这给我们带来了度数表示法,它以一种使聪的稀有性一目了然的方式明确地表示一个序" +"数: " #: src/overview.md:89 msgid "" @@ -448,8 +451,8 @@ msgid "" "Ordinal theorists often use the terms \"hour\", \"minute\", \"second\", and " "\"third\" for _A_, _B_, _C_, and _D_, respectively." msgstr "" -"序数理论家通常使用 \"小时\", \"分钟\", \"秒\", 以及 " -"\"第三\" 等专用词汇来对应的表示 _A_, _B_, _C_, 和 _D_。" +"序数理论家通常使用 \"小时\", \"分钟\", \"秒\", 以及 \"第三\" 等专用词汇来对应" +"的表示 _A_, _B_, _C_, 和 _D_。" #: src/overview.md:100 msgid "Now for some examples. This satoshi is common:" @@ -587,8 +590,7 @@ msgstr "" msgid "" "If the block offset is zero, it may be omitted. This is the uncommon satoshi " "from above:" -msgstr "" -"如果区块偏移量为零,则可以省略。这是对比以上的非普通的聪:" +msgstr "如果区块偏移量为零,则可以省略。这是对比以上的非普通的聪:" #: src/overview.md:164 msgid "" @@ -668,8 +670,8 @@ msgid "" "745,855 uncommon satoshis have been mined - one per 25.6 bitcoin in " "circulation." msgstr "" -"目前即使是非普通的聪也非常罕见。 截至撰写本文时, " -"已开采出 745,855 个非普通的聪-大约在每 25.6个流通比特币中会有一个。 " +"目前即使是非普通的聪也非常罕见。 截至撰写本文时, 已开采出 745,855 个非普通的" +"聪-大约在每 25.6个流通比特币中会有一个。 " #: src/overview.md:196 msgid "Names" @@ -682,9 +684,9 @@ msgid "" "short and get longer, but then all the good, short names would be trapped in " "the unspendable genesis block." msgstr "" -"每个聪都有一个名字,由字母 _A_ 到 _Z_构成 " -"随着聪被开采的时间越长,名字越短。 如果他们从短开始," -"然后变得更长,那么所有好的、短的名字都会被困在无法使用的创世块中。 " +"每个聪都有一个名字,由字母 _A_ 到 _Z_构成 随着聪被开采的时间越长,名字越短。 " +"如果他们从短开始,然后变得更长,那么所有好的、短的名字都会被困在无法使用的创" +"世块中。 " #: src/overview.md:204 msgid "" @@ -692,9 +694,8 @@ msgid "" "last satoshi to be mined is \"a\". Every combination of 10 characters or " "less is out there, or will be out there, someday." msgstr "" -msgid "" -"举个例子, 1905530482684727°'的名字是 \"iaiufjszmoba\".最后一个被挖掘的聪的名字会是\"a\"。10个字母" -"或更少字符的组合都会存在,或者总有一天会存在。" +"举个例子, 1905530482684727°'的名字是 \"iaiufjszmoba\".最后一个被挖掘的聪的名" +"字会是\"a\"。10个字母或更少字符的组合都会存在,或者总有一天会存在。" #: src/overview.md:208 msgid "Exotics" @@ -708,10 +709,10 @@ msgid "" "event, such as satoshis from block 477,120, the block in which SegWit " "activated, or 2099999997689999°, the last satoshi that will ever be mined." msgstr "" -"除了它们的名字或稀有性之外,聪可能还因为其他原因而受到重视。" -"这可能是由于数字本身的性质,比如具有整数的平方根或立方根。" -"或者它与某件历史事件有关,例如来自区块477,120的聪(SegWit激活的区块)" -"是 2099999997689999°,这是最后一个被挖出来的聪。" +"除了它们的名字或稀有性之外,聪可能还因为其他原因而受到重视。这可能是由于数字" +"本身的性质,比如具有整数的平方根或立方根。或者它与某件历史事件有关,例如来自" +"区块477,120的聪(SegWit激活的区块)是 2099999997689999°,这是最后一个被挖出来" +"的聪。" #: src/overview.md:217 msgid "" @@ -719,8 +720,8 @@ msgid "" "makes them so is subjective. Ordinal theorists are encouraged to seek out " "exotics based on criteria of their own devising." msgstr "" -"这种比特币被称为“奇特的”。哪些聪是“奇特的”?是什么让他们如此被重视?" -"序数理论家被鼓励根据他们自己设计的标准来寻找“奇特的”聪。" +"这种比特币被称为“奇特的”。哪些聪是“奇特的”?是什么让他们如此被重视?序数理论" +"家被鼓励根据他们自己设计的标准来寻找“奇特的”聪。" #: src/overview.md:224 msgid "" @@ -731,13 +732,10 @@ msgid "" "digital artifact that can be tracked, transferred, hoarded, bought, sold, " "lost, and rediscovered." msgstr "" -"" -"聪可以刻有任意内容,从而创建比特币原生的数字文物(数字艺术)。" -"铭刻是通过将要铭刻的内容发送到交易中来完成的,该交易会在链上显示铭文内容。" -"由于铭文内容与聪有着密不可分的联系,从将创造了一个不可改变的数字人工制品。" -"这个数字文物可以被追踪、转移、储存、购买、出售、丢失和重新发现。" - - +"聪可以刻有任意内容,从而创建比特币原生的数字文物(数字艺术)。铭刻是通过将要" +"铭刻的内容发送到交易中来完成的,该交易会在链上显示铭文内容。由于铭文内容与聪" +"有着密不可分的联系,从将创造了一个不可改变的数字人工制品。这个数字文物可以被" +"追踪、转移、储存、购买、出售、丢失和重新发现。" #: src/overview.md:231 msgid "Archaeology" @@ -750,8 +748,9 @@ msgid "" "Chainleft.](https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-" "N29oF4iwCgX3lacrvaG9Kjko)" msgstr "" -"致力于编目和收集早期 NFT 的活跃考古学家社区如雨后春笋般涌现 " -"[Chainleft对历史NFT的精彩总结](https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-N29oF4iwCgX3lacrvaG9Kjko)" +"致力于编目和收集早期 NFT 的活跃考古学家社区如雨后春笋般涌现 [Chainleft对历史" +"NFT的精彩总结](https://mirror.xyz/chainleft.eth/MzPWRsesC9mQflxlLo-" +"N29oF4iwCgX3lacrvaG9Kjko)" #: src/overview.md:238 msgid "" @@ -759,8 +758,8 @@ msgid "" "first ERC-721 contract, [SU SQUARES](https://tenthousandsu.com/), was " "deployed on Ethereum." msgstr "" -"普遍接受的古老NFT 的截止日期是 2018年3月19日,即 " -"第一个 ERC-721 合约,[SU SQUARES](https://tenthousandsu.com/), 被部署在以太坊上的时间 " +"普遍接受的古老NFT 的截止日期是 2018年3月19日,即 第一个 ERC-721 合约,[SU " +"SQUARES](https://tenthousandsu.com/), 被部署在以太坊上的时间 " #: src/overview.md:242 msgid "" @@ -769,8 +768,8 @@ msgid "" "Ordinals specification was finalized. In this sense, they are not of " "historical interest." msgstr "" -"NFT 考古学家是否对序数感兴趣是一个悬而未决的问题! " -"从某种意义上说,序数是在 2022 年初创建的,当时序数规范已定稿" +"NFT 考古学家是否对序数感兴趣是一个悬而未决的问题! 从某种意义上说,序数是在 " +"2022 年初创建的,当时序数规范已定稿" #: src/overview.md:247 msgid "" @@ -778,9 +777,9 @@ msgid "" "in 2009 when he mined the Bitcoin genesis block. In this sense, ordinals, " "and especially early ordinals, are certainly of historical interest." msgstr "" -"从这个意义上说,它们不具有历史意义。" -"但从另一种意义上说,序数实际上是由中本聪在 2009 年开采比特币创世块时创造的。" -"从这个意义上说,序数,尤其是早期的序数,当然具有历史意义。" +"从这个意义上说,它们不具有历史意义。但从另一种意义上说,序数实际上是由中本聪" +"在 2009 年开采比特币创世块时创造的。从这个意义上说,序数,尤其是早期的序数," +"当然具有历史意义。" #: src/overview.md:251 msgid "" @@ -788,8 +787,8 @@ msgid "" "ordinals were independently discovered on at least two separate occasions, " "long before the era of modern NFTs began." msgstr "" -"许多序数理论家赞成后一种观点。这不仅仅是因为序数是在至少两个不同的场合独立发现的," -"远早于现代 NFT 时代开始。" +"许多序数理论家赞成后一种观点。这不仅仅是因为序数是在至少两个不同的场合独立发" +"现的,远早于现代 NFT 时代开始。" #: src/overview.md:255 msgid "" @@ -798,9 +797,10 @@ msgid "" "topic=102355.0). This wasn't an asset scheme, but did use the ordinal " "algorithm, and was implemented but never deployed." msgstr "" -"2012 年 8 月 21 日,Charlie Lee 在 Charlie Lee [在Bitcoin Talk论坛上发布一项将" -"比特币权益证明Proof-of-stake添加的提案](https://bitcointalk.org/index.php?" -"topic=102355.0). 这不是资产方案,但确实使用了序数算法,并且已实施但从未部署过。" +"2012 年 8 月 21 日,Charlie Lee 在 Charlie Lee [在Bitcoin Talk论坛上发布一项" +"将比特币权益证明Proof-of-stake添加的提案](https://bitcointalk.org/index.php?" +"topic=102355.0). 这不是资产方案,但确实使用了序数算法,并且已实施但从未部署" +"过。" #: src/overview.md:261 msgid "" @@ -809,9 +809,9 @@ msgid "" "has all the important properties of ordinals. The scheme was discussed but " "never implemented." msgstr "" -"2012 年 10 月 8 日,jl2012 在[同一论坛上发布了一个方案](https://" -"bitcointalk.org/index.php?topic=117224.0) 该方案使用十进制表示法并" -"具有序数的所有重要属性。 该计划进行了讨论,但从未实施。" +"2012 年 10 月 8 日,jl2012 在[同一论坛上发布了一个方案](https://bitcointalk." +"org/index.php?topic=117224.0) 该方案使用十进制表示法并具有序数的所有重要属" +"性。 该计划进行了讨论,但从未实施。" #: src/overview.md:266 msgid "" @@ -822,10 +822,9 @@ msgid "" "sequence of events set in motion with the mining of the first block, so many " "years ago." msgstr "" -"这些序数的独立发明在某种程度上表明序数是被发现的, " -"或者是重新发现的,而不是发明的。 序数是比特币数学的必然性, " -"不是源于它们的现代文档,而是源于它们古老的起源。 " -"它们是许多年前随着第一个区块的开采而启动的一系列事件的高潮。" +"这些序数的独立发明在某种程度上表明序数是被发现的, 或者是重新发现的,而不是发" +"明的。 序数是比特币数学的必然性, 不是源于它们的现代文档,而是源于它们古老的" +"起源。 它们是许多年前随着第一个区块的开采而启动的一系列事件的高潮。" #: src/digital-artifacts.md:4 msgid "" @@ -833,15 +832,13 @@ msgid "" "the dark, secret clutch of a Viking hoard, now dug from the earth by your " "grasping hands. It…" msgstr "" -"想象有一个实体的人工制品。 比方说,一枚稀有的硬币," -"在维京人的宝库的黑暗中秘密保存了无数年," -"现在被你亲手从地下挖了出来。 它…" +"想象有一个实体的人工制品。 比方说,一枚稀有的硬币,在维京人的宝库的黑暗中秘密" +"保存了无数年,现在被你亲手从地下挖了出来。 它…" #: src/digital-artifacts.md:8 msgid "" "…has an owner. You. As long as you keep it safe, nobody can take it from you." -msgstr "" -"…有了一个主人. 那就是您. 只要您妥善保管,就没有人能从您手中夺走它。" +msgstr "…有了一个主人. 那就是您. 只要您妥善保管,就没有人能从您手中夺走它。" #: src/digital-artifacts.md:10 msgid "…is complete. It has no missing parts." @@ -852,23 +849,22 @@ msgid "" "…can only be changed by you. If you were a trader, and you made your way to " "18th century China, none but you could stamp it with your chop-mark." msgstr "" -"…只能由您来改变。如果您是一名商人,并且您来到了 " -"18世纪的中国,那么除您之外,无人可以在上面盖章。" +"…只能由您来改变。如果您是一名商人,并且您来到了 18世纪的中国,那么除您之外," +"无人可以在上面盖章。" #: src/digital-artifacts.md:15 msgid "" "…can only be disposed of by you. The sale, trade, or gift is yours to make, " "to whomever you wish." -msgstr "" -"……只能由您处置。 销售、交易或赠送都是您的决定,您想给谁就给谁。 " +msgstr "……只能由您处置。 销售、交易或赠送都是您的决定,您想给谁就给谁。 " #: src/digital-artifacts.md:18 msgid "" "What are digital artifacts? Simply put, they are the digital equivalent of " "physical artifacts." msgstr "" -"什么是数字文物(数字工件、数字人工制品)? " -"简而言之,它们是物理人工制品的数字等价物。" +"什么是数字文物(数字工件、数字人工制品)? 简而言之,它们是物理人工制品的数字" +"等价物。" #: src/digital-artifacts.md:21 msgid "" @@ -880,22 +876,23 @@ msgstr "要使数字化事物成为数字人工制品,它必须像您的那枚 msgid "" "Digital artifacts can have owners. A number is not a digital artifact, " "because nobody can own it." -msgstr "" -"数字文物可以有所有者,因此数字不同于数字文物,因为没有人可以拥有数字。" +msgstr "数字文物可以有所有者,因此数字不同于数字文物,因为没有人可以拥有数字。" #: src/digital-artifacts.md:27 msgid "" "Digital artifacts are complete. An NFT that points to off-chain content on " "IPFS or Arweave is incomplete, and thus not a digital artifact." msgstr "" -"数字文物是完整的,指向 IPFS 或 Arweave 上链下内容的 NFT 是不完整的,因此不是数字文物。" +"数字文物是完整的,指向 IPFS 或 Arweave 上链下内容的 NFT 是不完整的,因此不是" +"数字文物。" #: src/digital-artifacts.md:30 msgid "" "Digital artifacts are permissionless. An NFT which cannot be sold without " "paying a royalty is not permissionless, and thus not a digital artifact." msgstr "" -"数字文物是无需许可的,不支付版税就不能出售的 NFT 不是无需许可的,因此不是数字文物。" +"数字文物是无需许可的,不支付版税就不能出售的 NFT 不是无需许可的,因此不是数字" +"文物。" #: src/digital-artifacts.md:33 msgid "" @@ -903,15 +900,14 @@ msgid "" "on a centralized ledger today, but maybe not tomorrow, and thus one cannot " "be a digital artifact." msgstr "" -"数字文物是不可审查的, 也许你今天可以更改集中式分类账上的数据库条目,但明天可能不行 " -"因此一个不是数字文物" +"数字文物是不可审查的, 也许你今天可以更改集中式分类账上的数据库条目,但明天可" +"能不行 因此一个不是数字文物" #: src/digital-artifacts.md:37 msgid "" "Digital artifacts are immutable. An NFT with an upgrade key is not a digital " "artifact." -msgstr "" -"数字文物是不可篡改的,带有升级密钥的NFT不是数字文物。" +msgstr "数字文物是不可篡改的,带有升级密钥的NFT不是数字文物。" #: src/digital-artifacts.md:40 msgid "" @@ -919,8 +915,8 @@ msgid "" "_should_ be, sometimes are, and what inscriptions _always_ are, by their " "very nature." msgstr "" -"数字文物的定义旨在从其特定的本质上反映NFT " -"_应该_ 是什么, 有时是什么, 以及铭文_始终_ 是什么 " +"数字文物的定义旨在从其特定的本质上反映NFT _应该_ 是什么, 有时是什么, 以及铭文" +"_始终_ 是什么 " #: src/inscriptions.md:4 msgid "" @@ -928,8 +924,8 @@ msgid "" "digital artifacts, more commonly known as NFTs. Inscriptions do not require " "a sidechain or separate token." msgstr "" -"铭文里可刻有任意内容,从而创造了比特币原生的数字人工制品,通常被称为 NFT。" -"铭文不需要侧链或单独的代币。 " +"铭文里可刻有任意内容,从而创造了比特币原生的数字人工制品,通常被称为 NFT。铭" +"文不需要侧链或单独的代币。 " #: src/inscriptions.md:8 msgid "" @@ -941,8 +937,8 @@ msgid "" "according to ordinal theory." msgstr "" "这些铭刻的聪,可以使用比特币交易传输发送到比特币地址,保存在比特币 UTXO 中。" -"这些交易、地址 和 UTXO 在所有方面都是正常的比特币交易、地址和 UTXO。" -"除了为了发送单个聪,交易必须根据序数理论控制输入和输出的顺序和值。 " +"这些交易、地址 和 UTXO 在所有方面都是正常的比特币交易、地址和 UTXO。除了为了" +"发送单个聪,交易必须根据序数理论控制输入和输出的顺序和值。 " #: src/inscriptions.md:15 msgid "" @@ -952,8 +948,9 @@ msgid "" "server, and for creating HTML inscriptions that use and remix the content of " "other inscriptions." msgstr "" -"铭文内容是基于万维网标准的。铭文由内容类型(也称为 MIME 类型)和内容本身(字节串)组成。" -"这允许从 Web 服务器返回铭文内容,并用于创建和使用HTML铭文并重新混合其他铭文内容。" +"铭文内容是基于万维网标准的。铭文由内容类型(也称为 MIME 类型)和内容本身(字" +"节串)组成。这允许从 Web 服务器返回铭文内容,并用于创建和使用HTML铭文并重新混" +"合其他铭文内容。" #: src/inscriptions.md:21 msgid "" @@ -962,8 +959,8 @@ msgid "" "and additionally receive the witness discount, making inscription content " "storage relatively economical." msgstr "" -"铭文内容完全在链上,存储在taproot script-path spend脚本中。 " -"Taproot 脚本对其内容的限制很少,并且额外获得见证折扣,使得铭文内容存储相对经济。" +"铭文内容完全在链上,存储在taproot script-path spend脚本中。 Taproot 脚本对其" +"内容的限制很少,并且额外获得见证折扣,使得铭文内容存储相对经济。" #: src/inscriptions.md:26 msgid "" @@ -974,10 +971,10 @@ msgid "" "output created by the commit transaction is spent, revealing the inscription " "content on-chain." msgstr "" -"因为taproot script-path spend脚本只能从现有的 taproot 输出中产生," -"因此使用两阶段commit/reveal过程进行铭刻。首先,在commit中," -"创建一个提交到包含铭文内容的脚本的taproot 输出。 其次,在reveal交易中," -"使用commit交易产生的输出,来显示链上的铭文内容。" +"因为taproot script-path spend脚本只能从现有的 taproot 输出中产生,因此使用两" +"阶段commit/reveal过程进行铭刻。首先,在commit中,创建一个提交到包含铭文内容的" +"脚本的taproot 输出。 其次,在reveal交易中,使用commit交易产生的输出,来显示链" +"上的铭文内容。" #: src/inscriptions.md:33 msgid "" @@ -987,18 +984,15 @@ msgid "" "effectively no-ops, they do not change the semantics of the script in which " "they are included, and can be combined with any other locking script." msgstr "" -"铭文内容使用未执行条件中的数据推送进行序列化,称为“信封”。" -"信封由 OP_FALSE OP_IF … OP_ENDIF 组成,包装任意数量的数据推送。" -"因为信封实际上是空操作,所以它们不会改变包含它们的脚本的语义," -"并且可以与任何其他锁定脚本结合使用。" +"铭文内容使用未执行条件中的数据推送进行序列化,称为“信封”。信封由 OP_FALSE " +"OP_IF … OP_ENDIF 组成,包装任意数量的数据推送。因为信封实际上是空操作,所以它" +"们不会改变包含它们的脚本的语义,并且可以与任何其他锁定脚本结合使用。" #: src/inscriptions.md:39 msgid "" "A text inscription containing the string \"Hello, world!\" is serialized as " "follows:" -msgstr "" -"包含字符串“Hello, world!”的文本铭文 序列化如下:" - +msgstr "包含字符串“Hello, world!”的文本铭文 序列化如下:" #: src/inscriptions.md:42 msgid "" @@ -1018,197 +1012,357 @@ msgstr "" msgid "" "First the string `ord` is pushed, to disambiguate inscriptions from other " "uses of envelopes." -msgstr "" -"首先字符串`ord`被推送,以消除铭文与信封其他用途的歧义。" +msgstr "首先字符串`ord`被推送,以消除铭文与信封其他用途的歧义。" #: src/inscriptions.md:56 +#, fuzzy msgid "" "`OP_PUSH 1` indicates that the next push contains the content type, and " -"`OP_PUSH 0` indicates that subsequent data pushes contain the content " -"itself. Multiple data pushes must be used for large inscriptions, as one of " +"`OP_PUSH 0`indicates that subsequent data pushes contain the content itself. " +"Multiple data pushes must be used for large inscriptions, as one of " "taproot's few restrictions is that individual data pushes may not be larger " "than 520 bytes." msgstr "" -"`OP_PUSH 1` 表示下一次推送包含内容类型, " -"`OP_PUSH 0` 表示后续数据推送包含内容本身," -"大型铭文必须使用多次数据推送,因为 taproot 的少数限制之一是单个数据推送不得大于 520 字节。" - +"`OP_PUSH 1` 表示下一次推送包含内容类型, `OP_PUSH 0` 表示后续数据推送包含内容" +"本身,大型铭文必须使用多次数据推送,因为 taproot 的少数限制之一是单个数据推送" +"不得大于 520 字节。" -#: src/inscriptions.md:61 +#: src/inscriptions.md:62 msgid "" "The inscription content is contained within the input of a reveal " "transaction, and the inscription is made on the first sat of its input. This " "sat can then be tracked using the familiar rules of ordinal theory, allowing " "it to be transferred, bought, sold, lost to fees, and recovered." msgstr "" -"铭文内容包含在reveal交易的输入中,并且铭文是铭刻在其第一个输出的第一个聪(Satoshi)上。" -"我们可以使用熟悉的序数理论规则来跟踪这个聪 sat,允许它被转移、购买、出售、丢失和恢复。" +"铭文内容包含在reveal交易的输入中,并且铭文是铭刻在其第一个输出的第一个聪" +"(Satoshi)上。我们可以使用熟悉的序数理论规则来跟踪这个聪 sat,允许它被转移、" +"购买、出售、丢失和恢复。" -#: src/inscriptions.md:66 +#: src/inscriptions.md:67 msgid "Content" msgstr "内容" -#: src/inscriptions.md:69 +#: src/inscriptions.md:70 msgid "" "The data model of inscriptions is that of a HTTP response, allowing " "inscription content to be served by a web server and viewed in a web browser." msgstr "" -"铭文的数据模型是 HTTP 响应的数据模型,允许铭文由网络服务器提供服务并在网络浏览器中查看的内容。" +"铭文的数据模型是 HTTP 响应的数据模型,允许铭文由网络服务器提供服务并在网络浏" +"览器中查看的内容。" -#: src/inscriptions.md:72 +#: src/inscriptions.md:73 msgid "Fields" msgstr "字段" -#: src/inscriptions.md:75 +#: src/inscriptions.md:76 msgid "" "Inscriptions may include fields before an optional body. Each field consists " "of two data pushes, a tag and a value." msgstr "" -"铭文可以在可选主体之前包含字段。每个字段都包含" -"两个数据推送,一个标签和一个值。" +"铭文可以在可选主体之前包含字段。每个字段都包含两个数据推送,一个标签和一个" +"值。" -#: src/inscriptions.md:78 +#: src/inscriptions.md:79 msgid "" "Currently, the only defined field is `content-type`, with a tag of `1`, " "whose value is the MIME type of the body." msgstr "" -"目前,唯一定义的字段是‘content-type’,标签为‘1’," -"其值是正文的 MIME 类型。" +"目前,唯一定义的字段是‘content-type’,标签为‘1’,其值是正文的 MIME 类型。" -#: src/inscriptions.md:81 +#: src/inscriptions.md:82 msgid "" "The beginning of the body and end of fields is indicated with an empty data " "push." -msgstr "" -"正文的开头和字段的结尾用'空数据'指示推送。" +msgstr "正文的开头和字段的结尾用'空数据'指示推送。" -#: src/inscriptions.md:84 +#: src/inscriptions.md:85 msgid "" "Unrecognized tags are interpreted differently depending on whether they are " "even or odd, following the \"it's okay to be odd\" rule used by the " "Lightning Network." msgstr "" -"无法识别的标签的解释不同,取决于它们是否是偶数或奇数,遵循闪电网络\"可以是奇数\"的规则。" +"无法识别的标签的解释不同,取决于它们是否是偶数或奇数,遵循闪电网络\"可以是奇" +"数\"的规则。" -#: src/inscriptions.md:88 +#: src/inscriptions.md:89 msgid "" "Even tags are used for fields which may affect creation, initial assignment, " "or transfer of an inscription. Thus, inscriptions with unrecognized even " "fields must be displayed as \"unbound\", that is, without a location." msgstr "" -"甚至标签也用于可能影响创建、初始分配的字段,或铭文的转移。因此,即使无法识别的铭文," -"字段也必须显示为\"未绑定\",即没有位置。" - +"甚至标签也用于可能影响创建、初始分配的字段,或铭文的转移。因此,即使无法识别" +"的铭文,字段也必须显示为\"未绑定\",即没有位置。" -#: src/inscriptions.md:92 +#: src/inscriptions.md:93 msgid "" "Odd tags are used for fields which do not affect creation, initial " "assignment, or transfer, such as additional metadata, and thus are safe to " "ignore." msgstr "" -"奇数标签用于不影响创建、初始的字段,分配或转移,例如附加元数据,因此是选择忽略是安全的。" +"奇数标签用于不影响创建、初始的字段,分配或转移,例如附加元数据,因此是选择忽略" +"是安全的。" -#: src/inscriptions.md:95 +#: src/inscriptions.md:96 msgid "Inscription IDs" msgstr "铭文身份ID" -#: src/inscriptions.md:98 +#: src/inscriptions.md:99 msgid "" "The inscriptions are contained within the inputs of a reveal transaction. In " "order to uniquely identify them they are assigned an ID of the form:" msgstr "" -"铭文包含在揭示交易的输入中。为了唯一地识别他们,他们被分配了一个以下形式的 ID:" +"铭文包含在揭示交易的输入中。为了唯一地识别他们,他们被分配了一个以下形式的 " +"ID:" -#: src/inscriptions.md:101 +#: src/inscriptions.md:102 msgid "`521f8eccffa4c41a3a7728dd012ea5a4a02feed81f41159231251ecf1e5c79dai0`" msgstr "" -#: src/inscriptions.md:103 +#: src/inscriptions.md:104 msgid "" "The part in front of the `i` is the transaction ID (`txid`) of the reveal " "transaction. The number after the `i` defines the index (starting at 0) of " "new inscriptions being inscribed in the reveal transaction." msgstr "" -" `i` 的前面部分是交易ID (`txid`),在`i`之后的数字定义了" -"新的铭文在交易总被铭刻的索引的位置 (从 0 开始)" +" `i` 的前面部分是交易ID (`txid`),在`i`之后的数字定义了新的铭文在交易总被铭刻" +"的索引的位置 (从 0 开始)" -#: src/inscriptions.md:107 +#: src/inscriptions.md:108 msgid "" "Inscriptions can either be located in different inputs, within the same " "input or a combination of both. In any case the ordering is clear, since a " "parser would go through the inputs consecutively and look for all " "inscription `envelopes`." msgstr "" -"铭文可以位于同一输入中的不同输入中,可以是同一个输入或两者的组合。" -"在任何情况下,顺序都是明确的,因为解析器将连续检查输入并查找所有铭文`信封`" +"铭文可以位于同一输入中的不同输入中,可以是同一个输入或两者的组合。在任何情况" +"下,顺序都是明确的,因为解析器将连续检查输入并查找所有铭文`信封`" -#: src/inscriptions.md:111 +#: src/inscriptions.md:112 msgid "Input" msgstr "" -#: src/inscriptions.md:111 +#: src/inscriptions.md:112 msgid "Inscription Count" msgstr "" -#: src/inscriptions.md:111 +#: src/inscriptions.md:112 msgid "Indices" msgstr "" -#: src/inscriptions.md:113 src/inscriptions.md:116 +#: src/inscriptions.md:114 src/inscriptions.md:117 msgid "0" msgstr "" -#: src/inscriptions.md:113 src/inscriptions.md:115 +#: src/inscriptions.md:114 src/inscriptions.md:116 msgid "2" msgstr "" -#: src/inscriptions.md:113 +#: src/inscriptions.md:114 msgid "i0, i1" msgstr "" -#: src/inscriptions.md:114 src/inscriptions.md:117 +#: src/inscriptions.md:115 src/inscriptions.md:118 msgid "1" msgstr "" -#: src/inscriptions.md:114 +#: src/inscriptions.md:115 msgid "i2" msgstr "" -#: src/inscriptions.md:115 src/inscriptions.md:116 +#: src/inscriptions.md:116 src/inscriptions.md:117 msgid "3" msgstr "" -#: src/inscriptions.md:115 +#: src/inscriptions.md:116 msgid "i3, i4, i5" msgstr "" -#: src/inscriptions.md:117 +#: src/inscriptions.md:118 msgid "4" msgstr "" -#: src/inscriptions.md:117 +#: src/inscriptions.md:118 msgid "i6" msgstr "" -#: src/inscriptions.md:119 +#: src/inscriptions.md:120 msgid "Sandboxing" msgstr "沙盒化" -#: src/inscriptions.md:122 +#: src/inscriptions.md:123 msgid "" "HTML and SVG inscriptions are sandboxed in order to prevent references to " "off-chain content, thus keeping inscriptions immutable and self-contained." msgstr "" -"HTML 和 SVG 铭文被沙箱化,以防止引用链下内容,从而保持铭文的不可变性和独立性。" +"HTML 和 SVG 铭文被沙箱化,以防止引用链下内容,从而保持铭文的不可变性和独立" +"性。" -#: src/inscriptions.md:125 +#: src/inscriptions.md:126 msgid "" "This is accomplished by loading HTML and SVG inscriptions inside `iframes` " "with the `sandbox` attribute, as well as serving inscription content with " "`Content-Security-Policy` headers." msgstr "" -"这是通过在“iframes”中加载 HTML 和 SVG 铭文来完成的`sandbox` 属性,以及提供铭文内容" -"Content-Security-Policy”标头。" +"这是通过在“iframes”中加载 HTML 和 SVG 铭文来完成的`sandbox` 属性,以及提供铭" +"文内容Content-Security-Policy”标头。" + +#: src/inscriptions/provenance.md:4 +msgid "" +"The owner of an inscription can create child inscriptions, trustlessly " +"establishing the provenance of those children on-chain as having been " +"created by the owner of the parent inscription. This can be used for " +"collections, with the children of a parent inscription being members of the " +"same collection." +msgstr "" +"铭文的所有者可以创建子铭文,在链上无需信任地建立这些子铭文的'溯源'," +"证明它们是由父铭文的所有者创建的。这可以用于集合,父铭文的子铭文会成为同一集合的成员。" + +#: src/inscriptions/provenance.md:9 +msgid "" +"Children can themselves have children, allowing for complex hierarchies. For " +"example, an artist might create an inscription representing themselves, with " +"sub inscriptions representing collections that they create, with the " +"children of those sub inscriptions being items in those collections." +msgstr "" +"子铭文自己也可以有子铭文,从而形成复杂的层级结构。" +"例如,一位艺术家可能创建一个代表自己的铭文,子铭文代表他们创建的合辑,而那些子铭文的子项就是合辑中的项目。" + +#: src/inscriptions/provenance.md:14 +msgid "Specification" +msgstr "规范" + +#: src/inscriptions/provenance.md:16 +msgid "To create a child inscription C with parent inscription P:" +msgstr "为父系铭文P创建一个子铭文C:" + +#: src/inscriptions/provenance.md:18 +msgid "Create an inscribe transaction T as usual for C." +msgstr "像通常一样为C创建常用的铭刻交易T。" + +#: src/inscriptions/provenance.md:19 +msgid "Spend the parent P in one of the inputs of T." +msgstr "在其中的一个T输入中加入父系铭文P" + +#: src/inscriptions/provenance.md:20 +msgid "" +"Include tag `3`, i.e. `OP_PUSH 3`, in C, with the value of the serialized " +"binary inscription ID of P, serialized as the 32-byte `TXID`, followed by " +"the four-byte little-endian `INDEX`, with trailing zeroes omitted." +msgstr "" +"在C中包含标签`3`,即`OP_PUSH 3`,其值为P的序列化二进制铭文ID序列化为32字节的`TXID`," +"后跟四字节的小端`INDEX`,不含末尾的零。" + +#: src/inscriptions/provenance.md:24 +msgid "" +"_NB_ The bytes of a bitcoin transaction ID are reversed in their text " +"representation, so the serialized transaction ID will be in the opposite " +"order." +msgstr "" +"_请注意_,比特币交易ID的字节在文本中的表现形式是反向的,所以序列化的交易ID会以相反的顺序呈现。" + +#: src/inscriptions/provenance.md:27 src/guides/testing.md:18 +#: src/guides/reindexing.md:15 +msgid "Example" +msgstr "示例" + +#: src/inscriptions/provenance.md:29 +msgid "" +"An example of a child inscription of " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi0`:" +msgstr "" +"子铭文的一个示例 " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi0`:" + +#: src/inscriptions/provenance.md:32 +msgid "" +"```\n" +"OP_FALSE\n" +"OP_IF\n" +" OP_PUSH \"ord\"\n" +" OP_PUSH 1\n" +" OP_PUSH \"text/plain;charset=utf-8\"\n" +" OP_PUSH 3\n" +" OP_PUSH " +"0x1f1e1d1c1b1a191817161514131211100f0e0d0c0b0a09080706050403020100\n" +" OP_PUSH 0\n" +" OP_PUSH \"Hello, world!\"\n" +"OP_ENDIF\n" +"```" +msgstr "" + +#: src/inscriptions/provenance.md:45 +msgid "" +"Note that the value of tag `3` is binary, not hex, and that for the child " +"inscription to be recognized as a child, " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi0` must be " +"spent as one of the inputs of the inscribe transaction." +msgstr "" +"请注意,标签`3`的值是二进制的,而不是十六进制的,主要是为了让子铭文识别出来是个子铭文," +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi0`必须作为铭文交易的输入之一" + +#: src/inscriptions/provenance.md:50 +msgid "" +"Example encoding of inscription ID " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi255`:" +msgstr "" +"铭文ID的编码示例 " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi255`:" + +#: src/inscriptions/provenance.md:53 +msgid "" +"```\n" +"OP_FALSE\n" +"OP_IF\n" +" …\n" +" OP_PUSH 3\n" +" OP_PUSH " +"0x1f1e1d1c1b1a191817161514131211100f0e0d0c0b0a09080706050403020100ff\n" +" …\n" +"OP_ENDIF\n" +"```" +msgstr "" + +#: src/inscriptions/provenance.md:63 +msgid "" +"And of inscription ID " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi256`:" +msgstr "" +"以及铭文 ID " +"`000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1fi256`:" + +#: src/inscriptions/provenance.md:65 +msgid "" +"```\n" +"OP_FALSE\n" +"OP_IF\n" +" …\n" +" OP_PUSH 3\n" +" OP_PUSH " +"0x1f1e1d1c1b1a191817161514131211100f0e0d0c0b0a090807060504030201000001\n" +" …\n" +"OP_ENDIF\n" +"```" +msgstr "" + +#: src/inscriptions/provenance.md:75 +msgid "Notes" +msgstr "注释" + +#: src/inscriptions/provenance.md:77 +msgid "" +"The tag `3` is used because it is the first available odd tag. Unrecognized " +"odd tags do not make an inscription unbound, so child inscriptions would be " +"recognized and tracked by old versions of `ord`." +msgstr "" +"标签 `3` 被使用是因为它是第一个可用的奇数标签。" +"未识别的奇数标签不会使铭文无法进行绑定,因此,旧版本的ord仍可以识别和追踪子铭文。" + +#: src/inscriptions/provenance.md:81 +msgid "" +"A collection can be closed by burning the collection's parent inscription, " +"which guarantees that no more items in the collection can be issued." +msgstr "" +"通过销毁集合的父铭文,可以关闭一个集合,这保证了该集合中不能再发行更多的项目。 " #: src/inscriptions/recursion.md:4 msgid "" @@ -1217,56 +1371,58 @@ msgid "" "inscriptions to access the content of other inscriptions by requesting `/" "content/`." msgstr "" -"[沙盒化](../inscriptions.md#sandboxing)的一个重要例外是递归:" -"访问“ord”的“/content”允许端点,允许铭文访问" -"其他端点的内容通过请求 `/content/` 来获取铭文。" +"[沙盒化](../inscriptions.md#sandboxing)的一个重要例外是递归:访问“ord”的“/" +"content”允许端点,允许铭文访问其他端点的内容通过请求 `/content/" +"` 来获取铭文。" -#: src/inscriptions/recursion.md:8 +#: src/inscriptions/recursion.md:9 msgid "This has a number of interesting use-cases:" msgstr "这有许多有趣的用例:" -#: src/inscriptions/recursion.md:10 +#: src/inscriptions/recursion.md:11 msgid "Remixing the content of existing inscriptions." msgstr "重新混合现有铭文的内容。" -#: src/inscriptions/recursion.md:12 +#: src/inscriptions/recursion.md:13 msgid "" "Publishing snippets of code, images, audio, or stylesheets as shared public " "resources." msgstr "将代码、图像、音频或样式表片段发布为公共的共享资源。" -#: src/inscriptions/recursion.md:15 +#: src/inscriptions/recursion.md:16 msgid "" "Generative art collections where an algorithm is inscribed as JavaScript, " "and instantiated from multiple inscriptions with unique seeds." -msgstr "生成艺术收藏,其中算法使用JavaScript刻写,并从具有独特种子的多个铭文中实例化。" +msgstr "" +"生成艺术收藏,其中算法使用JavaScript刻写,并从具有独特种子的多个铭文中实例" +"化。" -#: src/inscriptions/recursion.md:18 +#: src/inscriptions/recursion.md:19 msgid "" "Generative profile picture collections where accessories and attributes are " "inscribed as individual images, or in a shared texture atlas, and then " "combined, collage-style, in unique combinations in multiple inscriptions." msgstr "" -"生成个人资料图片集,其中包含配件和属性刻录为单独的图像,或刻录在共享纹理图集中,然后" -"组合,拼贴风格,在多个铭文中以独特的组合。" +"生成个人资料图片集,其中包含配件和属性刻录为单独的图像,或刻录在共享纹理图集" +"中,然后组合,拼贴风格,在多个铭文中以独特的组合。" -#: src/inscriptions/recursion.md:22 +#: src/inscriptions/recursion.md:23 msgid "A few other endpoints that inscriptions may access are the following:" msgstr "铭文可以访问的其他几个端点如下:" -#: src/inscriptions/recursion.md:24 +#: src/inscriptions/recursion.md:25 msgid "`/blockheight`: latest block height." msgstr "`/blockheight`:最新区块高度。" -#: src/inscriptions/recursion.md:25 +#: src/inscriptions/recursion.md:26 msgid "`/blockhash`: latest block hash." msgstr "`/blockhash`:最新的块哈希。" -#: src/inscriptions/recursion.md:26 +#: src/inscriptions/recursion.md:27 msgid "`/blockhash/`: block hash at given block height." msgstr "`/blockhash/`:给定块高度的块哈希。" -#: src/inscriptions/recursion.md:27 +#: src/inscriptions/recursion.md:28 msgid "`/blocktime`: UNIX time stamp of latest block." msgstr "`/blocktime`:最新块的 UNIX 时间戳。" @@ -1284,23 +1440,22 @@ msgid "" "smallest subdivision of a bitcoin, and tracking those satoshis as they are " "spent by transactions." msgstr "" -"序数理论是一种为聪(satoshi,以下写作“聪”,比特币的最小单位)分配序列号的协议," -"并在交易中跟踪这些聪。" +"序数理论是一种为聪(satoshi,以下写作“聪”,比特币的最小单位)分配序列号的协" +"议,并在交易中跟踪这些聪。" #: src/faq.md:11 msgid "" "These serial numbers are large numbers, like this 804766073970493. Every " "satoshi, which is ¹⁄₁₀₀₀₀₀₀₀₀ of a bitcoin, has an ordinal number." msgstr "" -"这些序号都是很大的数字,比如,804766073970493. 每一个聪satoshi, " -"也都是比特币的 ¹⁄₁₀₀₀₀₀₀₀₀ 都有一个序数号号码" +"这些序号都是很大的数字,比如,804766073970493. 每一个聪satoshi, 也都是比特币" +"的 ¹⁄₁₀₀₀₀₀₀₀₀ 都有一个序数号号码" #: src/faq.md:14 msgid "" "Does ordinal theory require a side chain, a separate token, or changes to " "Bitcoin?" -msgstr "" -"序数理论是否需要一个侧链,一个单独的代币,或对比特币做出改变?" +msgstr "序数理论是否需要一个侧链,一个单独的代币,或对比特币做出改变?" #: src/faq.md:17 msgid "" @@ -1311,8 +1466,7 @@ msgstr "" #: src/faq.md:20 msgid "What is ordinal theory good for?" -msgstr "" -"序数理论有什么用?" +msgstr "序数理论有什么用?" #: src/faq.md:23 msgid "" @@ -1320,7 +1474,8 @@ msgid "" "individual satoshis, allowing them to be individually tracked and traded, as " "curios and for numismatic value." msgstr "" -"收集、交易和策划。序数理论将身份分配给单个聪,允许它们作为古玩和钱币价值被单独跟踪和交易。" +"收集、交易和策划。序数理论将身份分配给单个聪,允许它们作为古玩和钱币价值被单" +"独跟踪和交易。" #: src/faq.md:27 msgid "" @@ -1328,12 +1483,12 @@ msgid "" "content to individual satoshis, turning them into bitcoin-native digital " "artifacts." msgstr "" -"序数理论还赋能铭文,这是一种将任意内容附加到单个聪的协议,将它们变成比特币原生的数字文物。" +"序数理论还赋能铭文,这是一种将任意内容附加到单个聪的协议,将它们变成比特币原" +"生的数字文物。" #: src/faq.md:31 msgid "How does ordinal theory work?" -msgstr "" -"序数理论是如何运作的?" +msgstr "序数理论是如何运作的?" #: src/faq.md:34 msgid "" @@ -1351,13 +1506,12 @@ msgid "" "ones, so ordinal theory uses an algorithm to determine how satoshis hop from " "the inputs of a transaction to its outputs." msgstr "" -"聪存在于输出中,但交易会破坏输出并创建新的输出," -"因此序数理论使用一种算法来确定聪如何从交易的输入跳到其输出" +"聪存在于输出中,但交易会破坏输出并创建新的输出,因此序数理论使用一种算法来确" +"定聪如何从交易的输入跳到其输出" #: src/faq.md:43 msgid "Fortunately, that algorithm is very simple." -msgstr "" -"幸运的是,这个算法非常简单。" +msgstr "幸运的是,这个算法非常简单。" #: src/faq.md:45 msgid "" @@ -1367,9 +1521,9 @@ msgid "" "each satoshi in the inputs in order, and assign each to the first available " "slot in the outputs." msgstr "" -"聪按照先进先出的顺序进行转账。 将交易的输入视为聪列表,将输出视为插槽slot列表," -"等待接收聪。 要将输入聪分配给插槽,按顺序检查输入中的每个聪," -"并将每个聪分配给输出中的第一个可用插槽。" +"聪按照先进先出的顺序进行转账。 将交易的输入视为聪列表,将输出视为插槽slot列" +"表,等待接收聪。 要将输入聪分配给插槽,按顺序检查输入中的每个聪,并将每个聪分" +"配给输出中的第一个可用插槽。" #: src/faq.md:51 msgid "" @@ -1377,7 +1531,8 @@ msgid "" "are on the left of the arrow and the outputs are on the right, all labeled " "with their values:" msgstr "" -"让我们想象一个具有三个输入和两个输出的交易。 输入在箭头的左边,输出在右边,都标有它们的值:" +"让我们想象一个具有三个输入和两个输出的交易。 输入在箭头的左边,输出在右边,都" +"标有它们的值:" #: src/faq.md:55 msgid "" @@ -1392,8 +1547,8 @@ msgid "" "satoshis that each input contains, and question marks for each output slot. " "Ordinal numbers are large, so let's use letters to represent them:" msgstr "" -"现在,我们用每个输入包含的聪序数标记同一笔交易,并为每个输出插槽标记问号。 " -"序数号很大,所以我们用字母来表示它们:" +"现在,我们用每个输入包含的聪序数标记同一笔交易,并为每个输出插槽标记问号。 序" +"数号很大,所以我们用字母来表示它们:" #: src/faq.md:61 msgid "" @@ -1406,8 +1561,7 @@ msgstr "" msgid "" "To figure out which satoshi goes to which output, go through the input " "satoshis in order and assign each to a question mark:" -msgstr "" -"要弄清楚哪个聪到哪个输出,请按顺序检查输入聪并将每个聪分配给一个问号:" +msgstr "要弄清楚哪个聪到哪个输出,请按顺序检查输入聪并将每个聪分配给一个问号:" #: src/faq.md:66 msgid "" @@ -1423,9 +1577,9 @@ msgid "" "more satoshis in the inputs than are received by the outputs, so to make our " "transaction into one that pays fees, we'll remove the second output:" msgstr "" -"你可能会问交易费用呢? 好问题! 让我们想象一下同一笔交易,这次是两个聪的费用。" -"收费交易在输入中发送的聪 多于输出接收的聪,因此为了使我们的交易成为支付费用的交易," -"我们将删除第二个输出:" +"你可能会问交易费用呢? 好问题! 让我们想象一下同一笔交易,这次是两个聪的费" +"用。收费交易在输入中发送的聪 多于输出接收的聪,因此为了使我们的交易成为支付费" +"用的交易,我们将删除第二个输出:" #: src/faq.md:73 msgid "" @@ -1473,10 +1627,10 @@ msgid "" "ordered in the block. The coinbase transaction of the block might look like " "this:" msgstr "" -"所以他们作为“费用”去到挖这个区块的矿工那里。[The BIP]" -"(https://github.com/ordinals/ord/blob/master/bip.mediawiki) 有更详细的描述," -"但简而言之,交易支付的费用被视为对Coinbase交易的额外输入,并按照其对应的交易在区块中" -"的顺序进行排序。该区块的Coinbase交易可能是这样的:" +"所以他们作为“费用”去到挖这个区块的矿工那里。[The BIP](https://github.com/" +"ordinals/ord/blob/master/bip.mediawiki) 有更详细的描述,但简而言之,交易支付的" +"费用被视为对Coinbase交易的额外输入,并按照其对应的交易在区块中的顺序进行排" +"序。该区块的Coinbase交易可能是这样的:" #: src/faq.md:87 msgid "" @@ -1502,7 +1656,8 @@ msgstr "为什么聪的铭文被称为“数字文物”而不是“NFT”?" msgid "" "An inscription is an NFT, but the term \"digital artifact\" is used instead, " "because it's simple, suggestive, and familiar." -msgstr "铭文也是一种NFT,但使用术语“数字文物”代替,因为它简单、有启发性且熟悉。" +msgstr "" +"铭文也是一种NFT,但使用术语“数字文物”代替,因为它简单、有启发性且熟悉。" #: src/faq.md:100 msgid "" @@ -1511,9 +1666,9 @@ msgid "" "doesn't provide any indication of what it means if you haven't heard the " "term before." msgstr "" -"\"数字文物\"(数字工件,数字人工制品)这些词具有很强的暗示性," -"即使对以前从未听说过这个词的人来说也是如此" -"相比之下,NFT是一个首字母缩略词,如果你以前没有听过这个术语,它就无法说明它的意思。" +"\"数字文物\"(数字工件,数字人工制品)这些词具有很强的暗示性,即使对以前从未" +"听说过这个词的人来说也是如此相比之下,NFT是一个首字母缩略词,如果你以前没有听" +"过这个术语,它就无法说明它的意思。" #: src/faq.md:104 msgid "" @@ -1521,8 +1676,8 @@ msgid "" "\"fungible\" and sense of the word \"token\" as used in \"NFT\" is uncommon " "outside of financial contexts." msgstr "" -"此外,\"NFT\"感觉像是金融术语,\"NFT\"中使用的\"同质化\"一词和\"代币\"一词的含义" -"在金融语境之外并不常见。" +"此外,\"NFT\"感觉像是金融术语,\"NFT\"中使用的\"同质化\"一词和\"代币\"一词的" +"含义在金融语境之外并不常见。" #: src/faq.md:108 msgid "How do sat inscriptions compare to…" @@ -1540,15 +1695,15 @@ msgstr "_铭文永恒不变_" msgid "" "There is simply no way to for the creator of an inscription, or the owner of " "an inscription, to modify it after it has been created." -msgstr "" -"铭文的创建者或铭文的所有者根本无法在创建铭文后对其进行修改。" +msgstr "铭文的创建者或铭文的所有者根本无法在创建铭文后对其进行修改。" #: src/faq.md:118 msgid "" "Ethereum NFTs _can_ be immutable, but many are not, and can be changed or " "deleted by the NFT contract owner." msgstr "" -"以太坊NFTs_可以_是不可更改的,但很多都不是,且是可以由 NFT 合约所有者更改或删除。" +"以太坊NFTs_可以_是不可更改的,但很多都不是,且是可以由 NFT 合约所有者更改或删" +"除。" #: src/faq.md:121 msgid "" @@ -1556,7 +1711,8 @@ msgid "" "contract code must be audited, which requires detailed knowledge of the EVM " "and Solidity semantics." msgstr "" -"为了确保特定的以太坊 NFT 是不可变的,必须审计合约代码,这需要详细了解 EVM 和 Solidity 语义。" +"为了确保特定的以太坊 NFT 是不可变的,必须审计合约代码,这需要详细了解 EVM 和 " +"Solidity 语义。" #: src/faq.md:125 msgid "" @@ -1565,8 +1721,8 @@ msgid "" "effort to distinguish whether an NFT is mutable or immutable, and whether " "the contract source code is available and has been audited." msgstr "" -"对于非技术用户来说,很难确定某以太坊NFT是否可变,以太坊NFT平台也没有努力去区分NFT是否可变," -"以及合约源代码是否可用并已经过审计。" +"对于非技术用户来说,很难确定某以太坊NFT是否可变,以太坊NFT平台也没有努力去区" +"分NFT是否可变,以及合约源代码是否可用并已经过审计。" #: src/faq.md:130 msgid "_Inscription content is always on-chain._" @@ -1579,8 +1735,8 @@ msgid "" "because inscription creators must pay fees proportional to the size of the " "content." msgstr "" -"铭文无法引用链下内容。因为内容不会丢失,这使得铭文更加持久," -"也使得铭文创作者必须支付与内容大小成比例的费用。" +"铭文无法引用链下内容。因为内容不会丢失,这使得铭文更加持久,也使得铭文创作者" +"必须支付与内容大小成比例的费用。" #: src/faq.md:136 msgid "" @@ -1592,17 +1748,17 @@ msgid "" "catastrophically when these economic assumptions are no longer met. " "Centralized web servers may disappear at any time." msgstr "" -"一些以太坊 NFT 内容在链上的,但大部分内容在链下,存储在 IPFS 或 Arweave 等平台上," -"或传统完全中心化的网络服务器上。IPFS上的内容不保证继续可用,一些存储在IPFS上的NFT内容已经丢失。" -"像Arweave这样的平台依赖于薄弱的经济假设,当这些经济假设不再满足时," -"它们很可能会发生灾难性的失败。中心化的网络服务器随时可能消失。" +"一些以太坊 NFT 内容在链上的,但大部分内容在链下,存储在 IPFS 或 Arweave 等平" +"台上,或传统完全中心化的网络服务器上。IPFS上的内容不保证继续可用,一些存储在" +"IPFS上的NFT内容已经丢失。像Arweave这样的平台依赖于薄弱的经济假设,当这些经济" +"假设不再满足时,它们很可能会发生灾难性的失败。中心化的网络服务器随时可能消" +"失。" #: src/faq.md:144 msgid "" "It is very hard for a non-technical user to determine where the content of a " "given Ethereum NFT is stored." -msgstr "" -"对于非技术用户来说,很难确定某以太坊NFT的内容存储在哪里。" +msgstr "对于非技术用户来说,很难确定某以太坊NFT的内容存储在哪里。" #: src/faq.md:147 msgid "_Inscriptions are much simpler._" @@ -1614,7 +1770,8 @@ msgid "" "highly complex, constantly changing, and which introduce changes via " "backwards-incompatible hard forks." msgstr "" -"以太坊 NFT 依赖于以太坊网络和虚拟机,它们高度复杂、不断变化,并通过向后不兼容的硬分叉引入变化。" +"以太坊 NFT 依赖于以太坊网络和虚拟机,它们高度复杂、不断变化,并通过向后不兼容" +"的硬分叉引入变化。" #: src/faq.md:153 msgid "" @@ -1622,7 +1779,8 @@ msgid "" "relatively simple and conservative, and which introduces changes via " "backwards-compatible soft forks." msgstr "" -"相反,铭文依赖于比特币区块链,它相对简单和保守,并通过向后兼容的软分叉引入变化。" +"相反,铭文依赖于比特币区块链,它相对简单和保守,并通过向后兼容的软分叉引入变" +"化。" #: src/faq.md:157 msgid "_Inscriptions are more secure._" @@ -1636,8 +1794,9 @@ msgid "" "transactions, which don't require allowing a third party, such as an " "exchange or marketplace, to transfer them on the user's behalf." msgstr "" -"铭文继承了比特币的交易模型,允许用户在签名之前准确地看到交易中转移了哪些铭文。" -"铭文可以使用部分签名交易进行销售,不需要允许第三方(如交易所或市场)代表用户转让它们。" +"铭文继承了比特币的交易模型,允许用户在签名之前准确地看到交易中转移了哪些铭" +"文。铭文可以使用部分签名交易进行销售,不需要允许第三方(如交易所或市场)代表" +"用户转让它们。" #: src/faq.md:165 msgid "" @@ -1648,9 +1807,9 @@ msgid "" "hazards for Ethereum NFT users which are simply not a concern for ordinal " "theorists." msgstr "" -"相比之下,以太坊NFT受到终端用户安全漏洞的困扰。盲签交易、授予第三方应用程序对用户NFT" -"的无限权限,以及与复杂且不可预测的智能合约交互都是司空见惯的事情。这为以太坊 NFT 用户制造了" -"一个危险雷区,而这些对于序号理论家来说,根本毋需操心。" +"相比之下,以太坊NFT受到终端用户安全漏洞的困扰。盲签交易、授予第三方应用程序对" +"用户NFT的无限权限,以及与复杂且不可预测的智能合约交互都是司空见惯的事情。这为" +"以太坊 NFT 用户制造了一个危险雷区,而这些对于序号理论家来说,根本毋需操心。" #: src/faq.md:171 msgid "_Inscriptions are scarcer._" @@ -1662,7 +1821,8 @@ msgid "" "downside on the surface, but the raison d'etre of digital artifacts is to be " "scarce and thus valuable." msgstr "" -"铭文需要比特币来铸造、转移和存储。从表面上看,这似乎是一个阻碍,但数字文物存在的价值目的正是稀缺。" +"铭文需要比特币来铸造、转移和存储。从表面上看,这似乎是一个阻碍,但数字文物存" +"在的价值目的正是稀缺。" #: src/faq.md:177 msgid "" @@ -1670,8 +1830,8 @@ msgid "" "qualities with a single transaction, making them inherently less scarce, and " "thus, potentially less valuable." msgstr "" -"另一方面,以太坊 NFT 可以通过单笔交易以几乎无限的质量进行铸造,使它们本质上不那么稀缺," -"因此可能没太多价值。" +"另一方面,以太坊 NFT 可以通过单笔交易以几乎无限的质量进行铸造,使它们本质上不" +"那么稀缺,因此可能没太多价值。" #: src/faq.md:181 msgid "_Inscriptions do not pretend to support on-chain royalties._" @@ -1688,9 +1848,9 @@ msgid "" "remove royalty support." msgstr "" "“链上版税”理论上是个好主意,但在实践中却行不通。 如果没有复杂和侵入性的限制," -"就不能在链上强制执行版税支付。以太坊 NFT 生态系统正在努力地解决围绕版税的难题," -"并且也在共同面对一个现实:即向艺术家传达NFT 链上版税这个利器其实是不可行的," -"与此同时,多个平台则在竞相删除对版税的支持。" +"就不能在链上强制执行版税支付。以太坊 NFT 生态系统正在努力地解决围绕版税的难" +"题,并且也在共同面对一个现实:即向艺术家传达NFT 链上版税这个利器其实是不可行" +"的,与此同时,多个平台则在竞相删除对版税的支持。" #: src/faq.md:190 msgid "" @@ -1698,8 +1858,8 @@ msgid "" "supporting royalties on-chain, thus avoiding the confusion, chaos, and " "negativity of the Ethereum NFT situation." msgstr "" -"铭文完全避免了这种情况,不虚假地承诺支持链上版税," -"从而避免了和以太坊NFT一样混乱又消极的状况。" +"铭文完全避免了这种情况,不虚假地承诺支持链上版税,从而避免了和以太坊NFT一样混" +"乱又消极的状况。" #: src/faq.md:194 msgid "_Inscriptions unlock new markets._" @@ -1712,15 +1872,17 @@ msgid "" "since many Bitcoiners prefer not to interact with the Ethereum ecosystem due " "to concerns related to simplicity, security, and decentralization." msgstr "" -"比特币的市值和流动性都大大超越以太坊。以太坊NFT无法获得此类大部分的流动性," -"因为许多比特币使用者出于简单性、安全性和去中心化的考虑,不愿意与以太坊生态系统进行交互。" +"比特币的市值和流动性都大大超越以太坊。以太坊NFT无法获得此类大部分的流动性,因" +"为许多比特币使用者出于简单性、安全性和去中心化的考虑,不愿意与以太坊生态系统" +"进行交互。" #: src/faq.md:201 msgid "" "Such Bitcoiners may be more interested in inscriptions than Ethereum NFTs, " "unlocking new classes of collector." msgstr "" -"与以太坊 NFT 相比,此类比特币拥护者可能对铭文更感兴趣,从而解锁了新的类别的收藏家。" +"与以太坊 NFT 相比,此类比特币拥护者可能对铭文更感兴趣,从而解锁了新的类别的收" +"藏家。" #: src/faq.md:204 msgid "_Inscriptions have a richer data model._" @@ -1734,8 +1896,9 @@ msgid "" "to support any kind of content supported by web browsers, without requiring " "changes to the underlying protocol." msgstr "" -"铭文由内容类型(也称为MIME类型)和内容(任意字节字符串)组成。这相同于 web 使用的数据模型," -"允许铭文内容随着 web 的发展而发展,并支持 web 浏览器支持的任何类型的内容,而无需更改底层协议。" +"铭文由内容类型(也称为MIME类型)和内容(任意字节字符串)组成。这相同于 web 使" +"用的数据模型,允许铭文内容随着 web 的发展而发展,并支持 web 浏览器支持的任何" +"类型的内容,而无需更改底层协议。" #: src/faq.md:212 msgid "RGB and Taro assets?" @@ -1747,7 +1910,8 @@ msgid "" "Compared to inscriptions, they are much more complicated, but much more " "featureful." msgstr "" -"RGB 和 Taro 都是建立在比特币之上的二层资产协议。 与铭文相比,它们要复杂得多,但也更有特色。" +"RGB 和 Taro 都是建立在比特币之上的二层资产协议。 与铭文相比,它们要复杂得多," +"但也更有特色。" #: src/faq.md:217 msgid "" @@ -1756,8 +1920,8 @@ msgid "" "user experience for inscriptions is likely to be simpler and more polished " "than the user experience for RGB and Taro NFTs." msgstr "" -"序号理论是为数字人工制品而设计的,而 RGB 和 Taro 的主要用例是可替代代币," -"因此铭文的用户体验可能比 RGB 和 Taro NFT 的用户体验更简单、更完善 。" +"序号理论是为数字人工制品而设计的,而 RGB 和 Taro 的主要用例是可替代代币,因此" +"铭文的用户体验可能比 RGB 和 Taro NFT 的用户体验更简单、更完善 。" #: src/faq.md:222 msgid "" @@ -1765,8 +1929,8 @@ msgid "" "infrastructure, and which may be lost. By contrast, inscription content is " "stored on-chain, and cannot be lost." msgstr "" -"RGB 和 Taro 都在链下存储内容,这需要额外的基础设施,而且可能会丢失。" -"相比之下,铭文内容存储在链上,不会丢失。" +"RGB 和 Taro 都在链下存储内容,这需要额外的基础设施,而且可能会丢失。相比之" +"下,铭文内容存储在链上,不会丢失。" #: src/faq.md:226 msgid "" @@ -1775,9 +1939,9 @@ msgid "" "digital artifacts, including a better content model, and features like " "globally unique symbols." msgstr "" -"序数理论、RGB和Taro都是非常早期的,所以这只是推测,但序号理论的重点可能使其在数字艺术品" -"的特性方面具有优势,包括更好的内容模型,以及像全球唯一符号这样的特性。" - +"序数理论、RGB和Taro都是非常早期的,所以这只是推测,但序号理论的重点可能使其在" +"数字艺术品的特性方面具有优势,包括更好的内容模型,以及像全球唯一符号这样的特" +"性。" #: src/faq.md:231 msgid "Counterparty assets?" @@ -1789,15 +1953,16 @@ msgid "" "functionality, which makes most bitcoiners regard it as an altcoin, and not " "an extension or second layer for bitcoin." msgstr "" -"Counterparty 有自己的代币 XCP,它是某些功能所必需的,这使得大多数比特币持有者将其视为山寨币," -"而不是比特币的扩展或第二层。" +"Counterparty 有自己的代币 XCP,它是某些功能所必需的,这使得大多数比特币持有者" +"将其视为山寨币,而不是比特币的扩展或第二层。" #: src/faq.md:237 msgid "" "Ordinal theory has been designed from the ground up for digital artifacts, " "whereas Counterparty was primarily designed for financial token issuance." msgstr "" -"序数理论是为数字文物从头开始设计的,而Counterparty主要是为金融代币发行而设计的。" +"序数理论是为数字文物从头开始设计的,而Counterparty主要是为金融代币发行而设计" +"的。" #: src/faq.md:240 msgid "Inscriptions for…" @@ -1814,8 +1979,8 @@ msgid "" "guarantee that your art survives into the future, there is no better way to " "publish it than as inscriptions." msgstr "" -"_铭文在比特币上_ 比特币是目前地位最高、长期生存机会最大的数字货币。 " -"如果你想保证你的艺术作品能流传到未来,没有比铭文更好的发布方式了。" +"_铭文在比特币上_ 比特币是目前地位最高、长期生存机会最大的数字货币。 如果你想" +"保证你的艺术作品能流传到未来,没有比铭文更好的发布方式了。" #: src/faq.md:250 msgid "" @@ -1823,8 +1988,8 @@ msgid "" "1 satoshi per vbyte, publishing inscription content costs $50 per 1 million " "bytes." msgstr "" -"_链上存储更便宜_按每个比特币2万美元和每 vbyte 1聪的最低中继费用计算," -"发布铭文内容的成本为每100万字节50美元。" +"_链上存储更便宜_按每个比特币2万美元和每 vbyte 1聪的最低中继费用计算,发布铭文" +"内容的成本为每100万字节50美元。" #: src/faq.md:254 msgid "" @@ -1832,22 +1997,22 @@ msgid "" "not yet launched on mainnet. This gives you an opportunity to be an early " "adopter, and explore the medium as it evolves." msgstr "" -"_铭文还处于项目早期_ 铭文仍在开发中,尚未在主网上发布(建议更新)。 " -"这使您有机会成为早期采用者,并随着媒体的发展探索它。" +"_铭文还处于项目早期_ 铭文仍在开发中,尚未在主网上发布(建议更新)。 这使您有" +"机会成为早期采用者,并随着媒体的发展探索它。" #: src/faq.md:258 msgid "" "_Inscriptions are simple._ Inscriptions do not require writing or " "understanding smart contracts." -msgstr "" -"_铭文很简单_ 铭文不需要你编写或理解智能合约。" +msgstr "_铭文很简单_ 铭文不需要你编写或理解智能合约。" #: src/faq.md:261 msgid "" "_Inscriptions unlock new liquidity._ Inscriptions are more accessible and " "appealing to bitcoin holders, unlocking an entirely new class of collector." msgstr "" -"_铭文解锁新的流动性_对于比特币持有者来说,铭文更容易获得,也更有吸引力,从而带来全新的收藏者。" +"_铭文解锁新的流动性_对于比特币持有者来说,铭文更容易获得,也更有吸引力,从而" +"带来全新的收藏者。" #: src/faq.md:264 msgid "" @@ -1855,8 +2020,8 @@ msgid "" "from the ground up to support NFTs, and feature a better data model, and " "features like globally unique symbols and enhanced provenance." msgstr "" -"_铭文是为数字文物设计_ 全新设计的铭文是为了支持 NFT,并具有更好的数据模型," -"以及全球独特符号和增强来源等功能。" +"_铭文是为数字文物设计_ 全新设计的铭文是为了支持 NFT,并具有更好的数据模型,以" +"及全球独特符号和增强来源等功能。" #: src/faq.md:268 msgid "" @@ -1871,11 +2036,12 @@ msgid "" "sale, to benefit from future appreciation, or perhaps offer perks for users " "who respect optional royalties." msgstr "" -"_铭文不鼓励链上版税_ 这可能不是个好消息,但也取决于你如何看待它。链上版税一直是创作者的福音," -"但也在以太坊 NFT生态系统中造成了巨大的混乱。以太坊现在正努力解决这个问题,也是一场逐底竞赛," -"以实现一个“可选版税”的未来。铭文不支持链上版税,因为它们在技术上不可行。" -"如果您选择创建铭文,有许多方法可以绕过这个限制:保留一部分铭文供未来售卖,以受益于未来的升值," -"或者为尊重可选版税的用户提供额外津贴。" +"_铭文不鼓励链上版税_ 这可能不是个好消息,但也取决于你如何看待它。链上版税一直" +"是创作者的福音,但也在以太坊 NFT生态系统中造成了巨大的混乱。以太坊现在正努力" +"解决这个问题,也是一场逐底竞赛,以实现一个“可选版税”的未来。铭文不支持链上版" +"税,因为它们在技术上不可行。如果您选择创建铭文,有许多方法可以绕过这个限制:" +"保留一部分铭文供未来售卖,以受益于未来的升值,或者为尊重可选版税的用户提供额" +"外津贴。" #: src/faq.md:279 msgid "Collectors" @@ -1886,7 +2052,8 @@ msgid "" "_Inscriptions are simple, clear, and have no surprises._ They are always " "immutable and on-chain, with no special due diligence required." msgstr "" -"_铭文很简单_,清晰并无意外* 它们始终是不可变的并且在链上,不需要特殊的尽职调查。" +"_铭文很简单_,清晰并无意外* 它们始终是不可变的并且在链上,不需要特殊的尽职调" +"查。" #: src/faq.md:284 msgid "" @@ -1907,9 +2074,9 @@ msgid "" "understand and acknowledge this, and believe that ordinal theory helps, at " "least in a small way, Bitcoin's primary mission." msgstr "" -"让我在开头说明一下:比特币网络所做的最重要的事情是货币去中心化。" -"所有其他用例都是次要的,包括序数理论。序数理论的开发者理解并承认这一点," -"并相信序数理论至少在很小的程度上有助于比特币的主要任务。" +"让我在开头说明一下:比特币网络所做的最重要的事情是货币去中心化。所有其他用例" +"都是次要的,包括序数理论。序数理论的开发者理解并承认这一点,并相信序数理论至" +"少在很小的程度上有助于比特币的主要任务。" #: src/faq.md:295 msgid "" @@ -1920,9 +2087,9 @@ msgid "" "inception, and predates even trade and money, which are also ancient " "technologies." msgstr "" -"与其他山寨币领域的事物不同,数字文物有其优点。当然,有大量的NFT是丑陋、愚蠢和存在欺骗性的。" -"然而,还是有很多有奇妙的创意,创造和收藏艺术本来就是人类故事的一部分," -"甚至早于贸易和金钱这些同样古老的技术。" +"与其他山寨币领域的事物不同,数字文物有其优点。当然,有大量的NFT是丑陋、愚蠢和" +"存在欺骗性的。然而,还是有很多有奇妙的创意,创造和收藏艺术本来就是人类故事的" +"一部分,甚至早于贸易和金钱这些同样古老的技术。" #: src/faq.md:302 msgid "" @@ -1931,8 +2098,8 @@ msgid "" "the same way that it provides an amazing platform for sending and receiving " "value, and for all the same reasons." msgstr "" -"比特币提供了一个精彩的平台,以一种安全、去中心化的方式创造、收集数字文物," -"也以同样的方式保护了用户和艺术家,更同时提供了一个优秀的平台来发送和接收价值。" +"比特币提供了一个精彩的平台,以一种安全、去中心化的方式创造、收集数字文物,也" +"以同样的方式保护了用户和艺术家,更同时提供了一个优秀的平台来发送和接收价值。" #: src/faq.md:307 msgid "" @@ -1941,8 +2108,8 @@ msgid "" "Bitcoin's transition to a fee-dependent security model, as the block subsidy " "is halved into insignificance." msgstr "" -"序数和铭文增加了对比特币区块空间的需求,这也增加了比特币的安全预算。" -"这对于保障比特币向费用依赖型的安全模式过渡至关重要,因为区块补贴减半已少得微不足道。" +"序数和铭文增加了对比特币区块空间的需求,这也增加了比特币的安全预算。这对于保" +"障比特币向费用依赖型的安全模式过渡至关重要,因为区块补贴减半已少得微不足道。" #: src/faq.md:312 msgid "" @@ -1951,9 +2118,9 @@ msgid "" "for _all_ Bitcoin block space. This will help support a robust fee market, " "which ensures that Bitcoin remains secure." msgstr "" -"铭文内容存储在链上,因此对用于铭文区块空间的需求是无限的。" -"这就为所有比特币区块空间创造了一个最后买家。" -"这将有助于支持一个强大的收费市场,从而确保比特币一直安全。" +"铭文内容存储在链上,因此对用于铭文区块空间的需求是无限的。这就为所有比特币区" +"块空间创造了一个最后买家。这将有助于支持一个强大的收费市场,从而确保比特币一" +"直安全。" #: src/faq.md:317 msgid "" @@ -1964,9 +2131,9 @@ msgid "" "which targets a popular and proven use case, NFTs, which makes it highly " "legible." msgstr "" -"铭文还反驳了比特币不能扩展或用于新用例的说法。 如果你关注 DLC、Fedimint、Lightning、" -"Taro 和 RGB 等项目,你就会知道这种说法是错误的。铭文提供了一个易于理解的反论点," -"并且针对一个流行且经过验证的用例:NFT,这使得它非常易理解。" +"铭文还反驳了比特币不能扩展或用于新用例的说法。 如果你关注 DLC、Fedimint、" +"Lightning、Taro 和 RGB 等项目,你就会知道这种说法是错误的。铭文提供了一个易于" +"理解的反论点,并且针对一个流行且经过验证的用例:NFT,这使得它非常易理解。" #: src/faq.md:323 msgid "" @@ -1975,9 +2142,9 @@ msgid "" "for Bitcoin adoption: come for the fun, rich art, stay for the decentralized " "digital money." msgstr "" -"如果像作者所希望的那样,铭文被证明是具有丰富历史的数字文物,并且受到高度追捧," -"它们将会成为比特币采用的强大吸引力:被乐趣、丰富的艺术吸引而来," -"也为去中心化的数字货币而愿意留下来。" +"如果像作者所希望的那样,铭文被证明是具有丰富历史的数字文物,并且受到高度追" +"捧,它们将会成为比特币采用的强大吸引力:被乐趣、丰富的艺术吸引而来,也为去中" +"心化的数字货币而愿意留下来。" #: src/faq.md:327 msgid "" @@ -1988,9 +2155,10 @@ msgid "" "and collectables on Bitcoin, are unlikely to produce individual entities " "with enough power to corrupt Bitcoin. Art is decentralized." msgstr "" -"铭文是区块空间需求的一个极其良性的来源,不像稳定币,可能会让大型发行人对比特币的未来" -"发展产生影响;也不像DeFi,可能通过在比特币上引入MEV、数字艺术和收藏品的机会来集中挖矿。" -"艺术是去中心化的,任何实体都不可能运用权力去破坏得了比特币。" +"铭文是区块空间需求的一个极其良性的来源,不像稳定币,可能会让大型发行人对比特" +"币的未来发展产生影响;也不像DeFi,可能通过在比特币上引入MEV、数字艺术和收藏品" +"的机会来集中挖矿。艺术是去中心化的,任何实体都不可能运用权力去破坏得了比特" +"币。" #: src/faq.md:334 msgid "" @@ -1998,14 +2166,16 @@ msgid "" "nodes, to publish and track inscriptions, and thus throw their economic " "weight behind the honest chain." msgstr "" -"铭文用户和服务提供商被激励运行比特币全节点,以及发布跟踪铭文,从而将他们的经济权重投向诚实的链。" +"铭文用户和服务提供商被激励运行比特币全节点,以及发布跟踪铭文,从而将他们的经" +"济权重投向诚实的链。" #: src/faq.md:338 msgid "" "Ordinal theory and inscriptions do not meaningfully affect Bitcoin's " "fungibility. Bitcoin users can ignore both and be unaffected." msgstr "" -"序数理论和铭文不会对比特币的可替代性产生重大影响。比特币用户即使忽略这两者也不会受到影响。" +"序数理论和铭文不会对比特币的可替代性产生重大影响。比特币用户即使忽略这两者也" +"不会受到影响。" #: src/faq.md:341 msgid "" @@ -2013,8 +2183,8 @@ msgid "" "another dimension of appeal and functionality, enabling it more effectively " "serve its primary use case as humanity's decentralized store of value." msgstr "" -"我们希望序数理论能够加强、丰富比特币,并赋予它另一个维度的吸引力和功能," -"使其能够更有效地服务于其作为人类去中心化价值存储的主要用例。" +"我们希望序数理论能够加强、丰富比特币,并赋予它另一个维度的吸引力和功能,使其" +"能够更有效地服务于其作为人类去中心化价值存储的主要用例。" #: src/contributing.md:1 msgid "Contributing to `ord`" @@ -2065,13 +2235,11 @@ msgstr "" msgid "" "Mash the keyboard randomly until the tests pass, and refactor until the code " "is ready to submit." -msgstr "" -"随机敲击键盘直到测试通过,然后重构直到代码准备好提交。" +msgstr "随机敲击键盘直到测试通过,然后重构直到代码准备好提交。" #: src/contributing.md:23 msgid "Mark the PR as ready to review." -msgstr "" -"将 PR 标记为审查就绪。" +msgstr "将 PR 标记为审查就绪。" #: src/contributing.md:24 msgid "Revise the PR as needed." @@ -2089,7 +2257,9 @@ msgstr "集腋成裘" msgid "" "Small changes will allow you to make an impact quickly, and if you take the " "wrong tack, you won't have wasted much time." -msgstr "小的改变可以让你迅速的产生影响力,即便你采取了错误的策略,你也不会浪费太多的时间。" +msgstr "" +"小的改变可以让你迅速的产生影响力,即便你采取了错误的策略,你也不会浪费太多的" +"时间。" #: src/contributing.md:33 msgid "Ideas for small issues:" @@ -2117,7 +2287,9 @@ msgstr "找到一个过时的问题,并评论使其关闭" msgid "" "Find an issue that shouldn't be done, and provide constructive feedback " "detailing why you think that is the case" -msgstr "找到一个本不该做的问题,并提供建设性的反馈,详细说明您认为会出现这种情况的原因" +msgstr "" +"找到一个本不该做的问题,并提供建设性的反馈,详细说明您认为会出现这种情况的原" +"因" #: src/contributing.md:42 msgid "Merge early and often" @@ -2132,9 +2304,9 @@ msgid "" "a feature into small sub-features, and implement them one at a time." msgstr "" "将大大型的任务分成多个较小的步骤,这些步骤可以单独取的进展。如果有程序错误," -"您也可以打开一个PR,添加一个失败的忽略测试。这可以合并,下一步可以修复错误" -"并忽略测试。将你的研究或者测试结果进行报告。将一个大的功能分解为小的子功能" -"并一次一个的逐步实现它们。" +"您也可以打开一个PR,添加一个失败的忽略测试。这可以合并,下一步可以修复错误并" +"忽略测试。将你的研究或者测试结果进行报告。将一个大的功能分解为小的子功能并一" +"次一个的逐步实现它们。" #: src/contributing.md:51 msgid "" @@ -2148,8 +2320,7 @@ msgstr "" #: src/contributing.md:55 msgid "" "I strive to follow this advice myself, and am always better off when I do." -msgstr "" -"我自己努力遵循这个建议,而且当我这样做时,我总是可以做的更好。" +msgstr "我自己努力遵循这个建议,而且当我这样做时,我总是可以做的更好。" #: src/contributing.md:57 msgid "" @@ -2164,10 +2335,10 @@ msgid "" "will, the slow merge what they must._" msgstr "" "小的更改可以快速编写、审查和合并,这比为一个需要永远编写、审查和合并的大型的" -"PR工作要有趣得多。小的更改不会花费太多时间,因此如果您需要停止处理一个小的更改," -"与代表许多小时工作的较大更改相比,您不会浪费太多时间。 快速获得PR可以立即改进项目," -"而不必等待很长时间才能进行更大的改进。 小的更改不太可能累积合并冲突。" -"正如雅典人所说:_快者尽其所愿,慢者兼并其所必须。_" +"PR工作要有趣得多。小的更改不会花费太多时间,因此如果您需要停止处理一个小的更" +"改,与代表许多小时工作的较大更改相比,您不会浪费太多时间。 快速获得PR可以立即" +"改进项目,而不必等待很长时间才能进行更大的改进。 小的更改不太可能累积合并冲" +"突。正如雅典人所说:_快者尽其所愿,慢者兼并其所必须。_" #: src/contributing.md:67 msgid "Get help" @@ -2178,8 +2349,8 @@ msgid "" "If you're stuck for more than 15 minutes, ask for help, like a Rust Discord, " "Stack Exchange, or in a project issue or discussion." msgstr "" -"如果您遇到困难超过 15 分钟,请寻求帮助,例如 Rust Discord、Stack Exchange," -"或者在项目问题或讨论中寻求帮助。" +"如果您遇到困难超过 15 分钟,请寻求帮助,例如 Rust Discord、Stack Exchange,或" +"者在项目问题或讨论中寻求帮助。" #: src/contributing.md:73 msgid "Practice hypothesis-driven debugging" @@ -2192,8 +2363,9 @@ msgid "" "issue or now you know how to fix the issue. If not, repeat with a new " "hypothesis." msgstr "" -"就导致问题的原因提出假设。 弄清楚如何检验该假设。 执行该测试。 如果有效,那太好了," -"您解决了问题,或者现在您知道如何解决问题了。 如果不是,请重复一个新的假设。" +"就导致问题的原因提出假设。 弄清楚如何检验该假设。 执行该测试。 如果有效,那太" +"好了,您解决了问题,或者现在您知道如何解决问题了。 如果不是,请重复一个新的假" +"设。" #: src/contributing.md:81 msgid "Pay attention to error messages" @@ -2209,9 +2381,9 @@ msgid "" "`ord` is [raphjaph](https://github.com/raphjaph/). Raph's work on `ord` is " "entirely funded by donations. If you can, please consider donating!" msgstr "" -"Ordinals序数是开源的,由社区资助的项目。目前`ord`的首席维护者是" -"[raphjaph](https://github.com/raphjaph/)." -"Raph在 `ord` 上的维护工作全部由捐赠的资金完成。你如果可以的话,请考虑捐赠!" +"Ordinals序数是开源的,由社区资助的项目。目前`ord`的首席维护者是[raphjaph]" +"(https://github.com/raphjaph/).Raph在 `ord` 上的维护工作全部由捐赠的资金完" +"成。你如果可以的话,请考虑捐赠!" #: src/donate.md:8 msgid "" @@ -2224,13 +2396,11 @@ msgid "" "mempool.space/address/" "bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0)." msgstr "" -"捐赠地址为 " -"[bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt](https://" -"mempool.space/address/" -"bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt). " -"铭文的捐赠地址为 " -"[bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0](https://" -"mempool.space/address/" +"捐赠地址为 [bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt]" +"(https://mempool.space/address/" +"bc1q8kt9pyd6r27k2840l8g5d7zshz3cg9v6rfda0m248lva3ve5072q3sxelt). 铭文的捐赠地" +"址为 [bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0]" +"(https://mempool.space/address/" "bc1qn3map8m9hmk5jyqdkkwlwvt335g94zvxwd9aql7q3vdkdw9r5eyqvlvec0)." #: src/donate.md:11 @@ -2240,18 +2410,17 @@ msgid "" "[rodarmor](https://twitter.com/rodarmor), and [ordinally](https://twitter." "com/veryordinally)." msgstr "" -"上述两个地址是由以下多签人(2/4)持有管理: [raphjaph]" -"(https://twitter.com/raphjaph), [erin](https://twitter.com/realizingerin), " -"[rodarmor](https://twitter.com/rodarmor), and [ordinally](https://twitter." -"com/veryordinally)." +"上述两个地址是由以下多签人(2/4)持有管理: [raphjaph](https://twitter.com/" +"raphjaph), [erin](https://twitter.com/realizingerin), [rodarmor](https://" +"twitter.com/rodarmor), and [ordinally](https://twitter.com/veryordinally)." #: src/donate.md:17 msgid "" "Donations received will go towards funding maintenance and development of " "`ord`, as well as hosting costs for [ordinals.com](https://ordinals.com)." msgstr "" -"收到的捐赠款将用于资助 `ord`的维护和进一步开发," -"同时将支付[ordinals.com](https://ordinals.com)的托管费用。" +"收到的捐赠款将用于资助 `ord`的维护和进一步开发,同时将支付[ordinals.com]" +"(https://ordinals.com)的托管费用。" #: src/donate.md:20 msgid "Thank you for donating!" @@ -2265,8 +2434,7 @@ msgstr "序数理论指引" msgid "" "See the table of contents for a list of guides, including a guide to the " "explorer, a guide for sat hunters, and a guide to inscriptions." -msgstr "" -"请参阅目录以获取指南列表,包括区块浏览器指南、猎聪指南和铭文指南。" +msgstr "请参阅目录以获取指南列表,包括区块浏览器指南、猎聪指南和铭文指南。" #: src/guides/explorer.md:1 msgid "Ordinal Explorer" @@ -2278,8 +2446,9 @@ msgid "" "explorer on mainnet at [ordinals.com](https://ordinals.com), and on signet " "at [signet.ordinals.com](https://signet.ordinals.com)." msgstr "" -"`ord` 文件包含一个区块浏览器。我们的主网区块链器部署在 [ordinals.com](https://ordinals.com), " -"signet部署在[signet.ordinals.com](https://signet.ordinals.com)." +"`ord` 文件包含一个区块浏览器。我们的主网区块链器部署在 [ordinals.com]" +"(https://ordinals.com), signet部署在[signet.ordinals.com](https://signet." +"ordinals.com)." #: src/guides/explorer.md:8 msgid "Running The Explorer" @@ -2357,8 +2526,7 @@ msgstr "输出" msgid "" "Transaction outputs can searched by outpoint, for example, the only output " "of the genesis block coinbase transaction:" -msgstr "" -"可以通过outpoint搜索交易输出,例如创世块coinbase交易的唯一输出:" +msgstr "可以通过outpoint搜索交易输出,例如创世块coinbase交易的唯一输出:" #: src/guides/explorer.md:44 msgid "" @@ -2375,8 +2543,7 @@ msgstr "聪" msgid "" "Sats can be searched by integer, their position within the entire bitcoin " "supply:" -msgstr "" -"聪 可以按整数搜索,它们在整个比特币供应中的位置:" +msgstr "聪 可以按整数搜索,它们在整个比特币供应中的位置:" #: src/guides/explorer.md:51 msgid "[2099994106992659](https://ordinals.com/search/2099994106992659)" @@ -2394,7 +2561,9 @@ msgstr "" msgid "" "By degree, their cycle, blocks since the last halving, blocks since the last " "difficulty adjustment, and offset within their block:" -msgstr "按度数,他们的周期,自上次减半以来的区块,自上次难度调整以来的区块,以及区块内的偏移量:" +msgstr "" +"按度数,他们的周期,自上次减半以来的区块,自上次难度调整以来的区块,以及区块" +"内的偏移量:" #: src/guides/explorer.md:60 msgid "[1°0′0″0‴](https://ordinals.com/search/1°0′0″0‴)" @@ -2430,8 +2599,8 @@ msgid "" "transferred using Bitcoin transactions. Inscriptions are as durable, " "immutable, secure, and decentralized as Bitcoin itself." msgstr "" -"单个 聪 可以刻有任意内容,创建可以保存在比特币钱包中并使用比特币交易传输的比特币原生数字人工制品。" -"铭文与比特币本身一样持久、不变、安全和去中心化。" +"单个 聪 可以刻有任意内容,创建可以保存在比特币钱包中并使用比特币交易传输的比" +"特币原生数字人工制品。铭文与比特币本身一样持久、不变、安全和去中心化。" #: src/guides/inscriptions.md:9 msgid "" @@ -2440,8 +2609,8 @@ msgid "" "inscriptions and perform sat control when constructing transactions to send " "inscriptions to another wallet." msgstr "" -"使用铭文需要一个比特币完整节点,让您了解比特币区块链的当前状态,以及一个可以创建铭文并在构建交易" -"以将铭文发送到另一个钱包时执行 聪 控制的钱包。" +"使用铭文需要一个比特币完整节点,让您了解比特币区块链的当前状态,以及一个可以" +"创建铭文并在构建交易以将铭文发送到另一个钱包时执行 聪 控制的钱包。" #: src/guides/inscriptions.md:14 msgid "" @@ -2449,7 +2618,8 @@ msgid "" "Bitcoin Core wallet cannot create inscriptions and does not perform sat " "control." msgstr "" -"Bitcoin Core 提供比特币全节点和钱包。 但是,Bitcoin Core 钱包不能创建铭文,不执行 聪 控制。" +"Bitcoin Core 提供比特币全节点和钱包。 但是,Bitcoin Core 钱包不能创建铭文,不" +"执行 聪 控制。" #: src/guides/inscriptions.md:17 msgid "" @@ -2457,8 +2627,8 @@ msgid "" "`ord` doesn't implement its own wallet, so `ord wallet` subcommands interact " "with Bitcoin Core wallets." msgstr "" -"这需要[`ord`](https://github.com/ordinals/ord),序数实用程序。" -" `ord` 没有自己的钱包,因此 `ord wallet`子命令与 Bitcoin Core 钱包交互。" +"这需要[`ord`](https://github.com/ordinals/ord),序数实用程序。 `ord` 没有自己" +"的钱包,因此 `ord wallet`子命令与 Bitcoin Core 钱包交互。" #: src/guides/inscriptions.md:21 msgid "This guide covers:" @@ -2503,17 +2673,17 @@ msgid "" "[issues](https://github.com/ordinals/ord/issues) and [discussions](https://" "github.com/ordinals/ord/discussions)." msgstr "" -"如果你遇到困难,可以在[Ordinals Discord Server](https://discord.com/invite/87cjuz4FYg)," -"或者检查Github上的相关内容[问题](https://github.com/ordinals/ord/issues) 和" -"[讨论](https://github.com/ordinals/ord/discussions)." +"如果你遇到困难,可以在[Ordinals Discord Server](https://discord.com/" +"invite/87cjuz4FYg),或者检查Github上的相关内容[问题](https://github.com/" +"ordinals/ord/issues) 和[讨论](https://github.com/ordinals/ord/discussions)." #: src/guides/inscriptions.md:42 msgid "" "Bitcoin Core is available from [bitcoincore.org](https://bitcoincore.org/) " "on the [download page](https://bitcoincore.org/en/download/)." msgstr "" -"Bitcoin Core 可以在 [bitcoincore.org](https://bitcoincore.org/) 上" -"的[下载页面](https://bitcoincore.org/en/download/)." +"Bitcoin Core 可以在 [bitcoincore.org](https://bitcoincore.org/) 上的[下载页" +"面](https://bitcoincore.org/en/download/)." #: src/guides/inscriptions.md:45 msgid "Making inscriptions requires Bitcoin Core 24 or newer." @@ -2525,8 +2695,8 @@ msgid "" "Core is installed, you should be able to run `bitcoind -version` " "successfully from the command line." msgstr "" -"本指南不包括如何详细安装 Bitcoin Core;当你成功安装Bitcoin Core以后," -"你应该可以在命令行使用 `bitcoind -version`命令。" +"本指南不包括如何详细安装 Bitcoin Core;当你成功安装Bitcoin Core以后,你应该可" +"以在命令行使用 `bitcoind -version`命令。" #: src/guides/inscriptions.md:51 msgid "Configuring Bitcoin Core" @@ -2588,7 +2758,8 @@ msgid "" "`ord`." msgstr "" "像区块链浏览器[the mempool.space block explorer](https://mempool.space/)一样" -"对区块进行记述. `ord`同`bitcoind`进行交互, 所以你在使用`ord`时候需要让`bitcoind` 在后台运行。" +"对区块进行记述. `ord`同`bitcoind`进行交互, 所以你在使用`ord`时候需要让" +"`bitcoind` 在后台运行。" #: src/guides/inscriptions.md:88 msgid "Installing `ord`" @@ -2600,8 +2771,9 @@ msgid "" "github.com/ordinals/ord). Pre-built binaries are available on the [releases " "page](https://github.com/ordinals/ord/releases)." msgstr "" -"`ord` 程序使用Rust语言写成,可以从[源码](https://github.com/ordinals/ord)安装. " -"预制文件可以从[版本发布页](https://github.com/ordinals/ord/releases)下载。" +"`ord` 程序使用Rust语言写成,可以从[源码](https://github.com/ordinals/ord)安" +"装. 预制文件可以从[版本发布页](https://github.com/ordinals/ord/releases)下" +"载。" #: src/guides/inscriptions.md:95 msgid "You can install the latest pre-built binary from the command line with:" @@ -2659,7 +2831,9 @@ msgstr "接收聪" msgid "" "Inscriptions are made on individual sats, using normal Bitcoin transactions " "that pay fees in sats, so your wallet will need some sats." -msgstr "铭文是在单个聪上制作的,使用聪来支付费用的普通比特币交易,因此你的钱包将需要一些 聪(比特币)。" +msgstr "" +"铭文是在单个聪上制作的,使用聪来支付费用的普通比特币交易,因此你的钱包将需要" +"一些 聪(比特币)。" #: src/guides/inscriptions.md:127 msgid "Get a new address from your `ord` wallet by running:" @@ -2703,13 +2877,16 @@ msgstr "创建铭文内容" msgid "" "Sats can be inscribed with any kind of content, but the `ord` wallet only " "supports content types that can be displayed by the `ord` block explorer." -msgstr "聪上可以刻录任何类型的内容,但`ord`钱包只支持`ord`区块浏览器可以显示的内容类型。" +msgstr "" +"聪上可以刻录任何类型的内容,但`ord`钱包只支持`ord`区块浏览器可以显示的内容类" +"型。" #: src/guides/inscriptions.md:150 msgid "" "Additionally, inscriptions are included in transactions, so the larger the " "content, the higher the fee that the inscription transaction must pay." -msgstr "另外,铭文是包含在交易中的,所以内容越大,铭文交易需要支付的费用就越高。" +msgstr "" +"另外,铭文是包含在交易中的,所以内容越大,铭文交易需要支付的费用就越高。" #: src/guides/inscriptions.md:153 msgid "" @@ -2718,8 +2895,8 @@ msgid "" "transaction will pay, divide the content size by four and multiply by the " "fee rate." msgstr "" -"铭文内容包含在交易见证中,获得见证折扣。要计算写入交易将支付的大概费用," -"请将内容大小除以四,然后乘以费率。" +"铭文内容包含在交易见证中,获得见证折扣。要计算写入交易将支付的大概费用,请将" +"内容大小除以四,然后乘以费率。" #: src/guides/inscriptions.md:157 msgid "" @@ -2729,9 +2906,9 @@ msgid "" "inscription content, limit inscription content to less than 400,000 weight " "units. 390,000 weight units should be safe." msgstr "" -"铭文交易必须少于 400,000 个权重计量单位,否则不会被 Bitcoin Core 中继。" -"一个字节的铭文内容需要一个权重计量单位。 由于铭文交易不只是铭文内容," -"铭文内容限制在400,000权重计量单位以内。390,000 个权重计量单位应该是安全的。" +"铭文交易必须少于 400,000 个权重计量单位,否则不会被 Bitcoin Core 中继。一个字" +"节的铭文内容需要一个权重计量单位。 由于铭文交易不只是铭文内容,铭文内容限制在" +"400,000权重计量单位以内。390,000 个权重计量单位应该是安全的。" #: src/guides/inscriptions.md:163 msgid "Creating Inscriptions" @@ -2756,8 +2933,9 @@ msgid "" "transaction, and `N` is the index of the inscription in the reveal " "transaction." msgstr "" -"Ord会输出两个交易ID,一个是commit交易,一个是reveal交易,还有铭文ID。" -"铭文 ID 的格式为`TXIDiN`,其中`TXID` 是揭示交易的交易 ID,`N` 是揭示交易中铭文的索引。" +"Ord会输出两个交易ID,一个是commit交易,一个是reveal交易,还有铭文ID。铭文 ID " +"的格式为`TXIDiN`,其中`TXID` 是揭示交易的交易 ID,`N` 是揭示交易中铭文的索" +"引。" #: src/guides/inscriptions.md:177 msgid "" @@ -2782,8 +2960,7 @@ msgstr "" msgid "" "Once the reveal transaction has been mined, the inscription ID should be " "printed when you run:" -msgstr "" -"一旦reveal交易完成记账,你可以使用以下命令查询铭文ID:" +msgstr "一旦reveal交易完成记账,你可以使用以下命令查询铭文ID:" #: src/guides/inscriptions.md:189 src/guides/inscriptions.md:220 #: src/guides/inscriptions.md:246 @@ -2861,8 +3038,9 @@ msgid "" "supplied. Additionally, `ord` now has a built-in wallet that wraps a Bitcoin " "Core wallet. See `ord wallet --help`._" msgstr "" -"_本指南已过时。自编写以来,“ord”安装文件已更改仅当提供“--index-sats”标志时才构建完整的聪索引。" -"此外,“ord”现在有一个内置钱包,其中包含比特币核心钱包。请参阅`ord wallet --help`。_" +"_本指南已过时。自编写以来,“ord”安装文件已更改仅当提供“--index-sats”标志时才" +"构建完整的聪索引。此外,“ord”现在有一个内置钱包,其中包含比特币核心钱包。请参" +"阅`ord wallet --help`。_" #: src/guides/sat-hunting.md:9 msgid "" @@ -3265,8 +3443,9 @@ msgid "" "supporting sat-control and sat-selection, which are required to safely store " "and send rare sats and inscriptions, hereafter ordinals." msgstr "" -"目前,[ord](https://github.com/ordinals/ord/) 是唯一支持 sat-control 和 sat-selection " -"的钱包,这是安全存储和发送稀有 sats 和铭文(以下简称序数)所必需的。" +"目前,[ord](https://github.com/ordinals/ord/) 是唯一支持 sat-control 和 sat-" +"selection 的钱包,这是安全存储和发送稀有 sats 和铭文(以下简称序数)所必需" +"的。" #: src/guides/collecting.md:8 msgid "" @@ -3274,8 +3453,8 @@ msgid "" "if you are careful, it is possible to safely store, and in some cases send, " "ordinals with other wallets." msgstr "" -"发送、接收和存储序号的推荐方法是使用 `ord`,但如果你小心,可以安全地存储," -"在某些情况下,使用其他钱包发送序号。" +"发送、接收和存储序号的推荐方法是使用 `ord`,但如果你小心,可以安全地存储,在" +"某些情况下,使用其他钱包发送序号。" #: src/guides/collecting.md:12 msgid "" @@ -3285,9 +3464,9 @@ msgid "" "used to send bitcoin, it may select the UTXO containing the ordinal as an " "input, and send the inscription or spend it to fees." msgstr "" -"作为一般说明,在不受支持的钱包中接收序号并不危险。 序号可以发送到任何比特币地址," -"只要包含它们的 UTXO 没有被花费,它就是安全的。 但是,如果该钱包随后用于发送比特币," -"它可能会选择包含序号的 UTXO 作为输入,并发送铭文或将其用于费用。" +"作为一般说明,在不受支持的钱包中接收序号并不危险。 序号可以发送到任何比特币地" +"址,只要包含它们的 UTXO 没有被花费,它就是安全的。 但是,如果该钱包随后用于发" +"送比特币,它可能会选择包含序号的 UTXO 作为输入,并发送铭文或将其用于费用。" #: src/guides/collecting.md:18 msgid "" @@ -3295,8 +3474,8 @@ msgid "" "wallet with [Sparrow Wallet](https://sparrowwallet.com/), is available in " "this handbook." msgstr "" -"本手册提供了使用[Sparrow Wallet](https://sparrowwallet.com/)创建" -"与 `ord`兼容的钱包的[指南](./collecting/sparrow-wallet.md) 。" +"本手册提供了使用[Sparrow Wallet](https://sparrowwallet.com/)创建与 `ord`兼容" +"的钱包的[指南](./collecting/sparrow-wallet.md) 。" #: src/guides/collecting.md:21 msgid "" @@ -3304,7 +3483,8 @@ msgid "" "create to send BTC, unless you perform manual coin-selection to avoid " "sending ordinals." msgstr "" -"请注意,如果您遵循本指南,则不应使用您创建的钱包发送 BTC,除非您执行手动硬币选择以避免发送序号。" +"请注意,如果您遵循本指南,则不应使用您创建的钱包发送 BTC,除非您执行手动硬币" +"选择以避免发送序号。" #: src/guides/collecting/sparrow-wallet.md:1 msgid "Collecting Inscriptions and Ordinals with Sparrow Wallet" @@ -3317,16 +3497,17 @@ msgid "" "bitcoin wallets, as long as they are _very_ careful about how they spend " "from that wallet." msgstr "" -"那些无法活着尚未设置[ord](https://github.com/ordinals/ord) " -"钱包的用户可以使用其他比特币钱包接收铭文和序数,只要他们在使用该钱包时非常小心。" +"那些无法活着尚未设置[ord](https://github.com/ordinals/ord) 钱包的用户可以使用" +"其他比特币钱包接收铭文和序数,只要他们在使用该钱包时非常小心。" #: src/guides/collecting/sparrow-wallet.md:6 msgid "" "This guide gives some basic steps on how to create a wallet with [Sparrow " "Wallet](https://sparrowwallet.com/) which is compatible with `ord` and can " "be later imported into `ord`" -msgstr "本指南提供了一些基本步骤,说明如何使用 [Sparrow Wallet](https://sparrowwallet.com/) " -"创建一个与`ord`兼容的钱包,稍后可以将其导入到`ord`" +msgstr "" +"本指南提供了一些基本步骤,说明如何使用 [Sparrow Wallet](https://" +"sparrowwallet.com/) 创建一个与`ord`兼容的钱包,稍后可以将其导入到`ord`" #: src/guides/collecting/sparrow-wallet.md:8 msgid "⚠️⚠️ Warning!! ⚠️⚠️" @@ -3337,7 +3518,8 @@ msgid "" "As a general rule if you take this approach, you should use this wallet with " "the Sparrow software as a receive-only wallet." msgstr "" -"一般来说,如果你选择这种方法,你应该将这个钱包作为接收款项的钱包,使用Sparrow软件。" +"一般来说,如果你选择这种方法,你应该将这个钱包作为接收款项的钱包,使用Sparrow" +"软件。" #: src/guides/collecting/sparrow-wallet.md:11 msgid "" @@ -3345,8 +3527,8 @@ msgid "" "you are doing. You could very easily inadvertently lose access to your " "ordinals and inscriptions if you don't heed this warning." msgstr "" -"除非你确定知道自己在做什么,否则不要从这个钱包中花费任何比特币。如果你不注意这个警告," -"你可能会很容易无意间失去对序数和铭文的访问权限。" +"除非你确定知道自己在做什么,否则不要从这个钱包中花费任何比特币。如果你不注意" +"这个警告,你可能会很容易无意间失去对序数和铭文的访问权限。" #: src/guides/collecting/sparrow-wallet.md:13 msgid "Wallet Setup & Receiving" @@ -3357,12 +3539,12 @@ msgid "" "Download the Sparrow Wallet from the [releases page](https://sparrowwallet." "com/download/) for your particular operating system." msgstr "" -"根据你的操作系统从 [发布页面](https://sparrowwallet.com/download/) 下载Sparrow钱包。" +"根据你的操作系统从 [发布页面](https://sparrowwallet.com/download/) 下载" +"Sparrow钱包。" #: src/guides/collecting/sparrow-wallet.md:17 msgid "Select `File -> New Wallet` and create a new wallet called `ord`." -msgstr "" -"选择 `File -> New Wallet`并创建一个名为`ord`的新钱包。" +msgstr "选择 `File -> New Wallet`并创建一个名为`ord`的新钱包。" #: src/guides/collecting/sparrow-wallet.md:19 msgid "![](images/wallet_setup_01.png)" @@ -3373,7 +3555,8 @@ msgid "" "Change the `Script Type` to `Taproot (P2TR)` and select the `New or Imported " "Software Wallet` option." msgstr "" -"将`Script Type`更改为`Taproot (P2TR)`,然后选择`New or Imported Software Wallet`选项。" +"将`Script Type`更改为`Taproot (P2TR)`,然后选择`New or Imported Software " +"Wallet`选项。" #: src/guides/collecting/sparrow-wallet.md:23 msgid "![](images/wallet_setup_02.png)" @@ -3383,8 +3566,7 @@ msgstr "" msgid "" "Select `Use 12 Words` and then click `Generate New`. Leave the passphrase " "blank." -msgstr "" -"选择`Use 12 Words`,然后点击 `Generate New`。密码短语留空。" +msgstr "选择`Use 12 Words`,然后点击 `Generate New`。密码短语留空。" #: src/guides/collecting/sparrow-wallet.md:27 msgid "![](images/wallet_setup_03.png)" @@ -3396,8 +3578,8 @@ msgid "" "somewhere safe as this is your backup to get access to your wallet. NEVER " "share or show this seed phrase to anyone else." msgstr "" -"将为你生成一个新的12词BIP39种子短语。将此短语写在安全的地方," -"这是获取钱包访问权限的备份。切勿与他人分享或显示这个种子短语。" +"将为你生成一个新的12词BIP39种子短语。将此短语写在安全的地方,这是获取钱包访问" +"权限的备份。切勿与他人分享或显示这个种子短语。" #: src/guides/collecting/sparrow-wallet.md:31 msgid "Once you have written down the seed phrase click `Confirm Backup`." @@ -3411,8 +3593,7 @@ msgstr "" msgid "" "Re-enter the seed phrase which you wrote down, and then click `Create " "Keystore`." -msgstr "" -"重新输入你记下的种子短语,然后点击 `Create Keystore`." +msgstr "重新输入你记下的种子短语,然后点击 `Create Keystore`." #: src/guides/collecting/sparrow-wallet.md:37 msgid "![](images/wallet_setup_05.png)" @@ -3422,7 +3603,6 @@ msgstr "" msgid "Click `Import Keystore`." msgstr "点击 `Import Keystore`." - #: src/guides/collecting/sparrow-wallet.md:41 msgid "![](images/wallet_setup_06.png)" msgstr "" @@ -3441,8 +3621,8 @@ msgid "" "into `ord` using the BIP39 Seed Phrase. To receive ordinals or inscriptions, " "click on the `Receive` tab and copy a new address." msgstr "" -"你现在有了一个兼容`ord`的钱包,可以使用BIP39种子短语导入到 `ord`。要接收序数或铭文," -"点击 `Receive`选项卡并复制一个新地址。" +"你现在有了一个兼容`ord`的钱包,可以使用BIP39种子短语导入到 `ord`。要接收序数" +"或铭文,点击 `Receive`选项卡并复制一个新地址。" #: src/guides/collecting/sparrow-wallet.md:49 msgid "" @@ -3457,8 +3637,9 @@ msgid "" "generate a new address by clicking on the `Get Next Address` button. You can " "see all of your addresses in the `Addresses` tab of the app." msgstr "" -"注意,比特币与一些其他区块链钱包不同,这个钱包可以生成无限数量的新地址。" -"你可以通过点击获取下一个地址按钮生成新地址。你可以在应用程序的`Addresses`选项卡中看到所有的地址。" +"注意,比特币与一些其他区块链钱包不同,这个钱包可以生成无限数量的新地址。你可" +"以通过点击获取下一个地址按钮生成新地址。你可以在应用程序的`Addresses`选项卡中" +"看到所有的地址。" #: src/guides/collecting/sparrow-wallet.md:53 msgid "" @@ -3479,8 +3660,8 @@ msgid "" "Once you have received an inscription you will see a new transaction in the " "`Transactions` tab of Sparrow, as well as a new UTXO in the `UTXOs` tab." msgstr "" -"一旦你收到一条铭文,你将在 Sparrow 的 `Transactions` 选项卡中看到一个新的交易," -"以及在`UTXOs`选项卡中看到一个新的 UTXO。" +"一旦你收到一条铭文,你将在 Sparrow 的 `Transactions` 选项卡中看到一个新的交" +"易,以及在`UTXOs`选项卡中看到一个新的 UTXO。" #: src/guides/collecting/sparrow-wallet.md:61 msgid "" @@ -3488,7 +3669,8 @@ msgid "" "need to wait for it to be mined into a bitcoin block before it is fully " "received." msgstr "" -"最初,这笔交易可能有一个\"未确认\"的状态,你需要等待它被挖矿到一个比特币块中,才算真正收到。" +"最初,这笔交易可能有一个\"未确认\"的状态,你需要等待它被挖矿到一个比特币块" +"中,才算真正收到。" #: src/guides/collecting/sparrow-wallet.md:63 msgid "![](images/validating_viewing_01.png)" @@ -3500,8 +3682,8 @@ msgid "" "`Copy Transaction ID` and then paste that transaction id into [mempool.space]" "(https://mempool.space)." msgstr "" -"要跟踪你的交易状态,你可以右键点击它,选择`Copy Transaction ID`," -"然后将该交易 id 粘贴到 [mempool.space](https://mempool.space)。" +"要跟踪你的交易状态,你可以右键点击它,选择`Copy Transaction ID`,然后将该交" +"易 id 粘贴到 [mempool.space](https://mempool.space)。" #: src/guides/collecting/sparrow-wallet.md:67 msgid "![](images/validating_viewing_02.png)" @@ -3515,9 +3697,9 @@ msgid "" "Output`. This transaction output id can then be pasted into the [ordinals." "com](https://ordinals.com) search." msgstr "" -"一旦交易被确认,你可以通过前往`UTXOs`选项卡,找到你想要检查的 UTXO,右键点击 `Output` " -"并选择 `Copy Transaction Output` 来验证和查看你的铭文。然后,这个交易输出 id " -"可以粘贴到 [ordinals.com](https://ordinals.com) 搜索。" +"一旦交易被确认,你可以通过前往`UTXOs`选项卡,找到你想要检查的 UTXO,右键点击 " +"`Output` 并选择 `Copy Transaction Output` 来验证和查看你的铭文。然后,这个交" +"易输出 id 可以粘贴到 [ordinals.com](https://ordinals.com) 搜索。" #: src/guides/collecting/sparrow-wallet.md:72 msgid "Freezing UTXO's" @@ -3530,22 +3712,22 @@ msgid "" "spend your inscriptions, and one way to make it harder for this to happen is " "to freeze the UTXO." msgstr "" -"如上所述,你的每一条铭文都存储在一个未花费的交易输出 (UTXO) 中。你需要非常小心不要意外花费你的铭文," -"而冻结 UTXO 是使这种情况发生的难度增加的一种方式。" +"如上所述,你的每一条铭文都存储在一个未花费的交易输出 (UTXO) 中。你需要非常小" +"心不要意外花费你的铭文,而冻结 UTXO 是使这种情况发生的难度增加的一种方式。" #: src/guides/collecting/sparrow-wallet.md:75 msgid "" "To do this, go to the `UTXOs` tab, find the UTXO you want to freeze, right-" "click on the `Output` and select `Freeze UTXO`." msgstr "" -"要做到这一点,去 UTXOs 选项卡,找到你想要冻结的 `UTXOs`,右键点击 `Output` 并选择`Freeze UTXO`。" +"要做到这一点,去 UTXOs 选项卡,找到你想要冻结的 `UTXOs`,右键点击 `Output` " +"并选择`Freeze UTXO`。" #: src/guides/collecting/sparrow-wallet.md:77 msgid "" "This UTXO (Inscription) is now un-spendable within the Sparrow Wallet until " "you unfreeze it." -msgstr "" -"这个 UTXO (铭文) 现在在 Sparrow 钱包中是不可消费的,直到你解冻它。" +msgstr "这个 UTXO (铭文) 现在在 Sparrow 钱包中是不可消费的,直到你解冻它。" #: src/guides/collecting/sparrow-wallet.md:79 msgid "Importing into `ord` wallet" @@ -3556,7 +3738,8 @@ msgid "" "For details on setting up Bitcoin Core and the `ord` wallet check out the " "[Inscriptions Guide](../inscriptions.md)" msgstr "" -"关于设置比特币核心和 `ord` 钱包的详细信息,请查看[铭文指南](../inscriptions.md)" +"关于设置比特币核心和 `ord` 钱包的详细信息,请查看[铭文指南](../inscriptions." +"md)" #: src/guides/collecting/sparrow-wallet.md:83 msgid "" @@ -3565,9 +3748,9 @@ msgid "" "restore \"BIP39 SEED PHRASE\"` using the seed phrase you generated with " "Sparrow Wallet." msgstr "" -"设置 `ord` 时,你可以使用 `ord wallet restore \"BIP39 SEED PHRASE\"` 命令和" -"你用Sparrow Wallet生成的种子短语,导入你现有的钱包,而不是运行 `ord wallet create` " -"来创建一个全新的钱包。" +"设置 `ord` 时,你可以使用 `ord wallet restore \"BIP39 SEED PHRASE\"` 命令和你" +"用Sparrow Wallet生成的种子短语,导入你现有的钱包,而不是运行 `ord wallet " +"create` 来创建一个全新的钱包。" #: src/guides/collecting/sparrow-wallet.md:85 msgid "" @@ -3578,8 +3761,8 @@ msgid "" "rescanblockchain 767430`" msgstr "" "目前存在一个[程序错误](https://github.com/ordinals/ord/issues/1589) 导致导入" -"的钱包无法自动重新扫描区块链。为解决这个问题,你需要手动触发重新扫描," -"使用比特币核心命令行界面:" +"的钱包无法自动重新扫描区块链。为解决这个问题,你需要手动触发重新扫描,使用比" +"特币核心命令行界面:" #: src/guides/collecting/sparrow-wallet.md:88 msgid "" @@ -3594,8 +3777,8 @@ msgid "" "all `ord` commands to reference a different wallet, eg:" msgstr "" "注意,如果你之前已经用 `ord` 创建过一个钱包,那么你已经有一个默认名称的钱包," -"需要给你导入的钱包取一个不同的名称。你可以在所有的 `ord`命令中使用 `--wallet` 参数" -"来引用不同的钱包,例如:" +"需要给你导入的钱包取一个不同的名称。你可以在所有的 `ord`命令中使用 `--" +"wallet` 参数来引用不同的钱包,例如:" #: src/guides/collecting/sparrow-wallet.md:92 msgid "`ord --wallet ord_from_sparrow wallet restore \"BIP39 SEED PHRASE\"`" @@ -3625,8 +3808,9 @@ msgid "" "recommended, and you should only do this if you fully understand what you " "are doing." msgstr "" -"虽然强烈建议你设置一个比特币核心节点并运行 `ord` 软件,但是你可以通过一些安全的方式在" -" Sparrow 钱包中发送铭文。请注意,这并不推荐,只有在你完全理解你正在做什么的情况下才能这么做。" +"虽然强烈建议你设置一个比特币核心节点并运行 `ord` 软件,但是你可以通过一些安全" +"的方式在 Sparrow 钱包中发送铭文。请注意,这并不推荐,只有在你完全理解你正在做" +"什么的情况下才能这么做。" #: src/guides/collecting/sparrow-wallet.md:103 msgid "" @@ -3634,7 +3818,8 @@ msgid "" "describing here, as it is able to automatically and safely handle sending " "inscriptions in an easy way." msgstr "" -"使用 `ord` 软件将大大简化我们在这里描述的复杂性,因为它能以一种简单的方式自动并安全地处理发送铭文。"" +"使用 `ord` 软件将大大简化我们在这里描述的复杂性,因为它能以一种简单的方式自动" +"并安全地处理发送铭文。" #: src/guides/collecting/sparrow-wallet.md:105 msgid "⚠️⚠️ Additional Warning ⚠️⚠️" @@ -3647,8 +3832,8 @@ msgid "" "to do normal bitcoin transactions, and keep your inscriptions wallet " "separate." msgstr "" -"不要用你的sparrow麻雀铭文钱包去发送非铭文比特币。如果你需要进行普通的比特币交易," -"你可以在麻雀中设置一个单独的钱包,并保持你的铭文钱包独立。" +"不要用你的sparrow麻雀铭文钱包去发送非铭文比特币。如果你需要进行普通的比特币交" +"易,你可以在麻雀中设置一个单独的钱包,并保持你的铭文钱包独立。" #: src/guides/collecting/sparrow-wallet.md:108 msgid "Bitcoin's UTXO model" @@ -3668,12 +3853,12 @@ msgid "" "can select specific UTXO's which you want to spend, and you can choose not " "to spend certain UTXO's." msgstr "" -"在发送任何交易之前,你必须对比特币的未消费交易输出(UTXO)系统有一个良好的理解。" -"比特币的工作方式与以太坊等许多其他区块链有着根本的不同。在以太坊中,通常你有一个存储ETH的单一地址," -"你无法区分其中的任何ETH - 它们只是该地址中的总金额的单一值。" -"而比特币的工作方式完全不同,我们为每个接收生成一个新地址,每次你向钱包中的一个地址接收sats时," -"你都在创建一个新的UTXO。每个UTXO都可以单独查看和管理。" -"你可以选择想要花费的特定UTXO,也可以选择不花费某些UTXO。" +"在发送任何交易之前,你必须对比特币的未消费交易输出(UTXO)系统有一个良好的理" +"解。比特币的工作方式与以太坊等许多其他区块链有着根本的不同。在以太坊中,通常" +"你有一个存储ETH的单一地址,你无法区分其中的任何ETH - 它们只是该地址中的总金额" +"的单一值。而比特币的工作方式完全不同,我们为每个接收生成一个新地址,每次你向" +"钱包中的一个地址接收sats时,你都在创建一个新的UTXO。每个UTXO都可以单独查看和" +"管理。你可以选择想要花费的特定UTXO,也可以选择不花费某些UTXO。" #: src/guides/collecting/sparrow-wallet.md:111 msgid "" @@ -3682,8 +3867,8 @@ msgid "" "when sending inscriptions it is important that you use a wallet like Sparrow " "which allows for UTXO control." msgstr "" -"有些比特币钱包并不显示这个级别的详细信息,它们只向你显示钱包中所有比特币的单一总和值。" -"然而,当发送铭文时,使用如麻雀这样允许UTXO控制的钱包非常重要。" +"有些比特币钱包并不显示这个级别的详细信息,它们只向你显示钱包中所有比特币的单" +"一总和值。然而,当发送铭文时,使用如麻雀这样允许UTXO控制的钱包非常重要。" #: src/guides/collecting/sparrow-wallet.md:113 msgid "Inspecting your inscription before sending" @@ -3697,9 +3882,9 @@ msgid "" "not always) the inscription will be inscribed on the first satoshi in the " "UTXO." msgstr "" -"如我们之前所述,铭文是刻在聪上的,sats存储在UTXO中。" -"UTXO是具有某个特定数量的satoshi(输出值)的satoshi集合。通常(但不总是)铭文会被刻在UTXO中" -"的第一个satoshi上。" +"如我们之前所述,铭文是刻在聪上的,sats存储在UTXO中。UTXO是具有某个特定数量的" +"satoshi(输出值)的satoshi集合。通常(但不总是)铭文会被刻在UTXO中的第一个" +"satoshi上。" #: src/guides/collecting/sparrow-wallet.md:116 msgid "" @@ -3714,8 +3899,8 @@ msgid "" "(./sparrow-wallet.md#validating--viewing-received-inscriptions) described " "above to find the inscription page for your inscription on ordinals.com" msgstr "" -"为此,你可以按照上述 [验证/查看收到的铭文](./sparrow-wallet.md#validating--viewing-received-inscriptions)" -"来找到ordinals.com上你的铭文的铭文页面。" +"为此,你可以按照上述 [验证/查看收到的铭文](./sparrow-wallet.md#validating--" +"viewing-received-inscriptions)来找到ordinals.com上你的铭文的铭文页面。" #: src/guides/collecting/sparrow-wallet.md:120 msgid "" @@ -3749,15 +3934,16 @@ msgid "" "for sending the transaction. The exact amount you will need depends on the " "fee rate you will select for the transaction" msgstr "" -"`output_value` 有足够的sats来支付发送交易的交易费(邮资),您需要的确切金额取决于" -"您为交易选择的费率" +"`output_value` 有足够的sats来支付发送交易的交易费(邮资),您需要的确切金额取" +"决于您为交易选择的费率" #: src/guides/collecting/sparrow-wallet.md:129 msgid "" "If all of the above are true for your inscription, it should be safe for you " "to send it using the method below." msgstr "" -"如果以上所有内容对于您的铭文都是正确的,那么您应该可以安全地使用以下方法发送它。" +"如果以上所有内容对于您的铭文都是正确的,那么您应该可以安全地使用以下方法发送" +"它。" #: src/guides/collecting/sparrow-wallet.md:131 msgid "" @@ -3766,8 +3952,9 @@ msgid "" "case, as doing so you could accidentally send your inscription to a bitcoin " "miner unless you know what you are doing." msgstr "" -"⚠️⚠️ 发送铭文时要非常小心,特别是如果`offset` 值不是`0`。如果是这种情况," -"不建议使用这种方法,否则您可能会无意中将您的雕文发送给比特币矿工,除非您知道自己在做什么。" +"⚠️⚠️ 发送铭文时要非常小心,特别是如果`offset` 值不是`0`。如果是这种情况,不建议" +"使用这种方法,否则您可能会无意中将您的雕文发送给比特币矿工,除非您知道自己在" +"做什么。" #: src/guides/collecting/sparrow-wallet.md:133 msgid "Sending your inscription" @@ -3784,8 +3971,7 @@ msgstr "" msgid "" "If you previously froze the UXTO you will need to right-click on it and " "unfreeze it." -msgstr "" -"如果您之前冻结了UXTO,您将需要右键单击它并解冻它。" +msgstr "如果您之前冻结了UXTO,您将需要右键单击它并解冻它。" #: src/guides/collecting/sparrow-wallet.md:138 msgid "" @@ -3793,8 +3979,8 @@ msgid "" "selected. You should see `UTXOs 1/1` in the interface. Once you are sure " "this is the case you can hit `Send Selected`." msgstr "" -"选择您想要发送的UTXO,并确保这是唯一选中的UTXO。在界面中,您应该看到`UTXOs 1/1`。" -"确定这个后,您可以点击`Send Selected`。" +"选择您想要发送的UTXO,并确保这是唯一选中的UTXO。在界面中,您应该看到`UTXOs " +"1/1`。确定这个后,您可以点击`Send Selected`。" #: src/guides/collecting/sparrow-wallet.md:140 msgid "![](images/sending_02.png)" @@ -3806,21 +3992,20 @@ msgid "" "There is a few things you need to check here to make sure that this is a " "safe send:" msgstr "" -"然后,您将看到交易构建界面。在这里,您需要检查几件事以确保这是一个安全的发送:" +"然后,您将看到交易构建界面。在这里,您需要检查几件事以确保这是一个安全的发" +"送:" #: src/guides/collecting/sparrow-wallet.md:144 msgid "" "The transaction should have only 1 input, and this should be the UTXO with " "the label you want to send" -msgstr "" -"交易应该只有1个输入,这应该是您想要发送的带有标签的UTXO" +msgstr "交易应该只有1个输入,这应该是您想要发送的带有标签的UTXO" #: src/guides/collecting/sparrow-wallet.md:145 msgid "" "The transaction should have only 1 output, which is the address/label where " "you want to send the inscription" -msgstr "" -"交易应该只有1个输出,这是您想要发送铭文的地址/标签" +msgstr "交易应该只有1个输出,这是您想要发送铭文的地址/标签" #: src/guides/collecting/sparrow-wallet.md:147 msgid "" @@ -3829,8 +4014,8 @@ msgid "" "inscription, and you should abandon sending until you understand more, or " "can import into the `ord` wallet." msgstr "" -"如果您的交易看起来与此不同,例如您有多个输入或多个输出,那么这可能不是一种安全的铭文传输方式," -"您应该放弃发送,直到您更了解或可以导入到`ord`钱包。" +"如果您的交易看起来与此不同,例如您有多个输入或多个输出,那么这可能不是一种安" +"全的铭文传输方式,您应该放弃发送,直到您更了解或可以导入到`ord`钱包。" #: src/guides/collecting/sparrow-wallet.md:149 msgid "" @@ -3839,22 +4024,24 @@ msgid "" "mempool.space) to see what the recommended fee rate is for sending a " "transaction." msgstr "" -"您应该设置合适的交易费用,Sparrow通常会推荐一个合理的费用," -"但您也可以查看[mempool.space](https://mempool.space) 以查看发送交易的推荐费率。" +"您应该设置合适的交易费用,Sparrow通常会推荐一个合理的费用,但您也可以查看" +"[mempool.space](https://mempool.space) 以查看发送交易的推荐费率。" #: src/guides/collecting/sparrow-wallet.md:151 msgid "" "You should add a label for the recipient address, a label like `alice " "address for inscription #123` would be ideal." msgstr "" -"您应该为收件人地址添加一个标签,如`alice address for inscription #123`就很理想。" +"您应该为收件人地址添加一个标签,如`alice address for inscription #123`就很理" +"想。" #: src/guides/collecting/sparrow-wallet.md:153 msgid "" "Once you have checked the transaction is a safe transaction using the checks " "above, and you are confident to send it you can click `Create Transaction`." msgstr "" -"在使用上述检查确认交易是安全的交易,并且有信心发送它后,您可以点击`Create Transaction`。" +"在使用上述检查确认交易是安全的交易,并且有信心发送它后,您可以点击`Create " +"Transaction`。" #: src/guides/collecting/sparrow-wallet.md:155 msgid "![](images/sending_03.png)" @@ -3865,7 +4052,8 @@ msgid "" "Here again you can double check that your transaction looks safe, and once " "you are confident you can click `Finalize Transaction for Signing`." msgstr "" -"在这里,您可以再次确认您的交易是否安全,在确认后,您可以点击`Finalize Transaction for Signing`。" +"在这里,您可以再次确认您的交易是否安全,在确认后,您可以点击`Finalize " +"Transaction for Signing`。" #: src/guides/collecting/sparrow-wallet.md:159 msgid "![](images/sending_04.png)" @@ -3898,8 +4086,8 @@ msgid "" "`Transaction Id (Txid)` and paste that into [mempool.space](https://mempool." "space)" msgstr "" -"如果你想跟踪你的交易状态,你可以复制`Transaction Id (Txid)`并粘贴到" -"[mempool.space](https://mempool.space)" +"如果你想跟踪你的交易状态,你可以复制`Transaction Id (Txid)`并粘贴到[mempool." +"space](https://mempool.space)" #: src/guides/collecting/sparrow-wallet.md:171 msgid "" @@ -3907,8 +4095,8 @@ msgid "" "[ordinals.com](https://ordinals.com) to validate that it has moved to the " "new output location and address." msgstr "" -"一旦交易确认,你可以在[ordinals.com](https://ordinals.com) " -"的铭文页面上验证它是否已移动到新的输出位置和地址。" +"一旦交易确认,你可以在[ordinals.com](https://ordinals.com) 的铭文页面上验证它" +"是否已移动到新的输出位置和地址。" #: src/guides/collecting/sparrow-wallet.md:173 msgid "Troubleshooting" @@ -3926,8 +4114,8 @@ msgid "" "head into the `Preferences`\\-> `Server` settings, and click `Edit Existing " "Connection`." msgstr "" -"确保你的钱包连接到一个比特币节点。要验证这一点,转到`Preferences`\\-> `Server` 设置," -"并点击 `Edit Existing Connection`。" +"确保你的钱包连接到一个比特币节点。要验证这一点,转到`Preferences`\\-> " +"`Server` 设置,并点击 `Edit Existing Connection`。" #: src/guides/collecting/sparrow-wallet.md:179 msgid "![](images/troubleshooting_01.png)" @@ -3937,7 +4125,9 @@ msgstr "" msgid "" "From there you can select a node and click `Test Connection` to validate " "that Sparrow is able to connect successfully." -msgstr "从那里你可以选择一个节点并点击 `Test Connection` 来验证Sparrow是否能够成功连接。" +msgstr "" +"从那里你可以选择一个节点并点击 `Test Connection` 来验证Sparrow是否能够成功连" +"接。" #: src/guides/collecting/sparrow-wallet.md:183 msgid "![](images/troubleshooting_02.png)" @@ -3950,15 +4140,17 @@ msgid "" "developer documentation](https://developer.bitcoin.org/examples/testing." "html)." msgstr "" -"使用以下标志来指定测试网络,可以测试 Ord。有关运行比特币核心进行测试的更多信息," -"请参见[比特币的开发者文档](https://developer.bitcoin.org/examples/testing。" +"使用以下标志来指定测试网络,可以测试 Ord。有关运行比特币核心进行测试的更多信" +"息,请参见[比特币的开发者文档](https://developer.bitcoin.org/examples/" +"testing。" #: src/guides/testing.md:7 msgid "" "Most `ord` commands in [inscriptions](inscriptions.md) and [explorer]" "(explorer.md) can be run with the following network flags:" msgstr "" -"大多数在[铭文](inscriptions.md) 和 [浏览器](explorer.md) 中的 `ord`命令可以使用以下网络标志运行:" +"大多数在[铭文](inscriptions.md) 和 [浏览器](explorer.md) 中的 `ord`命令可以使" +"用以下网络标志运行:" #: src/guides/testing.md:10 msgid "Network" @@ -3996,10 +4188,6 @@ msgstr "" msgid "Regtest doesn't require downloading the blockchain or indexing ord." msgstr "Regtest不需要下载区块链或者建立ord索引" -#: src/guides/testing.md:18 src/guides/reindexing.md:15 -msgid "Example" -msgstr "示例" - #: src/guides/testing.md:21 msgid "Run bitcoind in regtest with:" msgstr "在regtest里运行bitcoind,使用:" @@ -4086,8 +4274,8 @@ msgid "" "When testing out [recursion](../inscriptions/recursion.md), inscribe the " "dependencies first (example with [p5.js](https://p5js.org):" msgstr "" -"测试 [recursion](../inscriptions/recursion.md) 时,首先记下依赖项" -"(以 [p5.js](https://p5js.org) 为例:" +"测试 [recursion](../inscriptions/recursion.md) 时,首先记下依赖项(以 [p5.js]" +"(https://p5js.org) 为例:" #: src/guides/testing.md:55 msgid "" @@ -4100,15 +4288,15 @@ msgstr "" msgid "" "This should return a `inscription_id` which you can then reference in your " "recursive inscription." -msgstr "" -"这应该返回一个`inscription_id`,然后您可以在递归铭文中引用它。" +msgstr "这应该返回一个`inscription_id`,然后您可以在递归铭文中引用它。" #: src/guides/testing.md:61 msgid "" "ATTENTION: These ids will be different when inscribing on mainnet or signet, " "so be sure to change those in your recursive inscription for each chain." msgstr "" -"请注意,在主网和signet上铭刻的时候这些id有所不同,因此请务必更改每个链的递归铭文中的内容。" +"请注意,在主网和signet上铭刻的时候这些id有所不同,因此请务必更改每个链的递归" +"铭文中的内容。" #: src/guides/testing.md:65 msgid "Then you can inscribe your recursive inscription with:" @@ -4136,8 +4324,7 @@ msgstr "" #: src/guides/moderation.md:4 msgid "" "`ord` includes a block explorer, which you can run locally with `ord server`." -msgstr "" -"`ord` 包含了一个区块浏览器,你可以在本地运行`ord server`." +msgstr "`ord` 包含了一个区块浏览器,你可以在本地运行`ord server`." #: src/guides/moderation.md:6 msgid "" @@ -4153,7 +4340,8 @@ msgid "" "unlawful content, and decide what moderation policy is appropriate for their " "instance." msgstr "" -"运行ord区块浏览器实例的每个人都有责任了解他们对非法内容的责任,并决定适合他们实例的审核政策。" +"运行ord区块浏览器实例的每个人都有责任了解他们对非法内容的责任,并决定适合他们" +"实例的审核政策。" #: src/guides/moderation.md:13 msgid "" @@ -4161,15 +4349,14 @@ msgid "" "instance, they can be included in a YAML config file, which is loaded with " "the `--config` option." msgstr "" -"为了防止特定的铭文显示在`ord`实例上,它们可以包含在 YAML 配置文件中," -"该文件使用 `--config`选项加载。" +"为了防止特定的铭文显示在`ord`实例上,它们可以包含在 YAML 配置文件中,该文件使" +"用 `--config`选项加载。" #: src/guides/moderation.md:17 msgid "" "To hide inscriptions, first create a config file, with the inscription ID " "you want to hide:" -msgstr "" -"要隐藏铭文,首先创建一个配置文件,其中包含要隐藏的铭文 ID:" +msgstr "要隐藏铭文,首先创建一个配置文件,其中包含要隐藏的铭文 ID:" #: src/guides/moderation.md:20 msgid "" @@ -4183,13 +4370,11 @@ msgstr "" msgid "" "The suggested name for `ord` config files is `ord.yaml`, but any filename " "can be used." -msgstr "" -"`ord` 配置文件的建议名称是 `ord.yaml`,但可以使用任何文件名。" +msgstr "`ord` 配置文件的建议名称是 `ord.yaml`,但可以使用任何文件名。" #: src/guides/moderation.md:28 msgid "Then pass the file to `--config` when starting the server:" -msgstr "" -"然后将文件在服务启动的使用使用 `--config` :" +msgstr "然后将文件在服务启动的使用使用 `--config` :" #: src/guides/moderation.md:30 msgid "`ord --config ord.yaml server`" @@ -4199,13 +4384,11 @@ msgstr "" msgid "" "Note that the `--config` option comes after `ord` but before the `server` " "subcommand." -msgstr "" -"请注意, `--config` 选项的位置在 `ord` 之后但是在 `server`子命令前。" +msgstr "请注意, `--config` 选项的位置在 `ord` 之后但是在 `server`子命令前。" #: src/guides/moderation.md:35 msgid "`ord` must be restarted in to load changes to the config file." -msgstr "" -"`ord` 必须重启才可以加载在配置文件中的更改。" +msgstr "`ord` 必须重启才可以加载在配置文件中的更改。" #: src/guides/moderation.md:37 msgid "`ordinals.com`" @@ -4216,8 +4399,8 @@ msgid "" "The `ordinals.com` instances use `systemd` to run the `ord server` service, " "which is called `ord`, with a config file located at `/var/lib/ord/ord.yaml`." msgstr "" -"`ordinals.com` 实例使用 `systemd` 运行名为 `ord`的 `ord server` 服务," -"配置文件在 `/var/lib/ord/ord.yaml`." +"`ordinals.com` 实例使用 `systemd` 运行名为 `ord`的 `ord server` 服务,配置文" +"件在 `/var/lib/ord/ord.yaml`." #: src/guides/moderation.md:43 msgid "To hide an inscription on `ordinals.com`:" @@ -4251,8 +4434,8 @@ msgid "" "database and restarting the indexing process with either `ord index run` or " "`ord server`. Reasons to reindex are:" msgstr "" -"有时必须重新索引‘ord’数据库,这意味着删除数据库并使用 `ord index run`或`ord server`" -"来重新索引数据库。重新索引的原因是:" +"有时必须重新索引‘ord’数据库,这意味着删除数据库并使用 `ord index run`或`ord " +"server`来重新索引数据库。重新索引的原因是:" #: src/guides/reindexing.md:8 msgid "A new major release of ord, which changes the database scheme" @@ -4268,9 +4451,9 @@ msgid "" "so we give the index the default file name `index.redb`. By default we store " "this file in different locations depending on your operating system." msgstr "" -"`ord` 使用的数据库称为 [redb](https://github.com/cberner/redb)," -"所以我们为索引指定默认文件名‘index.redb’。默认情况下我们存储" -"根据您的操作系统,此文件位于不同的位置。" +"`ord` 使用的数据库称为 [redb](https://github.com/cberner/redb),所以我们为索" +"引指定默认文件名‘index.redb’。默认情况下我们存储根据您的操作系统,此文件位于" +"不同的位置。" #: src/guides/reindexing.md:15 msgid "Platform" @@ -4320,8 +4503,7 @@ msgstr "" msgid "" "So to delete the database and reindex on MacOS you would have to run the " "following commands in the terminal:" -msgstr "" -"因此,要在 MacOS 上删除数据库并重新索引,您必须在终端中执行以下命令:" +msgstr "因此,要在 MacOS 上删除数据库并重新索引,您必须在终端中执行以下命令:" #: src/guides/reindexing.md:24 msgid "" @@ -4337,8 +4519,8 @@ msgid "" "`ord --data-dir index run` or give it a specific filename and path " "with `ord --index index run`." msgstr "" -"您当然也可以自己设置数据目录的位置,`ord --data-dir index run` 或" -"为其指定特定的文件名和路径,使用‘ord --index 索引运行’。" +"您当然也可以自己设置数据目录的位置,`ord --data-dir index run` 或为其指" +"定特定的文件名和路径,使用‘ord --index 索引运行’。" #: src/bounties.md:1 msgid "Ordinal Bounty Hunting Hints" @@ -4350,8 +4532,8 @@ msgid "" "ordinal theory is extremely simple. A clever hacker should be able to write " "code from scratch to manipulate satoshis using ordinal theory in no time." msgstr "" -"`ord` 钱包可以发送和接收特定的聪。此外序数理论非常简单。聪明的黑客应该能够很快的从头开始" -"编写代码,使用序数理论来操作聪;" +"`ord` 钱包可以发送和接收特定的聪。此外序数理论非常简单。聪明的黑客应该能够很" +"快的从头开始编写代码,使用序数理论来操作聪;" #: src/bounties.md:8 msgid "" @@ -4360,9 +4542,9 @@ msgid "" "mediawiki) for the technical details, and the [ord repo](https://github.com/" "ordinals/ord) for the `ord` wallet and block explorer." msgstr "" -"关于序数理论的更多信息,请查阅[FAQ](./faq.md) 来获取概述;" -"查阅[BIP](https://github.com/ordinals/ord/blob/master/bip.mediawiki) 来获取技术细节" -"查阅[ord repo](https://github.com/ordinals/ord)来获取`ord`钱包和浏览器的信息." +"关于序数理论的更多信息,请查阅[FAQ](./faq.md) 来获取概述;查阅[BIP](https://" +"github.com/ordinals/ord/blob/master/bip.mediawiki) 来获取技术细节查阅[ord " +"repo](https://github.com/ordinals/ord)来获取`ord`钱包和浏览器的信息." #: src/bounties.md:14 msgid "" @@ -4371,8 +4553,9 @@ msgid "" "and it was lost to the sands of time. This potent theory is only now being " "rediscovered. You can help by researching rare satoshis." msgstr "" -"中本聪是序数理论的原始开发者。然而,他知道其他人可能会认为这是异端邪说并且危险,因此他隐藏了自己的" -"知识,使其在时间的沙漠里失传。现在,这个强大的理论被重新发现。您可以通过研究稀有的聪来帮助我们。" +"中本聪是序数理论的原始开发者。然而,他知道其他人可能会认为这是异端邪说并且危" +"险,因此他隐藏了自己的知识,使其在时间的沙漠里失传。现在,这个强大的理论被重" +"新发现。您可以通过研究稀有的聪来帮助我们。" #: src/bounties.md:19 msgid "Good luck and godspeed!" @@ -4430,8 +4613,7 @@ msgid "" "Claimed by [@count_null](https://twitter.com/rodarmor/" "status/1560793241473400833)!" msgstr "" -"[@count_null](https://twitter.com/rodarmor/" -"status/1560793241473400833)! 赢得" +"[@count_null](https://twitter.com/rodarmor/status/1560793241473400833)! 赢得" #: src/bounty/1.md:1 msgid "Ordinal Bounty 1" @@ -4442,7 +4624,9 @@ msgid "" "The transaction that submits a UTXO containing the oldest sat, i.e., that " "with the lowest number, amongst all submitted UTXOs will be judged the " "winner." -msgstr "提交一个包含最古老的聪的UTXO,譬如在所有提交的UTXO中,最小的数字将被判定为获胜者;" +msgstr "" +"提交一个包含最古老的聪的UTXO,譬如在所有提交的UTXO中,最小的数字将被判定为获" +"胜者;" #: src/bounty/1.md:10 msgid "" @@ -4450,8 +4634,8 @@ msgid "" "difficulty adjustment period 374. Submissions included in block 753984 or " "later will not be considered." msgstr "" -"赏金在区块高度 753984 前有效,区块高度753984是第一个难度调整期374后的第一个区块。" -"包含或者晚于区块高度 753984 的,将不会被考虑。" +"赏金在区块高度 753984 前有效,区块高度753984是第一个难度调整期374后的第一个区" +"块。包含或者晚于区块高度 753984 的,将不会被考虑。" #: src/bounty/1.md:17 msgid "200,000 sats" @@ -4468,8 +4652,8 @@ msgid "" "Claimed by [@ordinalsindex](https://twitter.com/rodarmor/" "status/1569883266508853251)!" msgstr "" -"由 [@ordinalsindex](https://twitter.com/rodarmor/" -"status/1569883266508853251)赢得!" +"由 [@ordinalsindex](https://twitter.com/rodarmor/status/1569883266508853251)" +"赢得!" #: src/bounty/2.md:1 msgid "Ordinal Bounty 2" @@ -4500,7 +4684,8 @@ msgid "" "Confirm that the submission address has not received transactions before " "submitting your entry. Only the first successful submission will be rewarded." msgstr "" -"在提交之前确认上述地址并未在你之前收到其他的稀有聪,只有第一个成功的提交可以获得奖励;" +"在提交之前确认上述地址并未在你之前收到其他的稀有聪,只有第一个成功的提交可以" +"获得奖励;" #: src/bounty/2.md:18 msgid "300,000 sats" @@ -4517,8 +4702,7 @@ msgid "" "Claimed by [@utxoset](https://twitter.com/rodarmor/" "status/1582424455615172608)!" msgstr "" -"由[@utxoset](https://twitter.com/rodarmor/" -"status/1582424455615172608) 赢得!" +"由[@utxoset](https://twitter.com/rodarmor/status/1582424455615172608) 赢得!" #: src/bounty/3.md:1 msgid "Ordinal Bounty 3" @@ -4533,10 +4717,10 @@ msgid "" "sat 0, the first sat to be mined is `nvtdijuwxlp` and the name of sat " "2,099,999,997,689,999, the last sat to be mined, is `a`." msgstr "" -"任务3有两个部分,都是基于_序数名字_" -"序数名字是把序数数字用修改后的base-26进行的编码.为了避免将短名字锁定在不可花费的创世区块奖励中," -"随着序数的_变长_,序数名字将变得_更短_ 比如第一个开采的0号聪的名字是`nvtdijuwxlp`," -"而最后一个被开采的2,099,999,997,689,999号聪的名字,则是 `a`." +"任务3有两个部分,都是基于_序数名字_序数名字是把序数数字用修改后的base-26进行" +"的编码.为了避免将短名字锁定在不可花费的创世区块奖励中,随着序数的_变长_,序数" +"名字将变得_更短_ 比如第一个开采的0号聪的名字是`nvtdijuwxlp`,而最后一个被开采" +"的2,099,999,997,689,999号聪的名字,则是 `a`." #: src/bounty/3.md:14 msgid "" @@ -4544,7 +4728,8 @@ msgid "" "the fourth halvening. Submissions included in block 840000 or later will not " "be considered." msgstr "" -"赏金计划开放到区块高度840000-第四次减半后的第一个区块。区块高度840000以及以后的区块将不被考虑。" +"赏金计划开放到区块高度840000-第四次减半后的第一个区块。区块高度840000以及以后" +"的区块将不被考虑。" #: src/bounty/3.md:18 msgid "" @@ -4554,9 +4739,10 @@ msgid "" "include the names of sats which will have been mined by the end of the " "submission period, that appear at least 5000 times in the corpus." msgstr "" -"两个部分任务都使用 [frequency.tsv](frequency.tsv), 一个单词的清单以及他们" -"在 [Google Books Ngram dataset](http://storage.googleapis.com/books/ngrams/books/datasetsv2.html)" -"中出现的次数。过滤后仅包含在提交期结束时能被挖掘的聪的名字,这些名称在语料库中出现至少5000次。" +"两个部分任务都使用 [frequency.tsv](frequency.tsv), 一个单词的清单以及他们在 " +"[Google Books Ngram dataset](http://storage.googleapis.com/books/ngrams/" +"books/datasetsv2.html)中出现的次数。过滤后仅包含在提交期结束时能被挖掘的聪的" +"名字,这些名称在语料库中出现至少5000次。" #: src/bounty/3.md:24 msgid "" @@ -4565,24 +4751,24 @@ msgid "" "entries are sorted from least-frequently occurring to most-frequently " "occurring." msgstr "" -"`frequency.tsv` 制表符分割值的文件,第一列是单词,第二列是它在语料库里出现的次数。" -"这些条目从出现频率最低到出现频率最高的顺序进行排序。" +"`frequency.tsv` 制表符分割值的文件,第一列是单词,第二列是它在语料库里出现的" +"次数。这些条目从出现频率最低到出现频率最高的顺序进行排序。" #: src/bounty/3.md:29 msgid "" "`frequency.tsv` was compiled using [this program](https://github.com/casey/" "onegrams)." msgstr "" -"`frequency.tsv` 使用了[这个程序](https://github.com/casey/onegrams)进行的编译." - +"`frequency.tsv` 使用了[这个程序](https://github.com/casey/onegrams)进行的编" +"译." #: src/bounty/3.md:32 msgid "" "To search an `ord` wallet for sats with a name in `frequency.tsv`, use the " "following [`ord`](https://github.com/ordinals/ord) command:" msgstr "" -"在`ord`钱包里搜索`frequency.tsv`中所包含的聪的名字 , 使用下面的" -"[`ord`](https://github.com/ordinals/ord)命令: " +"在`ord`钱包里搜索`frequency.tsv`中所包含的聪的名字 , 使用下面的[`ord`]" +"(https://github.com/ordinals/ord)命令: " #: src/bounty/3.md:35 msgid "" @@ -4612,7 +4798,8 @@ msgid "" "with the lowest number of occurrences in `frequency.tsv` shall be the winner " "of part 0." msgstr "" -"提交的交易UTXO中包含的聪的名字,是`frequency.tsv`中出现的最低的频率者,即是第0部分的获胜者。" +"提交的交易UTXO中包含的聪的名字,是`frequency.tsv`中出现的最低的频率者,即是第" +"0部分的获胜者。" #: src/bounty/3.md:50 msgid "Part 1" @@ -4622,14 +4809,14 @@ msgstr "第1部分" msgid "_Popularity is the font of value._" msgstr "_人气是价值的源泉_" - #: src/bounty/3.md:54 msgid "" "The transaction that submits the UTXO containing the sat whose name appears " "with the highest number of occurrences in `frequency.tsv` shall be the " "winner of part 1." msgstr "" -"提交的交易UTXO中包含的聪的名字,是`frequency.tsv`中出现的最高的频率者,是第 1 部分的获胜者。" +"提交的交易UTXO中包含的聪的名字,是`frequency.tsv`中出现的最高的频率者,是第 " +"1 部分的获胜者。" #: src/bounty/3.md:58 msgid "Tie Breaking" diff --git a/docs/theme/index.hbs b/docs/theme/index.hbs index 2ada9ad874..04e3db03f7 100644 --- a/docs/theme/index.hbs +++ b/docs/theme/index.hbs @@ -175,6 +175,9 @@
  3. +