diff --git a/Cargo.lock b/Cargo.lock index dd4ca4f40..4b5f9a3dc 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -463,9 +463,12 @@ dependencies = [ "bitcoincore-rpc", "bitcoincore-rpc-json", "chainhook-types 1.0.7", + "clap 3.2.23", + "clap_generate", "clarinet-files", "clarinet-utils", "crossbeam-channel 0.5.8", + "ctrlc", "dashmap 5.4.0", "futures", "fxhash", @@ -475,7 +478,7 @@ dependencies = [ "rand 0.8.5", "reqwest", "rocket", - "schemars", + "schemars 0.8.12", "serde", "serde-hex", "serde_derive", @@ -483,6 +486,7 @@ dependencies = [ "stacks-rpc-client", "threadpool", "tokio", + "toml", "zeromq", ] @@ -493,7 +497,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "65268eb9aad0d567865cd13c5a5c7a9a301d99f6301f8f5cac1654744eda0a7c" dependencies = [ "hex", - "schemars", + "schemars 0.8.11", "serde", "serde_derive", "serde_json", @@ -505,7 +509,7 @@ name = "chainhook-types" version = "1.0.7" dependencies = [ "hex", - "schemars", + "schemars 0.8.12", "serde", "serde_derive", "serde_json", @@ -2437,11 +2441,10 @@ dependencies = [ [[package]] name = "okapi" version = "0.7.0-rc.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ce66b6366e049880a35c378123fddb630b1a1a3c37fa1ca70caaf4a09f6e2893" +source = "git+https://github.com/hirosystems/okapi.git?branch=feat-chainhook-fixes#21baab1052426159857572788c3b5774fcb41b91" dependencies = [ "log", - "schemars", + "schemars 0.8.12", "serde", "serde_json", ] @@ -3169,15 +3172,14 @@ dependencies = [ [[package]] name = "rocket_okapi" version = "0.8.0-rc.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "742098674565c8f0c35c77444f90344aafedebb71cfee9cdbf0185acc6b9cdb7" +source = "git+https://github.com/hirosystems/okapi.git?branch=feat-chainhook-fixes#21baab1052426159857572788c3b5774fcb41b91" dependencies = [ "either", "log", "okapi", "rocket", "rocket_okapi_codegen", - "schemars", + "schemars 0.8.12", "serde", "serde_json", ] @@ -3185,8 +3187,7 @@ dependencies = [ [[package]] name = "rocket_okapi_codegen" version = "0.8.0-rc.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8c43f8edc57d88750a220b0ec1870a36c1106204ec99cc35131b49de3b954a4a" +source = "git+https://github.com/hirosystems/okapi.git?branch=feat-chainhook-fixes#21baab1052426159857572788c3b5774fcb41b91" dependencies = [ "darling", "proc-macro2", @@ -3387,12 +3388,24 @@ name = "schemars" version = "0.8.11" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2a5fb6c61f29e723026dc8e923d94c694313212abbecbbe5f55a7748eec5b307" +dependencies = [ + "dyn-clone", + "schemars_derive 0.8.11", + "serde", + "serde_json", +] + +[[package]] +name = "schemars" +version = "0.8.12" +source = "git+https://github.com/hirosystems/schemars.git?branch=feat-chainhook-fixes#15fdd4711700114d57c090aad62516593bd4ca6d" dependencies = [ "dyn-clone", "indexmap", - "schemars_derive", + "schemars_derive 0.8.12", "serde", "serde_json", + "uuid 1.3.0", ] [[package]] @@ -3407,6 +3420,17 @@ dependencies = [ "syn 1.0.105", ] +[[package]] +name = "schemars_derive" +version = "0.8.12" +source = "git+https://github.com/hirosystems/schemars.git?branch=feat-chainhook-fixes#15fdd4711700114d57c090aad62516593bd4ca6d" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn 1.0.105", +] + [[package]] name = "scoped-tls" version = "1.0.1" diff --git a/components/chainhook-cli/Cargo.toml b/components/chainhook-cli/Cargo.toml index b10e9ed0a..abb4ab822 100644 --- a/components/chainhook-cli/Cargo.toml +++ b/components/chainhook-cli/Cargo.toml @@ -38,7 +38,7 @@ atty = "0.2.14" crossbeam-channel = "0.5.8" uuid = { version = "1.3.0", features = ["v4", "fast-rng"] } threadpool = "1.8.1" -rocket_okapi = "0.8.0-rc.3" +rocket_okapi = { version = "0.8.0-rc.3", git = "https://github.com/hirosystems/okapi.git", branch = "feat-chainhook-fixes"} rocket = { version = "=0.5.0-rc.3", features = ["json"] } [dependencies.rocksdb] diff --git a/components/chainhook-cli/src/cli/mod.rs b/components/chainhook-cli/src/cli/mod.rs index 221001685..96295baaa 100644 --- a/components/chainhook-cli/src/cli/mod.rs +++ b/components/chainhook-cli/src/cli/mod.rs @@ -3,6 +3,7 @@ use crate::config::generator::generate_config; use crate::config::{Config, PredicatesApi}; use crate::scan::bitcoin::scan_bitcoin_chainstate_via_rpc_using_predicate; use crate::scan::stacks::scan_stacks_chainstate_via_csv_using_predicate; +use crate::service::http_api::document_predicate_api_server; use crate::service::Service; use crate::storage::{ get_last_block_height_inserted, get_stacks_block_at_block_height, is_stacks_block_present, @@ -52,6 +53,9 @@ enum Command { /// Stacks related subcommands #[clap(subcommand)] Stacks(StacksCommand), + /// Generate documentation + #[clap(subcommand)] + Docs(DocsCommand), } #[derive(Subcommand, PartialEq, Clone, Debug)] @@ -248,6 +252,22 @@ struct InitHordDbCommand { pub config_path: Option, } +#[derive(Subcommand, PartialEq, Clone, Debug)] +#[clap(bin_name = "docs", aliases=&["doc"])] +enum DocsCommand { + /// Generate new documentation for the predicate registration API. + #[clap(subcommand)] + #[clap(name = "api")] + Api(ApiDocsCommand), +} + +#[derive(Subcommand, PartialEq, Clone, Debug)] +enum ApiDocsCommand { + /// Generate documentation for the predicate registration API. + #[clap(name = "new", bin_name = "new", aliases = &["generate"])] + Generate, +} + pub fn main() { let logger = hiro_system_kit::log::setup_logger(); let _guard = hiro_system_kit::log::setup_global_logger(logger.clone()); @@ -571,6 +591,25 @@ async fn handle_command(opts: Opts, ctx: Context) -> Result<(), String> { } } }, + Command::Docs(subcmd) => match subcmd { + DocsCommand::Api(api_docs_cmd) => match api_docs_cmd { + ApiDocsCommand::Generate => { + use std::fs::File; + use std::io::Write; + let spec = document_predicate_api_server() + .map_err(|e| format!("unable to generate API docs: {}", e))?; + let mut file_path = PathBuf::new(); + file_path.push("openapi.json"); + let mut file = File::create(&file_path).map_err(|e| { + format!("unable to open file {}\n{}", file_path.display(), e) + })?; + file.write_all(spec.as_bytes()).map_err(|e| { + format!("unable to write file {}\n{}", file_path.display(), e) + })?; + println!("Created file openapi.json"); + } + }, + }, } Ok(()) } diff --git a/components/chainhook-cli/src/service/http_api.rs b/components/chainhook-cli/src/service/http_api.rs index aeef58321..5d3b134b0 100644 --- a/components/chainhook-cli/src/service/http_api.rs +++ b/components/chainhook-cli/src/service/http_api.rs @@ -14,8 +14,7 @@ use redis::{Commands, Connection}; use rocket::config::{self, Config, LogLevel}; use rocket::serde::json::{json, Json, Value as JsonValue}; use rocket::State; -use rocket_okapi::openapi; -use rocket_okapi::openapi_get_routes; +use rocket_okapi::{okapi::openapi3::OpenApi, openapi, openapi_get_routes_spec}; use std::error::Error; use crate::config::PredicatesApiConfig; @@ -46,14 +45,7 @@ pub async fn start_predicate_api_server( ..Config::default() }; - let routes = openapi_get_routes![ - handle_ping, - handle_get_predicates, - handle_get_predicate, - handle_create_predicate, - handle_delete_bitcoin_predicate, - handle_delete_stacks_predicate - ]; + let (routes, _) = get_routes_spec(); let background_job_tx_mutex = Arc::new(Mutex::new(observer_commands_tx.clone())); @@ -372,3 +364,21 @@ pub fn load_predicates_from_redis( .map_err(|e| format!("unable to connect to redis: {}", e.to_string()))?; get_entries_from_predicates_db(&mut predicate_db_conn, ctx) } + +pub fn document_predicate_api_server() -> Result { + let (_, spec) = get_routes_spec(); + let json_spec = serde_json::to_string_pretty(&spec) + .map_err(|e| format!("failed to serialize openapi spec: {}", e.to_string()))?; + Ok(json_spec) +} + +pub fn get_routes_spec() -> (Vec, OpenApi) { + openapi_get_routes_spec![ + handle_ping, + handle_get_predicates, + handle_get_predicate, + handle_create_predicate, + handle_delete_bitcoin_predicate, + handle_delete_stacks_predicate + ] +} diff --git a/components/chainhook-cli/src/service/mod.rs b/components/chainhook-cli/src/service/mod.rs index b4e67fd9b..954f85605 100644 --- a/components/chainhook-cli/src/service/mod.rs +++ b/components/chainhook-cli/src/service/mod.rs @@ -1,4 +1,4 @@ -mod http_api; +pub(crate) mod http_api; mod runloops; use crate::config::{Config, PredicatesApi, PredicatesApiConfig}; diff --git a/components/chainhook-sdk/Cargo.toml b/components/chainhook-sdk/Cargo.toml index 7169132b7..70d73884e 100644 --- a/components/chainhook-sdk/Cargo.toml +++ b/components/chainhook-sdk/Cargo.toml @@ -32,7 +32,7 @@ reqwest = { version = "0.11", default-features = false, features = [ ] } tokio = { version = "1.24", features = ["full"] } base58 = "0.2.0" -schemars = { version = "0.8.10" } +schemars = { version = "0.8.10", git = "https://github.com/hirosystems/schemars.git", branch = "feat-chainhook-fixes" } crossbeam-channel = "0.5.6" futures = "0.3.21" hyper = { version = "0.14.24", features = ["http1", "client"] } diff --git a/components/chainhook-types-rs/Cargo.toml b/components/chainhook-types-rs/Cargo.toml index 7aef34adb..ca79561ce 100644 --- a/components/chainhook-types-rs/Cargo.toml +++ b/components/chainhook-types-rs/Cargo.toml @@ -12,5 +12,5 @@ serde = "1" serde_json = "1" serde_derive = "1" strum = { version = "0.23.0", features = ["derive"] } -schemars = { version = "0.8.10" } +schemars = { version = "0.8.10", git = "https://github.com/hirosystems/schemars.git", branch = "feat-chainhook-fixes" } hex = "0.4.3" diff --git a/docs/chainhook-openapi.json b/docs/chainhook-openapi.json index b98aea0fc..e720b9e3d 100644 --- a/docs/chainhook-openapi.json +++ b/docs/chainhook-openapi.json @@ -1,677 +1,738 @@ { - "openapi": "3.0.0", - "info": { - "title": "Chainhook", - "description": "Stateless Transaction Indexing Engine for Stacks and Bitcoin", - "version": "1.0.6" - }, - "paths": { - "/v1/chainhooks": { - "get": { - "tags": [ - "Chainhooks" - ], - "operationId": "handle_get_hooks", - "responses": { - "200": { - "description": "", - "content": { - "application/json": { - "schema": {} - } + "openapi": "3.0.0", + "info": { + "title": "chainhook", + "version": "0.16.0" + }, + "paths": { + "/ping": { + "get": { + "tags": ["Chainhooks"], + "operationId": "handle_ping", + "responses": { + "200": { + "description": "", + "content": { + "application/json": { + "schema": {} } } } - }, - "post": { - "tags": [ - "Chainhooks" - ], - "operationId": "handle_create_hook", - "requestBody": { + } + } + }, + "/v1/chainhooks": { + "get": { + "tags": ["Chainhooks"], + "operationId": "handle_get_predicates", + "responses": { + "200": { + "description": "", "content": { "application/json": { - "schema": { - "$ref": "#/components/schemas/ChainhookFullSpecification" - } - } - }, - "required": true - }, - "responses": { - "200": { - "description": "", - "content": { - "application/json": { - "schema": {} - } + "schema": {} } } } } }, - "/v1/chainhooks/bitcoin/{hook_uuid}": { - "delete": { - "tags": [ - "Chainhooks" - ], - "operationId": "handle_delete_bitcoin_hook", - "parameters": [ - { - "name": "hook_uuid", - "in": "path", - "required": true, + "post": { + "tags": ["Chainhooks"], + "operationId": "handle_create_predicate", + "requestBody": { + "content": { + "application/json": { "schema": { - "type": "string" + "$ref": "#/components/schemas/ChainhookFullSpecification" } } - ], - "responses": { - "200": { - "description": "", - "content": { - "application/json": { - "schema": {} - } + }, + "required": true + }, + "responses": { + "200": { + "description": "", + "content": { + "application/json": { + "schema": {} } } } } - }, - "/v1/chainhooks/stacks/{hook_uuid}": { - "delete": { - "tags": [ - "Chainhooks" - ], - "operationId": "handle_delete_stacks_hook", - "parameters": [ - { - "name": "hook_uuid", - "in": "path", - "required": true, - "schema": { - "type": "string" - } + } + }, + "/v1/chainhooks/{predicate_uuid}": { + "get": { + "tags": ["Chainhooks"], + "operationId": "handle_get_predicate", + "parameters": [ + { + "name": "predicate_uuid", + "in": "path", + "required": true, + "schema": { + "type": "string" } - ], - "responses": { - "200": { - "description": "", - "content": { - "application/json": { - "schema": {} - } + } + ], + "responses": { + "200": { + "description": "", + "content": { + "application/json": { + "schema": {} } } } } } }, - "components": { - "schemas": { - "ChainhookFullSpecification": { - "oneOf": [ - { - "type": "object", - "required": [ - "chain", - "name", - "networks", - "uuid", - "version" - ], - "properties": { - "chain": { - "type": "string", - "enum": [ - "bitcoin" - ] - }, - "uuid": { - "type": "string" - }, - "owner_uuid": { - "type": "string", - "nullable": true - }, - "name": { - "type": "string" - }, - "version": { - "type": "integer", - "format": "uint32", - "minimum": 0.0 - }, - "networks": { - "type": "object", - "additionalProperties": { - "$ref": "#/components/schemas/BitcoinChainhookNetworkSpecification" - } - } - } - }, - { - "type": "object", - "required": [ - "chain", - "name", - "networks", - "uuid", - "version" - ], - "properties": { - "chain": { - "type": "string", - "enum": [ - "stacks" - ] - }, - "uuid": { - "type": "string" - }, - "owner_uuid": { - "type": "string", - "nullable": true - }, - "name": { - "type": "string" - }, - "version": { - "type": "integer", - "format": "uint32", - "minimum": 0.0 - }, - "networks": { - "type": "object", - "additionalProperties": { - "$ref": "#/components/schemas/StacksChainhookNetworkSpecification" - } - } + "/v1/chainhooks/bitcoin/{predicate_uuid}": { + "delete": { + "tags": ["Chainhooks"], + "operationId": "handle_delete_bitcoin_predicate", + "parameters": [ + { + "name": "predicate_uuid", + "in": "path", + "required": true, + "schema": { + "type": "string" + } + } + ], + "responses": { + "200": { + "description": "", + "content": { + "application/json": { + "schema": {} } } - ] - }, - "BitcoinChainhookNetworkSpecification": { - "type": "object", - "required": [ - "if_this", - "then_that" - ], - "properties": { - "start_block": { - "type": "integer", - "format": "uint64", - "minimum": 0.0, - "nullable": true - }, - "end_block": { - "type": "integer", - "format": "uint64", - "minimum": 0.0, - "nullable": true - }, - "expire_after_occurrence": { - "type": "integer", - "format": "uint64", - "minimum": 0.0, - "nullable": true - }, - "if_this": { - "$ref": "#/components/schemas/BitcoinPredicateType" - }, - "then_that": { - "$ref": "#/components/schemas/HookAction" + } + } + } + }, + "/v1/chainhooks/stacks/{predicate_uuid}": { + "delete": { + "tags": ["Chainhooks"], + "operationId": "handle_delete_stacks_predicate", + "parameters": [ + { + "name": "predicate_uuid", + "in": "path", + "required": true, + "schema": { + "type": "string" } } - }, - "BitcoinPredicateType": { - "oneOf": [ - { - "type": "object", - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "block" - ] - } + ], + "responses": { + "200": { + "description": "", + "content": { + "application/json": { + "schema": {} } - }, - { - "type": "object", - "oneOf": [ - { - "type": "object", - "required": [ - "equals" - ], - "properties": { - "equals": { - "type": "string" + } + } + } + } + } + }, + "components": { + "schemas": { + "ChainhookFullSpecification": { + "oneOf": [ + { + "type": "object", + "required": ["chain", "name", "networks", "uuid", "version"], + "properties": { + "chain": { + "type": "string", + "enum": ["bitcoin"] + }, + "uuid": { + "type": "string" + }, + "owner_uuid": { + "type": "string", + "nullable": true + }, + "name": { + "type": "string" + }, + "version": { + "type": "integer", + "format": "uint32", + "minimum": 0.0 + }, + "networks": { + "oneOf": [ + { + "type": "object", + "required": ["regtest"], + "properties": { + "regtest": { + "$ref": "#/components/schemas/BitcoinChainhookNetworkSpecification" + } } }, - "additionalProperties": false - } - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "txid" - ] - } - } - }, - { - "type": "object", - "oneOf": [ - { - "type": "object", - "required": [ - "txid" - ], - "properties": { - "txid": { - "$ref": "#/components/schemas/TxinPredicate" + { + "type": "object", + "required": ["testnet"], + "properties": { + "testnet": { + "$ref": "#/components/schemas/BitcoinChainhookNetworkSpecification" + } } }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "witness_script" - ], - "properties": { - "witness_script": { - "$ref": "#/components/schemas/MatchingRule" + { + "type": "object", + "required": ["mainnet"], + "properties": { + "mainnet": { + "$ref": "#/components/schemas/BitcoinChainhookNetworkSpecification" + } } - }, - "additionalProperties": false - } - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "inputs" - ] - } + } + ] } - }, - { - "type": "object", - "oneOf": [ - { - "type": "object", - "required": [ - "op_return" - ], - "properties": { - "op_return": { - "$ref": "#/components/schemas/MatchingRule" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "p2pkh" - ], - "properties": { - "p2pkh": { - "$ref": "#/components/schemas/ExactMatchingRule" + } + }, + { + "type": "object", + "required": ["chain", "name", "networks", "uuid", "version"], + "properties": { + "chain": { + "type": "string", + "enum": ["stacks"] + }, + "uuid": { + "type": "string" + }, + "owner_uuid": { + "type": "string", + "nullable": true + }, + "name": { + "type": "string" + }, + "version": { + "type": "integer", + "format": "uint32", + "minimum": 0.0 + }, + "networks": { + "oneOf": [ + { + "type": "object", + "required": ["simnet"], + "properties": { + "simnet": { + "$ref": "#/components/schemas/StacksChainhookNetworkSpecification" + } } }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "p2sh" - ], - "properties": { - "p2sh": { - "$ref": "#/components/schemas/ExactMatchingRule" + { + "type": "object", + "required": ["devnet"], + "properties": { + "devnet": { + "$ref": "#/components/schemas/StacksChainhookNetworkSpecification" + } } }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "p2wpkh" - ], - "properties": { - "p2wpkh": { - "$ref": "#/components/schemas/ExactMatchingRule" + { + "type": "object", + "required": ["testnet"], + "properties": { + "testnet": { + "$ref": "#/components/schemas/StacksChainhookNetworkSpecification" + } } }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "p2wsh" - ], - "properties": { - "p2wsh": { - "$ref": "#/components/schemas/ExactMatchingRule" + { + "type": "object", + "required": ["mainnet"], + "properties": { + "mainnet": { + "$ref": "#/components/schemas/StacksChainhookNetworkSpecification" + } } - }, - "additionalProperties": false - } - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "outputs" - ] - } + } + ] } - }, - { - "type": "object", - "oneOf": [ - { - "type": "object", - "required": [ - "stacks" - ], - "properties": { - "stacks": { - "$ref": "#/components/schemas/StacksOperations" - } - }, - "additionalProperties": false + } + } + ] + }, + "BitcoinNetwork": { + "type": "string", + "enum": ["regtest", "testnet", "mainnet"] + }, + "BitcoinChainhookNetworkSpecification": { + "type": "object", + "required": ["if_this", "then_that"], + "properties": { + "start_block": { + "type": "integer", + "format": "uint64", + "minimum": 0.0, + "nullable": true + }, + "end_block": { + "type": "integer", + "format": "uint64", + "minimum": 0.0, + "nullable": true + }, + "expire_after_occurrence": { + "type": "integer", + "format": "uint64", + "minimum": 0.0, + "nullable": true + }, + "include_proof": { + "type": "boolean", + "nullable": true + }, + "include_inputs": { + "type": "boolean", + "nullable": true + }, + "include_outputs": { + "type": "boolean", + "nullable": true + }, + "include_witness": { + "type": "boolean", + "nullable": true + }, + "if_this": { + "$ref": "#/components/schemas/BitcoinPredicateType" + }, + "then_that": { + "$ref": "#/components/schemas/HookAction" + } + } + }, + "BitcoinPredicateType": { + "oneOf": [ + { + "type": "object", + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["block"] + } + } + }, + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["equals"], + "properties": { + "equals": { + "type": "string" + } }, - { - "type": "object", - "required": [ - "ordinal" - ], - "properties": { - "ordinal": { - "$ref": "#/components/schemas/OrdinalOperations" - } - }, - "additionalProperties": false - } - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "protocol" - ] - } + "additionalProperties": false + } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["txid"] } } - ] - }, - "TxinPredicate": { - "type": "object", - "required": [ - "txid", - "vout" - ], - "properties": { - "txid": { - "type": "string" - }, - "vout": { - "type": "integer", - "format": "uint32", - "minimum": 0.0 + }, + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["txid"], + "properties": { + "txid": { + "$ref": "#/components/schemas/TxinPredicate" + } + }, + "additionalProperties": false + }, + { + "type": "object", + "required": ["witness_script"], + "properties": { + "witness_script": { + "$ref": "#/components/schemas/MatchingRule" + } + }, + "additionalProperties": false + } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["inputs"] + } } - } - }, - "MatchingRule": { - "oneOf": [ - { - "type": "object", - "required": [ - "equals" - ], - "properties": { - "equals": { - "type": "string" - } + }, + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["op_return"], + "properties": { + "op_return": { + "$ref": "#/components/schemas/MatchingRule" + } + }, + "additionalProperties": false }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "starts_with" - ], - "properties": { - "starts_with": { - "type": "string" - } + { + "type": "object", + "required": ["p2pkh"], + "properties": { + "p2pkh": { + "$ref": "#/components/schemas/ExactMatchingRule" + } + }, + "additionalProperties": false }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "ends_with" - ], - "properties": { - "ends_with": { - "type": "string" - } + { + "type": "object", + "required": ["p2sh"], + "properties": { + "p2sh": { + "$ref": "#/components/schemas/ExactMatchingRule" + } + }, + "additionalProperties": false }, - "additionalProperties": false + { + "type": "object", + "required": ["p2wpkh"], + "properties": { + "p2wpkh": { + "$ref": "#/components/schemas/ExactMatchingRule" + } + }, + "additionalProperties": false + }, + { + "type": "object", + "required": ["p2wsh"], + "properties": { + "p2wsh": { + "$ref": "#/components/schemas/ExactMatchingRule" + } + }, + "additionalProperties": false + } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["outputs"] + } } - ] - }, - "ExactMatchingRule": { - "oneOf": [ - { - "type": "object", - "required": [ - "equals" - ], - "properties": { - "equals": { - "type": "string" + }, + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["operation"], + "properties": { + "operation": { + "type": "string", + "enum": ["stacker_rewarded"] + } } }, - "additionalProperties": false - } - ] - }, - "StacksOperations": { - "type": "string", - "enum": [ - "stacker_rewarded", - "block_committed", - "leader_registered", - "stx_transfered", - "stx_locked" - ] - }, - "OrdinalOperations": { - "type": "string", - "enum": [ - "inscription_feed" - ] - }, - "HookAction": { - "oneOf": [ - { - "type": "string", - "enum": [ - "noop" - ] - }, - { - "type": "object", - "required": [ - "http_post" - ], - "properties": { - "http_post": { - "$ref": "#/components/schemas/HttpHook" + { + "type": "object", + "required": ["operation"], + "properties": { + "operation": { + "type": "string", + "enum": ["block_committed"] + } } }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "file_append" - ], - "properties": { - "file_append": { - "$ref": "#/components/schemas/FileHook" + { + "type": "object", + "required": ["operation"], + "properties": { + "operation": { + "type": "string", + "enum": ["leader_registered"] + } + } + }, + { + "type": "object", + "required": ["operation"], + "properties": { + "operation": { + "type": "string", + "enum": ["stx_transferred"] + } } }, - "additionalProperties": false + { + "type": "object", + "required": ["operation"], + "properties": { + "operation": { + "type": "string", + "enum": ["stx_locked"] + } + } + } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["stacks_protocol"] + } } - ] - }, - "HttpHook": { - "type": "object", - "required": [ - "authorization_header", - "url" - ], - "properties": { - "url": { - "type": "string" - }, - "authorization_header": { - "type": "string" + }, + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["operation"], + "properties": { + "operation": { + "type": "string", + "enum": ["inscription_feed"] + } + } + } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["ordinals_protocol"] + } } } - }, - "FileHook": { - "type": "object", - "required": [ - "path" - ], - "properties": { - "path": { - "type": "string" - } + ] + }, + "TxinPredicate": { + "type": "object", + "required": ["txid", "vout"], + "properties": { + "txid": { + "type": "string" + }, + "vout": { + "type": "integer", + "format": "uint32", + "minimum": 0.0 } - }, - "StacksChainhookNetworkSpecification": { - "type": "object", - "required": [ - "if_this", - "then_that" - ], - "properties": { - "start_block": { - "type": "integer", - "format": "uint64", - "minimum": 0.0, - "nullable": true + } + }, + "MatchingRule": { + "oneOf": [ + { + "type": "object", + "required": ["equals"], + "properties": { + "equals": { + "type": "string" + } }, - "end_block": { - "type": "integer", - "format": "uint64", - "minimum": 0.0, - "nullable": true + "additionalProperties": false + }, + { + "type": "object", + "required": ["starts_with"], + "properties": { + "starts_with": { + "type": "string" + } }, - "expire_after_occurrence": { - "type": "integer", - "format": "uint64", - "minimum": 0.0, - "nullable": true + "additionalProperties": false + }, + { + "type": "object", + "required": ["ends_with"], + "properties": { + "ends_with": { + "type": "string" + } }, - "capture_all_events": { - "type": "boolean", - "nullable": true + "additionalProperties": false + } + ] + }, + "ExactMatchingRule": { + "oneOf": [ + { + "type": "object", + "required": ["equals"], + "properties": { + "equals": { + "type": "string" + } }, - "decode_clarity_values": { - "type": "boolean", - "nullable": true + "additionalProperties": false + } + ] + }, + "HookAction": { + "oneOf": [ + { + "type": "string", + "enum": ["noop"] + }, + { + "type": "object", + "required": ["http_post"], + "properties": { + "http_post": { + "$ref": "#/components/schemas/HttpHook" + } }, - "if_this": { - "$ref": "#/components/schemas/StacksPredicate" + "additionalProperties": false + }, + { + "type": "object", + "required": ["file_append"], + "properties": { + "file_append": { + "$ref": "#/components/schemas/FileHook" + } }, - "then_that": { - "$ref": "#/components/schemas/HookAction" - } + "additionalProperties": false } - }, - "StacksPredicate": { - "oneOf": [ - { - "type": "object", - "oneOf": [ - { - "type": "object", - "required": [ - "equals" - ], - "properties": { - "equals": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false + ] + }, + "HttpHook": { + "type": "object", + "required": ["authorization_header", "url"], + "properties": { + "url": { + "type": "string" + }, + "authorization_header": { + "type": "string" + } + } + }, + "FileHook": { + "type": "object", + "required": ["path"], + "properties": { + "path": { + "type": "string" + } + } + }, + "StacksNetwork": { + "type": "string", + "enum": ["simnet", "devnet", "testnet", "mainnet"] + }, + "StacksChainhookNetworkSpecification": { + "type": "object", + "required": ["if_this", "then_that"], + "properties": { + "start_block": { + "type": "integer", + "format": "uint64", + "minimum": 0.0, + "nullable": true + }, + "end_block": { + "type": "integer", + "format": "uint64", + "minimum": 0.0, + "nullable": true + }, + "expire_after_occurrence": { + "type": "integer", + "format": "uint64", + "minimum": 0.0, + "nullable": true + }, + "capture_all_events": { + "type": "boolean", + "nullable": true + }, + "decode_clarity_values": { + "type": "boolean", + "nullable": true + }, + "if_this": { + "$ref": "#/components/schemas/StacksPredicate" + }, + "then_that": { + "$ref": "#/components/schemas/HookAction" + } + } + }, + "StacksPredicate": { + "oneOf": [ + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["equals"], + "properties": { + "equals": { + "type": "integer", + "format": "uint64", + "minimum": 0.0 + } }, - { - "type": "object", - "required": [ - "higher_than" - ], - "properties": { - "higher_than": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false + "additionalProperties": false + }, + { + "type": "object", + "required": ["higher_than"], + "properties": { + "higher_than": { + "type": "integer", + "format": "uint64", + "minimum": 0.0 + } }, - { - "type": "object", - "required": [ - "lower_than" - ], - "properties": { - "lower_than": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false + "additionalProperties": false + }, + { + "type": "object", + "required": ["lower_than"], + "properties": { + "lower_than": { + "type": "integer", + "format": "uint64", + "minimum": 0.0 + } }, - { - "type": "object", - "required": [ - "between" - ], - "properties": { - "between": { - "type": "array", - "items": [ + "additionalProperties": false + }, + { + "type": "object", + "required": ["between"], + "properties": { + "between": { + "type": "array", + "items": { + "anyOf": [ { "type": "integer", "format": "uint64", @@ -682,195 +743,159 @@ "format": "uint64", "minimum": 0.0 } - ], - "maxItems": 2, - "minItems": 2 - } - }, - "additionalProperties": false - } - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "block_height" - ] - } - } - }, - { - "type": "object", - "oneOf": [ - { - "type": "string", - "enum": [ - "implement_sip09", - "implement_sip10" - ] + ] + }, + "maxItems": 2, + "minItems": 2 + } }, - { - "type": "object", - "required": [ - "deployer" - ], - "properties": { - "deployer": { - "type": "string" - } - }, - "additionalProperties": false - } - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "contract_deployment" - ] - } + "additionalProperties": false } - }, - { - "type": "object", - "required": [ - "contract_identifier", - "method", - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "contract_call" - ] - }, - "contract_identifier": { - "type": "string" - }, - "method": { - "type": "string" - } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["block_height"] } - }, - { - "type": "object", - "required": [ - "contains", - "contract_identifier", - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "print_event" - ] - }, - "contract_identifier": { - "type": "string" + } + }, + { + "type": "object", + "oneOf": [ + { + "type": "string", + "enum": ["implement_sip09", "implement_sip10"] + }, + { + "type": "object", + "required": ["deployer"], + "properties": { + "deployer": { + "type": "string" + } }, - "contains": { + "additionalProperties": false + } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["contract_deployment"] + } + } + }, + { + "type": "object", + "required": ["contract_identifier", "method", "scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["contract_call"] + }, + "contract_identifier": { + "type": "string" + }, + "method": { + "type": "string" + } + } + }, + { + "type": "object", + "required": ["contains", "contract_identifier", "scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["print_event"] + }, + "contract_identifier": { + "type": "string" + }, + "contains": { + "type": "string" + } + } + }, + { + "type": "object", + "required": ["actions", "asset_identifier", "scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["ft_event"] + }, + "asset_identifier": { + "type": "string" + }, + "actions": { + "type": "array", + "items": { "type": "string" } } - }, - { - "type": "object", - "required": [ - "actions", - "asset_identifier", - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "ft_event" - ] - }, - "asset_identifier": { + } + }, + { + "type": "object", + "required": ["actions", "asset_identifier", "scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["nft_event"] + }, + "asset_identifier": { + "type": "string" + }, + "actions": { + "type": "array", + "items": { "type": "string" - }, - "actions": { - "type": "array", - "items": { - "type": "string" - } } } - }, - { - "type": "object", - "required": [ - "actions", - "asset_identifier", - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "nft_event" - ] - }, - "asset_identifier": { + } + }, + { + "type": "object", + "required": ["actions", "scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["stx_event"] + }, + "actions": { + "type": "array", + "items": { "type": "string" - }, - "actions": { - "type": "array", - "items": { - "type": "string" - } } } - }, - { - "type": "object", - "required": [ - "actions", - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "stx_event" - ] - }, - "actions": { - "type": "array", - "items": { + } + }, + { + "type": "object", + "oneOf": [ + { + "type": "object", + "required": ["equals"], + "properties": { + "equals": { "type": "string" } - } + }, + "additionalProperties": false } - }, - { - "type": [ - "object", - "string" - ], - "required": [ - "scope" - ], - "properties": { - "scope": { - "type": "string", - "enum": [ - "txid" - ] - } + ], + "required": ["scope"], + "properties": { + "scope": { + "type": "string", + "enum": ["txid"] } } - ] - } + } + ] } } - } \ No newline at end of file + } +}