diff --git a/examples/abi/Cargo.lock b/examples/abi/Cargo.lock new file mode 100644 index 000000000..804319b8f --- /dev/null +++ b/examples/abi/Cargo.lock @@ -0,0 +1,1242 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 3 + +[[package]] +name = "Inflector" +version = "0.11.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fe438c63458706e03479442743baae6c88256498e6431708f6dfc520a26515d3" + +[[package]] +name = "abi" +version = "0.1.0" +dependencies = [ + "near-sdk", + "schemars", + "serde", +] + +[[package]] +name = "ahash" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fcb51a0695d8f838b1ee009b3fbf66bda078cd64590202a864a8f3e8c4315c47" +dependencies = [ + "getrandom 0.2.6", + "once_cell", + "version_check", +] + +[[package]] +name = "arrayref" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4c527152e37cf757a3f78aae5a06fbeefdb07ccc535c980a3208ee3060dd544" + +[[package]] +name = "arrayvec" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23b62fc65de8e4e7f52534fb52b0f3ed04746ae267519eef2a83941e8085068b" + +[[package]] +name = "arrayvec" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8da52d66c7071e2e3fa2a1e5c6d088fec47b593032b254f5e980de8ea54454d6" + +[[package]] +name = "autocfg" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" + +[[package]] +name = "base64" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b41b7ea54a0c9d92199de89e20e58d49f02f8e699814ef3fdf266f6f748d15c7" + +[[package]] +name = "base64" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "904dfeac50f3cdaba28fc6f57fdcddb75f49ed61346676a78c4ffe55877802fd" + +[[package]] +name = "bitvec" +version = "0.20.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7774144344a4faa177370406a7ff5f1da24303817368584c6206c8303eb07848" +dependencies = [ + "funty", + "radium", + "tap", + "wyz", +] + +[[package]] +name = "blake2" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a4e37d16930f5459780f5621038b6382b9bb37c19016f39fb6b5808d831f174" +dependencies = [ + "crypto-mac", + "digest 0.9.0", + "opaque-debug", +] + +[[package]] +name = "block-buffer" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4" +dependencies = [ + "generic-array", +] + +[[package]] +name = "block-buffer" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bf7fe51849ea569fd452f37822f606a5cabb684dc918707a0193fd4664ff324" +dependencies = [ + "generic-array", +] + +[[package]] +name = "borsh" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15bf3650200d8bffa99015595e10f1fbd17de07abbc25bb067da79e769939bfa" +dependencies = [ + "borsh-derive", + "hashbrown", +] + +[[package]] +name = "borsh-derive" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6441c552f230375d18e3cc377677914d2ca2b0d36e52129fe15450a2dce46775" +dependencies = [ + "borsh-derive-internal", + "borsh-schema-derive-internal", + "proc-macro-crate 0.1.5", + "proc-macro2", + "syn", +] + +[[package]] +name = "borsh-derive-internal" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5449c28a7b352f2d1e592a8a28bf139bc71afb0764a14f3c02500935d8c44065" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "borsh-schema-derive-internal" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdbd5696d8bfa21d53d9fe39a714a18538bad11492a42d066dbbc395fb1951c0" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "bs58" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "771fe0050b883fcc3ea2359b1a96bcfbc090b7116eae7c3c512c7a083fdf23d3" + +[[package]] +name = "byte-slice-cast" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87c5fdd0166095e1d463fc6cc01aa8ce547ad77a4e84d42eb6762b084e28067e" + +[[package]] +name = "byteorder" +version = "1.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" + +[[package]] +name = "bytesize" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c58ec36aac5066d5ca17df51b3e70279f5670a72102f5752cb7e7c856adfc70" + +[[package]] +name = "c2-chacha" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d27dae93fe7b1e0424dc57179ac396908c26b035a87234809f5c4dfd1b47dc80" +dependencies = [ + "cipher", + "ppv-lite86", +] + +[[package]] +name = "cc" +version = "1.0.73" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2fff2a6927b3bb87f9595d67196a70493f627687a71d87a0d692242c33f58c11" + +[[package]] +name = "cfg-if" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4785bdd1c96b2a846b2bd7cc02e86b6b3dbf14e7e53446c4f54c92a361040822" + +[[package]] +name = "cfg-if" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" + +[[package]] +name = "chrono" +version = "0.4.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "670ad68c9088c2a963aaa298cb369688cf3f9465ce5e2d4ca10e6e0098a1ce73" +dependencies = [ + "libc", + "num-integer", + "num-traits", + "serde", + "time", + "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 = "convert_case" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" + +[[package]] +name = "cpufeatures" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "59a6001667ab124aebae2a495118e11d30984c3a653e99d86d58971708cf5e4b" +dependencies = [ + "libc", +] + +[[package]] +name = "crunchy" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" + +[[package]] +name = "crypto-common" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57952ca27b5e3606ff4dd79b0020231aaf9d6aa76dc05fd30137538c50bd3ce8" +dependencies = [ + "generic-array", + "typenum", +] + +[[package]] +name = "crypto-mac" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b584a330336237c1eecd3e94266efb216c56ed91225d634cb2991c5f3fd1aeab" +dependencies = [ + "generic-array", + "subtle", +] + +[[package]] +name = "curve25519-dalek" +version = "3.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90f9d052967f590a76e62eb387bd0bbb1b000182c3cefe5364db6b7211651bc0" +dependencies = [ + "byteorder", + "digest 0.9.0", + "rand_core 0.5.1", + "subtle", + "zeroize", +] + +[[package]] +name = "derive_more" +version = "0.99.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4fb810d30a7c1953f91334de7244731fc3f3c10d7fe163338a35b9f640960321" +dependencies = [ + "convert_case", + "proc-macro2", + "quote", + "rustc_version", + "syn", +] + +[[package]] +name = "digest" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066" +dependencies = [ + "generic-array", +] + +[[package]] +name = "digest" +version = "0.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2fb860ca6fafa5552fb6d0e816a69c8e49f0908bf524e30a90d97c85892d506" +dependencies = [ + "block-buffer 0.10.2", + "crypto-common", +] + +[[package]] +name = "dyn-clone" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21e50f3adc76d6a43f5ed73b698a87d0760ca74617f60f7c3b879003536fdd28" + +[[package]] +name = "easy-ext" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53aff6fdc1b181225acdcb5b14c47106726fd8e486707315b1b138baed68ee31" + +[[package]] +name = "ed25519" +version = "1.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e9c280362032ea4203659fc489832d0204ef09f247a0506f170dafcac08c369" +dependencies = [ + "signature", +] + +[[package]] +name = "ed25519-dalek" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d" +dependencies = [ + "curve25519-dalek", + "ed25519", + "rand 0.7.3", + "serde", + "sha2 0.9.9", + "zeroize", +] + +[[package]] +name = "fixed-hash" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cfcf0ed7fe52a17a03854ec54a9f76d6d84508d1c0e66bc1793301c73fc8493c" +dependencies = [ + "byteorder", + "rand 0.8.5", + "rustc-hex", + "static_assertions", +] + +[[package]] +name = "funty" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fed34cd105917e91daa4da6b3728c47b068749d6a62c59811f06ed2ac71d9da7" + +[[package]] +name = "generic-array" +version = "0.14.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd48d33ec7f05fbfa152300fdad764757cbded343c1aa1cff2fbaf4134851803" +dependencies = [ + "typenum", + "version_check", +] + +[[package]] +name = "getrandom" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" +dependencies = [ + "cfg-if 1.0.0", + "libc", + "wasi 0.9.0+wasi-snapshot-preview1", +] + +[[package]] +name = "getrandom" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9be70c98951c83b8d2f8f60d7065fa6d5146873094452a1008da8c2f1e4205ad" +dependencies = [ + "cfg-if 1.0.0", + "libc", + "wasi 0.10.2+wasi-snapshot-preview1", +] + +[[package]] +name = "hashbrown" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e" +dependencies = [ + "ahash", +] + +[[package]] +name = "heck" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2540771e65fc8cb83cd6e8a237f70c319bd5c29f78ed1084ba5d50eeac86f7f9" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "impl-codec" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "161ebdfec3c8e3b52bf61c4f3550a1eea4f9579d10dc1b936f3171ebdcd6c443" +dependencies = [ + "parity-scale-codec", +] + +[[package]] +name = "impl-trait-for-tuples" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11d7a9f6330b71fea57921c9b61c47ee6e84f72d394754eff6163ae67e7395eb" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "itoa" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "112c678d4050afce233f4f2852bb2eb519230b3cf12f33585275537d7e41578d" + +[[package]] +name = "keccak" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9b7d56ba4a8344d6be9729995e6b06f928af29998cdf79fe390cbf6b1fee838" + +[[package]] +name = "libc" +version = "0.2.126" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "349d5a591cd28b49e1d1037471617a32ddcda5731b99419008085f72d5a53836" + +[[package]] +name = "memory_units" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8452105ba047068f40ff7093dd1d9da90898e63dd61736462e9cdda6a90ad3c3" + +[[package]] +name = "near-account-id" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de83d74a9241be8cc4eb3055216966b58bf8c463e8e285c0dc553925acdd19fa" +dependencies = [ + "borsh", + "serde", +] + +[[package]] +name = "near-crypto" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8ecf0b8b31aa7f4e60f629f72213a2617ca4a5f45cd1ae9ed2cf7cecfebdbb7" +dependencies = [ + "arrayref", + "blake2", + "borsh", + "bs58", + "c2-chacha", + "curve25519-dalek", + "derive_more", + "ed25519-dalek", + "libc", + "near-account-id", + "once_cell", + "parity-secp256k1", + "primitive-types", + "rand 0.7.3", + "rand_core 0.5.1", + "serde", + "serde_json", + "subtle", + "thiserror", +] + +[[package]] +name = "near-primitives" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a2ba19282e79a4485a77736b679d276b09870bbf8042a18e0f0ae36347489c5" +dependencies = [ + "borsh", + "byteorder", + "bytesize", + "chrono", + "derive_more", + "easy-ext", + "hex", + "near-crypto", + "near-primitives-core", + "near-rpc-error-macro", + "near-vm-errors", + "num-rational", + "once_cell", + "primitive-types", + "rand 0.7.3", + "reed-solomon-erasure", + "serde", + "serde_json", + "smart-default", + "strum", + "thiserror", +] + +[[package]] +name = "near-primitives-core" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb561feb392bb8c4f540256073446e6689af087bf6356e8dddcf75fc279f201f" +dependencies = [ + "base64 0.11.0", + "borsh", + "bs58", + "derive_more", + "near-account-id", + "num-rational", + "serde", + "sha2 0.10.2", + "strum", +] + +[[package]] +name = "near-rpc-error-core" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77fdd7ea8d8f786878651c37691515d5053f827ae60894aa40c16882b78f77c9" +dependencies = [ + "quote", + "serde", + "syn", +] + +[[package]] +name = "near-rpc-error-macro" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e521842b6ae864dfe5391afbbe2df9e9d8427c26e9333b2e0b65cd42094f7607" +dependencies = [ + "near-rpc-error-core", + "serde", + "syn", +] + +[[package]] +name = "near-sdk" +version = "4.0.0" +dependencies = [ + "base64 0.13.0", + "borsh", + "bs58", + "near-crypto", + "near-primitives", + "near-primitives-core", + "near-sdk-macros", + "near-sys", + "near-vm-logic", + "schemars", + "serde", + "serde_json", + "wee_alloc", +] + +[[package]] +name = "near-sdk-macros" +version = "4.0.0" +dependencies = [ + "Inflector", + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "near-sys" +version = "0.2.0" + +[[package]] +name = "near-vm-errors" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e02faf2bc1f6ef82b965cfe44389808fb5594f7aca4b596766117f4ce74df20" +dependencies = [ + "borsh", + "near-account-id", + "near-rpc-error-macro", + "serde", +] + +[[package]] +name = "near-vm-logic" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f024d90451cd3c24d7a0a5cabf3636b192a60eb8e3ff0456f6c18b91152c346d" +dependencies = [ + "base64 0.13.0", + "borsh", + "bs58", + "byteorder", + "near-account-id", + "near-crypto", + "near-primitives", + "near-primitives-core", + "near-vm-errors", + "ripemd", + "serde", + "sha2 0.10.2", + "sha3", +] + +[[package]] +name = "num-bigint" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f6f7833f2cbf2360a6cfd58cd41a53aa7a90bd4c202f5b1c7dd2ed73c57b2c3" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-integer" +version = "0.1.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "225d3389fb3509a24c93f5c29eb6bde2586b98d9f016636dff58d7c6f7569cd9" +dependencies = [ + "autocfg", + "num-traits", +] + +[[package]] +name = "num-rational" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "12ac428b1cb17fce6f731001d307d351ec70a6d202fc2e60f7d4c5e42d8f4f07" +dependencies = [ + "autocfg", + "num-bigint", + "num-integer", + "num-traits", + "serde", +] + +[[package]] +name = "num-traits" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "578ede34cf02f8924ab9447f50c28075b4d3e5b269972345e7e0372b38c6cdcd" +dependencies = [ + "autocfg", +] + +[[package]] +name = "once_cell" +version = "1.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7709cef83f0c1f58f666e746a08b21e0085f7440fa6a29cc194d68aac97a4225" + +[[package]] +name = "opaque-debug" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" + +[[package]] +name = "parity-scale-codec" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "373b1a4c1338d9cd3d1fa53b3a11bdab5ab6bd80a20f7f7becd76953ae2be909" +dependencies = [ + "arrayvec 0.7.2", + "bitvec", + "byte-slice-cast", + "impl-trait-for-tuples", + "parity-scale-codec-derive", + "serde", +] + +[[package]] +name = "parity-scale-codec-derive" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1557010476e0595c9b568d16dcfb81b93cdeb157612726f5170d31aa707bed27" +dependencies = [ + "proc-macro-crate 1.1.3", + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "parity-secp256k1" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4fca4f82fccae37e8bbdaeb949a4a218a1bbc485d11598f193d2a908042e5fc1" +dependencies = [ + "arrayvec 0.5.2", + "cc", + "cfg-if 0.1.10", + "rand 0.7.3", +] + +[[package]] +name = "ppv-lite86" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872" + +[[package]] +name = "primitive-types" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05e4722c697a58a99d5d06a08c30821d7c082a4632198de1eaa5a6c22ef42373" +dependencies = [ + "fixed-hash", + "impl-codec", + "uint", +] + +[[package]] +name = "proc-macro-crate" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d6ea3c4595b96363c13943497db34af4460fb474a95c43f4446ad341b8c9785" +dependencies = [ + "toml", +] + +[[package]] +name = "proc-macro-crate" +version = "1.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e17d47ce914bf4de440332250b0edd23ce48c005f59fab39d3335866b114f11a" +dependencies = [ + "thiserror", + "toml", +] + +[[package]] +name = "proc-macro2" +version = "1.0.39" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c54b25569025b7fc9651de43004ae593a75ad88543b17178aa5e1b9c4f15f56f" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "quote" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "radium" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "643f8f41a8ebc4c5dc4515c82bb8abd397b527fc20fd681b7c011c2aee5d44fb" + +[[package]] +name = "rand" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03" +dependencies = [ + "getrandom 0.1.16", + "libc", + "rand_chacha 0.2.2", + "rand_core 0.5.1", + "rand_hc", +] + +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha 0.3.1", + "rand_core 0.6.3", +] + +[[package]] +name = "rand_chacha" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402" +dependencies = [ + "ppv-lite86", + "rand_core 0.5.1", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core 0.6.3", +] + +[[package]] +name = "rand_core" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" +dependencies = [ + "getrandom 0.1.16", +] + +[[package]] +name = "rand_core" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7" +dependencies = [ + "getrandom 0.2.6", +] + +[[package]] +name = "rand_hc" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c" +dependencies = [ + "rand_core 0.5.1", +] + +[[package]] +name = "reed-solomon-erasure" +version = "4.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a415a013dd7c5d4221382329a5a3482566da675737494935cbbbcdec04662f9d" +dependencies = [ + "smallvec", +] + +[[package]] +name = "ripemd" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1facec54cb5e0dc08553501fa740091086d0259ad0067e0d4103448e4cb22ed3" +dependencies = [ + "digest 0.10.3", +] + +[[package]] +name = "rustc-hex" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e75f6a532d0fd9f7f13144f392b6ad56a32696bfcd9c78f797f16bbb6f072d6" + +[[package]] +name = "rustc_version" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa0f585226d2e68097d4f95d113b15b83a82e819ab25717ec0590d9584ef366" +dependencies = [ + "semver", +] + +[[package]] +name = "rustversion" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2cc38e8fa666e2de3c4aba7edeb5ffc5246c1c2ed0e3d17e560aeeba736b23f" + +[[package]] +name = "ryu" +version = "1.0.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3f6f92acf49d1b98f7a81226834412ada05458b7364277387724a237f062695" + +[[package]] +name = "schemars" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1847b767a3d62d95cbf3d8a9f0e421cf57a0d8aa4f411d4b16525afb0284d4ed" +dependencies = [ + "dyn-clone", + "schemars_derive", + "serde", + "serde_json", +] + +[[package]] +name = "schemars_derive" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af4d7e1b012cb3d9129567661a63755ea4b8a7386d339dc945ae187e403c6743" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn", +] + +[[package]] +name = "semver" +version = "1.0.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8cb243bdfdb5936c8dc3c45762a19d12ab4550cdc753bc247637d4ec35a040fd" + +[[package]] +name = "serde" +version = "1.0.137" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61ea8d54c77f8315140a05f4c7237403bf38b72704d031543aa1d16abbf517d1" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde_derive" +version = "1.0.137" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f26faba0c3959972377d3b2d306ee9f71faee9714294e41bb777f83f88578be" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "serde_derive_internals" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85bf8229e7920a9f636479437026331ce11aa132b4dde37d121944a44d6e5f3c" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "serde_json" +version = "1.0.81" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b7ce2b32a1aed03c558dc61a5cd328f15aff2dbc17daad8fb8af04d2100e15c" +dependencies = [ + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "sha2" +version = "0.9.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800" +dependencies = [ + "block-buffer 0.9.0", + "cfg-if 1.0.0", + "cpufeatures", + "digest 0.9.0", + "opaque-debug", +] + +[[package]] +name = "sha2" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55deaec60f81eefe3cce0dc50bda92d6d8e88f2a27df7c5033b42afeb1ed2676" +dependencies = [ + "cfg-if 1.0.0", + "cpufeatures", + "digest 0.10.3", +] + +[[package]] +name = "sha3" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "881bf8156c87b6301fc5ca6b27f11eeb2761224c7081e69b409d5a1951a70c86" +dependencies = [ + "digest 0.10.3", + "keccak", +] + +[[package]] +name = "signature" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f054c6c1a6e95179d6f23ed974060dcefb2d9388bb7256900badad682c499de4" + +[[package]] +name = "smallvec" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2dd574626839106c320a323308629dcb1acfc96e32a8cba364ddc61ac23ee83" + +[[package]] +name = "smart-default" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "133659a15339456eeeb07572eb02a91c91e9815e9cbc89566944d2c8d3efdbf6" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "static_assertions" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" + +[[package]] +name = "strum" +version = "0.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e96acfc1b70604b8b2f1ffa4c57e59176c7dbb05d556c71ecd2f5498a1dee7f8" +dependencies = [ + "strum_macros", +] + +[[package]] +name = "strum_macros" +version = "0.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6878079b17446e4d3eba6192bb0a2950d5b14f0ed8424b852310e5a94345d0ef" +dependencies = [ + "heck", + "proc-macro2", + "quote", + "rustversion", + "syn", +] + +[[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.96" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0748dd251e24453cb8717f0354206b91557e4ec8703673a4b30208f2abaf1ebf" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "synstructure" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f36bdaa60a83aca3921b5259d5400cbf5e90fc51931376a9bd4a0eb79aa7210f" +dependencies = [ + "proc-macro2", + "quote", + "syn", + "unicode-xid", +] + +[[package]] +name = "tap" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369" + +[[package]] +name = "thiserror" +version = "1.0.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd829fe32373d27f76265620b5309d0340cb8550f523c1dda251d6298069069a" +dependencies = [ + "thiserror-impl", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0396bc89e626244658bef819e22d0cc459e795a5ebe878e6ec336d1674a8d79a" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "time" +version = "0.1.43" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca8a50ef2360fbd1eeb0ecd46795a87a19024eb4b53c5dc916ca1fd95fe62438" +dependencies = [ + "libc", + "winapi", +] + +[[package]] +name = "toml" +version = "0.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8d82e1a7758622a465f8cee077614c73484dac5b836c02ff6a40d5d1010324d7" +dependencies = [ + "serde", +] + +[[package]] +name = "typenum" +version = "1.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dcf81ac59edc17cc8697ff311e8f5ef2d99fcbd9817b34cec66f90b6c3dfd987" + +[[package]] +name = "uint" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "12f03af7ccf01dd611cc450a0d10dbc9b745770d096473e2faf0ca6e2d66d1e0" +dependencies = [ + "byteorder", + "crunchy", + "hex", + "static_assertions", +] + +[[package]] +name = "unicode-ident" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d22af068fba1eb5edcb4aea19d382b2a3deb4c8f9d475c589b6ada9e0fd493ee" + +[[package]] +name = "unicode-xid" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "957e51f3646910546462e67d5f7599b9e4fb8acdd304b087a6494730f9eebf04" + +[[package]] +name = "version_check" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" + +[[package]] +name = "wasi" +version = "0.9.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" + +[[package]] +name = "wasi" +version = "0.10.2+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6" + +[[package]] +name = "wee_alloc" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dbb3b5a6b2bb17cb6ad44a2e68a43e8d2722c997da10e928665c72ec6c0a0b8e" +dependencies = [ + "cfg-if 0.1.10", + "libc", + "memory_units", + "winapi", +] + +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + +[[package]] +name = "wyz" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85e60b0d1b5f99db2556934e21937020776a5d31520bf169e851ac44e6420214" + +[[package]] +name = "zeroize" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4756f7db3f7b5574938c3eb1c117038b8e07f95ee6718c0efad4ac21508f1efd" +dependencies = [ + "zeroize_derive", +] + +[[package]] +name = "zeroize_derive" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f8f187641dad4f680d25c4bfc4225b418165984179f26ca76ec4fb6441d3a17" +dependencies = [ + "proc-macro2", + "quote", + "syn", + "synstructure", +] diff --git a/examples/abi/Cargo.toml b/examples/abi/Cargo.toml new file mode 100644 index 000000000..424e46095 --- /dev/null +++ b/examples/abi/Cargo.toml @@ -0,0 +1,24 @@ +[package] +name = "abi" +version = "0.1.0" +authors = ["Near Inc "] +edition = "2021" + +[lib] +crate-type = ["cdylib"] + +[dependencies] +near-sdk = { path = "../../near-sdk", features = ["abi"] } +serde = { version = "1", features = ["derive"] } +schemars = "0.8" + +[profile.release] +codegen-units = 1 +# Tell `rustc` to optimize for small code size. +opt-level = "z" +lto = true +debug = false +panic = "abort" + +[workspace] +members = [] diff --git a/examples/abi/README.md b/examples/abi/README.md new file mode 100644 index 000000000..8171a43ff --- /dev/null +++ b/examples/abi/README.md @@ -0,0 +1,12 @@ +# ABI + +Showcases how to generate ABI from a NEAR contract. + +## Generating ABI +To generate the ABI file run: + +```bash +cargo near abi +``` + +This will generate a file located at `target/near/abi.json`. diff --git a/examples/abi/build.sh b/examples/abi/build.sh new file mode 100755 index 000000000..e91c09a0b --- /dev/null +++ b/examples/abi/build.sh @@ -0,0 +1,8 @@ +#!/bin/bash +TARGET="${CARGO_TARGET_DIR:-target}" +set -e +cd "$(dirname $0)" + +cargo build --target wasm32-unknown-unknown --release +cp $TARGET/wasm32-unknown-unknown/release/abi.wasm ./res/ +#wasm-opt -Oz --output ./res/status_message.wasm ./res/status_message.wasm diff --git a/examples/abi/res/abi.json b/examples/abi/res/abi.json new file mode 100644 index 000000000..d19613f36 --- /dev/null +++ b/examples/abi/res/abi.json @@ -0,0 +1,112 @@ +{ + "abi_schema_version": "0.1.0", + "metadata": { + "name": "abi", + "version": "0.1.0", + "authors": [ + "Near Inc " + ] + }, + "abi": { + "functions": [ + { + "name": "add", + "is_view": true, + "params": [ + { + "name": "a", + "type_schema": { + "$ref": "#/definitions/Pair" + }, + "serialization_type": "json" + }, + { + "name": "b", + "type_schema": { + "$ref": "#/definitions/Pair" + }, + "serialization_type": "json" + } + ], + "result": { + "type_schema": { + "$ref": "#/definitions/Pair" + }, + "serialization_type": "json" + } + }, + { + "name": "add_callback", + "is_view": true, + "callbacks": [ + { + "type_schema": { + "$ref": "#/definitions/DoublePair" + }, + "serialization_type": "json" + }, + { + "type_schema": { + "$ref": "#/definitions/DoublePair" + }, + "serialization_type": "json" + } + ], + "callbacks_vec": { + "type_schema": { + "type": "array", + "items": { + "$ref": "#/definitions/DoublePair" + } + }, + "serialization_type": "json" + }, + "result": { + "type_schema": { + "$ref": "#/definitions/DoublePair" + }, + "serialization_type": "json" + } + } + ], + "root_schema": { + "$schema": "http://json-schema.org/draft-07/schema#", + "title": "String", + "type": "string", + "definitions": { + "DoublePair": { + "type": "object", + "required": [ + "first", + "second" + ], + "properties": { + "first": { + "$ref": "#/definitions/Pair" + }, + "second": { + "$ref": "#/definitions/Pair" + } + } + }, + "Pair": { + "type": "array", + "items": [ + { + "type": "integer", + "format": "uint32", + "minimum": 0.0 + }, + { + "type": "integer", + "format": "uint32", + "minimum": 0.0 + } + ], + "maxItems": 2, + "minItems": 2 + } + } + } + } +} \ No newline at end of file diff --git a/examples/abi/res/abi.wasm b/examples/abi/res/abi.wasm new file mode 100755 index 000000000..41718f45b Binary files /dev/null and b/examples/abi/res/abi.wasm differ diff --git a/examples/abi/src/lib.rs b/examples/abi/src/lib.rs new file mode 100644 index 000000000..4041151ba --- /dev/null +++ b/examples/abi/src/lib.rs @@ -0,0 +1,40 @@ +use near_sdk::__private::schemars::JsonSchema; +use near_sdk::borsh::{self, BorshDeserialize, BorshSerialize}; +use near_sdk::near_bindgen; +use near_sdk::serde::{Deserialize, Serialize}; + +#[derive(JsonSchema, Serialize, Deserialize)] +pub struct Pair(u32, u32); + +#[derive(JsonSchema, Serialize, Deserialize)] +pub struct DoublePair { + first: Pair, + second: Pair, +} + +#[near_bindgen] +#[derive(Default, BorshDeserialize, BorshSerialize)] +pub struct Adder {} + +#[near_bindgen] +impl Adder { + pub fn add(&self, a: Pair, b: Pair) -> Pair { + sum_pair(&a, &b) + } + + pub fn add_callback( + &self, + #[callback_unwrap] a: DoublePair, + #[callback_unwrap] b: DoublePair, + #[callback_vec] others: Vec, + ) -> DoublePair { + Some(b).iter().chain(others.iter()).fold(a, |acc, el| DoublePair { + first: sum_pair(&acc.first, &el.first), + second: sum_pair(&acc.second, &el.second), + }) + } +} + +fn sum_pair(a: &Pair, b: &Pair) -> Pair { + Pair(a.0 + b.0, a.1 + b.1) +} diff --git a/examples/fungible-token/Cargo.lock b/examples/fungible-token/Cargo.lock index e769990fe..b74649d96 100644 --- a/examples/fungible-token/Cargo.lock +++ b/examples/fungible-token/Cargo.lock @@ -621,6 +621,7 @@ version = "0.0.1" dependencies = [ "near-contract-standards", "near-sdk", + "schemars", ] [[package]] @@ -686,6 +687,12 @@ dependencies = [ "winapi", ] +[[package]] +name = "dyn-clone" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21e50f3adc76d6a43f5ed73b698a87d0760ca74617f60f7c3b879003536fdd28" + [[package]] name = "easy-ext" version = "0.2.9" @@ -834,6 +841,7 @@ version = "1.1.0" dependencies = [ "near-contract-standards", "near-sdk", + "schemars", ] [[package]] @@ -1393,6 +1401,7 @@ name = "near-contract-standards" version = "4.0.0" dependencies = [ "near-sdk", + "schemars", "serde", "serde_json", ] @@ -1783,6 +1792,7 @@ dependencies = [ "near-sdk-macros", "near-sys", "near-vm-logic", + "schemars", "serde", "serde_json", "wee_alloc", @@ -2509,6 +2519,30 @@ dependencies = [ "windows-sys", ] +[[package]] +name = "schemars" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1847b767a3d62d95cbf3d8a9f0e421cf57a0d8aa4f411d4b16525afb0284d4ed" +dependencies = [ + "dyn-clone", + "schemars_derive", + "serde", + "serde_json", +] + +[[package]] +name = "schemars_derive" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af4d7e1b012cb3d9129567661a63755ea4b8a7386d339dc945ae187e403c6743" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn", +] + [[package]] name = "scopeguard" version = "1.1.0" @@ -2564,6 +2598,17 @@ dependencies = [ "syn", ] +[[package]] +name = "serde_derive_internals" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85bf8229e7920a9f636479437026331ce11aa132b4dde37d121944a44d6e5f3c" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "serde_json" version = "1.0.81" diff --git a/examples/fungible-token/ft/Cargo.toml b/examples/fungible-token/ft/Cargo.toml index 628a1589d..5197ca9ea 100644 --- a/examples/fungible-token/ft/Cargo.toml +++ b/examples/fungible-token/ft/Cargo.toml @@ -8,5 +8,6 @@ edition = "2021" crate-type = ["cdylib", "rlib"] [dependencies] -near-sdk = { path = "../../../near-sdk" } -near-contract-standards = { path = "../../../near-contract-standards" } +near-sdk = { path = "../../../near-sdk", features = ["abi"] } +near-contract-standards = { path = "../../../near-contract-standards", features = ["abi"] } +schemars = "0.8" diff --git a/examples/fungible-token/res/defi.wasm b/examples/fungible-token/res/defi.wasm index 550e107e7..462bba577 100755 Binary files a/examples/fungible-token/res/defi.wasm and b/examples/fungible-token/res/defi.wasm differ diff --git a/examples/fungible-token/res/fungible_token.wasm b/examples/fungible-token/res/fungible_token.wasm index ab5e9b9c4..a4a392d8c 100755 Binary files a/examples/fungible-token/res/fungible_token.wasm and b/examples/fungible-token/res/fungible_token.wasm differ diff --git a/examples/fungible-token/test-contract-defi/Cargo.toml b/examples/fungible-token/test-contract-defi/Cargo.toml index 921115b57..e27fae75d 100644 --- a/examples/fungible-token/test-contract-defi/Cargo.toml +++ b/examples/fungible-token/test-contract-defi/Cargo.toml @@ -8,6 +8,6 @@ edition = "2021" crate-type = ["cdylib", "rlib"] [dependencies] -near-sdk = { path = "../../../near-sdk" } -near-contract-standards = { path = "../../../near-contract-standards" } - +near-sdk = { path = "../../../near-sdk", features = ["abi"] } +near-contract-standards = { path = "../../../near-contract-standards", features = ["abi"] } +schemars = "0.8" diff --git a/examples/non-fungible-token/Cargo.lock b/examples/non-fungible-token/Cargo.lock index 2ff465b69..fc76661fb 100644 --- a/examples/non-fungible-token/Cargo.lock +++ b/examples/non-fungible-token/Cargo.lock @@ -686,6 +686,12 @@ dependencies = [ "winapi", ] +[[package]] +name = "dyn-clone" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21e50f3adc76d6a43f5ed73b698a87d0760ca74617f60f7c3b879003536fdd28" + [[package]] name = "easy-ext" version = "0.2.9" @@ -1370,6 +1376,7 @@ name = "near-contract-standards" version = "4.0.0" dependencies = [ "near-sdk", + "schemars", "serde", "serde_json", ] @@ -2508,6 +2515,30 @@ dependencies = [ "windows-sys", ] +[[package]] +name = "schemars" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1847b767a3d62d95cbf3d8a9f0e421cf57a0d8aa4f411d4b16525afb0284d4ed" +dependencies = [ + "dyn-clone", + "schemars_derive", + "serde", + "serde_json", +] + +[[package]] +name = "schemars_derive" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af4d7e1b012cb3d9129567661a63755ea4b8a7386d339dc945ae187e403c6743" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn", +] + [[package]] name = "scopeguard" version = "1.1.0" @@ -2563,6 +2594,17 @@ dependencies = [ "syn", ] +[[package]] +name = "serde_derive_internals" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85bf8229e7920a9f636479437026331ce11aa132b4dde37d121944a44d6e5f3c" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "serde_json" version = "1.0.81" diff --git a/examples/non-fungible-token/res/approval_receiver.wasm b/examples/non-fungible-token/res/approval_receiver.wasm index 127747894..551da3f38 100755 Binary files a/examples/non-fungible-token/res/approval_receiver.wasm and b/examples/non-fungible-token/res/approval_receiver.wasm differ diff --git a/examples/non-fungible-token/res/non_fungible_token.wasm b/examples/non-fungible-token/res/non_fungible_token.wasm index cdbfde6f9..450ef5d80 100755 Binary files a/examples/non-fungible-token/res/non_fungible_token.wasm and b/examples/non-fungible-token/res/non_fungible_token.wasm differ diff --git a/examples/non-fungible-token/res/token_receiver.wasm b/examples/non-fungible-token/res/token_receiver.wasm index a1141cc90..c8c2d1265 100755 Binary files a/examples/non-fungible-token/res/token_receiver.wasm and b/examples/non-fungible-token/res/token_receiver.wasm differ diff --git a/near-contract-standards/Cargo.toml b/near-contract-standards/Cargo.toml index 9ef621320..8d2c5b28d 100644 --- a/near-contract-standards/Cargo.toml +++ b/near-contract-standards/Cargo.toml @@ -16,3 +16,7 @@ NEAR smart contracts standard library. near-sdk = { path = "../near-sdk", version = "=4.0.0" } serde = "1" serde_json = "1" +schemars = "0.8" + +[features] +abi = ["near-sdk/abi"] diff --git a/near-contract-standards/src/fungible_token/metadata.rs b/near-contract-standards/src/fungible_token/metadata.rs index d57dd1df7..3f1e1c231 100644 --- a/near-contract-standards/src/fungible_token/metadata.rs +++ b/near-contract-standards/src/fungible_token/metadata.rs @@ -7,6 +7,7 @@ pub const FT_METADATA_SPEC: &str = "ft-1.0.0"; #[derive(BorshDeserialize, BorshSerialize, Clone, Deserialize, Serialize)] #[serde(crate = "near_sdk::serde")] +#[cfg_attr(feature = "abi", derive(schemars::JsonSchema))] pub struct FungibleTokenMetadata { pub spec: String, pub name: String, diff --git a/near-contract-standards/src/storage_management/mod.rs b/near-contract-standards/src/storage_management/mod.rs index e8eddb24a..13d024791 100644 --- a/near-contract-standards/src/storage_management/mod.rs +++ b/near-contract-standards/src/storage_management/mod.rs @@ -5,6 +5,7 @@ use near_sdk::AccountId; #[derive(BorshDeserialize, BorshSerialize, Serialize, Deserialize)] #[serde(crate = "near_sdk::serde")] +#[cfg_attr(feature = "abi", derive(schemars::JsonSchema))] pub struct StorageBalance { pub total: U128, pub available: U128, @@ -12,6 +13,7 @@ pub struct StorageBalance { #[derive(BorshDeserialize, BorshSerialize, Serialize, Deserialize)] #[serde(crate = "near_sdk::serde")] +#[cfg_attr(feature = "abi", derive(schemars::JsonSchema))] pub struct StorageBalanceBounds { pub min: U128, pub max: Option, diff --git a/near-sdk-macros/Cargo.toml b/near-sdk-macros/Cargo.toml index 8bccfe61c..7f9910e7a 100644 --- a/near-sdk-macros/Cargo.toml +++ b/near-sdk-macros/Cargo.toml @@ -19,3 +19,6 @@ proc-macro2 = "1.0" syn = {version = "1", features = ["full", "fold", "extra-traits", "visit"] } quote = "1.0" Inflector = { version = "0.11.4", default-features = false, features = [] } + +[features] +abi = [] diff --git a/near-sdk-macros/src/core_impl/abi/abi_generator.rs b/near-sdk-macros/src/core_impl/abi/abi_generator.rs new file mode 100644 index 000000000..7b5fc95e1 --- /dev/null +++ b/near-sdk-macros/src/core_impl/abi/abi_generator.rs @@ -0,0 +1,169 @@ +use crate::core_impl::utils; +use crate::core_impl::{BindgenArgType, SerializerType}; +use crate::{ImplItemMethodInfo, MethodType}; +use proc_macro2::{Span, TokenStream as TokenStream2}; +use quote::quote; +use syn::ReturnType; + +impl ImplItemMethodInfo { + /// Generates ABI struct for this function. + /// + /// # Example: + /// The following function: + /// ```ignore + /// fn f3(&mut self, arg0: FancyStruct, arg1: u64) -> Result { } + /// ``` + /// will produce this struct: + /// ```ignore + /// near_sdk::__private::AbiFunction { + /// name: "f3".to_string(), + /// is_view: false, + /// is_init: false, + /// params: vec![ + /// near_sdk::__private::AbiParameter { + /// type_id: 0, + /// serialization_type: "json", + /// }, + /// near_sdk::__private::AbiParameter { + /// type_id: 1, + /// serialization_type: "json", + /// } + /// ], + /// callbacks: vec![], + /// callbacks_vec: None, + /// result: near_sdk::__private::AbiParameter { + /// type_id: 2, + /// serialization_type: "json", + /// } + /// } + /// ``` + /// If args are serialized with Borsh it will not include `#[derive(borsh::BorshSchema)]`. + pub fn abi_struct(&self) -> TokenStream2 { + let function_name_str = self.attr_signature_info.ident.to_string(); + let is_view = matches!(&self.attr_signature_info.method_type, &MethodType::View); + let is_init = matches!( + &self.attr_signature_info.method_type, + &MethodType::Init | &MethodType::InitIgnoreState + ); + let is_payable = self.attr_signature_info.is_payable; + let is_private = self.attr_signature_info.is_private; + + let mut params = Vec::::new(); + let mut callbacks = Vec::::new(); + let mut callback_vec: Option = None; + for arg in &self.attr_signature_info.args { + let typ = &arg.ty; + let serialization_type = abi_serialization_type(&arg.serializer_ty); + let arg_name = arg.ident.to_string(); + match arg.bindgen_ty { + BindgenArgType::Regular => { + params.push(quote! { + near_sdk::__private::AbiParameter { + name: #arg_name.to_string(), + type_schema: gen.subschema_for::<#typ>(), + serialization_type: #serialization_type, + } + }); + } + BindgenArgType::CallbackArg => { + callbacks.push(quote! { + near_sdk::__private::AbiType { + type_schema: gen.subschema_for::<#typ>(), + serialization_type: #serialization_type, + } + }); + } + BindgenArgType::CallbackResultArg => { + let typ = if let Some(ok_type) = utils::extract_ok_type(typ) { + ok_type + } else { + return syn::Error::new_spanned( + &arg.ty, + "Function parameters marked with \ + #[callback_result] should have type Result", + ) + .into_compile_error(); + }; + callbacks.push(quote! { + near_sdk::__private::AbiType { + type_schema: gen.subschema_for::<#typ>(), + serialization_type: #serialization_type, + } + }); + } + BindgenArgType::CallbackArgVec => { + if callback_vec.is_none() { + callback_vec = Some(quote! { + Some( + near_sdk::__private::AbiType { + type_schema: gen.subschema_for::<#typ>(), + serialization_type: #serialization_type, + } + ) + }) + } else { + return syn::Error::new( + Span::call_site(), + "A function can only have one #[callback_vec] parameter.", + ) + .to_compile_error(); + } + } + }; + } + let callback_vec = callback_vec.unwrap_or(quote! { None }); + + let result = match self.attr_signature_info.method_type { + MethodType::Init | MethodType::InitIgnoreState => { + // Init methods must return the contract state, so the return type does not matter + quote! { + None + } + } + _ => match &self.attr_signature_info.returns { + ReturnType::Default => { + quote! { + None + } + } + ReturnType::Type(_, ty) => { + let serialization_type = + abi_serialization_type(&self.attr_signature_info.result_serializer); + quote! { + Some( + near_sdk::__private::AbiType { + type_schema: gen.subschema_for::<#ty>(), + serialization_type: #serialization_type, + } + ) + } + } + }, + }; + + quote! { + near_sdk::__private::AbiFunction { + name: #function_name_str.to_string(), + is_view: #is_view, + is_init: #is_init, + is_payable: #is_payable, + is_private: #is_private, + params: vec![#(#params),*], + callbacks: vec![#(#callbacks),*], + callbacks_vec: #callback_vec, + result: #result + } + } + } +} + +fn abi_serialization_type(serializer_type: &SerializerType) -> TokenStream2 { + match serializer_type { + SerializerType::JSON => quote! { + near_sdk::__private::AbiSerializationType::Json + }, + SerializerType::Borsh => quote! { + near_sdk::__private::AbiSerializationType::Borsh + }, + } +} diff --git a/near-sdk-macros/src/core_impl/abi/abi_visitor.rs b/near-sdk-macros/src/core_impl/abi/abi_visitor.rs new file mode 100644 index 000000000..0a5d53608 --- /dev/null +++ b/near-sdk-macros/src/core_impl/abi/abi_visitor.rs @@ -0,0 +1,67 @@ +use crate::core_impl::ImplItemMethodInfo; +use crate::ItemImplInfo; +use proc_macro2::TokenStream as TokenStream2; +use quote::{format_ident, quote}; +use syn::visit::Visit; +use syn::{Error, ItemImpl}; + +/// Information relevant to ABI extracted from the `impl` section decorated with `#[near_bindgen]`. +#[derive(Default)] +pub struct AbiVisitor { + impl_item_infos: Vec, + /// Errors that occured while extracting the data. + errors: Vec, +} + +impl<'ast> Visit<'ast> for AbiVisitor { + fn visit_item_impl(&mut self, i: &'ast ItemImpl) { + match ItemImplInfo::new(&mut i.clone()) { + Ok(info) => self.impl_item_infos.push(info), + Err(err) => self.errors.push(err), + } + syn::visit::visit_item_impl(self, i); + } +} +impl AbiVisitor { + pub fn new() -> Self { + Default::default() + } + + pub fn generate_abi_function(&self) -> syn::Result { + if !self.errors.is_empty() { + return Err(self.errors[0].clone()); + } + + let public_functions: Vec<&ImplItemMethodInfo> = self + .impl_item_infos + .iter() + .flat_map(|i| { + i.methods.iter().filter(|m| m.is_public || i.is_trait_impl).collect::>() + }) + .collect(); + if public_functions.is_empty() { + // Short-circuit if there are not public functions to export to ABI + return Ok(TokenStream2::new()); + } + + let functions: Vec = + public_functions.iter().map(|m| m.abi_struct()).collect(); + let first_function_name = &public_functions[0].attr_signature_info.ident; + let near_abi_symbol = format_ident!("__near_abi_{}", &first_function_name); + Ok(quote! { + #[cfg(not(target_arch = "wasm32"))] + const _: () = { + #[no_mangle] + pub fn #near_abi_symbol() -> near_sdk::__private::AbiRoot { + let mut gen = near_sdk::__private::schemars::gen::SchemaGenerator::default(); + near_sdk::__private::AbiRoot::new( + near_sdk::__private::Abi { + functions: vec![#(#functions),*], + root_schema: gen.into_root_schema_for::(), + } + ) + } + }; + }) + } +} diff --git a/near-sdk-macros/src/core_impl/abi/mod.rs b/near-sdk-macros/src/core_impl/abi/mod.rs new file mode 100644 index 000000000..7cae05dac --- /dev/null +++ b/near-sdk-macros/src/core_impl/abi/mod.rs @@ -0,0 +1,2 @@ +pub mod abi_generator; +pub mod abi_visitor; diff --git a/near-sdk-macros/src/core_impl/mod.rs b/near-sdk-macros/src/core_impl/mod.rs index ff6047706..d14299ba5 100644 --- a/near-sdk-macros/src/core_impl/mod.rs +++ b/near-sdk-macros/src/core_impl/mod.rs @@ -1,7 +1,11 @@ +#[cfg(feature = "abi")] +mod abi; mod code_generator; mod info_extractor; mod metadata; mod utils; +#[cfg(feature = "abi")] +pub use abi::abi_visitor::AbiVisitor; pub use code_generator::*; pub use info_extractor::*; pub use metadata::metadata_visitor::MetadataVisitor; diff --git a/near-sdk-macros/src/lib.rs b/near-sdk-macros/src/lib.rs index 387ad5985..3a0aea379 100644 --- a/near-sdk-macros/src/lib.rs +++ b/near-sdk-macros/src/lib.rs @@ -58,6 +58,18 @@ pub fn near_bindgen(_attr: TokenStream, item: TokenStream) -> TokenStream { #ext_gen }) } else if let Ok(mut input) = syn::parse::(item) { + #[cfg(not(feature = "abi"))] + let abi_generated = proc_macro2::TokenStream::new(); + #[cfg(feature = "abi")] + let abi_generated = { + let mut visitor = AbiVisitor::new(); + visitor.visit_item_impl(&input); + match visitor.generate_abi_function() { + Ok(x) => x, + Err(err) => return TokenStream::from(err.to_compile_error()), + } + }; + let item_impl_info = match ItemImplInfo::new(&mut input) { Ok(x) => x, Err(err) => { @@ -72,6 +84,7 @@ pub fn near_bindgen(_attr: TokenStream, item: TokenStream) -> TokenStream { #ext_generated_code #input #generated_code + #abi_generated }) } else { TokenStream::from( diff --git a/near-sdk/Cargo.toml b/near-sdk/Cargo.toml index b2d426860..a542ecc5d 100644 --- a/near-sdk/Cargo.toml +++ b/near-sdk/Cargo.toml @@ -25,6 +25,7 @@ near-sys = { path = "../sys", version = "0.2" } base64 = "0.13" borsh = { version = "0.9", features = ["const-generics"] } bs58 = "0.4" +schemars = { version = "0.8.8", optional = true } # Export dependencies for contracts wee_alloc = { version = "0.4.5", default-features = false, optional = true } @@ -50,3 +51,4 @@ hex = { version = "0.4.3", features = ["serde"] } default = ["wee_alloc"] expensive-debug = [] unstable = ["once_cell"] +abi = ["schemars", "near-sdk-macros/abi"] diff --git a/near-sdk/src/json_types/integers.rs b/near-sdk/src/json_types/integers.rs index b3fecb1ee..039a89c4a 100644 --- a/near-sdk/src/json_types/integers.rs +++ b/near-sdk/src/json_types/integers.rs @@ -11,6 +11,7 @@ macro_rules! impl_str_type { #[derive( Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, BorshDeserialize, BorshSerialize, )] + #[cfg_attr(feature = "abi", derive(schemars::JsonSchema))] pub struct $iden(pub $ty); impl From<$ty> for $iden { diff --git a/near-sdk/src/json_types/vector.rs b/near-sdk/src/json_types/vector.rs index cfa509cd3..705eb3fff 100644 --- a/near-sdk/src/json_types/vector.rs +++ b/near-sdk/src/json_types/vector.rs @@ -17,6 +17,17 @@ impl From for Vec { } } +#[cfg(feature = "abi")] +impl schemars::JsonSchema for Base64VecU8 { + fn schema_name() -> String { + "Base64VecU8".to_string() + } + + fn json_schema(gen: &mut schemars::gen::SchemaGenerator) -> schemars::schema::Schema { + String::json_schema(gen) + } +} + /// Convenience module to allow anotating a serde structure as base64 bytes. /// /// # Example diff --git a/near-sdk/src/private/abi.rs b/near-sdk/src/private/abi.rs new file mode 100644 index 000000000..0414b84ea --- /dev/null +++ b/near-sdk/src/private/abi.rs @@ -0,0 +1,170 @@ +use schemars::schema::{RootSchema, Schema}; +use serde::{Deserialize, Serialize}; +use std::collections::HashMap; + +/// Current version of the ABI schema format. +const ABI_SCHEMA_SEMVER: &str = "0.1.0"; + +/// Contract ABI. +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq)] +pub struct AbiRoot { + /// Semver of the ABI schema format. + pub abi_schema_version: String, + /// Metadata information about the contract. + pub metadata: AbiMetadata, + /// Core ABI information (functions and types). + pub abi: Abi, +} + +impl AbiRoot { + pub fn new(abi: Abi) -> Self { + Self { + abi_schema_version: ABI_SCHEMA_SEMVER.to_string(), + metadata: Default::default(), + abi, + } + } + + pub fn combine>(abi_roots: I) -> AbiRoot { + let mut abi_schema_version = "".to_string(); + let mut abis = Vec::::new(); + + for abi_root in abi_roots { + // Set common schema version to the current schema version. Only happens with the + // first element. + if abi_schema_version.is_empty() { + abi_schema_version = abi_root.abi_schema_version.clone(); + } + + // Check that all ABIs conform to the same version + if abi_root.abi_schema_version != abi_schema_version { + panic!( + "Conflicting ABI schema versions: {} and {}", + &abi_root.abi_schema_version, abi_schema_version + ); + } + + // Check that all metadata is empty. + if !abi_root.metadata.name.is_none() + || !abi_root.metadata.version.is_none() + || !abi_root.metadata.authors.is_empty() + || !abi_root.metadata.other.is_empty() + { + panic!("Non-empty metadata: {:?}", &abi_root.metadata); + } + + abis.push(abi_root.abi); + } + + let abi = Abi::combine(abis); + AbiRoot { abi_schema_version, metadata: AbiMetadata::default(), abi } + } +} + +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq, Default)] +pub struct AbiMetadata { + /// The name of the smart contract. + #[serde(default, skip_serializing_if = "Option::is_none")] + pub name: Option, + /// The version of the smart contract. + #[serde(default, skip_serializing_if = "Option::is_none")] + pub version: Option, + /// The authors of the smart contract. + #[serde(default, skip_serializing_if = "Vec::is_empty")] + pub authors: Vec, + /// Other arbitrary metadata. + #[serde(default, flatten, skip_serializing_if = "HashMap::is_empty")] + pub other: HashMap, +} + +/// Core ABI information. +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq)] +pub struct Abi { + /// ABIs of all contract's functions. + pub functions: Vec, + /// Root JSON Schema containing all types referenced in the functions. + pub root_schema: RootSchema, +} + +impl Abi { + pub fn combine>(abis: I) -> Abi { + let mut functions = Vec::::new(); + let mut gen = schemars::gen::SchemaGenerator::default(); + let definitions = gen.definitions_mut(); + + for abi in abis { + // Update resulting JSON Schema + definitions.extend(abi.root_schema.definitions.to_owned()); + + // Update resulting function list + functions.extend(abi.functions); + } + // Sort the function list for readability + functions.sort_by(|x, y| x.name.cmp(&y.name)); + + Abi { functions, root_schema: gen.into_root_schema_for::() } + } +} + +/// ABI of a single function. +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq)] +pub struct AbiFunction { + pub name: String, + /// Whether function does not modify the state. + #[serde(default, skip_serializing_if = "is_false")] + pub is_view: bool, + /// Whether function can be used to initialize the state. + #[serde(default, skip_serializing_if = "is_false")] + pub is_init: bool, + /// Whether function is accepting $NEAR. + #[serde(default, skip_serializing_if = "is_false")] + pub is_payable: bool, + /// Whether function can only accept calls from self (current account). + #[serde(default, skip_serializing_if = "is_false")] + pub is_private: bool, + /// Type identifiers of the function parameters. + #[serde(default, skip_serializing_if = "Vec::is_empty")] + pub params: Vec, + /// Type identifiers of the callbacks of the function. + #[serde(default, skip_serializing_if = "Vec::is_empty")] + pub callbacks: Vec, + /// Type identifier of the vararg callbacks of the function. + #[serde(default, skip_serializing_if = "Option::is_none")] + pub callbacks_vec: Option, + /// Return type identifier. + #[serde(default, skip_serializing_if = "Option::is_none")] + pub result: Option, +} + +/// Information about a single named function parameter. +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq)] +pub struct AbiParameter { + /// Parameter name (e.g. `p1` in `fn foo(p1: u32) {}`). + pub name: String, + /// JSON Subschema representing the type of the parameter. + pub type_schema: Schema, + /// How the parameter is serialized (either JSON or Borsh). + pub serialization_type: AbiSerializationType, +} + +/// Information about a single type (e.g. return type). +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq)] +pub struct AbiType { + /// JSON Subschema that represents this type. + pub type_schema: Schema, + /// How the type instance is serialized (either JSON or Borsh). + pub serialization_type: AbiSerializationType, +} + +/// Represents how instances of a certain type are serialized in a certain context. Same type +/// can have different serialization types associated with it depending on where they occur. +#[derive(Clone, Serialize, Deserialize, Debug, PartialEq)] +#[serde(rename_all = "lowercase")] +pub enum AbiSerializationType { + Json, + Borsh, +} + +fn is_false(b: &bool) -> bool { + !b +} diff --git a/near-sdk/src/private/mod.rs b/near-sdk/src/private/mod.rs index 122adf710..102d8cccf 100644 --- a/near-sdk/src/private/mod.rs +++ b/near-sdk/src/private/mod.rs @@ -1,5 +1,15 @@ -mod metadata; +#[cfg(feature = "abi")] +mod abi; + +#[cfg(feature = "abi")] +pub use abi::{ + Abi, AbiFunction, AbiMetadata, AbiParameter, AbiRoot, AbiSerializationType, AbiType, +}; +#[cfg(feature = "abi")] +pub use schemars; + +mod metadata; pub use metadata::{Metadata, MethodMetadata}; use crate::IntoStorageKey; diff --git a/near-sdk/src/promise.rs b/near-sdk/src/promise.rs index 5487d23e0..441e33d8c 100644 --- a/near-sdk/src/promise.rs +++ b/near-sdk/src/promise.rs @@ -474,6 +474,19 @@ impl borsh::BorshSerialize for Promise { } } +#[cfg(feature = "abi")] +impl schemars::JsonSchema for Promise { + fn schema_name() -> String { + "Promise".to_string() + } + + fn json_schema(_gen: &mut schemars::gen::SchemaGenerator) -> schemars::schema::Schema { + // Since promises are untyped, for now we represent Promise results with the schema + // `true` which matches everything (i.e. always passes validation) + schemars::schema::Schema::Bool(true) + } +} + /// When the method can return either a promise or a value, it can be called with `PromiseOrValue::Promise` /// or `PromiseOrValue::Value` to specify which one should be returned. /// # Example @@ -529,3 +542,14 @@ impl borsh::BorshSerialize for PromiseOrValue { } } } + +#[cfg(feature = "abi")] +impl schemars::JsonSchema for PromiseOrValue { + fn schema_name() -> String { + format!("PromiseOrValue{}", T::schema_name()) + } + + fn json_schema(gen: &mut schemars::gen::SchemaGenerator) -> schemars::schema::Schema { + T::json_schema(gen) + } +} diff --git a/near-sdk/src/types/account_id.rs b/near-sdk/src/types/account_id.rs index 325c0dedf..a338b5ca9 100644 --- a/near-sdk/src/types/account_id.rs +++ b/near-sdk/src/types/account_id.rs @@ -36,6 +36,7 @@ use crate::env::is_valid_account_id; #[derive( Debug, Clone, PartialEq, PartialOrd, Ord, Eq, BorshSerialize, Serialize, Hash, BorshSchema, )] +#[cfg_attr(feature = "abi", derive(schemars::JsonSchema))] pub struct AccountId(String); impl AccountId {