diff --git a/.github/workflows/release-please.yml b/.github/workflows/release-please.yml index a18e5c8..13e8091 100644 --- a/.github/workflows/release-please.yml +++ b/.github/workflows/release-please.yml @@ -49,12 +49,6 @@ jobs: with: repo-token: ${{ secrets.GITHUB_TOKEN }} - - uses: Swatinem/rust-cache@v2 - if: ${{ needs.release-please.outputs.release_created }} - with: - cache-directories: | - ~/.cargo/registry/src/**/librocksdb-sys-* - - name: Publish crate if: ${{ needs.release-please.outputs.release_created }} uses: actions-rs/cargo@v1 diff --git a/.github/workflows/rust.yml b/.github/workflows/rust.yml index 26fdd45..0193fdd 100644 --- a/.github/workflows/rust.yml +++ b/.github/workflows/rust.yml @@ -79,11 +79,6 @@ jobs: with: repo-token: ${{ secrets.GITHUB_TOKEN }} - - uses: Swatinem/rust-cache@v2 - with: - cache-directories: | - ~/.cargo/registry/src/**/librocksdb-sys-* - - name: Run cargo-tarpaulin uses: actions-rs/tarpaulin@v0.1 with: @@ -96,3 +91,29 @@ jobs: verbose: true fail_ci_if_error: true token: ${{ secrets.CODECOV_TOKEN }} + + Schema-Unchanged: # ensure schema is not changed + needs: Formatting + steps: + - name: Checkout repository + uses: actions/checkout@v3 + with: + lfs: true + + - name: Install stable toolchain + uses: actions-rs/toolchain@v1 + with: + toolchain: stable + override: true + + - name: Setup protoc + uses: arduino/setup-protoc@v1.1.2 + with: + repo-token: ${{ secrets.GITHUB_TOKEN }} + + - name: Check schema + run: | + cargo run --bin check-schema + cargo run -- server schema --output-file /tmp/openapi.yaml + set -e + diff openapi.yaml /tmp/openapi.yaml diff --git a/Cargo.lock b/Cargo.lock index ea4c723..414db58 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -30,7 +30,7 @@ dependencies = [ "actix-service", "actix-utils", "ahash", - "base64", + "base64 0.21.5", "bitflags 2.4.1", "brotli", "bytes", @@ -40,7 +40,7 @@ dependencies = [ "flate2", "futures-core", "h2", - "http", + "http 0.2.11", "httparse", "httpdate", "itoa", @@ -75,7 +75,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d66ff4d247d2b160861fa2866457e85706833527840e4133f8f49aa423a38799" dependencies = [ "bytestring", - "http", + "http 0.2.11", "regex", "serde", "tracing", @@ -315,6 +315,15 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "arbitrary" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7d5a26814d8dcb93b0e5a0ff3c6d80a8843bafb21b39e8e18a6f05471870e110" +dependencies = [ + "derive_arbitrary", +] + [[package]] name = "arc-swap" version = "1.6.0" @@ -359,6 +368,12 @@ version = "0.21.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "35636a1494ede3b646cc98f74f8e62c773a38a659ebc777a2cf26b9b74171df9" +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + [[package]] name = "bitflags" version = "1.3.2" @@ -583,9 +598,9 @@ dependencies = [ [[package]] name = "crc32fast" -version = "1.3.2" +version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b540bd8bc810d3885c6ea91e2018302f68baba2129ab3e88f32389ee9370880d" +checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" dependencies = [ "cfg-if", ] @@ -625,12 +640,9 @@ dependencies = [ [[package]] name = "crossbeam-utils" -version = "0.8.17" +version = "0.8.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c06d96137f14f244c37f989d9fff8f95e6c18b918e71f36638f8c49112e4c78f" -dependencies = [ - "cfg-if", -] +checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" [[package]] name = "crunchy" @@ -714,6 +726,17 @@ dependencies = [ "serde", ] +[[package]] +name = "derive_arbitrary" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67e77553c4162a157adbf834ebae5b415acbecbeafc7a74b0e886657506a7611" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "derive_more" version = "0.99.17" @@ -758,6 +781,17 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "displaydoc" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "downcast-rs" version = "1.2.0" @@ -1071,7 +1105,7 @@ dependencies = [ "futures-core", "futures-sink", "futures-util", - "http", + "http 0.2.11", "indexmap", "slab", "tokio", @@ -1146,6 +1180,40 @@ dependencies = [ "itoa", ] +[[package]] +name = "http" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21b9ddb458710bc376481b842f5da65cdf31522de232c1ca8146abce2a358258" +dependencies = [ + "bytes", + "fnv", + "itoa", +] + +[[package]] +name = "http-body" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1cac85db508abc24a2e48553ba12a996e87244a0395ce011e62b37158745d643" +dependencies = [ + "bytes", + "http 1.1.0", +] + +[[package]] +name = "http-body-util" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" +dependencies = [ + "bytes", + "futures-util", + "http 1.1.0", + "http-body", + "pin-project-lite", +] + [[package]] name = "httparse" version = "1.8.0" @@ -1164,6 +1232,63 @@ version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" +[[package]] +name = "hyper" +version = "1.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50dfd22e0e76d0f662d429a5f80fcaf3855009297eab6a0a9f8543834744ba05" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http 1.1.0", + "http-body", + "httparse", + "itoa", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.27.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ee4be2c948921a1a5320b629c4193916ed787a7f7f293fd3f7f5a6c9de74155" +dependencies = [ + "futures-util", + "http 1.1.0", + "hyper", + "hyper-util", + "rustls", + "rustls-pki-types", + "tokio", + "tokio-rustls", + "tower-service", + "webpki-roots", +] + +[[package]] +name = "hyper-util" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3ab92f4f49ee4fb4f997c784b7a2e0fa70050211e0b6a287f898c3c9785ca956" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http 1.1.0", + "http-body", + "hyper", + "pin-project-lite", + "socket2", + "tokio", + "tower", + "tower-service", + "tracing", +] + [[package]] name = "ident_case" version = "1.0.1" @@ -1231,6 +1356,12 @@ dependencies = [ "web-sys", ] +[[package]] +name = "ipnet" +version = "2.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f518f335dce6725a761382244631d86cf0ccb2863413590b31338feb467f9c3" + [[package]] name = "is-terminal" version = "0.4.9" @@ -1467,6 +1598,16 @@ version = "0.3.17" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" +[[package]] +name = "mime_guess" +version = "2.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7c44f8e672c00fe5308fa235f821cb4198414e1c77935c1ab6948d3fd78550e" +dependencies = [ + "mime", + "unicase", +] + [[package]] name = "minimal-lexical" version = "0.2.1" @@ -1605,6 +1746,27 @@ dependencies = [ "libc", ] +[[package]] +name = "num_enum" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02339744ee7253741199f897151b38e72257d13802d4ee837285cc2990a90845" +dependencies = [ + "num_enum_derive", +] + +[[package]] +name = "num_enum_derive" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "681030a937600a36906c185595136d26abfebb4aa9c65701cefcaf8578bb982b" +dependencies = [ + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "number_prefix" version = "0.4.0" @@ -1755,6 +1917,26 @@ dependencies = [ "indexmap", ] +[[package]] +name = "pin-project" +version = "1.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6bf43b791c5b9e34c3d182969b4abb522f9343702850a2e57f460d00d09b4b3" +dependencies = [ + "pin-project-internal", +] + +[[package]] +name = "pin-project-internal" +version = "1.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f38a4412a78282e09a2cf38d195ea5420d15ba0602cb375210efbc877243965" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "pin-project-lite" version = "0.2.13" @@ -1897,6 +2079,53 @@ dependencies = [ "prost", ] +[[package]] +name = "quinn" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e4ceeeeabace7857413798eb1ffa1e9c905a9946a57d81fb69b4b71c4d8eb3ad" +dependencies = [ + "bytes", + "pin-project-lite", + "quinn-proto", + "quinn-udp", + "rustc-hash", + "rustls", + "thiserror", + "tokio", + "tracing", +] + +[[package]] +name = "quinn-proto" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ddf517c03a109db8100448a4be38d498df8a210a99fe0e1b9eaf39e78c640efe" +dependencies = [ + "bytes", + "rand 0.8.5", + "ring", + "rustc-hash", + "rustls", + "slab", + "thiserror", + "tinyvec", + "tracing", +] + +[[package]] +name = "quinn-udp" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9096629c45860fc7fb143e125eb826b5e721e10be3263160c7d60ca832cf8c46" +dependencies = [ + "libc", + "once_cell", + "socket2", + "tracing", + "windows-sys 0.52.0", +] + [[package]] name = "quote" version = "1.0.35" @@ -2088,6 +2317,64 @@ dependencies = [ "winapi", ] +[[package]] +name = "reqwest" +version = "0.12.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7d6d2a27d57148378eb5e111173f4276ad26340ecc5c49a4a2152167a2d6a37" +dependencies = [ + "base64 0.22.1", + "bytes", + "futures-channel", + "futures-core", + "futures-util", + "http 1.1.0", + "http-body", + "http-body-util", + "hyper", + "hyper-rustls", + "hyper-util", + "ipnet", + "js-sys", + "log", + "mime", + "once_cell", + "percent-encoding", + "pin-project-lite", + "quinn", + "rustls", + "rustls-pemfile", + "rustls-pki-types", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper", + "tokio", + "tokio-rustls", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "webpki-roots", + "winreg", +] + +[[package]] +name = "ring" +version = "0.17.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c17fa4cb658e3583423e915b9f3acc01cceaee1860e33d59ebae66adc3a2dc0d" +dependencies = [ + "cc", + "cfg-if", + "getrandom", + "libc", + "spin", + "untrusted", + "windows-sys 0.52.0", +] + [[package]] name = "rstest" version = "0.21.0" @@ -2118,6 +2405,40 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "rust-embed" +version = "8.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa66af4a4fdd5e7ebc276f115e895611a34739a9c1c01028383d612d550953c0" +dependencies = [ + "rust-embed-impl", + "rust-embed-utils", + "walkdir", +] + +[[package]] +name = "rust-embed-impl" +version = "8.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6125dbc8867951125eec87294137f4e9c2c96566e61bf72c45095a7c77761478" +dependencies = [ + "proc-macro2", + "quote", + "rust-embed-utils", + "syn 2.0.48", + "walkdir", +] + +[[package]] +name = "rust-embed-utils" +version = "8.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e5347777e9aacb56039b0e1f28785929a8a3b709e87482e7442c72e7c12529d" +dependencies = [ + "sha2", + "walkdir", +] + [[package]] name = "rust-stemmers" version = "1.2.0" @@ -2162,6 +2483,47 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "rustls" +version = "0.23.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4828ea528154ae444e5a642dbb7d5623354030dc9822b83fd9bb79683c7399d0" +dependencies = [ + "once_cell", + "ring", + "rustls-pki-types", + "rustls-webpki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls-pemfile" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29993a25686778eb88d4189742cd713c9bce943bc54251a33509dc63cbacf73d" +dependencies = [ + "base64 0.22.1", + "rustls-pki-types", +] + +[[package]] +name = "rustls-pki-types" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "976295e77ce332211c0d24d92c0e83e50f5c5f046d11082cea19f3df13a3562d" + +[[package]] +name = "rustls-webpki" +version = "0.102.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9a6fccd794a42c2c105b513a2f62bc3fd8f3ba57a4593677ceb0bd035164d78" +dependencies = [ + "ring", + "rustls-pki-types", + "untrusted", +] + [[package]] name = "rustversion" version = "1.0.14" @@ -2183,6 +2545,15 @@ dependencies = [ "bytemuck", ] +[[package]] +name = "same-file" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502" +dependencies = [ + "winapi-util", +] + [[package]] name = "scoped-tls" version = "1.0.1" @@ -2259,7 +2630,7 @@ version = "3.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1b0ed1662c5a68664f45b76d18deb0e234aff37207086803165c961eb695e981" dependencies = [ - "base64", + "base64 0.21.5", "chrono", "hex", "indexmap", @@ -2281,6 +2652,19 @@ dependencies = [ "syn 2.0.48", ] +[[package]] +name = "serde_yaml" +version = "0.9.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1bf28c79a99f70ee1f1d83d10c875d2e70618417fda01ad1785e027579d9d38" +dependencies = [ + "indexmap", + "itoa", + "ryu", + "serde", + "unsafe-libyaml", +] + [[package]] name = "sha1" version = "0.10.6" @@ -2369,9 +2753,9 @@ dependencies = [ [[package]] name = "smallvec" -version = "1.11.2" +version = "1.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4dccd0940a2dcdf68d092b8cbab7dc0ad8fa938bf95787e1b916b0e3d0e8e970" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" [[package]] name = "smartstring" @@ -2394,6 +2778,12 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "spin" +version = "0.9.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" + [[package]] name = "stable_deref_trait" version = "1.2.0" @@ -2447,6 +2837,12 @@ dependencies = [ "syn 2.0.48", ] +[[package]] +name = "subtle" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" + [[package]] name = "syn" version = "1.0.109" @@ -2469,6 +2865,12 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "sync_wrapper" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" + [[package]] name = "tantivy" version = "0.21.1" @@ -2478,7 +2880,7 @@ dependencies = [ "aho-corasick", "arc-swap", "async-trait", - "base64", + "base64 0.21.5", "bitpacking", "byteorder", "census", @@ -2738,6 +3140,17 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "tokio-rustls" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c7bc40d0e5a97695bb96e27995cd3a08538541b0a846f65bba7a359f36700d4" +dependencies = [ + "rustls", + "rustls-pki-types", + "tokio", +] + [[package]] name = "tokio-util" version = "0.7.10" @@ -2769,6 +3182,33 @@ dependencies = [ "winnow", ] +[[package]] +name = "tower" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8fa9be0de6cf49e536ce1851f987bd21a43b771b09473c3549a6c853db37c1c" +dependencies = [ + "futures-core", + "futures-util", + "pin-project", + "pin-project-lite", + "tokio", + "tower-layer", + "tower-service", +] + +[[package]] +name = "tower-layer" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c20c8dbed6283a09604c3e69b4b7eeb54e298b8a600d4d5ecb5ad39de609f1d0" + +[[package]] +name = "tower-service" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6bc1c9ce2b5135ac7f93c72918fc37feb872bdc6a5533a8b85eb4b86bfdae52" + [[package]] name = "tracing" version = "0.1.40" @@ -2831,6 +3271,12 @@ dependencies = [ "tracing-log", ] +[[package]] +name = "try-lock" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" + [[package]] name = "typenum" version = "1.17.0" @@ -2843,6 +3289,15 @@ version = "0.1.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ed646292ffc8188ef8ea4d1e0e0150fb15a5c2e12ad9b8fc191ae7a8a7f3c4b9" +[[package]] +name = "unicase" +version = "2.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7d2d4dafb69621809a81864c9c1b864479e1235c0dd4e199924b9742439ed89" +dependencies = [ + "version_check", +] + [[package]] name = "unicode-bidi" version = "0.3.14" @@ -2870,6 +3325,18 @@ version = "0.1.11" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e51733f11c9c4f72aa0c160008246859e340b00807569a0da0e7a1079b27ba85" +[[package]] +name = "unsafe-libyaml" +version = "0.2.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "673aac59facbab8a9007c7f6108d11f63b603f7cabff99fabf650fea5c32b861" + +[[package]] +name = "untrusted" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" + [[package]] name = "url" version = "2.5.0" @@ -2893,6 +3360,50 @@ version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "711b9620af191e0cdc7468a8d14e709c3dcdb115b36f838e601583af800a370a" +[[package]] +name = "utoipa" +version = "4.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c5afb1a60e207dca502682537fefcfd9921e71d0b83e9576060f09abc6efab23" +dependencies = [ + "indexmap", + "serde", + "serde_json", + "serde_yaml", + "utoipa-gen", +] + +[[package]] +name = "utoipa-gen" +version = "4.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7bf0e16c02bc4bf5322ab65f10ab1149bdbcaa782cba66dc7057370a3f8190be" +dependencies = [ + "proc-macro-error", + "proc-macro2", + "quote", + "regex", + "syn 2.0.48", +] + +[[package]] +name = "utoipa-swagger-ui" +version = "7.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "943e0ff606c6d57d410fd5663a4d7c074ab2c5f14ab903b9514565e59fa1189e" +dependencies = [ + "actix-web", + "mime_guess", + "regex", + "reqwest", + "rust-embed", + "serde", + "serde_json", + "url", + "utoipa", + "zip", +] + [[package]] name = "uuid" version = "1.6.1" @@ -2957,6 +3468,27 @@ dependencies = [ "thiserror", "tracing", "tracing-subscriber", + "utoipa", + "utoipa-swagger-ui", +] + +[[package]] +name = "walkdir" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b" +dependencies = [ + "same-file", + "winapi-util", +] + +[[package]] +name = "want" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" +dependencies = [ + "try-lock", ] [[package]] @@ -2990,6 +3522,18 @@ dependencies = [ "wasm-bindgen-shared", ] +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.39" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac36a15a220124ac510204aec1c3e5db8a22ab06fd6706d881dc6149f8ed9a12" +dependencies = [ + "cfg-if", + "js-sys", + "wasm-bindgen", + "web-sys", +] + [[package]] name = "wasm-bindgen-macro" version = "0.2.89" @@ -3029,6 +3573,15 @@ dependencies = [ "wasm-bindgen", ] +[[package]] +name = "webpki-roots" +version = "0.26.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd7c23921eeb1713a4e851530e9b9756e4fb0e89978582942612524cf09f01cd" +dependencies = [ + "rustls-pki-types", +] + [[package]] name = "which" version = "4.4.2" @@ -3232,6 +3785,16 @@ dependencies = [ "memchr", ] +[[package]] +name = "winreg" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a277a57398d4bfa075df44f501a17cfdf8542d224f0d36095a2adc7aee4ef0a5" +dependencies = [ + "cfg-if", + "windows-sys 0.48.0", +] + [[package]] name = "yaml-rust" version = "0.4.5" @@ -3261,6 +3824,28 @@ dependencies = [ "syn 2.0.48", ] +[[package]] +name = "zeroize" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" + +[[package]] +name = "zip" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9cc23c04387f4da0374be4533ad1208cbb091d5c11d070dfef13676ad6497164" +dependencies = [ + "arbitrary", + "crc32fast", + "crossbeam-utils", + "displaydoc", + "flate2", + "indexmap", + "num_enum", + "thiserror", +] + [[package]] name = "zstd" version = "0.12.4" diff --git a/Cargo.toml b/Cargo.toml index 5a23ad7..2647785 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -3,16 +3,16 @@ name = "viguno" version = "0.2.1" edition = "2021" authors = ["Manuel Holtgrewe "] -description = "Rust template repository" +description = "Phenotype/disease for VarFish" license = "Apache-2.0" -homepage = "https://github.com/varfish-org/viguno" +repository = "https://github.com/varfish-org/viguno" readme = "README.md" [dependencies] actix-web = "4.5" anyhow = "1.0" -clap = { version = "4.4", features = ["derive", "help", "env"] } clap-verbosity-flag = "2.1" +clap = { version = "4.4", features = ["derive", "help", "env"] } console = "0.15" csv = "1.3" derive_more = "0.99" @@ -30,17 +30,19 @@ once_cell = "1.18" prost = "0.12" rayon = "1.8" regex = "1.10" -serde = { version = "1.0", features = ["serde_derive"] } serde_json = "1.0" +serde = { version = "1.0", features = ["serde_derive"] } serde_with = { version = "3.6", features=["alloc", "macros", "indexmap_2"], default-features = false } shellexpand = "3.0" -strum = { version = "0.26", features = ["strum_macros", "derive"] } strum_macros = "0.26" +strum = { version = "0.26", features = ["strum_macros", "derive"] } tantivy = "0.21.1" tempdir = "0.3.7" thiserror = "1.0" tracing = "0.1" tracing-subscriber = "0.3" +utoipa = { version = "4.2", features = ["actix_extras", "chrono", "indexmap", "preserve_order", "yaml"] } +utoipa-swagger-ui = { version = "7.1.0", features = ["actix-web"] } [build-dependencies] prost-build = "0.12" diff --git a/openapi.yaml b/openapi.yaml new file mode 100644 index 0000000..97c4e34 --- /dev/null +++ b/openapi.yaml @@ -0,0 +1,375 @@ +openapi: 3.0.3 +info: + title: viguno + description: Phenotype/disease for VarFish + contact: + name: Manuel Holtgrewe + email: manuel.holtgrewe@bih-charite.de + license: + name: Apache-2.0 + version: 0.2.1 +paths: + /hpo/genes: + get: + tags: + - hpo_genes + summary: Query for genes in the HPO database. + operationId: handle + parameters: + - name: gene_id + in: query + description: The gene ID to search for. + required: false + schema: + type: string + nullable: true + - name: gene_symbol + in: query + description: The gene symbol to search for. + required: false + schema: + type: string + nullable: true + - name: match_ + in: query + description: The match mode. + required: false + schema: + allOf: + - $ref: '#/components/schemas/Match' + nullable: true + - name: max_results + in: query + description: Maximal number of results to return. + required: false + schema: + type: integer + minimum: 0 + - name: hpo_terms + in: query + description: Whether to include HPO terms. + required: false + schema: + type: boolean + responses: + '200': + description: The query was successful. + content: + application/json: + schema: + $ref: '#/components/schemas/Result' + /hpo/omims: + get: + tags: + - hpo_omims + summary: Query for OMIM diseases in the HPO database. + operationId: handle + parameters: + - name: omim_id + in: query + description: The OMIM ID to search for. + required: false + schema: + type: string + nullable: true + - name: name + in: query + description: The disease name to search for. + required: false + schema: + type: string + nullable: true + - name: match_ + in: query + description: The match mode. + required: false + schema: + allOf: + - $ref: '#/components/schemas/Match' + nullable: true + - name: max_results + in: query + description: Maximal number of results to return. + required: false + schema: + type: integer + minimum: 0 + - name: hpo_terms + in: query + description: Whether to include HPO terms. + required: false + schema: + type: boolean + responses: + '200': + description: The query was successful. + content: + application/json: + schema: + $ref: '#/components/schemas/Result' + /hpo/terms: + get: + tags: + - hpo_terms + summary: Query for terms in the HPO database. + description: |- + # Errors + + In the case that there is an error running the server. + operationId: handle + parameters: + - name: term_id + in: query + description: The term ID to search for. + required: false + schema: + type: string + nullable: true + - name: name + in: query + description: The term name to search for. + required: false + schema: + type: string + nullable: true + - name: max_results + in: query + description: Maximal number of results to return. + required: false + schema: + type: integer + minimum: 0 + - name: genes + in: query + description: Whether to include genes. + required: false + schema: + type: boolean + responses: + '200': + description: The query was successful. + content: + application/json: + schema: + $ref: '#/components/schemas/Result' +components: + schemas: + HpoTerm: + type: object + description: Struct for loading an HPO term from JSON. + required: + - term_id + properties: + term_id: + type: string + description: The term ID. + term_name: + type: string + description: The term name (optional). + nullable: true + IcBasedOn: + type: string + description: |- + Enum for representing the information content kind. + + We replicate what is in the `hpo` create so we can put them on the command line and use + them in HTTP queries more easily. + enum: + - gene + - omim + Match: + type: string + description: Specify how to perform query matches in the API calls. + enum: + - exact + - prefix + - suffix + - contains + Query: + type: object + title: HpoSimTermGeneQuery + description: |- + Parameters for `handle`. + + This allows to compute differences between + + - `terms` -- set of terms to use as query + - `gene_ids` -- set of ids for genes to use as "database", can be NCBI\ + gene ID or HGNC gene ID. + - `gene_symbols` -- set of symbols for genes to use as + "database" + required: + - terms + properties: + terms: + type: array + items: + type: string + description: Set of terms to use as query. + gene_ids: + type: array + items: + type: string + description: The set of ids for genes to use as "database". + nullable: true + gene_symbols: + type: array + items: + type: string + description: The set of symbols for genes to use as "database". + nullable: true + ResponseQuery: + type: object + title: HpoSimTermTermQuery + description: |- + Request as sent together with the response. + + The difference is that the `lhs` and `rhs` fields are replaced by vecs. + required: + - lhs + - rhs + properties: + lhs: + type: array + items: + type: string + description: The one set of HPO terms to compute similarity for. + rhs: + type: array + items: + type: string + description: The second set of HPO terms to compute similarity for. + ic_base: + $ref: '#/components/schemas/IcBasedOn' + similarity: + $ref: '#/components/schemas/SimilarityMethod' + combiner: + $ref: '#/components/schemas/ScoreCombiner' + Result: + type: object + title: HpoSimTermTermResult + description: Result container. + required: + - version + - query + - result + properties: + version: + $ref: '#/components/schemas/Version' + query: + $ref: '#/components/schemas/ResponseQuery' + result: + type: array + items: + $ref: '#/components/schemas/ResultEntry' + description: The resulting records for the scored genes. + ResultEntry: + type: object + title: HpoSimTermTermResultEntry + description: Result entry for `handle`. + required: + - lhs + - rhs + - score + properties: + lhs: + type: string + description: The lhs entry. + rhs: + type: string + description: The rhs entry. + score: + type: number + format: float + description: The similarity score. + ResultGene: + type: object + description: Representation of a gene. + required: + - ncbi_gene_id + - gene_symbol + properties: + ncbi_gene_id: + type: integer + format: int32 + description: The HPO ID. + minimum: 0 + gene_symbol: + type: string + description: The description. + hgnc_id: + type: string + description: The HGNC ID. + nullable: true + ResultHpoTerm: + type: object + description: Representation of an HPO term. + required: + - term_id + - name + properties: + term_id: + type: string + description: The HPO ID. + name: + type: string + description: The term name. + ScoreCombiner: + type: string + description: |- + Representation of the standard combiners from HPO. + + We replicate what is in the `hpo` create so we can put them on the command line and use + them in HTTP queries more easily. + enum: + - fun-sim-avg + - fun-sim-max + - bma + SimilarityMethod: + type: string + description: |- + Enum for representing similarity method to use. + + We replicate what is in the `hpo` create so we can put them on the command line and use + them in HTTP queries more easily. + enum: + - distance-gene + - graph-ic + - information-coefficient + - jc + - lin + - mutation + - relevance + - resnik + TermDetails: + type: object + title: HpoSimTermGeneTermDetails + description: Detailed term scores. + required: + - term_gene + - score + properties: + term_query: + allOf: + - $ref: '#/components/schemas/HpoTerm' + nullable: true + term_gene: + $ref: '#/components/schemas/HpoTerm' + score: + type: number + format: float + description: The similarity score. + Version: + type: object + description: Version information that is returned by the HTTP server. + required: + - hpo + - viguno + properties: + hpo: + type: string + description: Version of the HPO. + viguno: + type: string + description: Version of the `viguno` package. + diff --git a/src/common.rs b/src/common.rs index 7332479..05539b4 100644 --- a/src/common.rs +++ b/src/common.rs @@ -60,6 +60,7 @@ pub fn load_hpo>(path: P) -> Result &'static str { } /// Version information that is returned by the HTTP server. -#[derive(serde::Serialize, serde::Deserialize, Default, Debug, Clone)] +#[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Default, Debug, Clone)] pub struct Version { /// Version of the HPO. pub hpo: String, diff --git a/src/main.rs b/src/main.rs index da01c15..81d95fd 100644 --- a/src/main.rs +++ b/src/main.rs @@ -40,7 +40,24 @@ struct Cli { enum Commands { Convert(crate::convert::Args), Query(crate::query::Args), - RunServer(crate::server::Args), + Server(Server), +} + +/// Clap sub command below "server". +#[derive(Debug, Parser)] +struct Server { + /// The sub command to run + #[clap(subcommand)] + pub command: ServerSubCommands, +} + +/// Sub commands for "server". +#[derive(Debug, Subcommand)] +enum ServerSubCommands { + /// Run the server. + Run(crate::server::run::Args), + /// Dump the schema. + Schema(crate::server::schema::Args), } fn main() -> Result<(), anyhow::Error> { @@ -71,9 +88,14 @@ fn main() -> Result<(), anyhow::Error> { Commands::Query(args) => { query::run(&cli.common, args)?; } - Commands::RunServer(args) => { - server::run(&cli.common, args)?; - } + Commands::Server(cmd_server) => match &cmd_server.command { + ServerSubCommands::Run(args) => { + server::run::run(&cli.common, args)?; + } + ServerSubCommands::Schema(args) => { + server::schema::run(&cli.common, args)?; + } + }, } Ok::<(), anyhow::Error>(()) diff --git a/src/query/mod.rs b/src/query/mod.rs index 8c66915..ad043b3 100644 --- a/src/query/mod.rs +++ b/src/query/mod.rs @@ -38,7 +38,7 @@ pub struct Gene { } /// Struct for loading an HPO term from JSON. -#[derive(Serialize, Deserialize, Debug, Clone)] +#[derive(Serialize, Deserialize, utoipa::ToSchema, Debug, Clone)] pub struct HpoTerm { /// The term ID. pub term_id: String, @@ -49,6 +49,8 @@ pub struct HpoTerm { /// Query result records. pub mod query_result { + use crate::common::Version; + use super::HpoTerm; /// Struct for storing gene information in the result. @@ -66,7 +68,8 @@ pub mod query_result { } /// The performed query. - #[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] + #[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug, Clone)] + #[schema(title = "HpoSimTermGeneQuery")] pub struct Query { /// The query HPO terms. pub terms: Vec, @@ -75,19 +78,21 @@ pub mod query_result { } /// Result container data structure. - #[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] - pub struct Container { + #[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug, Clone)] + #[schema(title = "HpoSimTermGeneResult")] + pub struct Result { /// Version information. - pub version: crate::common::Version, + pub version: Version, /// The original query records. pub query: Query, /// The resulting records for the scored genes. - pub result: Vec, + pub result: Vec, } /// Store score for a record with information on individual terms. - #[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] - pub struct Record { + #[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug, Clone)] + #[schema(title = "HpoSimTermGeneResultEntry")] + pub struct ResultEntry { /// The gene symbol. pub gene_symbol: String, /// The raw Phenomizer score. @@ -98,7 +103,8 @@ pub mod query_result { } /// Detailed term scores. - #[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] + #[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug, Clone)] + #[schema(title = "HpoSimTermGeneTermDetails")] pub struct TermDetails { /// The query HPO term. pub term_query: Option, @@ -138,7 +144,7 @@ pub fn run_query( genes: &Vec<&hpo::annotations::Gene>, hpo: &Ontology, ncbi_to_hgnc: &HashMap, -) -> Result +) -> Result where S: std::hash::BuildHasher, { @@ -155,7 +161,7 @@ where .collect(), genes: Vec::new(), }; - let mut result = query_result::Container { + let mut result = query_result::Result { version: crate::common::Version::new(&hpo.hpo_version()), query, result: Vec::new(), @@ -224,7 +230,7 @@ where hgnc_id: ncbi_to_hgnc.get(&ncbi_gene_id).cloned(), }); - result.result.push(query_result::Record { + result.result.push(query_result::ResultEntry { gene_symbol: gene.name().to_string(), raw_score, terms: Some(terms), diff --git a/src/server/actix_server/mod.rs b/src/server/actix_server/mod.rs deleted file mode 100644 index 6a1dd36..0000000 --- a/src/server/actix_server/mod.rs +++ /dev/null @@ -1,115 +0,0 @@ -//! Implementation of the Actix server. - -pub mod hpo_genes; -pub mod hpo_omims; -pub mod hpo_sim; -pub mod hpo_terms; - -use actix_web::{middleware::Logger, web::Data, App, HttpServer, ResponseError}; -use serde::{Deserialize, Deserializer, Serialize}; - -use super::{Args, WebServerData}; - -#[derive(Debug)] -struct CustomError { - err: anyhow::Error, -} - -impl std::fmt::Display for CustomError { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!(f, "{:?}", self.err) - } -} - -impl CustomError { - fn new(err: anyhow::Error) -> Self { - CustomError { err } - } -} - -impl ResponseError for CustomError {} - -/// Specify how to perform query matches in the API calls. -#[derive(Serialize, Deserialize, Debug, Clone, Copy, Default, PartialEq, Eq)] -#[serde(rename_all = "lowercase")] -enum Match { - #[default] - Exact, - Prefix, - Suffix, - Contains, -} - -/// Representation of a gene. -#[derive( - serde::Deserialize, serde::Serialize, Default, Debug, Clone, PartialOrd, Ord, PartialEq, Eq, -)] -#[serde_with::skip_serializing_none] -struct ResultGene { - /// The HPO ID. - pub ncbi_gene_id: u32, - /// The description. - pub gene_symbol: String, - /// The HGNC ID. - pub hgnc_id: Option, -} - -/// Representation of an HPO term. -#[derive( - serde::Deserialize, serde::Serialize, Default, Debug, Clone, PartialOrd, Ord, PartialEq, Eq, -)] -struct ResultHpoTerm { - /// The HPO ID. - pub term_id: String, - /// The term name. - pub name: String, -} - -/// Helper to deserialize a comma-separated list of strings. -fn vec_str_deserialize<'de, D>(deserializer: D) -> Result, D::Error> -where - D: Deserializer<'de>, -{ - let str_sequence = String::deserialize(deserializer)?; - Ok(str_sequence - .split(',') - .map(std::borrow::ToOwned::to_owned) - .collect()) -} - -/// Helper to deserialize a comma-separated list of strings. -fn option_vec_str_deserialize<'de, D>(deserializer: D) -> Result>, D::Error> -where - D: Deserializer<'de>, -{ - let str_sequence = String::deserialize(deserializer)?; - if str_sequence.is_empty() { - Ok(None) - } else { - Ok(Some( - str_sequence - .split(',') - .map(std::borrow::ToOwned::to_owned) - .collect(), - )) - } -} - -/// Main entry point for running the REST server. -#[allow(clippy::unused_async)] -#[actix_web::main] -pub async fn main(args: &Args, dbs: Data) -> std::io::Result<()> { - HttpServer::new(move || { - App::new() - .app_data(dbs.clone()) - .service(hpo_genes::handle) - .service(hpo_terms::handle) - .service(hpo_omims::handle) - .service(hpo_sim::term_term::handle) - .service(hpo_sim::term_gene::handle) - .wrap(Logger::default()) - }) - .bind((args.listen_host.as_str(), args.listen_port))? - .run() - .await -} diff --git a/src/server/mod.rs b/src/server/mod.rs index 828d17f..e2c4d7a 100644 --- a/src/server/mod.rs +++ b/src/server/mod.rs @@ -1,166 +1,4 @@ -//! Code for running the REST API server. +//! Code for the REST API server. -pub mod actix_server; - -use std::collections::HashMap; - -use clap::Parser; -use hpo::Ontology; - -use crate::common::load_hpo; - -/// Data structure for the web server data. -pub struct WebServerData { - /// The HPO ontology (`hpo` crate). - pub ontology: Ontology, - /// Xlink map from NCBI gene ID to HGNC gene ID. - pub ncbi_to_hgnc: HashMap, - /// Xlink map from HGNC gene ID to NCBI gene ID. - pub hgnc_to_ncbi: HashMap, - /// The full text index over the HPO OBO document. - pub full_text_index: crate::index::Index, -} - -/// Command line arguments for `server pheno` sub command. -#[derive(Parser, Debug)] -#[command(author, version, about = "Run viguno REST API server", long_about = None)] -pub struct Args { - /// Path to the directory with the HPO files. - #[arg(long, required = true)] - pub path_hpo_dir: String, - /// Path to the TSV file with the HGNC xlink data. - #[arg(long, required = true)] - pub path_hgnc_xlink: String, - - /// Whether to suppress printing hints. - #[arg(long, default_value_t = false)] - pub suppress_hints: bool, - - /// IP to listen on. - #[arg(long, default_value = "127.0.0.1")] - pub listen_host: String, - /// Port to listen on. - #[arg(long, default_value_t = 8080)] - pub listen_port: u16, -} - -/// Print some hints via `tracing::info!`. -pub fn print_hints(args: &Args) { - tracing::info!( - "Launching server main on http://{}:{} ...", - args.listen_host.as_str(), - args.listen_port - ); - - // Short-circuit if no hints are to be - if args.suppress_hints { - return; - } - - // The endpoint `/hpo/genes` provides information related to genes by symbol. - tracing::info!( - " try: http://{}:{}/hpo/genes?gene_symbol=TGDS", - args.listen_host.as_str(), - args.listen_port - ); - // Also, you can query `/hpo/genes` by NCBI gene ID and return the HPO terms of the gene. - tracing::info!( - " try: http://{}:{}/hpo/genes?gene_id=23483&hpo_terms=true", - args.listen_host.as_str(), - args.listen_port - ); - // The `/hpo/omims` term provides information on OMIM terms and can include HPO terms for - // the disease. - tracing::info!( - " try: http://{}:{}/hpo/omims?omim_id=616145&hpo_terms=true", - args.listen_host.as_str(), - args.listen_port - ); - // The `/hpo/terms` endpoint allows to query by HPO term ID and optionally return a list of - // genes that are linked to the term. - tracing::info!( - " try: http://{}:{}/hpo/terms?term_id=HP:0000023&genes=true", - args.listen_host.as_str(), - args.listen_port - ); - // We can use `/hpo/sim/term-term` to compute similarity between two HPO term sets `lhs` - // and `rhs` using a similarity metric. - tracing::info!( - " try: http://{}:{}/hpo/sim/term-term?lhs=HP:0001166,HP:0040069&rhs=HP:0005918,\ - HP:0004188", - args.listen_host.as_str(), - args.listen_port - ); - // The endpoint `/hpo/sim/term-gene` allows to compute the same for a list of `terms` and - // `gene_symbols`. - tracing::info!( - " try: http://{}:{}/hpo/sim/term-gene?terms=HP:0001166,HP:0000098&gene_symbols=FBN1,TGDS,TTN", - args.listen_host.as_str(), - args.listen_port - ); -} - -/// Main entry point for `run-server` sub command. -/// -/// # Errors -/// -/// In the case that there is an error running the server. -pub fn run(args_common: &crate::common::Args, args: &Args) -> Result<(), anyhow::Error> { - tracing::info!("args_common = {:?}", &args_common); - tracing::info!("args = {:?}", &args); - - if let Some(level) = args_common.verbose.log_level() { - match level { - log::Level::Trace | log::Level::Debug => { - std::env::set_var("RUST_LOG", "debug"); - env_logger::init_from_env(env_logger::Env::new().default_filter_or("info")); - } - _ => (), - } - } - - // Load data that we need for running the server. - tracing::info!("Loading HPO..."); - let before_loading = std::time::Instant::now(); - let ontology = load_hpo(&args.path_hpo_dir)?; - tracing::info!("...done loading HPO in {:?}", before_loading.elapsed()); - - tracing::info!("Loading HGNC xlink..."); - let before_load_xlink = std::time::Instant::now(); - let ncbi_to_hgnc = crate::common::hgnc_xlink::load_ncbi_to_hgnc(&args.path_hgnc_xlink)?; - let hgnc_to_ncbi = crate::common::hgnc_xlink::inverse_hashmap(&ncbi_to_hgnc); - tracing::info!( - "... done loading HGNC xlink in {:?}", - before_load_xlink.elapsed() - ); - - tracing::info!("Loading HPO OBO..."); - let before_load_obo = std::time::Instant::now(); - let hpo_doc = fastobo::from_file(format!("{}/{}", &args.path_hpo_dir, "hp.obo")) - .map_err(|e| anyhow::anyhow!("Error loading HPO OBO: {}", e))?; - tracing::info!( - "... done loading HPO OBO in {:?}", - before_load_obo.elapsed() - ); - - tracing::info!("Indexing OBO..."); - let before_index_obo = std::time::Instant::now(); - let full_text_index = crate::index::Index::new(hpo_doc) - .map_err(|e| anyhow::anyhow!("Error indexing HPO OBO: {}", e))?; - tracing::info!("... done indexing OBO in {:?}", before_index_obo.elapsed()); - - let data = actix_web::web::Data::new(WebServerData { - ontology, - ncbi_to_hgnc, - hgnc_to_ncbi, - full_text_index, - }); - - // Print the server URL and some hints (the latter: unless suppressed). - print_hints(args); - // Launch the Actix web server. - actix_server::main(args, data)?; - - tracing::info!("All done. Have a nice day!"); - Ok(()) -} +pub mod run; +pub mod schema; diff --git a/src/server/actix_server/hpo_genes.rs b/src/server/run/hpo_genes.rs similarity index 86% rename from src/server/actix_server/hpo_genes.rs rename to src/server/run/hpo_genes.rs index 0e3afdc..a048fc9 100644 --- a/src/server/actix_server/hpo_genes.rs +++ b/src/server/run/hpo_genes.rs @@ -12,7 +12,7 @@ use hpo::{ Ontology, }; -use crate::server::WebServerData; +use crate::{common::Version, server::run::WebServerData}; use super::{CustomError, Match, ResultHpoTerm}; @@ -29,8 +29,11 @@ use super::{CustomError, Match, ResultHpoTerm}; /// The following propery defines how matches are performed: /// /// - `match` -- how to match -#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] -struct Query { +#[derive( + serde::Serialize, serde::Deserialize, utoipa::ToSchema, utoipa::IntoParams, Debug, Clone, +)] +#[schema(title = "HpoGenesQuery")] +pub struct Query { /// The gene ID to search for. pub gene_id: Option, /// The gene symbol to search for. @@ -57,9 +60,20 @@ fn _default_hpo_terms() -> bool { } /// Result entry for `handle`. -#[derive(serde::Serialize, serde::Deserialize, Debug, Clone, Eq, PartialEq, Ord, PartialOrd)] +#[derive( + serde::Serialize, + serde::Deserialize, + utoipa::ToSchema, + Debug, + Clone, + Eq, + PartialEq, + Ord, + PartialOrd, +)] #[serde_with::skip_serializing_none] -struct ResultEntry { +#[schema(title = "HpoGenesResultEntry")] +pub struct ResultEntry { /// The gene's NCBI ID. pub gene_ncbi_id: u32, /// The gene's HGNC symbol. @@ -72,6 +86,7 @@ struct ResultEntry { } impl ResultEntry { + /// Create a `ResultEntry` from a `Gene` with an `Ontology`. pub fn from_gene_with_ontology( gene: &Gene, ontology: &Ontology, @@ -103,10 +118,11 @@ impl ResultEntry { } /// Container for the result. -#[derive(Debug, serde::Serialize, serde::Deserialize)] -struct Container { +#[derive(Debug, serde::Serialize, serde::Deserialize, utoipa::ToSchema)] +#[schema(title = "HpoGenesResult")] +pub struct Result { /// Version information. - pub version: crate::common::Version, + pub version: Version, /// The original query records. pub query: Query, /// The resulting records for the scored genes. @@ -115,6 +131,12 @@ struct Container { /// Query for genes in the HPO database. #[allow(clippy::unused_async)] +#[utoipa::path( + params(Query), + responses( + (status = 200, description = "The query was successful.", body = Result), + ) +)] #[get("/hpo/genes")] async fn handle( data: Data, @@ -174,8 +196,8 @@ async fn handle( result.sort(); - let result = Container { - version: crate::common::Version::new(&data.ontology.hpo_version()), + let result = Result { + version: Version::new(&data.ontology.hpo_version()), query: query.into_inner(), result, }; @@ -187,7 +209,7 @@ async fn handle( mod test { /// Helper function for running a query. #[allow(dead_code)] - async fn run_query(uri: &str) -> Result { + async fn run_query(uri: &str) -> Result { let ontology = crate::common::load_hpo("tests/data/hpo")?; let ncbi_to_hgnc = crate::common::hgnc_xlink::load_ncbi_to_hgnc("tests/data/hgnc_xlink.tsv")?; @@ -195,17 +217,19 @@ mod test { let hpo_doc = fastobo::from_file("tests/data/hpo/hp.obo")?; let app = actix_web::test::init_service( actix_web::App::new() - .app_data(actix_web::web::Data::new(crate::server::WebServerData { - ontology, - ncbi_to_hgnc, - hgnc_to_ncbi, - full_text_index: crate::index::Index::new(hpo_doc)?, - })) + .app_data(actix_web::web::Data::new( + crate::server::run::WebServerData { + ontology, + ncbi_to_hgnc, + hgnc_to_ncbi, + full_text_index: crate::index::Index::new(hpo_doc)?, + }, + )) .service(super::handle), ) .await; let req = actix_web::test::TestRequest::get().uri(uri).to_request(); - let resp: super::Container = actix_web::test::call_and_read_body_json(&app, req).await; + let resp: super::Result = actix_web::test::call_and_read_body_json(&app, req).await; Ok(resp) } diff --git a/src/server/actix_server/hpo_omims.rs b/src/server/run/hpo_omims.rs similarity index 87% rename from src/server/actix_server/hpo_omims.rs rename to src/server/run/hpo_omims.rs index 266e3e7..4c736b6 100644 --- a/src/server/actix_server/hpo_omims.rs +++ b/src/server/run/hpo_omims.rs @@ -11,7 +11,7 @@ use hpo::{ Ontology, }; -use crate::server::WebServerData; +use crate::{common::Version, server::run::WebServerData}; use super::{CustomError, Match, ResultHpoTerm}; @@ -28,8 +28,11 @@ use super::{CustomError, Match, ResultHpoTerm}; /// The following propery defines how matches are performed: /// /// - `match` -- how to match -#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] -struct Query { +#[derive( + serde::Serialize, serde::Deserialize, utoipa::ToSchema, utoipa::IntoParams, Debug, Clone, +)] +#[schema(title = "HpoOmimsQuery")] +pub struct Query { /// The OMIM ID to search for. pub omim_id: Option, /// The disease name to search for. @@ -56,8 +59,9 @@ fn _default_hpo_terms() -> bool { } /// Result entry for `handle`. -#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] -struct ResultEntry { +#[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug, Clone)] +#[schema(title = "HpoOmimsResultEntry")] +pub struct ResultEntry { /// The OMIM ID. pub omim_id: String, /// The OMIM disease name. @@ -92,6 +96,7 @@ impl Ord for ResultEntry { } impl ResultEntry { + /// Create a `ResultEntry` from an `OmimDisease`. pub fn from_omim_disease_with_ontology( omim_disease: &OmimDisease, ontology: &Ontology, @@ -128,10 +133,11 @@ impl ResultEntry { } /// Container for the result. -#[derive(Debug, serde::Serialize, serde::Deserialize)] -struct Container { +#[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug)] +#[schema(title = "HpoOmimsResult")] +pub struct Result { /// Version information. - pub version: crate::common::Version, + pub version: Version, /// The original query records. pub query: Query, /// The resulting records for the scored genes. @@ -140,6 +146,12 @@ struct Container { /// Query for OMIM diseases in the HPO database. #[allow(clippy::unused_async)] +#[utoipa::path( + params(Query), + responses( + (status = 200, description = "The query was successful.", body = Result), + ) +)] #[get("/hpo/omims")] async fn handle( data: Data, @@ -201,8 +213,8 @@ async fn handle( result.sort(); - let result = Container { - version: crate::common::Version::new(&data.ontology.hpo_version()), + let result = Result { + version: Version::new(&data.ontology.hpo_version()), query: query.into_inner(), result, }; @@ -214,7 +226,7 @@ async fn handle( mod test { /// Helper function for running a query. #[allow(dead_code)] - async fn run_query(uri: &str) -> Result { + async fn run_query(uri: &str) -> Result { let ontology = crate::common::load_hpo("tests/data/hpo")?; let ncbi_to_hgnc = crate::common::hgnc_xlink::load_ncbi_to_hgnc("tests/data/hgnc_xlink.tsv")?; @@ -222,17 +234,19 @@ mod test { let hpo_doc = fastobo::from_file("tests/data/hpo/hp.obo")?; let app = actix_web::test::init_service( actix_web::App::new() - .app_data(actix_web::web::Data::new(crate::server::WebServerData { - ontology, - ncbi_to_hgnc, - hgnc_to_ncbi, - full_text_index: crate::index::Index::new(hpo_doc)?, - })) + .app_data(actix_web::web::Data::new( + crate::server::run::WebServerData { + ontology, + ncbi_to_hgnc, + hgnc_to_ncbi, + full_text_index: crate::index::Index::new(hpo_doc)?, + }, + )) .service(super::handle), ) .await; let req = actix_web::test::TestRequest::get().uri(uri).to_request(); - let resp: super::Container = actix_web::test::call_and_read_body_json(&app, req).await; + let resp: super::Result = actix_web::test::call_and_read_body_json(&app, req).await; Ok(resp) } diff --git a/src/server/actix_server/hpo_sim/mod.rs b/src/server/run/hpo_sim/mod.rs similarity index 100% rename from src/server/actix_server/hpo_sim/mod.rs rename to src/server/run/hpo_sim/mod.rs diff --git a/src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_gene_ids.snap b/src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_gene_ids.snap similarity index 100% rename from src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_gene_ids.snap rename to src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_gene_ids.snap diff --git a/src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_hgnc_gene_ids.snap b/src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_hgnc_gene_ids.snap similarity index 100% rename from src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_hgnc_gene_ids.snap rename to src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_hgnc_gene_ids.snap diff --git a/src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_ncbi_gene_ids.snap b/src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_ncbi_gene_ids.snap similarity index 100% rename from src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_ncbi_gene_ids.snap rename to src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_ncbi_gene_ids.snap diff --git a/src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_symbols.snap b/src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_symbols.snap similarity index 100% rename from src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_symbols.snap rename to src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_gene__test__hpo_sim_term_gene_terms_symbols.snap diff --git a/src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_one_one.snap b/src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_one_one.snap similarity index 100% rename from src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_one_one.snap rename to src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_one_one.snap diff --git a/src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_two_two.snap b/src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_two_two.snap similarity index 100% rename from src/server/actix_server/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_two_two.snap rename to src/server/run/hpo_sim/snapshots/viguno__server__actix_server__hpo_sim__term_term__test__hpo_sim_term_term_two_two.snap diff --git a/src/server/actix_server/hpo_sim/term_gene.rs b/src/server/run/hpo_sim/term_gene.rs similarity index 86% rename from src/server/actix_server/hpo_sim/term_gene.rs rename to src/server/run/hpo_sim/term_gene.rs index 57fd8a9..3ab9bed 100644 --- a/src/server/actix_server/hpo_sim/term_gene.rs +++ b/src/server/run/hpo_sim/term_gene.rs @@ -10,7 +10,7 @@ use actix_web::{ use hpo::{annotations::GeneId, term::HpoGroup, HpoTermId, Ontology}; use super::super::CustomError; -use crate::{query, server::WebServerData}; +use crate::{query, server::run::WebServerData}; /// Parameters for `handle`. /// @@ -21,8 +21,9 @@ use crate::{query, server::WebServerData}; /// gene ID or HGNC gene ID. /// - `gene_symbols` -- set of symbols for genes to use as /// "database" -#[derive(serde::Deserialize, Debug, Clone)] -struct Query { +#[derive(serde::Deserialize, Debug, Clone, utoipa::ToSchema, utoipa::IntoParams)] +#[schema(title = "HpoSimTermGeneQuery")] +pub struct Query { /// Set of terms to use as query. #[serde(deserialize_with = "super::super::vec_str_deserialize")] pub terms: Vec, @@ -45,6 +46,12 @@ struct Query { /// Query for similarity between a set of terms to each entry in a /// list of genes. #[allow(clippy::unused_async)] +#[utoipa::path( + params(Query), + responses( + (status = 200, description = "The query was successful.", body = Result), + ) +)] #[get("/hpo/sim/term-gene")] async fn handle( data: Data, @@ -100,7 +107,7 @@ async fn handle( mod test { /// Helper function for running a query. #[allow(dead_code)] - async fn run_query(uri: &str) -> Result { + async fn run_query(uri: &str) -> Result { let hpo_path = "tests/data/hpo"; let ontology = crate::common::load_hpo(hpo_path)?; let ncbi_to_hgnc = @@ -110,17 +117,19 @@ mod test { let app = actix_web::test::init_service( actix_web::App::new() - .app_data(actix_web::web::Data::new(crate::server::WebServerData { - ontology, - ncbi_to_hgnc, - hgnc_to_ncbi, - full_text_index: crate::index::Index::new(hpo_doc)?, - })) + .app_data(actix_web::web::Data::new( + crate::server::run::WebServerData { + ontology, + ncbi_to_hgnc, + hgnc_to_ncbi, + full_text_index: crate::index::Index::new(hpo_doc)?, + }, + )) .service(super::handle), ) .await; let req = actix_web::test::TestRequest::get().uri(uri).to_request(); - let resp: crate::query::query_result::Container = + let resp: crate::query::query_result::Result = actix_web::test::call_and_read_body_json(&app, req).await; Ok(resp) diff --git a/src/server/actix_server/hpo_sim/term_term.rs b/src/server/run/hpo_sim/term_term.rs similarity index 82% rename from src/server/actix_server/hpo_sim/term_term.rs rename to src/server/run/hpo_sim/term_term.rs index 076062e..3503586 100644 --- a/src/server/actix_server/hpo_sim/term_term.rs +++ b/src/server/run/hpo_sim/term_term.rs @@ -12,8 +12,8 @@ use hpo::{ }; use itertools::Itertools; -use crate::common::{to_pairwise_sim, IcBasedOn, ScoreCombiner, SimilarityMethod}; -use crate::server::{actix_server::CustomError, WebServerData}; +use crate::common::{to_pairwise_sim, IcBasedOn, ScoreCombiner, SimilarityMethod, Version}; +use crate::server::{run::CustomError, run::WebServerData}; /// Parameters for `handle`. /// @@ -21,7 +21,7 @@ use crate::server::{actix_server::CustomError, WebServerData}; /// /// - `lhs` -- first set of terms to compute similarity for /// - `rhs` -- econd set of terms to compute similarity for -#[derive(serde::Serialize, serde::Deserialize, Default, Debug, Clone)] +#[derive(serde::Serialize, serde::Deserialize, utoipa::IntoParams, Default, Debug, Clone)] pub struct RequestQuery { /// The one set of HPO terms to compute similarity for. #[serde(deserialize_with = "super::super::vec_str_deserialize")] @@ -43,7 +43,8 @@ pub struct RequestQuery { /// Request as sent together with the response. /// /// The difference is that the `lhs` and `rhs` fields are replaced by vecs. -#[derive(serde::Serialize, serde::Deserialize, Default, Debug, Clone)] +#[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Default, Debug, Clone)] +#[schema(title = "HpoSimTermTermQuery")] pub struct ResponseQuery { /// The one set of HPO terms to compute similarity for. pub lhs: Vec, @@ -61,10 +62,11 @@ pub struct ResponseQuery { } /// Result container. -#[derive(serde::Serialize, serde::Deserialize, Default, Debug, Clone)] -pub struct Container { +#[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Default, Debug, Clone)] +#[schema(title = "HpoSimTermTermResult")] +pub struct Result { /// Version information. - pub version: crate::common::Version, + pub version: Version, /// The original query records. pub query: ResponseQuery, /// The resulting records for the scored genes. @@ -72,7 +74,17 @@ pub struct Container { } /// Result entry for `handle`. -#[derive(serde::Serialize, serde::Deserialize, Default, Debug, Clone, PartialEq, PartialOrd)] +#[derive( + serde::Serialize, + serde::Deserialize, + utoipa::ToSchema, + Default, + Debug, + Clone, + PartialEq, + PartialOrd, +)] +#[schema(title = "HpoSimTermTermResultEntry")] pub struct ResultEntry { /// The lhs entry. pub lhs: String, @@ -90,6 +102,12 @@ pub struct ResultEntry { /// /// In the case that there is an error running the server. #[allow(clippy::unused_async)] +#[utoipa::path( + params(RequestQuery), + responses( + (status = 200, description = "The query was successful.", body = Result), + ) +)] #[get("/hpo/sim/term-term")] async fn handle( data: Data, @@ -140,8 +158,8 @@ async fn handle( combiner, } = query.into_inner(); - let result = Container { - version: crate::common::Version::new(&data.ontology.hpo_version()), + let result = Result { + version: Version::new(&data.ontology.hpo_version()), query: ResponseQuery { lhs, rhs, @@ -161,7 +179,7 @@ async fn handle( mod test { /// Helper function for running a query. #[allow(dead_code)] - async fn run_query(uri: &str) -> Result { + async fn run_query(uri: &str) -> Result { let ontology = crate::common::load_hpo("tests/data/hpo")?; let ncbi_to_hgnc = crate::common::hgnc_xlink::load_ncbi_to_hgnc("tests/data/hgnc_xlink.tsv")?; @@ -170,12 +188,14 @@ mod test { let app = actix_web::test::init_service( actix_web::App::new() - .app_data(actix_web::web::Data::new(crate::server::WebServerData { - ontology, - ncbi_to_hgnc, - hgnc_to_ncbi, - full_text_index: crate::index::Index::new(hpo_doc)?, - })) + .app_data(actix_web::web::Data::new( + crate::server::run::WebServerData { + ontology, + ncbi_to_hgnc, + hgnc_to_ncbi, + full_text_index: crate::index::Index::new(hpo_doc)?, + }, + )) .service(super::handle), ) .await; diff --git a/src/server/actix_server/hpo_terms.rs b/src/server/run/hpo_terms.rs similarity index 90% rename from src/server/actix_server/hpo_terms.rs rename to src/server/run/hpo_terms.rs index ada04e7..f0cbb17 100644 --- a/src/server/actix_server/hpo_terms.rs +++ b/src/server/run/hpo_terms.rs @@ -9,7 +9,7 @@ use actix_web::{ }; use hpo::{annotations::AnnotationId, HpoTerm, HpoTermId, Ontology}; -use crate::server::WebServerData; +use crate::{common::Version, server::run::WebServerData}; use super::{CustomError, ResultGene}; @@ -22,8 +22,11 @@ use super::{CustomError, ResultGene}; /// - `gene_symbol` -- specify the gene symbol /// - `max_results` -- the maximum number of records to return /// - `genes` -- whether to include `"genes"` in result -#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] -struct Query { +#[derive( + serde::Serialize, serde::Deserialize, utoipa::ToSchema, utoipa::IntoParams, Debug, Clone, +)] +#[schema(title = "HpoTermsQuery")] +pub struct Query { /// The term ID to search for. pub term_id: Option, /// The term name to search for. @@ -47,8 +50,9 @@ fn _default_genes() -> bool { } /// Result entry for `fetch_hpo_genes`. -#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)] -struct ResultEntry { +#[derive(serde::Serialize, serde::Deserialize, utoipa::ToSchema, Debug, Clone)] +#[schema(title = "HpoTermsResultEntry")] +pub struct ResultEntry { /// The HPO term's ID. pub term_id: String, /// The HPO term's name. @@ -89,6 +93,7 @@ impl Ord for ResultEntry { } impl ResultEntry { + /// Create a `ResultEntry` from an `HpoTerm`. pub fn from_term_with_ontology( term: &HpoTerm, ontology: &Ontology, @@ -183,10 +188,11 @@ impl ResultEntry { } /// Container for the result. -#[derive(Debug, serde::Serialize, serde::Deserialize)] -struct Container { +#[derive(Debug, serde::Serialize, serde::Deserialize, utoipa::ToSchema)] +#[schema(title = "HpoTermsResult")] +pub struct Result { /// Version information. - pub version: crate::common::Version, + pub version: Version, /// The original query records. pub query: Query, /// The resulting records for the scored genes. @@ -200,6 +206,12 @@ struct Container { /// In the case that there is an error running the server. #[allow(clippy::unused_async)] #[allow(clippy::too_many_lines)] +#[utoipa::path( + params(Query), + responses( + (status = 200, description = "The query was successful.", body = Result), + ) +)] #[get("/hpo/terms")] async fn handle( data: Data, @@ -319,8 +331,8 @@ async fn handle( } }; - let result = Container { - version: crate::common::Version::new(&data.ontology.hpo_version()), + let result = Result { + version: Version::new(&data.ontology.hpo_version()), query: query.into_inner(), result, }; @@ -332,7 +344,7 @@ async fn handle( mod test { /// Helper function for running a query. #[allow(dead_code)] - async fn run_query(uri: &str) -> Result { + async fn run_query(uri: &str) -> Result { let ontology = crate::common::load_hpo("tests/data/hpo")?; let ncbi_to_hgnc = crate::common::hgnc_xlink::load_ncbi_to_hgnc("tests/data/hgnc_xlink.tsv")?; @@ -340,17 +352,19 @@ mod test { let hpo_doc = fastobo::from_file("tests/data/hpo/hp.obo")?; let app = actix_web::test::init_service( actix_web::App::new() - .app_data(actix_web::web::Data::new(crate::server::WebServerData { - ontology, - ncbi_to_hgnc, - hgnc_to_ncbi, - full_text_index: crate::index::Index::new(hpo_doc)?, - })) + .app_data(actix_web::web::Data::new( + crate::server::run::WebServerData { + ontology, + ncbi_to_hgnc, + hgnc_to_ncbi, + full_text_index: crate::index::Index::new(hpo_doc)?, + }, + )) .service(super::handle), ) .await; let req = actix_web::test::TestRequest::get().uri(uri).to_request(); - let resp: super::Container = actix_web::test::call_and_read_body_json(&app, req).await; + let resp: super::Result = actix_web::test::call_and_read_body_json(&app, req).await; Ok(resp) } diff --git a/src/server/run/mod.rs b/src/server/run/mod.rs new file mode 100644 index 0000000..a66e257 --- /dev/null +++ b/src/server/run/mod.rs @@ -0,0 +1,341 @@ +//! Implementation of the Actix server. + +pub mod hpo_genes; +pub mod hpo_omims; +pub mod hpo_sim; +pub mod hpo_terms; + +use std::collections::HashMap; + +use actix_web::{middleware::Logger, web::Data, App, HttpServer, ResponseError}; +use serde::{Deserialize, Deserializer, Serialize}; +use utoipa::OpenApi; + +use crate::common::load_hpo; + +/// Data structure for the web server data. +pub struct WebServerData { + /// The HPO ontology (`hpo` crate). + pub ontology: hpo::Ontology, + /// Xlink map from NCBI gene ID to HGNC gene ID. + pub ncbi_to_hgnc: HashMap, + /// Xlink map from HGNC gene ID to NCBI gene ID. + pub hgnc_to_ncbi: HashMap, + /// The full text index over the HPO OBO document. + pub full_text_index: crate::index::Index, +} + +/// Command line arguments for `server run` sub command. +#[derive(clap::Parser, Debug)] +#[command(author, version, about = "Run viguno REST API server", long_about = None)] +pub struct Args { + /// Path to the directory with the HPO files. + #[arg(long, required = true)] + pub path_hpo_dir: String, + /// Path to the TSV file with the HGNC xlink data. + #[arg(long, required = true)] + pub path_hgnc_xlink: String, + + /// Whether to suppress printing hints. + #[arg(long, default_value_t = false)] + pub suppress_hints: bool, + + /// IP to listen on. + #[arg(long, default_value = "127.0.0.1")] + pub listen_host: String, + /// Port to listen on. + #[arg(long, default_value_t = 8080)] + pub listen_port: u16, +} + +#[derive(Debug)] +struct CustomError { + err: anyhow::Error, +} + +impl std::fmt::Display for CustomError { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!(f, "{:?}", self.err) + } +} + +impl CustomError { + fn new(err: anyhow::Error) -> Self { + CustomError { err } + } +} + +impl ResponseError for CustomError {} + +/// Specify how to perform query matches in the API calls. +#[derive(Serialize, Deserialize, utoipa::ToSchema, Debug, Clone, Copy, Default, PartialEq, Eq)] +#[serde(rename_all = "lowercase")] +pub enum Match { + #[default] + /// Exact string match. + Exact, + /// Prefix string match. + Prefix, + /// Suffix string match. + Suffix, + /// String containment. + Contains, +} + +/// Representation of a gene. +#[derive( + serde::Deserialize, + serde::Serialize, + utoipa::ToSchema, + Default, + Debug, + Clone, + PartialOrd, + Ord, + PartialEq, + Eq, +)] +#[serde_with::skip_serializing_none] +pub struct ResultGene { + /// The HPO ID. + pub ncbi_gene_id: u32, + /// The description. + pub gene_symbol: String, + /// The HGNC ID. + pub hgnc_id: Option, +} + +/// Representation of an HPO term. +#[derive( + serde::Deserialize, + serde::Serialize, + utoipa::ToSchema, + Default, + Debug, + Clone, + PartialOrd, + Ord, + PartialEq, + Eq, +)] +pub struct ResultHpoTerm { + /// The HPO ID. + pub term_id: String, + /// The term name. + pub name: String, +} + +/// Helper to deserialize a comma-separated list of strings. +fn vec_str_deserialize<'de, D>(deserializer: D) -> Result, D::Error> +where + D: Deserializer<'de>, +{ + let str_sequence = String::deserialize(deserializer)?; + Ok(str_sequence + .split(',') + .map(std::borrow::ToOwned::to_owned) + .collect()) +} + +/// Helper to deserialize a comma-separated list of strings. +fn option_vec_str_deserialize<'de, D>(deserializer: D) -> Result>, D::Error> +where + D: Deserializer<'de>, +{ + let str_sequence = String::deserialize(deserializer)?; + if str_sequence.is_empty() { + Ok(None) + } else { + Ok(Some( + str_sequence + .split(',') + .map(std::borrow::ToOwned::to_owned) + .collect(), + )) + } +} + +/// Utoipa-based OpenAPI generation helper. +#[derive(utoipa::OpenApi)] +#[openapi( + paths(hpo_genes::handle, hpo_terms::handle, hpo_omims::handle,), + components(schemas( + hpo_genes::Query, + hpo_genes::Result, + hpo_genes::ResultEntry, + hpo_omims::Query, + hpo_omims::Result, + hpo_omims::ResultEntry, + hpo_terms::Query, + hpo_terms::Result, + hpo_terms::ResultEntry, + hpo_sim::term_gene::Query, + crate::query::query_result::Result, + crate::query::query_result::ResultEntry, + crate::query::query_result::TermDetails, + crate::query::HpoTerm, + hpo_sim::term_term::ResponseQuery, + hpo_sim::term_term::Result, + hpo_sim::term_term::ResultEntry, + ResultGene, + ResultHpoTerm, + Match, + crate::common::Version, + crate::common::IcBasedOn, + crate::common::SimilarityMethod, + crate::common::ScoreCombiner, + )) +)] +pub struct ApiDoc; + +/// Main entry point for running the REST server. +#[allow(clippy::unused_async)] +#[actix_web::main] +pub async fn main(args: &Args, dbs: Data) -> std::io::Result<()> { + let openapi = ApiDoc::openapi(); + + HttpServer::new(move || { + App::new() + .app_data(dbs.clone()) + .service(hpo_genes::handle) + .service(hpo_terms::handle) + .service(hpo_omims::handle) + .service(hpo_sim::term_term::handle) + .service(hpo_sim::term_gene::handle) + .service( + utoipa_swagger_ui::SwaggerUi::new("/swagger-ui/{_:.*}") + .url("/api-docs/openapi.json", openapi.clone()), + ) + .wrap(Logger::default()) + }) + .bind((args.listen_host.as_str(), args.listen_port))? + .run() + .await +} + +/// Print some hints via `tracing::info!`. +pub fn print_hints(args: &Args) { + tracing::info!( + "Launching server main on http://{}:{} ...", + args.listen_host.as_str(), + args.listen_port + ); + + // Short-circuit if no hints are to be + if args.suppress_hints { + return; + } + + tracing::info!( + " ==> for Swagger UI, see: http://{}:{}/swagger-ui/", + args.listen_host.as_str(), + args.listen_port + ); + + // The endpoint `/hpo/genes` provides information related to genes by symbol. + tracing::info!( + " try: http://{}:{}/hpo/genes?gene_symbol=TGDS", + args.listen_host.as_str(), + args.listen_port + ); + // Also, you can query `/hpo/genes` by NCBI gene ID and return the HPO terms of the gene. + tracing::info!( + " try: http://{}:{}/hpo/genes?gene_id=23483&hpo_terms=true", + args.listen_host.as_str(), + args.listen_port + ); + // The `/hpo/omims` term provides information on OMIM terms and can include HPO terms for + // the disease. + tracing::info!( + " try: http://{}:{}/hpo/omims?omim_id=616145&hpo_terms=true", + args.listen_host.as_str(), + args.listen_port + ); + // The `/hpo/terms` endpoint allows to query by HPO term ID and optionally return a list of + // genes that are linked to the term. + tracing::info!( + " try: http://{}:{}/hpo/terms?term_id=HP:0000023&genes=true", + args.listen_host.as_str(), + args.listen_port + ); + // We can use `/hpo/sim/term-term` to compute similarity between two HPO term sets `lhs` + // and `rhs` using a similarity metric. + tracing::info!( + " try: http://{}:{}/hpo/sim/term-term?lhs=HP:0001166,HP:0040069&rhs=HP:0005918,\ + HP:0004188", + args.listen_host.as_str(), + args.listen_port + ); + // The endpoint `/hpo/sim/term-gene` allows to compute the same for a list of `terms` and + // `gene_symbols`. + tracing::info!( + " try: http://{}:{}/hpo/sim/term-gene?terms=HP:0001166,HP:0000098&gene_symbols=FBN1,TGDS,TTN", + args.listen_host.as_str(), + args.listen_port + ); +} + +/// Main entry point for `run-server` sub command. +/// +/// # Errors +/// +/// In the case that there is an error running the server. +pub fn run(args_common: &crate::common::Args, args: &Args) -> Result<(), anyhow::Error> { + tracing::info!("args_common = {:?}", &args_common); + tracing::info!("args = {:?}", &args); + + if let Some(level) = args_common.verbose.log_level() { + match level { + log::Level::Trace | log::Level::Debug => { + std::env::set_var("RUST_LOG", "debug"); + env_logger::init_from_env(env_logger::Env::new().default_filter_or("info")); + } + _ => (), + } + } + + // Load data that we need for running the server. + tracing::info!("Loading HPO..."); + let before_loading = std::time::Instant::now(); + let ontology = load_hpo(&args.path_hpo_dir)?; + tracing::info!("...done loading HPO in {:?}", before_loading.elapsed()); + + tracing::info!("Loading HGNC xlink..."); + let before_load_xlink = std::time::Instant::now(); + let ncbi_to_hgnc = crate::common::hgnc_xlink::load_ncbi_to_hgnc(&args.path_hgnc_xlink)?; + let hgnc_to_ncbi = crate::common::hgnc_xlink::inverse_hashmap(&ncbi_to_hgnc); + tracing::info!( + "... done loading HGNC xlink in {:?}", + before_load_xlink.elapsed() + ); + + tracing::info!("Loading HPO OBO..."); + let before_load_obo = std::time::Instant::now(); + let hpo_doc = fastobo::from_file(format!("{}/{}", &args.path_hpo_dir, "hp.obo")) + .map_err(|e| anyhow::anyhow!("Error loading HPO OBO: {}", e))?; + tracing::info!( + "... done loading HPO OBO in {:?}", + before_load_obo.elapsed() + ); + + tracing::info!("Indexing OBO..."); + let before_index_obo = std::time::Instant::now(); + let full_text_index = crate::index::Index::new(hpo_doc) + .map_err(|e| anyhow::anyhow!("Error indexing HPO OBO: {}", e))?; + tracing::info!("... done indexing OBO in {:?}", before_index_obo.elapsed()); + + let data = actix_web::web::Data::new(WebServerData { + ontology, + ncbi_to_hgnc, + hgnc_to_ncbi, + full_text_index, + }); + + // Print the server URL and some hints (the latter: unless suppressed). + print_hints(args); + // Launch the Actix web server. + main(args, data)?; + + tracing::info!("All done. Have a nice day!"); + Ok(()) +} diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_contains_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_exact_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_prefix_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_gene_symbol_suffix_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_hgnc_gene_id_exact_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_genes__test__hpo_genes_ncbi_gene_id_exact_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_contains_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_exact_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_prefix_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_name_suffix_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_no_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_no_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_no_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_no_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_with_hpo_terms.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_with_hpo_terms.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_with_hpo_terms.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_omims__test__hpo_omims_omim_id_exact_with_hpo_terms.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_no_genes.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_no_genes.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_no_genes.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_no_genes.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_with_genes.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_with_genes.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_with_genes.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_exact_with_genes.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_no_genes.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_no_genes.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_no_genes.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_no_genes.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_with_genes.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_with_genes.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_with_genes.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_name_fuzzy_with_genes.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_no_genes.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_no_genes.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_no_genes.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_no_genes.snap diff --git a/src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_with_genes.snap b/src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_with_genes.snap similarity index 100% rename from src/server/actix_server/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_with_genes.snap rename to src/server/run/snapshots/viguno__server__actix_server__hpo_terms__test__hpo_terms_term_id_exact_with_genes.snap diff --git a/src/server/schema.rs b/src/server/schema.rs index e69de29..f94b8b6 100644 --- a/src/server/schema.rs +++ b/src/server/schema.rs @@ -0,0 +1,49 @@ +//! Dump schema of the REST API server. + +use std::{fs::File, io::{self, Write}}; + +use utoipa::OpenApi as _; + +use crate::server::run::ApiDoc; + +/// Command line arguments for `server schema` sub command. +#[derive(clap::Parser, Debug)] +#[command(author, version, about = "Dump REST API schema", long_about = None)] +pub struct Args { + /// Path to the output file. Use stdout if missing. + #[arg(long)] + pub output_file: Option, +} + +impl Args { + /// Get writeable output file or stdout. + fn get_output(&self) -> Result, io::Error> { + match self.output_file { + Some(ref path) => File::create(path).map(|f| Box::new(f) as Box), + None => Ok(Box::new(io::stdout())), + } + } +} + +/// Main entry point for `run-server` sub command. +/// +/// # Errors +/// +/// In the case that there is an error running the server. +pub fn run(args_common: &crate::common::Args, args: &Args) -> Result<(), anyhow::Error> { + tracing::info!("args_common = {:?}", &args_common); + tracing::info!("args = {:?}", &args); + + let schema_yaml = ApiDoc::openapi().to_yaml().map_err(|e| { + anyhow::anyhow!("Failed to convert OpenAPI to YAML: {}", e) + })?; + let mut output = args.get_output().map_err(|e| { + anyhow::anyhow!("Failed to open output file: {}", e) + })?; + write!(output, "{}", &schema_yaml).map_err(|e| { + anyhow::anyhow!("Failed to write output: {}", e) + })?; + + tracing::info!("All done. Have a nice day!"); + Ok(()) +}