From 4ab7d44548322a2bd3634a731904bed8d8073f67 Mon Sep 17 00:00:00 2001 From: plotchy Date: Tue, 11 Jun 2024 22:46:28 -0400 Subject: [PATCH 1/6] feat: debug site --- .gitignore | 3 + Cargo.lock | 791 +++++++++++++++++- Cargo.toml | 6 +- crates/analyzers/Cargo.toml | 1 + crates/analyzers/src/func_analyzer/mod.rs | 1 + .../src/func_analyzer/report_display.rs | 1 + .../src/var_analyzer/report_display.rs | 1 + crates/cli/Cargo.toml | 4 +- crates/cli/src/main.rs | 43 +- crates/pyrometer/Cargo.toml | 3 + crates/pyrometer/src/analyzer.rs | 65 +- crates/pyrometer/src/analyzer_backend.rs | 17 +- crates/pyrometer/src/graph_backend.rs | 148 +++- crates/shared/Cargo.toml | 6 +- crates/shared/src/analyzer_like.rs | 1 + 15 files changed, 1046 insertions(+), 45 deletions(-) diff --git a/.gitignore b/.gitignore index 113d56e2..aaec7a35 100644 --- a/.gitignore +++ b/.gitignore @@ -8,3 +8,6 @@ **/dot.dot **/flamegraph.svg **/.swp +.vscode +.env +notes.md \ No newline at end of file diff --git a/Cargo.lock b/Cargo.lock index bd01078d..07f1ae80 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2,6 +2,21 @@ # It is not intended for manual editing. version = 3 +[[package]] +name = "addr2line" +version = "0.22.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e4503c46a5c0c7844e948c9a4d6acd9f50cccb4de1c48eb9e291ea17470c678" +dependencies = [ + "gimli", +] + +[[package]] +name = "adler" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" + [[package]] name = "ahash" version = "0.8.10" @@ -32,6 +47,7 @@ dependencies = [ "graph", "shared", "solang-parser", + "tracing", ] [[package]] @@ -42,23 +58,24 @@ checksum = "4b46cbb362ab8752921c97e041f5e366ee6297bd428a31275b9fcf1e380f7299" [[package]] name = "anstream" -version = "0.6.4" +version = "0.6.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2ab91ebe16eb252986481c5b62f6098f3b698a45e34b5b98200cf20dd2484a44" +checksum = "418c75fa768af9c03be99d17643f93f79bbba589895012a80e3452a19ddda15b" dependencies = [ "anstyle", "anstyle-parse", "anstyle-query", "anstyle-wincon", "colorchoice", + "is_terminal_polyfill", "utf8parse", ] [[package]] name = "anstyle" -version = "1.0.4" +version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7079075b41f533b8c61d2a4d073c4676e1f8b249ff94a393b0595db304e0dd87" +checksum = "038dfcf04a5feb68e9c60b21c9625a54c2c0616e79b72b0fd87075a056ae1d1b" [[package]] name = "anstyle-parse" @@ -90,8 +107,9 @@ dependencies = [ [[package]] name = "ariadne" -version = "0.2.0" -source = "git+https://github.com/brockelmore/ariadne#c2e3a8e79f369e8a785e1ca337e144ebc946f115" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44055e597c674aef7cb903b2b9f6e4cba1277ed0d2d61dae7cd52d7ffa81f8e2" dependencies = [ "unicode-width", "yansi", @@ -112,6 +130,12 @@ dependencies = [ "term", ] +[[package]] +name = "atomic-waker" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0" + [[package]] name = "atty" version = "0.2.14" @@ -141,12 +165,33 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" +[[package]] +name = "backtrace" +version = "0.3.72" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17c6a35df3749d2e8bb1b7b21a976d82b15548788d2735b9d82f329268f71a11" +dependencies = [ + "addr2line", + "cc", + "cfg-if", + "libc", + "miniz_oxide", + "object", + "rustc-demangle", +] + [[package]] name = "base16ct" version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4c7f02d4ea65f2c1853089ffd8d2787bdbc63de2f0d29dedbcf8ccdfa0ccd4cf" +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + [[package]] name = "base64ct" version = "1.6.0" @@ -234,6 +279,12 @@ version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "37b2a672a2cb129a2e41c10b1224bb368f9f37a2b16b612598138befd7b37eb5" +[[package]] +name = "cc" +version = "1.0.98" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41c270e7540d725e65ac7f1b212ac8ce349719624d7bcff99f8e2e488e8cf03f" + [[package]] name = "cfg-if" version = "1.0.0" @@ -348,7 +399,9 @@ dependencies = [ "graph", "petgraph", "pyrometer", + "reqwest", "shared", + "tokio", "tracing", "tracing-subscriber", "tracing-tree", @@ -379,6 +432,22 @@ version = "0.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "28c122c3980598d243d63d9a704629a2d748d101f278052ff068be5a4423ab6f" +[[package]] +name = "core-foundation" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "core-foundation-sys" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06ea2b9bc92be3c2baa9334a323ebca2d6f074ff852cd1d7b11064035cd3868f" + [[package]] name = "cpufeatures" version = "0.2.11" @@ -593,6 +662,15 @@ dependencies = [ "log", ] +[[package]] +name = "encoding_rs" +version = "0.8.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b45de904aa0b010bce2ab45264d0631681847fa7b6f2eaa7dab7619943bc4f59" +dependencies = [ + "cfg-if", +] + [[package]] name = "equivalent" version = "1.0.1" @@ -718,12 +796,81 @@ version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0ce7134b9999ecaf8bcd65542e436736ef32ddca1b3e06094cb6ec5755203b80" +[[package]] +name = "fnv" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" + +[[package]] +name = "foreign-types" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1" +dependencies = [ + "foreign-types-shared", +] + +[[package]] +name = "foreign-types-shared" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" + +[[package]] +name = "form_urlencoded" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456" +dependencies = [ + "percent-encoding", +] + [[package]] name = "funty" version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c" +[[package]] +name = "futures-channel" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eac8f7d7865dcb88bd4373ab671c8cf4508703796caa2b1985a9ca867b3fcb78" +dependencies = [ + "futures-core", +] + +[[package]] +name = "futures-core" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dfc6580bb841c5a68e9ef15c77ccc837b40a7504914d52e47b8b0e9bbda25a1d" + +[[package]] +name = "futures-sink" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9fb8e00e87438d937621c1c6269e53f536c14d3fbd6a042bb24879e57d474fb5" + +[[package]] +name = "futures-task" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38d84fa142264698cdce1a9f9172cf383a0c82de1bddcf3092901442c4097004" + +[[package]] +name = "futures-util" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" +dependencies = [ + "futures-core", + "futures-task", + "pin-project-lite", + "pin-utils", +] + [[package]] name = "generic-array" version = "0.14.7" @@ -746,6 +893,12 @@ dependencies = [ "wasi", ] +[[package]] +name = "gimli" +version = "0.29.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "40ecd4077b5ae9fd2e9e169b102c6c330d0605168eb0e8bf79952b256dbefffd" + [[package]] name = "graph" version = "0.2.0" @@ -773,6 +926,25 @@ dependencies = [ "subtle", ] +[[package]] +name = "h2" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa82e28a107a8cc405f0839610bdc9b15f1e25ec7d696aa5cf173edbcb1486ab" +dependencies = [ + "atomic-waker", + "bytes", + "fnv", + "futures-core", + "futures-sink", + "http", + "indexmap 2.1.0", + "slab", + "tokio", + "tokio-util", + "tracing", +] + [[package]] name = "half" version = "1.8.2" @@ -827,6 +999,112 @@ dependencies = [ "digest", ] +[[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", +] + +[[package]] +name = "http-body-util" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0475f8b2ac86659c21b64320d5d653f9efe42acd2a4e560073ec61a155a34f1d" +dependencies = [ + "bytes", + "futures-core", + "http", + "http-body", + "pin-project-lite", +] + +[[package]] +name = "httparse" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d897f394bad6a705d5f4104762e116a75639e470d80901eed05a860a95cb1904" + +[[package]] +name = "hyper" +version = "1.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fe575dd17d0862a9a33781c8c4696a55c320909004a67a00fb286ba8b1bc496d" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "h2", + "http", + "http-body", + "httparse", + "itoa", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-tls" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70206fc6890eaca9fde8a0bf71caa2ddfc9fe045ac9e5c70df101a7dbde866e0" +dependencies = [ + "bytes", + "http-body-util", + "hyper", + "hyper-util", + "native-tls", + "tokio", + "tokio-native-tls", + "tower-service", +] + +[[package]] +name = "hyper-util" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b875924a60b96e5d7b9ae7b066540b1dd1cbd90d1828f54c92e02a283351c56" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http", + "http-body", + "hyper", + "pin-project-lite", + "socket2", + "tokio", + "tower", + "tower-service", + "tracing", +] + +[[package]] +name = "idna" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +dependencies = [ + "unicode-bidi", + "unicode-normalization", +] + [[package]] name = "impl-codec" version = "0.6.0" @@ -885,6 +1163,12 @@ dependencies = [ "hashbrown 0.14.3", ] +[[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" @@ -896,6 +1180,12 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "is_terminal_polyfill" +version = "1.70.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8478577c03552c21db0e2724ffb8986a5ce7af88107e6be5d2ee6e158c12800" + [[package]] name = "itertools" version = "0.10.5" @@ -1030,9 +1320,9 @@ dependencies = [ [[package]] name = "log" -version = "0.4.20" +version = "0.4.21" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b5e6163cb8c49088c2c36f57875e58ccd8c87c7427f7fbd50ea6710b2f3f2e8f" +checksum = "90ed8c1e510134f979dbc4f070f87d4313098b704861a105fe34231c70a3901c" [[package]] name = "matchers" @@ -1058,6 +1348,49 @@ dependencies = [ "autocfg", ] +[[package]] +name = "mime" +version = "0.3.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" + +[[package]] +name = "miniz_oxide" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87dfd01fe195c66b572b37921ad8803d010623c0aca821bea2302239d155cdae" +dependencies = [ + "adler", +] + +[[package]] +name = "mio" +version = "0.8.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" +dependencies = [ + "libc", + "wasi", + "windows-sys 0.48.0", +] + +[[package]] +name = "native-tls" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8614eb2c83d59d1c8cc974dd3f920198647674a0a035e1af1fa58707e317466" +dependencies = [ + "libc", + "log", + "openssl", + "openssl-probe", + "openssl-sys", + "schannel", + "security-framework", + "security-framework-sys", + "tempfile", +] + [[package]] name = "new_debug_unreachable" version = "1.0.4" @@ -1093,6 +1426,16 @@ dependencies = [ "libm", ] +[[package]] +name = "num_cpus" +version = "1.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" +dependencies = [ + "hermit-abi 0.3.3", + "libc", +] + [[package]] name = "num_enum" version = "0.7.1" @@ -1114,6 +1457,15 @@ dependencies = [ "syn 2.0.39", ] +[[package]] +name = "object" +version = "0.35.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8ec7ab813848ba4522158d5517a6093db1ded27575b070f4177b8d12b41db5e" +dependencies = [ + "memchr", +] + [[package]] name = "once_cell" version = "1.19.0" @@ -1151,6 +1503,50 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "openssl" +version = "0.10.64" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95a0481286a310808298130d22dd1fef0fa571e05a8f44ec801801e84b216b1f" +dependencies = [ + "bitflags 2.4.1", + "cfg-if", + "foreign-types", + "libc", + "once_cell", + "openssl-macros", + "openssl-sys", +] + +[[package]] +name = "openssl-macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.39", +] + +[[package]] +name = "openssl-probe" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" + +[[package]] +name = "openssl-sys" +version = "0.9.102" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c597637d56fbc83893a35eb0dd04b2b8e7a50c91e64e9493e398b5df4fb45fa2" +dependencies = [ + "cc", + "libc", + "pkg-config", + "vcpkg", +] + [[package]] name = "os_str_bytes" version = "6.6.1" @@ -1212,6 +1608,12 @@ dependencies = [ "windows-targets 0.48.5", ] +[[package]] +name = "percent-encoding" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" + [[package]] name = "petgraph" version = "0.6.4" @@ -1273,12 +1675,38 @@ dependencies = [ "siphasher", ] +[[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.39", +] + [[package]] name = "pin-project-lite" version = "0.2.13" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8afb450f006bf6385ca15ef45d71d2288452bc3683ce2e2cacc0d18e4be60b58" +[[package]] +name = "pin-utils" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" + [[package]] name = "pkcs8" version = "0.10.2" @@ -1289,6 +1717,12 @@ dependencies = [ "spki", ] +[[package]] +name = "pkg-config" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d231b230927b5e4ad203db57bbcbee2802f6bce620b1e4a9024a07d94e2907ec" + [[package]] name = "plotters" version = "0.3.5" @@ -1433,10 +1867,13 @@ dependencies = [ "ethers-core", "graph", "petgraph", + "reqwest", + "serde", "serde_json", "shared", "solang-parser", "solc-expressions", + "tokio", "tracing", "tracing-subscriber", ] @@ -1593,6 +2030,48 @@ version = "0.8.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c08c74e62047bb2de4ff487b251e4a92e24f48745648451635cec7d591162d9f" +[[package]] +name = "reqwest" +version = "0.12.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "566cafdd92868e0939d3fb961bd0dc25fcfaaed179291093b3d43e6b3150ea10" +dependencies = [ + "base64", + "bytes", + "encoding_rs", + "futures-core", + "futures-util", + "h2", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-tls", + "hyper-util", + "ipnet", + "js-sys", + "log", + "mime", + "native-tls", + "once_cell", + "percent-encoding", + "pin-project-lite", + "rustls-pemfile", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper", + "system-configuration", + "tokio", + "tokio-native-tls", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "winreg", +] + [[package]] name = "rfc6979" version = "0.4.0" @@ -1625,6 +2104,12 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "rustc-demangle" +version = "0.1.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f" + [[package]] name = "rustc-hex" version = "2.1.0" @@ -1644,6 +2129,22 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "rustls-pemfile" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29993a25686778eb88d4189742cd713c9bce943bc54251a33509dc63cbacf73d" +dependencies = [ + "base64", + "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 = "rustversion" version = "1.0.14" @@ -1689,6 +2190,15 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "schannel" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fbc91545643bcf3a0bbb6569265615222618bdf33ce4ffbbd13c4bbd4c093534" +dependencies = [ + "windows-sys 0.52.0", +] + [[package]] name = "scopeguard" version = "1.2.0" @@ -1709,6 +2219,29 @@ dependencies = [ "zeroize", ] +[[package]] +name = "security-framework" +version = "2.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "770452e37cad93e0a50d5abc3990d2bc351c36d0328f86cefec2f2fb206eaef6" +dependencies = [ + "bitflags 1.3.2", + "core-foundation", + "core-foundation-sys", + "libc", + "security-framework-sys", +] + +[[package]] +name = "security-framework-sys" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "317936bbbd05227752583946b9e66d7ce3b489f84e11a94a510b4437fef407d7" +dependencies = [ + "core-foundation-sys", + "libc", +] + [[package]] name = "serde" version = "1.0.193" @@ -1740,6 +2273,18 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa", + "ryu", + "serde", +] + [[package]] name = "sha2" version = "0.10.8" @@ -1778,11 +2323,23 @@ dependencies = [ "ethers-core", "hex", "petgraph", + "reqwest", + "serde", "solang-parser", + "tokio", "tracing", "tracing-subscriber", ] +[[package]] +name = "signal-hook-registry" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a9e9e0b4211b72e7b8b6e85c807d36c212bdb33ea8587f7569562a84df5465b1" +dependencies = [ + "libc", +] + [[package]] name = "signature" version = "2.2.0" @@ -1799,11 +2356,30 @@ version = "0.3.11" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d" +[[package]] +name = "slab" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67" +dependencies = [ + "autocfg", +] + [[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 = "socket2" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] [[package]] name = "solang-parser" @@ -1921,6 +2497,33 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "sync_wrapper" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" + +[[package]] +name = "system-configuration" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba3a3adc5c275d719af8cb4272ea1c4a6d668a777f37e115f6d11ddbc1c8e0e7" +dependencies = [ + "bitflags 1.3.2", + "core-foundation", + "system-configuration-sys", +] + +[[package]] +name = "system-configuration-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75fb188eb626b924683e3b95e3a48e63551fcfb51949de2f06a9d91dbee93c9" +dependencies = [ + "core-foundation-sys", + "libc", +] + [[package]] name = "tap" version = "1.0.1" @@ -2006,6 +2609,74 @@ dependencies = [ "serde_json", ] +[[package]] +name = "tinyvec" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" +dependencies = [ + "tinyvec_macros", +] + +[[package]] +name = "tinyvec_macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" + +[[package]] +name = "tokio" +version = "1.38.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba4f4a02a7a80d6f274636f0aa95c7e383b912d41fe721a31f29e29698585a4a" +dependencies = [ + "backtrace", + "bytes", + "libc", + "mio", + "num_cpus", + "parking_lot", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "windows-sys 0.48.0", +] + +[[package]] +name = "tokio-macros" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f5ae998a069d4b5aba8ee9dad856af7d520c3699e6159b185c2acd48155d39a" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.39", +] + +[[package]] +name = "tokio-native-tls" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbae76ab933c85776efabc971569dd6119c580d8f5d448769dec1764bf796ef2" +dependencies = [ + "native-tls", + "tokio", +] + +[[package]] +name = "tokio-util" +version = "0.7.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9cf6b47b3771c49ac75ad09a6162f53ad4b8088b76ac60e8ec1455b31a189fe1" +dependencies = [ + "bytes", + "futures-core", + "futures-sink", + "pin-project-lite", + "tokio", +] + [[package]] name = "toml_datetime" version = "0.6.3" @@ -2034,6 +2705,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" @@ -2107,6 +2805,12 @@ dependencies = [ "tracing-subscriber", ] +[[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" @@ -2131,12 +2835,27 @@ version = "0.1.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "eaea85b334db583fe3274d12b4cd1880032beab409c0d774be044d4480ab9a94" +[[package]] +name = "unicode-bidi" +version = "0.3.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08f95100a766bf4f8f28f90d77e0a5461bbdb219042e7679bebe79004fed8d75" + [[package]] name = "unicode-ident" version = "1.0.12" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" +[[package]] +name = "unicode-normalization" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a56d1686db2308d901306f92a263857ef59ea39678a5458e7cb17f01415101f5" +dependencies = [ + "tinyvec", +] + [[package]] name = "unicode-width" version = "0.1.11" @@ -2149,6 +2868,17 @@ version = "0.2.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f962df74c8c05a667b5ee8bcf162993134c104e96440b663c8daa176dc772d8c" +[[package]] +name = "url" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31e6302e3bb753d46e83516cae55ae196fc0c309407cf11ab35cc51a4c2a4633" +dependencies = [ + "form_urlencoded", + "idna", + "percent-encoding", +] + [[package]] name = "utf8parse" version = "0.2.1" @@ -2161,6 +2891,12 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d" +[[package]] +name = "vcpkg" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" + [[package]] name = "version_check" version = "0.9.4" @@ -2177,6 +2913,15 @@ dependencies = [ "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]] name = "wasi" version = "0.11.0+wasi-snapshot-preview1" @@ -2208,6 +2953,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" @@ -2419,6 +3176,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 = "wyz" version = "0.5.1" @@ -2430,9 +3197,9 @@ dependencies = [ [[package]] name = "yansi" -version = "0.5.1" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "09041cd90cf85f7f8b2df60c646f853b7f535ce68f85244eb6731cf89fa498ec" +checksum = "cfe53a6657fd280eaa890a3bc59152892ffa3e30101319d168b781ed6529b049" [[package]] name = "zerocopy" diff --git a/Cargo.toml b/Cargo.toml index 5bc8a535..e3e5e410 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -52,7 +52,7 @@ tracing-subscriber = { version = "0.3", features = [ tracing-tree = "0.3.0" ethers-core = "*" hex = "0.4.3" -ariadne = "0.2.0" +ariadne = "0.4.1" petgraph = "0.6.2" ahash = "0.8.10" @@ -61,10 +61,6 @@ ahash = "0.8.10" # [workspace] # members = ["cli", "shared"] -# we patch ariadne to allow for counting by bytes because solang uses byte locations not char locations -[patch.crates-io] -ariadne = { git = "https://github.com/brockelmore/ariadne" } - # ###################################### # # Benchmarks # ###################################### diff --git a/crates/analyzers/Cargo.toml b/crates/analyzers/Cargo.toml index 136ace93..a715b698 100644 --- a/crates/analyzers/Cargo.toml +++ b/crates/analyzers/Cargo.toml @@ -13,6 +13,7 @@ repository.workspace = true [dependencies] graph.workspace = true shared.workspace = true +tracing.workspace = true solang-parser.workspace = true ariadne.workspace = true \ No newline at end of file diff --git a/crates/analyzers/src/func_analyzer/mod.rs b/crates/analyzers/src/func_analyzer/mod.rs index dc3c5bcf..0eb3b0e3 100644 --- a/crates/analyzers/src/func_analyzer/mod.rs +++ b/crates/analyzers/src/func_analyzer/mod.rs @@ -114,6 +114,7 @@ impl<'a> FunctionVarsBoundAnalysis { .with_cross_gap(false) .with_underlines(true) .with_tab_width(4) + .with_index_type(ariadne::IndexType::Byte) .with_multiline_arrows(false), ); diff --git a/crates/analyzers/src/func_analyzer/report_display.rs b/crates/analyzers/src/func_analyzer/report_display.rs index be106ff4..828d8c9f 100644 --- a/crates/analyzers/src/func_analyzer/report_display.rs +++ b/crates/analyzers/src/func_analyzer/report_display.rs @@ -66,6 +66,7 @@ impl<'a> ReportDisplay for CLIFunctionVarsBoundAnalysis<'a> { Config::default() .with_cross_gap(false) .with_underlines(true) + .with_index_type(ariadne::IndexType::Byte) .with_tab_width(4), ); diff --git a/crates/analyzers/src/var_analyzer/report_display.rs b/crates/analyzers/src/var_analyzer/report_display.rs index a4b1e48a..403454bb 100644 --- a/crates/analyzers/src/var_analyzer/report_display.rs +++ b/crates/analyzers/src/var_analyzer/report_display.rs @@ -76,6 +76,7 @@ impl ReportDisplay for VarBoundAnalysis { Config::default() .with_cross_gap(false) .with_underlines(true) + .with_index_type(ariadne::IndexType::Byte) .with_tab_width(4), ); diff --git a/crates/cli/Cargo.toml b/crates/cli/Cargo.toml index 5029e551..4b76ced0 100644 --- a/crates/cli/Cargo.toml +++ b/crates/cli/Cargo.toml @@ -16,7 +16,6 @@ analyzers.workspace = true shared.workspace = true pyrometer.workspace = true graph.workspace = true -# queries.workspace = true ariadne.workspace = true tracing.workspace = true @@ -25,8 +24,9 @@ tracing-tree.workspace = true petgraph.workspace = true ethers-core.workspace = true - clap = { version = "4.1.4", features = ["derive"] } +tokio = { version = "1", features = ["full"] } +reqwest = { version = "0.12", features = ["json"] } [[bin]] name = "pyrometer" diff --git a/crates/cli/src/main.rs b/crates/cli/src/main.rs index a55d8426..372dbdab 100644 --- a/crates/cli/src/main.rs +++ b/crates/cli/src/main.rs @@ -5,12 +5,14 @@ use graph::{ Edge, }; use pyrometer::{Analyzer, Root, SourcePath}; -use shared::GraphDot; +use reqwest::Client; use shared::Search; +use shared::{GraphDot, USE_DEBUG_SITE}; use ariadne::sources; use clap::{ArgAction, Parser, ValueHint}; +use tracing::{debug, error, info, trace, warn}; use tracing_subscriber::{prelude::*, Registry}; use std::{ @@ -19,6 +21,7 @@ use std::{ fs, path::PathBuf, }; +use tokio::runtime::Runtime; #[derive(Parser, Debug)] #[command(author, version, about, long_about = None)] @@ -100,6 +103,10 @@ struct Args { /// Print stats about the IR #[clap(long)] pub stats: bool, + + /// Post pyrometer debugging information to debugging site + #[clap(long)] + pub debug_site: bool, } pub fn subscriber() { @@ -114,11 +121,12 @@ pub fn tree_subscriber() { .with( tracing_tree::HierarchicalLayer::default() .with_indent_lines(true) - .with_indent_amount(2) - .with_thread_names(true), // .with_thread_ids(true) - // .with_verbose_exit(true) - // .with_verbose_entry(true) - // .with_targets(true) + .with_indent_amount(1) + // .with_targets(true) + .with_thread_names(false), // .with_thread_ids(true) + // .with_verbose_exit(true) + // .with_verbose_entry(true) + // .with_targets(true) ) .with(tracing_subscriber::filter::EnvFilter::from_default_env()); tracing::subscriber::set_global_default(subscriber).unwrap(); @@ -226,6 +234,29 @@ fn main() { show_nonreverts: args.show_nonreverts.unwrap_or(true), }, }; + + if args.debug_site { + unsafe { + USE_DEBUG_SITE = true; + } + + let rt = Runtime::new().unwrap(); + rt.block_on(async { + let client = Client::new(); + let res = client + .post("http://127.0.0.1:8545/clear") + .send() + .await + .expect("Failed to send request"); + + if res.status().is_success() { + trace!("Successfully cleared history of site"); + } else { + error!("Failed to clear history of site: {:?}", res.status()); + } + }); + } + let mut analyzer = Analyzer { max_depth: args.max_stack_depth, root: Root::RemappingsDirectory(env::current_dir().unwrap()), diff --git a/crates/pyrometer/Cargo.toml b/crates/pyrometer/Cargo.toml index fd35e3e8..7a279ba5 100644 --- a/crates/pyrometer/Cargo.toml +++ b/crates/pyrometer/Cargo.toml @@ -24,6 +24,9 @@ tracing.workspace = true tracing-subscriber.workspace = true ahash.workspace = true serde_json = "1" +tokio = { version = "1", features = ["full"] } +serde = { version = "1", features = ["derive"] } +reqwest = { version = "0.12", features = ["json"] } diff --git a/crates/pyrometer/src/analyzer.rs b/crates/pyrometer/src/analyzer.rs index 8dee588d..2bc1e924 100644 --- a/crates/pyrometer/src/analyzer.rs +++ b/crates/pyrometer/src/analyzer.rs @@ -1,11 +1,15 @@ use crate::builtin_fns; use graph::elem::Elem; -use shared::RangeArena; - +use reqwest::Client; +use serde::{Deserialize, Serialize}; +use shared::{RangeArena, USE_DEBUG_SITE}; +use shared::GraphDot; use analyzers::LocStrSpan; use graph::{nodes::*, ContextEdge, Edge, Node, VarType}; use shared::{AnalyzerLike, GraphLike, JoinStats, NodeIdx, Search}; use solc_expressions::{ExprErr, FnCallBuilder, IntoExprErr, StatementParser}; +use tokio::runtime::Runtime; +use tracing::{debug, error, info, trace, warn}; use ahash::AHashMap; use ariadne::{Cache, Color, Config, Fmt, Label, Report, ReportKind, Source, Span}; @@ -160,7 +164,7 @@ impl Default for Analyzer { builtin_fn_inputs: Default::default(), expr_errs: Default::default(), max_depth: 200, - max_width: 2_i32.pow(14) as usize, + max_width: 2_i32.pow(14) as usize, // 14 splits == 16384 contexts parse_fn: NodeIdx::from(0).into(), debug_panic: false, fn_calls_fns: Default::default(), @@ -474,7 +478,8 @@ impl Analyzer { Config::default() .with_cross_gap(false) .with_underlines(true) - .with_tab_width(4), + .with_tab_width(4) + .with_index_type(ariadne::IndexType::Byte), ) .with_label( Label::new(str_span) @@ -497,6 +502,9 @@ impl Analyzer { let file_no = self.file_no; self.sources .push((current_path.clone(), src.to_string(), Some(file_no), None)); + if unsafe { USE_DEBUG_SITE } { + Self::post_source_to_site(file_no, ¤t_path.path_to_solidity_source(), src); + } match solang_parser::parse(src, file_no) { Ok((source_unit, _comments)) => { let parent = @@ -971,7 +979,7 @@ impl Analyzer { let normalized_remapped = normalize_path(remapped.path_to_solidity_source()); // take self.sources entry with the same path as remapped and update the file_no - if let Some((_, _, optional_file_no, _)) = + if let Some((source_path, source, optional_file_no, _)) = self.sources.iter_mut().find(|(path, _, _, _)| { normalize_path(path.path_to_solidity_source()) == normalized_remapped }) @@ -983,6 +991,10 @@ impl Analyzer { self.file_no += 1; let file_no = self.file_no; *optional_file_no = Some(file_no); + if unsafe { USE_DEBUG_SITE } { + // send the source to the site + Self::post_source_to_site(file_no, &source_path.path_to_solidity_source(), source); + } } let maybe_entry = self.parse(arena, &sol, &remapped, false); @@ -1390,6 +1402,42 @@ impl Analyzer { }; ty_node } + + fn post_source_to_site(file_no: usize, path: &PathBuf, source: &str) + where + Self: std::marker::Sized, + Self: AnalyzerLike, + { + let rt = Runtime::new().unwrap(); + rt.block_on(async { + Self::post_source_to_site_async(file_no, path, source).await; + }); + } + + async fn post_source_to_site_async(file_no: usize, path: &PathBuf, source: &str) where + Self: std::marker::Sized, + Self: AnalyzerLike, + { + let client = Client::new(); + let source_msg = SourceMessage { + file_number: file_no, + path: path.to_path_buf(), + source: source.to_string(), + }; + let res = client + .post("http://127.0.0.1:8545/addsource") + .json(&source_msg) + .send() + .await + .expect("Failed to send request"); + + if res.status().is_success() { + trace!("Successfully posted source to site"); + } else { + error!("Failed to post source to site: {:?}", res.status()); + } + + } } /// Print the report of parser's diagnostics @@ -1450,3 +1498,10 @@ pub fn normalize_path>(path: P) -> PathBuf { normalized_path } + +#[derive(Serialize, Deserialize, Debug, PartialEq, Eq, Hash)] +struct SourceMessage { + file_number: usize, + path: PathBuf, + source: String, +} \ No newline at end of file diff --git a/crates/pyrometer/src/analyzer_backend.rs b/crates/pyrometer/src/analyzer_backend.rs index b4834d50..1c3fbe2c 100644 --- a/crates/pyrometer/src/analyzer_backend.rs +++ b/crates/pyrometer/src/analyzer_backend.rs @@ -1,5 +1,5 @@ use crate::Analyzer; - +use shared::GraphDot; use graph::{ elem::Elem, nodes::{ @@ -18,7 +18,7 @@ use solang_parser::{ pt::{Expression, Loc}, }; -use std::collections::BTreeMap; +use std::{collections::BTreeMap, path::PathBuf}; impl AnalyzerBackend for Analyzer { fn add_concrete_var( @@ -276,4 +276,17 @@ impl AnalyzerLike for Analyzer { fn handled_funcs_mut(&mut self) -> &mut Vec { &mut self.handled_funcs } + + fn file_mapping(&self) -> BTreeMap { + let mut file_mapping: BTreeMap = BTreeMap::new(); + for (source_path, _, o_file_no, _) in self.sources.iter() { + if let Some(file_no) = o_file_no { + file_mapping.insert( + *file_no, + source_path.path_to_solidity_source().display().to_string(), + ); + } + } + file_mapping + } } diff --git a/crates/pyrometer/src/graph_backend.rs b/crates/pyrometer/src/graph_backend.rs index 3512cbdd..176729bf 100644 --- a/crates/pyrometer/src/graph_backend.rs +++ b/crates/pyrometer/src/graph_backend.rs @@ -5,7 +5,7 @@ use shared::RangeArena; // use std::collections::hash_map::DefaultHasher; // use std::hash::Hash; // use std::hash::Hasher; - +use tracing::{trace, debug, error, warn}; use graph::{ as_dot_str, nodes::ContextNode, AnalyzerBackend, AsDotStr, ContextEdge, Edge, GraphBackend, Node, @@ -128,6 +128,10 @@ impl GraphDot for Analyzer { return None; } + if handled_nodes.lock().unwrap().contains(child) { + return None; + } + let post_str = match self.node(*child) { Node::Context(c) => { *cluster_num += 2; @@ -260,6 +264,7 @@ impl GraphDot for Analyzer { )) } else { { + if handled_nodes.lock().unwrap().contains(child) { if handled_nodes.lock().unwrap().contains(child) { return None; } else { @@ -291,6 +296,7 @@ impl GraphDot for Analyzer { format!("{indent}{from:} -->|\"{edge_str}\"| {to:}\n{indent}class {to} linkSource{edge_idx}\n{indent}class {from} linkTarget{edge_idx}") } else { format!("{indent}{from:} -> {to:} [label = \"{edge_str}\"]",) + format!("{indent}{from:} -> {to:} [label = \"{edge_str}\"]",) } }) .collect::>() @@ -363,7 +369,7 @@ impl GraphDot for Analyzer { let raw_start_str = r##"digraph G { node [shape=box, style="filled, rounded", color="#565f89", fontcolor="#d5daf0", fontname="Helvetica", fillcolor="#24283b"]; edge [color="#414868", fontcolor="#c0caf5", fontname="Helvetica"]; - bgcolor="#1a1b26"; rankdir="BT"; splines=ortho;"##; + bgcolor="#1a1b26"; rankdir="BT"; splines=ortho; size="6,6"; ratio="fill";layout="fdp";"##; dot_str.push(raw_start_str.to_string()); let handled_edges = Arc::new(Mutex::new(BTreeSet::new())); let handled_nodes = Arc::new(Mutex::new(BTreeSet::new())); @@ -461,9 +467,9 @@ impl GraphDot for Analyzer { ); let mut dot_str = Vec::new(); let raw_start_str = r##"digraph G { - node [shape=box, style="filled, rounded", color="#565f89", fontcolor="#d5daf0", fontname="Helvetica", fillcolor="#24283b"]; - edge [color="#414868", fontcolor="#c0caf5", fontname="Helvetica"]; - bgcolor="#1a1b26";"##; + node [shape=box, style="filled, rounded", color="#565f89", fontcolor="#d5daf0", fontname="Helvetica", fillcolor="#24283b"]; + edge [color="#414868", fontcolor="#c0caf5", fontname="Helvetica"]; + bgcolor="#1a1b26"; rankdir="BT"; splines=ortho; size="6,6"; ratio="fill";layout="fdp";"##; dot_str.push(raw_start_str.to_string()); let nodes_and_edges_str = format!( "{:?}", @@ -646,6 +652,44 @@ impl GraphLike for G<'_> { } } +impl GraphDot for G<'_> { + fn cluster_str( + &self, + node: NodeIdx, + cluster_num: &mut usize, + is_killed: bool, + handled_nodes: Arc>>, + handled_edges: Arc>>>, + depth: usize, + as_mermaid: bool, + ) -> Option + where + Self: std::marker::Sized { + todo!() + } + + fn dot_str(&self) -> String + where + Self: std::marker::Sized, + Self: shared::AnalyzerLike { + todo!() + } + + fn dot_str_no_tmps(&self) -> String + where + Self: std::marker::Sized, + Self: GraphLike + shared::AnalyzerLike { + todo!() + } + + fn mermaid_str(&self) -> String + where + Self: std::marker::Sized, + Self: shared::AnalyzerLike { + todo!() + } +} + impl GraphBackend for G<'_> {} pub fn mermaid_node( @@ -655,6 +699,7 @@ pub fn mermaid_node( node: NodeIdx, style: bool, loc: bool, + loc: bool, class: Option<&str>, ) -> String { let mut node_str = format!( @@ -671,19 +716,54 @@ pub fn mermaid_node( )); } + // if loc { + // match g.node(node) { + // Node::ContextVar(..) => { + // match graph::nodes::ContextVarNode::from(node).loc(g).unwrap() { + // solang_parser::pt::Loc::File(f, s, e) => { + // node_str.push_str(&format!( + // "\n{indent}class {} loc_{f}_{s}_{e}", + // petgraph::graph::GraphIndex::index(&node) + // )); + // } + // _ => {} + // } + // }, + // _ => {} + // } + // } if loc { - match g.node(node) { + let mut current_node = node; + match g.node(current_node) { Node::ContextVar(..) => { - if let solang_parser::pt::Loc::File(f, s, e) = - graph::nodes::ContextVarNode::from(node).loc(g).unwrap() - { + // highlight self + if let Ok(loc) = graph::nodes::ContextVarNode::from(current_node).loc(g) { + if let solang_parser::pt::Loc::File(f, s, e) = loc { + node_str.push_str(&format!( + "\n{indent}class {} loc_{f}_{s}_{e}", + petgraph::graph::GraphIndex::index(¤t_node) + )); + } + } + + // color the forks + let ctx_node = graph::nodes::ContextVarNode::from(current_node).ctx(g); + gather_context_info(g, indent, ctx_node, current_node, &mut node_str); + }, + Node::Context(ctx) => { + // highlight self + if let solang_parser::pt::Loc::File(f, s, e) = ctx.loc { node_str.push_str(&format!( "\n{indent}class {} loc_{f}_{s}_{e}", - petgraph::graph::GraphIndex::index(&node) + petgraph::graph::GraphIndex::index(¤t_node) )); } - } - _ => {} + + // color the forks + let ctx_node = graph::nodes::ContextNode::from(current_node); + gather_context_info(g, indent, ctx_node, current_node, &mut node_str); + }, + _ => {}, } } @@ -696,3 +776,47 @@ pub fn mermaid_node( node_str } + +fn gather_context_info( + g: &impl GraphBackend, + indent: &str, + mut ctx_node: ContextNode, + original_cvar_node: NodeIdx, + node_str: &mut String, +) { + + loop { + // warn!("in loop, {:?}", ctx_node); + + let mut found_continue = false; + let mut current_loc = ctx_node.underlying(g).unwrap().loc; + for edge in g.graph().edges_directed(ctx_node.into(), Direction::Outgoing) { + if let Edge::Context(ContextEdge::Continue(true_or_false)) = edge.weight() { + let target_node = edge.target(); + if let Node::Context(ctx) = g.node(target_node) { + // error!("found continue pointing to node"); + ctx_node = target_node.into(); + found_continue = true; + // Gather the edge weight and the loc of the Context node it points to + if let solang_parser::pt::Loc::File(f, s, e) = current_loc { + let fork_str = format!( + "\n{indent}class {} fork_{f}_{s}_{e}_{}", + petgraph::graph::GraphIndex::index(&original_cvar_node), + match *true_or_false { + "fork_true" => true, + "fork_false" => false, + _ => false, + } + ); + // error!("in gather_context_info, {:?}", fork_str); + node_str.push_str(&fork_str); + } + break; + } + } + } + if !found_continue { + break; + } + } +} \ No newline at end of file diff --git a/crates/shared/Cargo.toml b/crates/shared/Cargo.toml index 22e1c17b..a69f834d 100644 --- a/crates/shared/Cargo.toml +++ b/crates/shared/Cargo.toml @@ -17,4 +17,8 @@ ethers-core.workspace = true hex.workspace = true tracing.workspace = true tracing-subscriber.workspace = true -ahash.workspace = true \ No newline at end of file +ahash.workspace = true + +tokio = { version = "1", features = ["full"] } +serde = { version = "1", features = ["derive"] } +reqwest = { version = "0.12", features = ["json"] } diff --git a/crates/shared/src/analyzer_like.rs b/crates/shared/src/analyzer_like.rs index e4cd7b25..1145e4af 100644 --- a/crates/shared/src/analyzer_like.rs +++ b/crates/shared/src/analyzer_like.rs @@ -183,4 +183,5 @@ pub trait AnalyzerLike: GraphLike { fn join_stats_mut(&mut self) -> &mut JoinStats; fn handled_funcs(&self) -> &[Self::FunctionNode]; fn handled_funcs_mut(&mut self) -> &mut Vec; + fn file_mapping(&self) -> BTreeMap; } From a20e8e09b59b18a1f54ad19e878791b9c2773414 Mon Sep 17 00:00:00 2001 From: plotchy Date: Tue, 11 Jun 2024 22:51:18 -0400 Subject: [PATCH 2/6] feat: debug site --- crates/graph/src/graph_elements.rs | 8 ++- crates/shared/src/graph_like.rs | 86 +++++++++++++++++++++++++++++- 2 files changed, 90 insertions(+), 4 deletions(-) diff --git a/crates/graph/src/graph_elements.rs b/crates/graph/src/graph_elements.rs index 652d643d..d5946a2d 100644 --- a/crates/graph/src/graph_elements.rs +++ b/crates/graph/src/graph_elements.rs @@ -1,7 +1,7 @@ use crate::elem::Elem; use crate::{nodes::*, VarType}; -use shared::{AnalyzerLike, GraphLike, Heirarchical, NodeIdx, RangeArena}; +use shared::{AnalyzerLike, GraphDot, GraphLike, Heirarchical, NodeIdx, RangeArena}; use lazy_static::lazy_static; use petgraph::{Directed, Graph}; @@ -9,7 +9,11 @@ use solang_parser::pt::{Identifier, Loc}; use std::collections::HashMap; -pub trait GraphBackend: GraphLike> {} + +pub trait GraphBackend: + GraphLike> + + GraphDot +{} pub trait AnalyzerBackend: AnalyzerLike< Builtin = Builtin, diff --git a/crates/shared/src/graph_like.rs b/crates/shared/src/graph_like.rs index 041f99aa..6b717ed5 100644 --- a/crates/shared/src/graph_like.rs +++ b/crates/shared/src/graph_like.rs @@ -12,6 +12,14 @@ use std::{ sync::{Arc, Mutex}, }; +use reqwest::Client; +use serde::{Deserialize, Serialize}; +use std::time::{SystemTime, UNIX_EPOCH}; +use tokio::runtime::Runtime; +use tracing::{error, trace}; + +pub static mut USE_DEBUG_SITE: bool = false; + pub type NodeIdx = NodeIndex; pub type EdgeIdx = EdgeIndex; pub type RangeArenaIdx = usize; @@ -32,7 +40,7 @@ pub trait GraphLike { /// Get a reference to the graph fn graph(&self) -> &Graph; /// Add a node to the graph - fn add_node(&mut self, node: impl Into) -> NodeIdx { + fn _add_node(&mut self, node: impl Into) -> NodeIdx { let res = self.graph_mut().add_node(node.into()); res } @@ -49,7 +57,7 @@ pub trait GraphLike { .expect("Index not in graph") } /// Add an edge to the graph - fn add_edge( + fn _add_edge( &mut self, from_node: impl Into, to_node: impl Into, @@ -154,6 +162,74 @@ pub trait GraphDot: GraphLike { .expect("failed to execute process"); } + fn add_node(&mut self, node: impl Into) -> NodeIdx + where + Self: std::marker::Sized, + Self: AnalyzerLike, + { + let res = self.graph_mut().add_node(node.into()); + if unsafe { USE_DEBUG_SITE } { + self.post_to_site(); + } + res + } + + fn add_edge( + &mut self, + from_node: impl Into, + to_node: impl Into, + edge: impl Into, + ) where + Self: std::marker::Sized, + Self: AnalyzerLike, + { + self.graph_mut() + .add_edge(from_node.into(), to_node.into(), edge.into()); + + if unsafe { USE_DEBUG_SITE } { + self.post_to_site(); + } + } + + fn post_to_site(&self) + where + Self: std::marker::Sized, + Self: AnalyzerLike, + { + let rt = Runtime::new().unwrap(); + rt.block_on(async { + self.post_to_site_async().await; + }); + } + + async fn post_to_site_async(&self) + where + Self: std::marker::Sized, + Self: AnalyzerLike, + { + let client = Client::new(); + let graph_msg = GraphMessage { + graph: self.mermaid_str(), + timestamp: SystemTime::now() + .duration_since(UNIX_EPOCH) + .expect("Time went backwards") + .as_secs(), + }; + + let res = client + .post("http://127.0.0.1:8545/addgraph") + .json(&graph_msg) + .send() + .await + .expect("Failed to send request"); + + if res.status().is_success() { + trace!("Successfully posted dot to site"); + } else { + error!("Failed to post graph to site: {:?}", res.status()); + } + } + /// Creates a subgraph for visually identifying contexts and subcontexts fn cluster_str( &self, @@ -186,3 +262,9 @@ pub trait GraphDot: GraphLike { Self: std::marker::Sized, Self: AnalyzerLike; } + +#[derive(Serialize, Deserialize, Debug)] +struct GraphMessage { + graph: String, + timestamp: u64, +} From 6524e0c94cbc804b56096420a54dec1eed09596d Mon Sep 17 00:00:00 2001 From: plotchy Date: Tue, 18 Jun 2024 11:44:35 -0400 Subject: [PATCH 3/6] feat: arena debugging enabled --- crates/cli/src/main.rs | 18 + crates/graph/src/graph_elements.rs | 1 + .../graph/src/range/elem/elem_enum/impls.rs | 27 + crates/pyrometer/src/graph_backend.rs | 651 +++++++++++++++++- crates/pyrometer/src/lib.rs | 2 +- 5 files changed, 662 insertions(+), 37 deletions(-) diff --git a/crates/cli/src/main.rs b/crates/cli/src/main.rs index 372dbdab..0df1f60f 100644 --- a/crates/cli/src/main.rs +++ b/crates/cli/src/main.rs @@ -295,6 +295,24 @@ fn main() { println!("DONE ANALYZING IN: {parse_time}ms. Writing to cli..."); + // println!("Arena: {:#?}", analyzer.range_arena); + if unsafe { USE_DEBUG_SITE } { + use shared::GraphLike; + use pyrometer::graph_backend::mermaid_str; + use pyrometer::graph_backend::post_to_site_arena; + use pyrometer::graph_backend::Elems; + match Elems::try_from(analyzer.range_arena()) { + Ok(elems) => { + let elems_graph = elems.to_graph(&analyzer); + let elems_graph_mermaid_str = mermaid_str(&elems_graph); + post_to_site_arena(elems_graph_mermaid_str); + } + Err(e) => { + eprintln!("Can't post arena, error creating Elems: {:?}", e); + } + }; + } + if args.stats { println!("{}", analyzer.stats(t_end, arena)); } diff --git a/crates/graph/src/graph_elements.rs b/crates/graph/src/graph_elements.rs index d5946a2d..92087dae 100644 --- a/crates/graph/src/graph_elements.rs +++ b/crates/graph/src/graph_elements.rs @@ -230,6 +230,7 @@ lazy_static! { m.insert("bg", "#1a1b26"); m.insert("font", "#c0caf5"); m.insert("deepred", "#703440"); + m.insert("deeporange", "#b5683c"); m.insert("default", "#565f89"); m }; diff --git a/crates/graph/src/range/elem/elem_enum/impls.rs b/crates/graph/src/range/elem/elem_enum/impls.rs index f9aa87d8..a592d16a 100644 --- a/crates/graph/src/range/elem/elem_enum/impls.rs +++ b/crates/graph/src/range/elem/elem_enum/impls.rs @@ -57,6 +57,33 @@ impl Elem { _ => None, } } + + pub fn arena_graph_node_label(&self) -> String { + match self { + Elem::Reference(reference) => { + format!("Ref-CVar{}", reference.idx.index()) + } + Elem::ConcreteDyn(range_dyn) => { + format!("concdyn-{}", self) + } + Elem::Concrete(range_concrete) => { + format!("conc-{}", self) + } + Elem::Expr(range_expr) => { + // Unbox and check the lhs and rhs to see if they are arena indices + let lhs_str = range_expr.lhs.arena_graph_node_label(); + let rhs_str = range_expr.rhs.arena_graph_node_label(); + let op = range_expr.op.clone(); + format!("expr-{}", &self) + }, + Elem::Arena(arena_idx) => { + format!("Arena({})", arena_idx) + }, + Elem::Null => { + todo!() + }, + } + } } impl Elem { diff --git a/crates/pyrometer/src/graph_backend.rs b/crates/pyrometer/src/graph_backend.rs index 176729bf..a5b1d894 100644 --- a/crates/pyrometer/src/graph_backend.rs +++ b/crates/pyrometer/src/graph_backend.rs @@ -1,10 +1,17 @@ use crate::Analyzer; -use graph::elem::Elem; +use std::{collections::HashMap, fmt::Display}; +use graph::{elem::Elem, nodes::ContextVarNode, range_string::ToRangeString, SolcRange, TOKYO_NIGHT_COLORS}; +use graph::elem::RangeElem; use graph::nodes::Concrete; -use shared::RangeArena; -// use std::collections::hash_map::DefaultHasher; -// use std::hash::Hash; -// use std::hash::Hasher; +use reqwest::Client; +use serde::{Deserialize, Serialize}; +use shared::{RangeArena, USE_DEBUG_SITE}; +use tokio::runtime::Runtime; +use std::cell::RefCell; +use std::collections::hash_map::DefaultHasher; +use std::hash::Hash; +use std::hash::Hasher; +use std::rc::Rc; use tracing::{trace, debug, error, warn}; use graph::{ as_dot_str, nodes::ContextNode, AnalyzerBackend, AsDotStr, ContextEdge, Edge, GraphBackend, @@ -13,7 +20,7 @@ use graph::{ use shared::{GraphDot, GraphLike, NodeIdx, Search}; use petgraph::{dot::Dot, graph::EdgeIndex, visit::EdgeRef, Directed, Direction, Graph}; - +use std::convert::TryFrom; use std::{ collections::BTreeSet, sync::{Arc, Mutex}, @@ -39,20 +46,610 @@ impl GraphLike for Analyzer { &mut self.range_arena } - // fn range_arena_idx(&self, elem: &Self::RangeElem) -> Option { - // if let Elem::Arena(idx) = elem { - // Some(*idx) - // } else { - // self.range_arena().map.get(elem).copied() - // } - // } + fn range_arena_idx(&self, elem: &Self::RangeElem) -> Option { + if let Elem::Arena(idx) = elem { + Some(*idx) + } else { + self.range_arena().map.get(elem).copied() + } + } + + fn range_arena_idx_or_upsert(&mut self, elem: Self::RangeElem) -> usize { + // tracing::trace!("arenaizing: {}", elem); + if let Elem::Arena(idx) = elem { + return idx; + } + + let res_idx = if let Some(idx) = self.range_arena_idx(&elem) { + let existing = &self.range_arena().ranges[idx]; + let Ok(existing) = existing.try_borrow_mut() else { + return idx; + }; + let (min_cached, max_cached) = existing.is_min_max_cached(self); + let mut existing_count = 0; + if min_cached { + existing_count += 1; + } + if max_cached { + existing_count += 1; + } + if existing.is_flatten_cached(self) { + existing_count += 1; + } + + let (min_cached, max_cached) = elem.is_min_max_cached(self); + let mut new_count = 0; + if min_cached { + new_count += 1; + } + if max_cached { + new_count += 1; + } + if elem.is_flatten_cached(self) { + new_count += 1; + } + + drop(existing); + + if new_count >= existing_count { + self.range_arena_mut().ranges[idx] = Rc::new(RefCell::new(elem)); + } + + idx + } else { + let idx = self.range_arena().ranges.len(); + self.range_arena_mut() + .ranges + .push(Rc::new(RefCell::new(elem.clone()))); + self.range_arena_mut().map.insert(elem, idx); + idx + }; + + // if unsafe { USE_DEBUG_SITE } { + // let elems = Elems::from(self.range_arena()); + // let elems_graph = elems.to_graph(self); + // let elems_graph_mermaid_str = mermaid_str(&elems_graph); + // post_to_site_arena(elems_graph_mermaid_str); + // } + res_idx + } +} + +pub fn post_to_site_arena(arena_str: String) { + let rt = Runtime::new().unwrap(); + rt.block_on(async { + post_to_site_arena_async(arena_str).await; + }); +} + +pub async fn post_to_site_arena_async(arena_str: String) { + let client = Client::new(); + let graph_msg = ArenaMessage { + arena: arena_str.to_string(), + }; + + let res = client + .post("http://127.0.0.1:8545/updatearena") + .json(&graph_msg) + .send() + .await + .expect("Failed to send request"); + + if res.status().is_success() { + trace!("Successfully posted arena to site"); + } else { + error!("Failed to post arena to site: {:?}", res.status()); + } +} + +#[derive(Serialize, Deserialize, Debug, PartialEq, Eq, Hash)] +struct ArenaMessage { + arena: String, +} + +#[derive(Serialize, Deserialize, Debug, PartialEq, Eq, Hash)] +pub enum ArenaNode { + /// Arena node (index) + ARENA(usize), + /// ContextVar node (label is the string representation of the node) + CVAR(String), + /// Elem node (e.g. an expression) (label is the string representation of the node) + ELEM(String), +} + +impl Display for ArenaNode { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + // warning: changing these will impact the pyro-debug site when rendering edge styling + let variant_name = match self { + ArenaNode::ARENA(idx) => format!("index {}", idx), + ArenaNode::ELEM(label) => label.to_string(), + ArenaNode::CVAR(label) => label.to_string(), + }; + write!(f, "{}", variant_name) + } +} + +impl ArenaNode { + pub fn dot_str_color(&self) -> String { + let c = match self { + ArenaNode::ARENA(_) => TOKYO_NIGHT_COLORS.get("red1").unwrap(), + ArenaNode::CVAR(_) => TOKYO_NIGHT_COLORS.get("deeporange").unwrap(), + ArenaNode::ELEM(_) => TOKYO_NIGHT_COLORS.get("blue0").unwrap(), + _ => TOKYO_NIGHT_COLORS.get("default").unwrap(), + }; + c.to_string() + } +} + + +#[derive(Serialize, Deserialize, Debug, PartialEq, Eq, Hash)] +pub enum ArenaEdge { + LHS, + RHS, + ARENA, + VAR, + REF, + MIN, + MAX, + NONE, +} + +impl Display for ArenaEdge { + + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + // warning: changing these will impact the pyro-debug site when rendering edge styling + let variant_name = match self { + ArenaEdge::LHS => "LHS", + ArenaEdge::RHS => "RHS", + ArenaEdge::ARENA => "ARENA", + ArenaEdge::VAR => "VAR", + ArenaEdge::MIN => "MIN", + ArenaEdge::MAX => "MAX", + ArenaEdge::REF => "REF", + ArenaEdge::NONE => "", + }; + write!(f, "{}", variant_name) + } +} + +#[derive(Debug)] +pub enum ElemsError { + BorrowError(String), + MissingMap(String), +} + +impl std::fmt::Display for ElemsError { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + ElemsError::BorrowError(msg) => write!(f, "BorrowError: {}", msg), + ElemsError::MissingMap(msg) => write!(f, "MissingMap: {}", msg), + } + } +} +impl std::error::Error for ElemsError {} +pub struct Elems { + /// arena_idx, elem + pub inner: Vec<(usize, Elem)> +} + +impl TryFrom<&RangeArena>> for Elems { + type Error = ElemsError; + fn try_from(arena: &RangeArena>) -> Result { + // Collect the elements and their indices first from ranges + // Collect the elements and their indices first from ranges + let mut inner = Vec::new(); + for rc in &arena.ranges { + // Attempt to borrow the RefCell + match rc.try_borrow() { + Ok(borrowed) => { + let elem = borrowed.clone(); + // Get the map value + if let Some(map_value) = arena.map.get(&elem).copied() { + // println!("Adding idx {} to elems {}", map_value, elem); + inner.push((map_value, elem)); + } else { + // println!("NONE REF elem: {:?}", elem); + return Err(ElemsError::MissingMap(format!("elem {:?}", elem))); + } + }, + Err(e) => { + // Print an error message if borrowing fails + // eprintln!("Failed to borrow RefCell: {:?}", e); + return Err(ElemsError::BorrowError(format!("error {:?}", e))); + } + } + } + + // Search .map for any entries that werent in .ranges + let inner_indices: BTreeSet<_> = inner.iter().map(|(idx, _)| *idx).collect(); + let missing_entries: Vec<_> = arena.map.iter() + .filter(|(_, &v)| !inner_indices.contains(&v)) + .collect(); + + { + // Log out missing entries + let missing_entries_str = missing_entries.iter() + .map(|(idx, elem)| format!("\telem {}: {}", idx, elem)) + .collect::>() + .join("\n"); + warn!( + "`RangeArena.ranges` is missing {} entries from the map:\n{}", + missing_entries.len(), + missing_entries_str + ); + } + + // Add any missing entries to inner + for (elem, &idx) in missing_entries { + if let Some(range_elem) = arena.ranges.get(idx) { + if let Ok(borrowed_elem) = range_elem.try_borrow() { + inner.push((idx, borrowed_elem.clone())); + } + } + } + + // Sort the collected elements by their indices + inner.sort_by(|a, b| a.0.cmp(&b.0)); + // dedup is needed as there are duplicate indices in the inner vec. TODO @brock is this a bug? arena has duplicate elems + inner.dedup(); + + // Print out elems + // for (idx, elem) in inner.iter() { + // println!("elem {}: {}", idx, elem); + // } + + Ok(Elems { inner }) + } } -// fn calculate_hash(t: &T) -> u64 { -// let mut s = DefaultHasher::new(); -// t.hash(&mut s); -// s.finish() -// } +impl Elems { + + /// Convert Elems into a Graph + /// + /// First pass: + /// - create nodes for each arena index + /// - this is needed separately, since when making edges some earlier nodes point to later nodes (that havent been made yet) + /// + /// Second pass: + /// - create an edge between each arena index and the elem that it represents, make an edge between them + /// - if elem is a reference, create nodes for the ContextVar that it depends on, make an edge from the elem to the ContextVar + /// + /// Third pass: + /// - for each ContextVar node, create edges to the arena indices that it depends on + /// + pub fn to_graph(&self, graph_backend: &impl GraphBackend) -> Graph { + let mut graph = Graph::default(); + let mut arena_idx_to_node_idx = HashMap::new(); + let mut dependency_map: HashMap> = HashMap::new(); + + // FIRST PASS: create nodes for each arena index + self.inner.iter().for_each(|(arena_idx, elem)| { + // add an arena node to the graph for the index + let arena_node_idx = graph.add_node(ArenaNode::ARENA(*arena_idx)); + arena_idx_to_node_idx.insert(arena_idx, arena_node_idx); + }); + + // SECOND PASS: create an edge between each arena index and the elem that it represents + self.inner.iter().for_each(|(arena_idx, elem)| { + // get the arena_node_idx for the arena index + let arena_node_idx = arena_idx_to_node_idx.get(arena_idx).unwrap(); + + // add a node for what the arena index has underlying it, and maybe edges for that elem + let underlying_node_idx = match elem { + Elem::Reference(reference) => { + let node_str = elem.arena_graph_node_label(); + let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); + + // attempt to add in the ContextVar node that the elem is referencing + let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); + context_var_nodes.into_iter().for_each(|dep_elem| { + let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { + // don't make a new ContextVar node, just use the existing one + existing_node_idx + } else { + // make a new ContextVar Node for the Arena graph + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend)))); + dependency_map.insert(dep_elem.clone(), new_node_idx); + new_node_idx + }; + // add an edge from the node to its dependency node + graph.add_edge(node_idx, dep_node_idx, ArenaEdge::VAR); + }); + + node_idx + }, + Elem::ConcreteDyn(range_dyn) => { + let node_str = elem.arena_graph_node_label(); + let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); + node_idx + }, + Elem::Concrete(range_concrete) => { + let node_str = elem.arena_graph_node_label(); + let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); + node_idx + }, + Elem::Expr(range_expr) => { + let node_str = elem.arena_graph_node_label(); + let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); + + // Unbox and check the lhs and rhs to see if they are arena indices + let lhs_arena = match *range_expr.lhs.clone() { + Elem::Arena(lhs) => Some(lhs), + Elem::Reference(lhs) => { + // println!("LHS is a reference: {}", range_expr.lhs); + // attempt to add in the ContextVar node that the elem is referencing + let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); + context_var_nodes.iter().for_each(|dep_elem| { + let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { + // don't make a new ContextVar node, just use the existing one + existing_node_idx + } else { + // make a new ContextVar Node for the Arena graph + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend)))); + dependency_map.insert(dep_elem.clone(), new_node_idx); + new_node_idx + }; + // use `update_edge` to avoid adding duplicate edges + graph.update_edge(node_idx, dep_node_idx, ArenaEdge::VAR); + }); + None + }, + _ => None, + }; + let rhs_arena = match *range_expr.rhs.clone() { + Elem::Arena(rhs) => { + // println!("RHS is an arena index: {}", range_expr.rhs); + Some(rhs) + }, + Elem::Reference(rhs) => { + // println!("RHS is a reference: {}", range_expr.rhs); + // attempt to add in the ContextVar node that the elem is referencing + let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); + context_var_nodes.iter().for_each(|dep_elem| { + let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { + // don't make a new ContextVar node, just use the existing one + existing_node_idx + } else { + // make a new ContextVar Node for the Arena graph + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend)))); + dependency_map.insert(dep_elem.clone(), new_node_idx); + new_node_idx + }; + // use `update_edge` to avoid adding duplicate edges + graph.update_edge(node_idx, dep_node_idx, ArenaEdge::VAR); + }); + None + }, + _ => { + println!("RHS is not an arena index: {}", range_expr.rhs); + None + }, + }; + + // Add edges to the arena indices if they exist + if let Some(lhs_idx) = lhs_arena { + if let Some(&lhs_node_idx) = arena_idx_to_node_idx.get(&lhs_idx) { + graph.add_edge(node_idx, lhs_node_idx, ArenaEdge::LHS); + } + } + + if let Some(rhs_idx) = rhs_arena { + if let Some(&rhs_node_idx) = arena_idx_to_node_idx.get(&rhs_idx) { + graph.add_edge(node_idx, rhs_node_idx, ArenaEdge::RHS); + } + } + node_idx + }, + Elem::Arena(range_arena_idx) => { + panic!("Arena index in elems: {:?}. This should not happen!", range_arena_idx); + }, + Elem::Null => { + let node_str = format!("null"); + let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); + node_idx + }, + }; + + // draw edge from the arena_node to the underlying node + graph.add_edge(*arena_node_idx, underlying_node_idx, ArenaEdge::NONE); + }); + + // THIRD PASS - iterate over ContextVarNodes + // iterate over the dependency map and add edges between the ContextVar nodes and the arena nodes + // println!("dependency map: {:?}", dependency_map); + for (cvar_node, &node_idx) in dependency_map.iter() { + // println!("cvar node: {:?}, node idx: {:?}", cvar_node, node_idx); + // Find the appropriate arena_idx for range.min and range.max using Elems.inner + if let Ok(Some(range_min)) = cvar_node.range_min(graph_backend) { + // println!(" range min: {:?}", range_min); + match range_min { + Elem::Arena(arena_idx) => { + // Make a direct edge to the arena node + // println!(" arena idx: {}", arena_idx); + if let Some(&min_node_idx) = arena_idx_to_node_idx.get(&arena_idx) { + graph.add_edge(node_idx, min_node_idx, ArenaEdge::MIN); + } + }, + _ => { + // attempt to find the elem in our `inner` and get the associated arena_graph index + let min_arena_idx = self.inner.iter() + .find(|(_, elem)| elem == &range_min) + .map(|(idx, _)| *idx); + // Add edges to the min arena indices + if let Some(min_idx) = min_arena_idx { + // println!(" min idx: {:?}", min_idx); + if let Some(&min_node_idx) = arena_idx_to_node_idx.get(&min_idx) { + // println!(" min node idx: {:?}", min_node_idx); + graph.add_edge(node_idx, min_node_idx, ArenaEdge::MIN); + } + } + }, + } + } + + if let Ok(Some(range_max)) = cvar_node.range_max(graph_backend) { + // println!(" range max: {:?}", range_max); + match range_max { + Elem::Arena(arena_idx) => { + // Make a direct edge to the arena node + // println!(" arena idx: {}", arena_idx); + if let Some(&max_node_idx) = arena_idx_to_node_idx.get(&arena_idx) { + graph.add_edge(node_idx, max_node_idx, ArenaEdge::MAX); + } + }, + _ => { + // attempt to find the elem in our `inner` and get the associated arena_graph index + let max_arena_idx = self.inner.iter() + .find(|(_, elem)| elem == &range_max) + .map(|(idx, _)| *idx); + // Add edges to the min arena indices + if let Some(max_idx) = max_arena_idx { + // println!(" max idx: {:?}", max_idx); + if let Some(&max_node_idx) = arena_idx_to_node_idx.get(&max_idx) { + // println!(" max node idx: {:?}", max_node_idx); + graph.add_edge(node_idx, max_node_idx, ArenaEdge::MAX); + } + } + }, + } + } + } + + // Ensure the graph does not have a cycle + debug_assert!(!petgraph::algo::is_cyclic_directed(&graph), "The graph contains a cycle!"); + + graph + } +} + +pub fn mermaid_str(graph: &Graph) -> String { + let mut mermaid_str = Vec::new(); + // let raw_start_str = "flowchart TB\n"; + let raw_start_str = r#" +%%{ +init : { +'theme': 'base', +'themeVariables': { + 'primaryColor': '#1a1b26', + 'primaryTextColor': '#d5daf0', + 'primaryBorderColor': '#4c4c4c', + 'lineColor': '#414868', + 'secondaryColor': '#24283b', + 'tertiaryColor': '#24283b' +}, +"flowchart" : { + "defaultRenderer": "elk" +} +} +}%% + +flowchart TB +"#; + mermaid_str.push(raw_start_str.to_string()); + + let nodes_str = graph.node_indices() + .map(|idx| { + let node_str = arena_mermaid_node(graph, "\t", idx, true, true, None); + node_str + + }) + .collect::>() + .join("\n"); + + let edges_str = graph.edge_indices() + .enumerate() + .map(|(i, edge)| { + let (from, to) = graph.edge_endpoints(edge).unwrap(); + let edge_label = format!("{}", graph[edge]); + if edge_label == "" { + // don't do a label + format!( + " {} --> {}", + from.index(), + to.index(), + ) + } else { + // do an edge label + format!( + " {} -->|\"{}\"| {}", + from.index(), + edge_label, + to.index(), + ) + } + }) + .collect::>() + .join("\n"); + + mermaid_str.push(nodes_str); + mermaid_str.push(edges_str); + + // Make an invisible node that holds all our edge information for coloring later on frontend + let data_str = graph.edge_indices() + .enumerate() + .map(|(i, edge)| { + let (from, to) = graph.edge_endpoints(edge).unwrap(); + format!( + "LS-{}_LE-{}_{}", + from.index(), + to.index(), + &graph[edge] + ) + }) + .collect::>() + .join(";"); + + let invis_data = format!(" {}(\"{}\"):::INVIS\n classDef INVIS display:none", graph.node_count(), data_str); + mermaid_str.push(invis_data); + mermaid_str.join("\n") +} + +pub fn arena_mermaid_node( + graph: &Graph, + indent: &str, + idx: NodeIdx, + style: bool, + loc: bool, + class: Option<&str>, +) -> String { + + + let node = &graph[idx]; + let mut node_str = match node { + ArenaNode::ARENA(arena_idx) => { + format!("{indent}{}{{{{\"{}\"}}}}", idx.index(), arena_idx) + }, + ArenaNode::ELEM(label) => { + format!("{indent}{}(\"{}\")", idx.index(), label.replace("\"", "'")) + }, + ArenaNode::CVAR(label) => { + format!("{indent}{}(\"{}\")", idx.index(), label.replace("\"", "'")) + } + }; + + if style { + node_str.push_str(&format!( + "\n{indent}style {} fill:{}", + idx.index(), + node.dot_str_color() + )); + } + + if let Some(class) = class { + node_str.push_str(&format!( + "\n{indent}class {} {class}", + idx.index(), + )); + } + + node_str +} + +fn calculate_hash(t: &T) -> u64 { + let mut s = DefaultHasher::new(); + t.hash(&mut s); + s.finish() +} impl GraphBackend for Analyzer {} @@ -716,22 +1313,6 @@ pub fn mermaid_node( )); } - // if loc { - // match g.node(node) { - // Node::ContextVar(..) => { - // match graph::nodes::ContextVarNode::from(node).loc(g).unwrap() { - // solang_parser::pt::Loc::File(f, s, e) => { - // node_str.push_str(&format!( - // "\n{indent}class {} loc_{f}_{s}_{e}", - // petgraph::graph::GraphIndex::index(&node) - // )); - // } - // _ => {} - // } - // }, - // _ => {} - // } - // } if loc { let mut current_node = node; match g.node(current_node) { @@ -786,8 +1367,6 @@ fn gather_context_info( ) { loop { - // warn!("in loop, {:?}", ctx_node); - let mut found_continue = false; let mut current_loc = ctx_node.underlying(g).unwrap().loc; for edge in g.graph().edges_directed(ctx_node.into(), Direction::Outgoing) { diff --git a/crates/pyrometer/src/lib.rs b/crates/pyrometer/src/lib.rs index bedfe8d6..be40bd1c 100644 --- a/crates/pyrometer/src/lib.rs +++ b/crates/pyrometer/src/lib.rs @@ -1,6 +1,6 @@ mod analyzer; mod analyzer_backend; mod builtin_fns; -mod graph_backend; +pub mod graph_backend; pub use analyzer::*; From d82e21a6a3c0b530917d747830e42a241c93b0b9 Mon Sep 17 00:00:00 2001 From: plotchy Date: Tue, 18 Jun 2024 16:41:36 -0400 Subject: [PATCH 4/6] refactor: preserve default add_node and add_edge fns --- crates/pyrometer/src/graph_backend.rs | 73 +++++++++++------- crates/shared/src/graph_like.rs | 103 +++++++++----------------- 2 files changed, 81 insertions(+), 95 deletions(-) diff --git a/crates/pyrometer/src/graph_backend.rs b/crates/pyrometer/src/graph_backend.rs index a5b1d894..d240c8b1 100644 --- a/crates/pyrometer/src/graph_backend.rs +++ b/crates/pyrometer/src/graph_backend.rs @@ -1,18 +1,18 @@ use crate::Analyzer; use std::{collections::HashMap, fmt::Display}; -use graph::{elem::Elem, nodes::ContextVarNode, range_string::ToRangeString, SolcRange, TOKYO_NIGHT_COLORS}; +use graph::{elem::Elem, nodes::ContextVarNode, TOKYO_NIGHT_COLORS}; use graph::elem::RangeElem; use graph::nodes::Concrete; use reqwest::Client; use serde::{Deserialize, Serialize}; -use shared::{RangeArena, USE_DEBUG_SITE}; +use shared::{post_to_site, RangeArena, USE_DEBUG_SITE}; use tokio::runtime::Runtime; use std::cell::RefCell; use std::collections::hash_map::DefaultHasher; use std::hash::Hash; use std::hash::Hasher; use std::rc::Rc; -use tracing::{trace, debug, error, warn}; +use tracing::{trace, error, warn}; use graph::{ as_dot_str, nodes::ContextNode, AnalyzerBackend, AsDotStr, ContextEdge, Edge, GraphBackend, Node, @@ -105,14 +105,37 @@ impl GraphLike for Analyzer { idx }; - // if unsafe { USE_DEBUG_SITE } { - // let elems = Elems::from(self.range_arena()); - // let elems_graph = elems.to_graph(self); - // let elems_graph_mermaid_str = mermaid_str(&elems_graph); - // post_to_site_arena(elems_graph_mermaid_str); - // } res_idx } + + fn add_node(&mut self, node: impl Into) -> NodeIdx + where + Self: std::marker::Sized, + Self: GraphLike, + { + let res = self.graph_mut().add_node(node.into()); + if unsafe { USE_DEBUG_SITE } { + post_to_site(self); + } + res + } + + fn add_edge( + &mut self, + from_node: impl Into, + to_node: impl Into, + edge: impl Into, + ) where + Self: std::marker::Sized, + Self: GraphLike, + { + self.graph_mut() + .add_edge(from_node.into(), to_node.into(), edge.into()); + + if unsafe { USE_DEBUG_SITE } { + post_to_site(self); + } + } } pub fn post_to_site_arena(arena_str: String) { @@ -122,7 +145,7 @@ pub fn post_to_site_arena(arena_str: String) { }); } -pub async fn post_to_site_arena_async(arena_str: String) { +async fn post_to_site_arena_async(arena_str: String) { let client = Client::new(); let graph_msg = ArenaMessage { arena: arena_str.to_string(), @@ -175,7 +198,6 @@ impl ArenaNode { ArenaNode::ARENA(_) => TOKYO_NIGHT_COLORS.get("red1").unwrap(), ArenaNode::CVAR(_) => TOKYO_NIGHT_COLORS.get("deeporange").unwrap(), ArenaNode::ELEM(_) => TOKYO_NIGHT_COLORS.get("blue0").unwrap(), - _ => TOKYO_NIGHT_COLORS.get("default").unwrap(), }; c.to_string() } @@ -280,7 +302,7 @@ impl TryFrom<&RangeArena>> for Elems { } // Add any missing entries to inner - for (elem, &idx) in missing_entries { + for (_elem, &idx) in missing_entries { if let Some(range_elem) = arena.ranges.get(idx) { if let Ok(borrowed_elem) = range_elem.try_borrow() { inner.push((idx, borrowed_elem.clone())); @@ -323,7 +345,7 @@ impl Elems { let mut dependency_map: HashMap> = HashMap::new(); // FIRST PASS: create nodes for each arena index - self.inner.iter().for_each(|(arena_idx, elem)| { + self.inner.iter().for_each(|(arena_idx, _elem)| { // add an arena node to the graph for the index let arena_node_idx = graph.add_node(ArenaNode::ARENA(*arena_idx)); arena_idx_to_node_idx.insert(arena_idx, arena_node_idx); @@ -336,7 +358,7 @@ impl Elems { // add a node for what the arena index has underlying it, and maybe edges for that elem let underlying_node_idx = match elem { - Elem::Reference(reference) => { + Elem::Reference(_reference) => { let node_str = elem.arena_graph_node_label(); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); @@ -358,12 +380,12 @@ impl Elems { node_idx }, - Elem::ConcreteDyn(range_dyn) => { + Elem::ConcreteDyn(_range_dyn) => { let node_str = elem.arena_graph_node_label(); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); node_idx }, - Elem::Concrete(range_concrete) => { + Elem::Concrete(_range_concrete) => { let node_str = elem.arena_graph_node_label(); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); node_idx @@ -375,7 +397,7 @@ impl Elems { // Unbox and check the lhs and rhs to see if they are arena indices let lhs_arena = match *range_expr.lhs.clone() { Elem::Arena(lhs) => Some(lhs), - Elem::Reference(lhs) => { + Elem::Reference(_lhs) => { // println!("LHS is a reference: {}", range_expr.lhs); // attempt to add in the ContextVar node that the elem is referencing let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); @@ -401,7 +423,7 @@ impl Elems { // println!("RHS is an arena index: {}", range_expr.rhs); Some(rhs) }, - Elem::Reference(rhs) => { + Elem::Reference(_rhs) => { // println!("RHS is a reference: {}", range_expr.rhs); // attempt to add in the ContextVar node that the elem is referencing let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); @@ -421,7 +443,6 @@ impl Elems { None }, _ => { - println!("RHS is not an arena index: {}", range_expr.rhs); None }, }; @@ -558,7 +579,7 @@ flowchart TB let edges_str = graph.edge_indices() .enumerate() - .map(|(i, edge)| { + .map(|(_i, edge)| { let (from, to) = graph.edge_endpoints(edge).unwrap(); let edge_label = format!("{}", graph[edge]); if edge_label == "" { @@ -587,7 +608,7 @@ flowchart TB // Make an invisible node that holds all our edge information for coloring later on frontend let data_str = graph.edge_indices() .enumerate() - .map(|(i, edge)| { + .map(|(_i, edge)| { let (from, to) = graph.edge_endpoints(edge).unwrap(); format!( "LS-{}_LE-{}_{}", @@ -609,7 +630,7 @@ pub fn arena_mermaid_node( indent: &str, idx: NodeIdx, style: bool, - loc: bool, + _loc: bool, class: Option<&str>, ) -> String { @@ -645,7 +666,7 @@ pub fn arena_mermaid_node( node_str } -fn calculate_hash(t: &T) -> u64 { +fn _calculate_hash(t: &T) -> u64 { let mut s = DefaultHasher::new(); t.hash(&mut s); s.finish() @@ -1314,7 +1335,7 @@ pub fn mermaid_node( } if loc { - let mut current_node = node; + let current_node = node; match g.node(current_node) { Node::ContextVar(..) => { // highlight self @@ -1368,11 +1389,11 @@ fn gather_context_info( loop { let mut found_continue = false; - let mut current_loc = ctx_node.underlying(g).unwrap().loc; + let current_loc = ctx_node.underlying(g).unwrap().loc; for edge in g.graph().edges_directed(ctx_node.into(), Direction::Outgoing) { if let Edge::Context(ContextEdge::Continue(true_or_false)) = edge.weight() { let target_node = edge.target(); - if let Node::Context(ctx) = g.node(target_node) { + if let Node::Context(_ctx) = g.node(target_node) { // error!("found continue pointing to node"); ctx_node = target_node.into(); found_continue = true; diff --git a/crates/shared/src/graph_like.rs b/crates/shared/src/graph_like.rs index 6b717ed5..89626b58 100644 --- a/crates/shared/src/graph_like.rs +++ b/crates/shared/src/graph_like.rs @@ -40,7 +40,7 @@ pub trait GraphLike { /// Get a reference to the graph fn graph(&self) -> &Graph; /// Add a node to the graph - fn _add_node(&mut self, node: impl Into) -> NodeIdx { + fn add_node(&mut self, node: impl Into) -> NodeIdx { let res = self.graph_mut().add_node(node.into()); res } @@ -57,7 +57,7 @@ pub trait GraphLike { .expect("Index not in graph") } /// Add an edge to the graph - fn _add_edge( + fn add_edge( &mut self, from_node: impl Into, to_node: impl Into, @@ -162,73 +162,6 @@ pub trait GraphDot: GraphLike { .expect("failed to execute process"); } - fn add_node(&mut self, node: impl Into) -> NodeIdx - where - Self: std::marker::Sized, - Self: AnalyzerLike, - { - let res = self.graph_mut().add_node(node.into()); - if unsafe { USE_DEBUG_SITE } { - self.post_to_site(); - } - res - } - - fn add_edge( - &mut self, - from_node: impl Into, - to_node: impl Into, - edge: impl Into, - ) where - Self: std::marker::Sized, - Self: AnalyzerLike, - { - self.graph_mut() - .add_edge(from_node.into(), to_node.into(), edge.into()); - - if unsafe { USE_DEBUG_SITE } { - self.post_to_site(); - } - } - - fn post_to_site(&self) - where - Self: std::marker::Sized, - Self: AnalyzerLike, - { - let rt = Runtime::new().unwrap(); - rt.block_on(async { - self.post_to_site_async().await; - }); - } - - async fn post_to_site_async(&self) - where - Self: std::marker::Sized, - Self: AnalyzerLike, - { - let client = Client::new(); - let graph_msg = GraphMessage { - graph: self.mermaid_str(), - timestamp: SystemTime::now() - .duration_since(UNIX_EPOCH) - .expect("Time went backwards") - .as_secs(), - }; - - let res = client - .post("http://127.0.0.1:8545/addgraph") - .json(&graph_msg) - .send() - .await - .expect("Failed to send request"); - - if res.status().is_success() { - trace!("Successfully posted dot to site"); - } else { - error!("Failed to post graph to site: {:?}", res.status()); - } - } /// Creates a subgraph for visually identifying contexts and subcontexts fn cluster_str( @@ -268,3 +201,35 @@ struct GraphMessage { graph: String, timestamp: u64, } + + +pub fn post_to_site(graph: &(impl GraphDot + AnalyzerLike)) { + let rt = Runtime::new().unwrap(); + rt.block_on(async { + post_to_site_async(graph).await; + }); +} + +async fn post_to_site_async(graph: &(impl GraphDot + AnalyzerLike)) { + let client = Client::new(); + let graph_msg = GraphMessage { + graph: graph.mermaid_str(), + timestamp: SystemTime::now() + .duration_since(UNIX_EPOCH) + .expect("Time went backwards") + .as_secs(), + }; + + let res = client + .post("http://127.0.0.1:8545/addgraph") + .json(&graph_msg) + .send() + .await + .expect("Failed to send request"); + + if res.status().is_success() { + trace!("Successfully posted dot to site"); + } else { + error!("Failed to post graph to site: {:?}", res.status()); + } +} \ No newline at end of file From 79fc49f554379497acc4b2d3f60d2f7d4443a2ee Mon Sep 17 00:00:00 2001 From: plotchy Date: Tue, 25 Jun 2024 22:51:28 -0400 Subject: [PATCH 5/6] fix: merge conflict resolutions --- Cargo.lock | 10 +- crates/cli/src/main.rs | 55 ++++---- crates/graph/src/graph_elements.rs | 48 +++++-- crates/pyrometer/src/analyzer_backend.rs | 3 +- crates/pyrometer/src/graph_backend.rs | 158 +++++------------------ crates/shared/src/graph_like.rs | 30 ++--- 6 files changed, 119 insertions(+), 185 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 07f1ae80..431407f4 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -112,7 +112,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "44055e597c674aef7cb903b2b9f6e4cba1277ed0d2d61dae7cd52d7ffa81f8e2" dependencies = [ "unicode-width", - "yansi", + "yansi 1.0.1", ] [[package]] @@ -1770,7 +1770,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "af7cee1a6c8a5b9208b3cb1061f10c0cb689087b3d8ce85fb9d2dd7a29b6ba66" dependencies = [ "diff", - "yansi", + "yansi 0.5.1", ] [[package]] @@ -3195,6 +3195,12 @@ dependencies = [ "tap", ] +[[package]] +name = "yansi" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09041cd90cf85f7f8b2df60c646f853b7f535ce68f85244eb6731cf89fa498ec" + [[package]] name = "yansi" version = "1.0.1" diff --git a/crates/cli/src/main.rs b/crates/cli/src/main.rs index 0df1f60f..e6571151 100644 --- a/crates/cli/src/main.rs +++ b/crates/cli/src/main.rs @@ -6,7 +6,7 @@ use graph::{ }; use pyrometer::{Analyzer, Root, SourcePath}; use reqwest::Client; -use shared::Search; +use shared::{post_to_site, Search}; use shared::{GraphDot, USE_DEBUG_SITE}; use ariadne::sources; @@ -235,27 +235,7 @@ fn main() { }, }; - if args.debug_site { - unsafe { - USE_DEBUG_SITE = true; - } - - let rt = Runtime::new().unwrap(); - rt.block_on(async { - let client = Client::new(); - let res = client - .post("http://127.0.0.1:8545/clear") - .send() - .await - .expect("Failed to send request"); - if res.status().is_success() { - trace!("Successfully cleared history of site"); - } else { - error!("Failed to clear history of site: {:?}", res.status()); - } - }); - } let mut analyzer = Analyzer { max_depth: args.max_stack_depth, @@ -288,6 +268,30 @@ fn main() { let mut arena_base = Default::default(); let arena = &mut arena_base; + + if args.debug_site { + unsafe { + USE_DEBUG_SITE = true; + } + + let rt = Runtime::new().unwrap(); + rt.block_on(async { + let client = Client::new(); + let res = client + .post("http://127.0.0.1:8545/clear") + .send() + .await + .expect("Failed to send request"); + + if res.status().is_success() { + trace!("Successfully cleared history of site"); + } else { + error!("Failed to clear history of site: {:?}", res.status()); + } + }); + post_to_site(&analyzer, arena); + } + let t0 = std::time::Instant::now(); let maybe_entry = analyzer.parse(arena, &sol, ¤t_path, true); let t_end = t0.elapsed(); @@ -297,13 +301,13 @@ fn main() { // println!("Arena: {:#?}", analyzer.range_arena); if unsafe { USE_DEBUG_SITE } { - use shared::GraphLike; use pyrometer::graph_backend::mermaid_str; use pyrometer::graph_backend::post_to_site_arena; use pyrometer::graph_backend::Elems; - match Elems::try_from(analyzer.range_arena()) { + let elems = Elems::try_from(&*arena); + match elems { Ok(elems) => { - let elems_graph = elems.to_graph(&analyzer); + let elems_graph = elems.to_graph(&analyzer, arena); let elems_graph_mermaid_str = mermaid_str(&elems_graph); post_to_site_arena(elems_graph_mermaid_str); } @@ -311,6 +315,9 @@ fn main() { eprintln!("Can't post arena, error creating Elems: {:?}", e); } }; + + // post the graph to the site + post_to_site(&analyzer, arena); } if args.stats { diff --git a/crates/graph/src/graph_elements.rs b/crates/graph/src/graph_elements.rs index 92087dae..494c7622 100644 --- a/crates/graph/src/graph_elements.rs +++ b/crates/graph/src/graph_elements.rs @@ -402,9 +402,7 @@ pub enum ContextEdge { } #[derive(Default)] -pub(crate) struct DummyGraph { - pub range_arena: RangeArena>, -} +pub(crate) struct DummyGraph {} impl GraphLike for DummyGraph { type Node = Node; @@ -417,12 +415,44 @@ impl GraphLike for DummyGraph { fn graph(&self) -> &Graph { panic!("Dummy Graph") } - fn range_arena(&self) -> &RangeArena> { - &self.range_arena - } - fn range_arena_mut(&mut self) -> &mut RangeArena> { - &mut self.range_arena - } } impl GraphBackend for DummyGraph {} +impl GraphDot for DummyGraph { + type T = Elem; + + fn dot_str(&self, _arena: &mut RangeArena<::T>) -> String { + // Provide a basic implementation or a placeholder + "digraph DummyGraph {}".to_string() + } + + fn cluster_str( + &self, + _arena: &mut RangeArena, + _node: NodeIdx, + _cluster_num: &mut usize, + _is_killed: bool, + _handled_nodes: std::sync::Arc>>, + _handled_edges: std::sync::Arc>>>, + _depth: usize, + _as_mermaid: bool, + ) -> Option + where + Self: std::marker::Sized { + todo!() + } + + fn dot_str_no_tmps(&self, _arena: &mut RangeArena) -> String + where + Self: std::marker::Sized, + { + todo!() + } + + fn mermaid_str(&self, _arena: &mut RangeArena) -> String + where + Self: std::marker::Sized, + { + todo!() + } +} \ No newline at end of file diff --git a/crates/pyrometer/src/analyzer_backend.rs b/crates/pyrometer/src/analyzer_backend.rs index 1c3fbe2c..d9171ebb 100644 --- a/crates/pyrometer/src/analyzer_backend.rs +++ b/crates/pyrometer/src/analyzer_backend.rs @@ -1,5 +1,4 @@ use crate::Analyzer; -use shared::GraphDot; use graph::{ elem::Elem, nodes::{ @@ -18,7 +17,7 @@ use solang_parser::{ pt::{Expression, Loc}, }; -use std::{collections::BTreeMap, path::PathBuf}; +use std::collections::BTreeMap; impl AnalyzerBackend for Analyzer { fn add_concrete_var( diff --git a/crates/pyrometer/src/graph_backend.rs b/crates/pyrometer/src/graph_backend.rs index d240c8b1..2c6d387e 100644 --- a/crates/pyrometer/src/graph_backend.rs +++ b/crates/pyrometer/src/graph_backend.rs @@ -5,13 +5,11 @@ use graph::elem::RangeElem; use graph::nodes::Concrete; use reqwest::Client; use serde::{Deserialize, Serialize}; -use shared::{post_to_site, RangeArena, USE_DEBUG_SITE}; +use shared::RangeArena; use tokio::runtime::Runtime; -use std::cell::RefCell; use std::collections::hash_map::DefaultHasher; use std::hash::Hash; use std::hash::Hasher; -use std::rc::Rc; use tracing::{trace, error, warn}; use graph::{ as_dot_str, nodes::ContextNode, AnalyzerBackend, AsDotStr, ContextEdge, Edge, GraphBackend, @@ -38,85 +36,12 @@ impl GraphLike for Analyzer { &self.graph } - fn range_arena(&self) -> &RangeArena> { - &self.range_arena - } - - fn range_arena_mut(&mut self) -> &mut RangeArena> { - &mut self.range_arena - } - - fn range_arena_idx(&self, elem: &Self::RangeElem) -> Option { - if let Elem::Arena(idx) = elem { - Some(*idx) - } else { - self.range_arena().map.get(elem).copied() - } - } - - fn range_arena_idx_or_upsert(&mut self, elem: Self::RangeElem) -> usize { - // tracing::trace!("arenaizing: {}", elem); - if let Elem::Arena(idx) = elem { - return idx; - } - - let res_idx = if let Some(idx) = self.range_arena_idx(&elem) { - let existing = &self.range_arena().ranges[idx]; - let Ok(existing) = existing.try_borrow_mut() else { - return idx; - }; - let (min_cached, max_cached) = existing.is_min_max_cached(self); - let mut existing_count = 0; - if min_cached { - existing_count += 1; - } - if max_cached { - existing_count += 1; - } - if existing.is_flatten_cached(self) { - existing_count += 1; - } - - let (min_cached, max_cached) = elem.is_min_max_cached(self); - let mut new_count = 0; - if min_cached { - new_count += 1; - } - if max_cached { - new_count += 1; - } - if elem.is_flatten_cached(self) { - new_count += 1; - } - - drop(existing); - - if new_count >= existing_count { - self.range_arena_mut().ranges[idx] = Rc::new(RefCell::new(elem)); - } - - idx - } else { - let idx = self.range_arena().ranges.len(); - self.range_arena_mut() - .ranges - .push(Rc::new(RefCell::new(elem.clone()))); - self.range_arena_mut().map.insert(elem, idx); - idx - }; - - res_idx - } - fn add_node(&mut self, node: impl Into) -> NodeIdx where Self: std::marker::Sized, Self: GraphLike, { let res = self.graph_mut().add_node(node.into()); - if unsafe { USE_DEBUG_SITE } { - post_to_site(self); - } res } @@ -131,10 +56,6 @@ impl GraphLike for Analyzer { { self.graph_mut() .add_edge(from_node.into(), to_node.into(), edge.into()); - - if unsafe { USE_DEBUG_SITE } { - post_to_site(self); - } } } @@ -260,25 +181,14 @@ impl TryFrom<&RangeArena>> for Elems { // Collect the elements and their indices first from ranges // Collect the elements and their indices first from ranges let mut inner = Vec::new(); - for rc in &arena.ranges { - // Attempt to borrow the RefCell - match rc.try_borrow() { - Ok(borrowed) => { - let elem = borrowed.clone(); - // Get the map value - if let Some(map_value) = arena.map.get(&elem).copied() { - // println!("Adding idx {} to elems {}", map_value, elem); - inner.push((map_value, elem)); - } else { - // println!("NONE REF elem: {:?}", elem); - return Err(ElemsError::MissingMap(format!("elem {:?}", elem))); - } - }, - Err(e) => { - // Print an error message if borrowing fails - // eprintln!("Failed to borrow RefCell: {:?}", e); - return Err(ElemsError::BorrowError(format!("error {:?}", e))); - } + for elem in &arena.ranges { + // Get the map value + if let Some(map_value) = arena.map.get(&elem).copied() { + // println!("Adding idx {} to elems {}", map_value, elem); + inner.push((map_value, elem.clone())); + } else { + // println!("NONE REF elem: {:?}", elem); + return Err(ElemsError::MissingMap(format!("elem {:?}", elem))); } } @@ -304,9 +214,7 @@ impl TryFrom<&RangeArena>> for Elems { // Add any missing entries to inner for (_elem, &idx) in missing_entries { if let Some(range_elem) = arena.ranges.get(idx) { - if let Ok(borrowed_elem) = range_elem.try_borrow() { - inner.push((idx, borrowed_elem.clone())); - } + inner.push((idx, range_elem.clone())); } } @@ -339,7 +247,7 @@ impl Elems { /// Third pass: /// - for each ContextVar node, create edges to the arena indices that it depends on /// - pub fn to_graph(&self, graph_backend: &impl GraphBackend) -> Graph { + pub fn to_graph(&self, graph_backend: &impl GraphBackend, arena: &mut RangeArena>) -> Graph { let mut graph = Graph::default(); let mut arena_idx_to_node_idx = HashMap::new(); let mut dependency_map: HashMap> = HashMap::new(); @@ -363,14 +271,14 @@ impl Elems { let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); // attempt to add in the ContextVar node that the elem is referencing - let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); + let context_var_nodes = elem.dependent_on(graph_backend, arena).into_iter().collect::>(); context_var_nodes.into_iter().for_each(|dep_elem| { let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { // don't make a new ContextVar node, just use the existing one existing_node_idx } else { // make a new ContextVar Node for the Arena graph - let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend)))); + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend, arena)))); dependency_map.insert(dep_elem.clone(), new_node_idx); new_node_idx }; @@ -400,14 +308,14 @@ impl Elems { Elem::Reference(_lhs) => { // println!("LHS is a reference: {}", range_expr.lhs); // attempt to add in the ContextVar node that the elem is referencing - let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); + let context_var_nodes = elem.dependent_on(graph_backend, arena).into_iter().collect::>(); context_var_nodes.iter().for_each(|dep_elem| { let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { // don't make a new ContextVar node, just use the existing one existing_node_idx } else { // make a new ContextVar Node for the Arena graph - let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend)))); + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend, arena)))); dependency_map.insert(dep_elem.clone(), new_node_idx); new_node_idx }; @@ -426,14 +334,14 @@ impl Elems { Elem::Reference(_rhs) => { // println!("RHS is a reference: {}", range_expr.rhs); // attempt to add in the ContextVar node that the elem is referencing - let context_var_nodes = elem.dependent_on(graph_backend).into_iter().collect::>(); + let context_var_nodes = elem.dependent_on(graph_backend, arena).into_iter().collect::>(); context_var_nodes.iter().for_each(|dep_elem| { let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { // don't make a new ContextVar node, just use the existing one existing_node_idx } else { // make a new ContextVar Node for the Arena graph - let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend)))); + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend, arena)))); dependency_map.insert(dep_elem.clone(), new_node_idx); new_node_idx }; @@ -882,7 +790,6 @@ impl GraphDot for Analyzer { )) } else { { - if handled_nodes.lock().unwrap().contains(child) { if handled_nodes.lock().unwrap().contains(child) { return None; } else { @@ -914,7 +821,6 @@ impl GraphDot for Analyzer { format!("{indent}{from:} -->|\"{edge_str}\"| {to:}\n{indent}class {to} linkSource{edge_idx}\n{indent}class {from} linkTarget{edge_idx}") } else { format!("{indent}{from:} -> {to:} [label = \"{edge_str}\"]",) - format!("{indent}{from:} -> {to:} [label = \"{edge_str}\"]",) } }) .collect::>() @@ -1262,45 +1168,42 @@ impl GraphLike for G<'_> { fn graph(&self) -> &Graph { self.graph } - fn range_arena(&self) -> &RangeArena> { - panic!("Should not call this") - } - fn range_arena_mut(&mut self) -> &mut RangeArena> { - panic!("Should not call this") - } } impl GraphDot for G<'_> { + type T = Elem; + fn cluster_str( &self, - node: NodeIdx, - cluster_num: &mut usize, - is_killed: bool, - handled_nodes: Arc>>, - handled_edges: Arc>>>, - depth: usize, - as_mermaid: bool, + _arena: &mut RangeArena, + _node: NodeIdx, + _cluster_num: &mut usize, + _is_killed: bool, + _handled_nodes: Arc>>, + _handled_edges: Arc>>>, + _depth: usize, + _as_mermaid: bool, ) -> Option where Self: std::marker::Sized { todo!() } - fn dot_str(&self) -> String + fn dot_str(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, Self: shared::AnalyzerLike { todo!() } - fn dot_str_no_tmps(&self) -> String + fn dot_str_no_tmps(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, Self: GraphLike + shared::AnalyzerLike { todo!() } - fn mermaid_str(&self) -> String + fn mermaid_str(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, Self: shared::AnalyzerLike { @@ -1317,7 +1220,6 @@ pub fn mermaid_node( node: NodeIdx, style: bool, loc: bool, - loc: bool, class: Option<&str>, ) -> String { let mut node_str = format!( diff --git a/crates/shared/src/graph_like.rs b/crates/shared/src/graph_like.rs index 89626b58..2330042e 100644 --- a/crates/shared/src/graph_like.rs +++ b/crates/shared/src/graph_like.rs @@ -66,22 +66,6 @@ pub trait GraphLike { self.graph_mut() .add_edge(from_node.into(), to_node.into(), edge.into()); } - - fn range_arena(&self) -> &RangeArena; - fn range_arena_mut(&mut self) -> &mut RangeArena; - fn try_take_range_arena(&mut self) -> Option> { - let arena = self.range_arena_mut(); - if !arena.ranges.is_empty() { - Some(std::mem::take(arena)) - } else { - None - } - } - - fn take_range_arena(&mut self) -> RangeArena { - let arena = self.range_arena_mut(); - std::mem::take(arena) - } } /// A trait that constructs dot-like visualization strings (either mermaid or graphviz) @@ -203,17 +187,23 @@ struct GraphMessage { } -pub fn post_to_site(graph: &(impl GraphDot + AnalyzerLike)) { +pub fn post_to_site(graph: &G, arena: &mut RangeArena) +where + G: GraphDot + AnalyzerLike, +{ let rt = Runtime::new().unwrap(); rt.block_on(async { - post_to_site_async(graph).await; + post_to_site_async(graph, arena).await; }); } -async fn post_to_site_async(graph: &(impl GraphDot + AnalyzerLike)) { +async fn post_to_site_async(graph: &G, arena: &mut RangeArena) +where + G: GraphDot + AnalyzerLike, +{ let client = Client::new(); let graph_msg = GraphMessage { - graph: graph.mermaid_str(), + graph: graph.mermaid_str(arena), timestamp: SystemTime::now() .duration_since(UNIX_EPOCH) .expect("Time went backwards") From ed0b78b94de9aceb981029fb7400d7b955b24cc9 Mon Sep 17 00:00:00 2001 From: plotchy Date: Tue, 25 Jun 2024 22:58:00 -0400 Subject: [PATCH 6/6] fix: lint --- crates/cli/src/main.rs | 4 +- crates/graph/src/graph_elements.rs | 22 +- .../graph/src/range/elem/elem_enum/impls.rs | 6 +- crates/pyrometer/src/analyzer.rs | 14 +- crates/pyrometer/src/analyzer_backend.rs | 2 +- crates/pyrometer/src/graph_backend.rs | 224 ++++++++++-------- crates/shared/src/graph_like.rs | 4 +- 7 files changed, 154 insertions(+), 122 deletions(-) diff --git a/crates/cli/src/main.rs b/crates/cli/src/main.rs index e6571151..b4761afe 100644 --- a/crates/cli/src/main.rs +++ b/crates/cli/src/main.rs @@ -235,8 +235,6 @@ fn main() { }, }; - - let mut analyzer = Analyzer { max_depth: args.max_stack_depth, root: Root::RemappingsDirectory(env::current_dir().unwrap()), @@ -273,7 +271,7 @@ fn main() { unsafe { USE_DEBUG_SITE = true; } - + let rt = Runtime::new().unwrap(); rt.block_on(async { let client = Client::new(); diff --git a/crates/graph/src/graph_elements.rs b/crates/graph/src/graph_elements.rs index 494c7622..5b255e99 100644 --- a/crates/graph/src/graph_elements.rs +++ b/crates/graph/src/graph_elements.rs @@ -9,11 +9,10 @@ use solang_parser::pt::{Identifier, Loc}; use std::collections::HashMap; - pub trait GraphBackend: - GraphLike> - + GraphDot -{} + GraphLike> + GraphDot +{ +} pub trait AnalyzerBackend: AnalyzerLike< Builtin = Builtin, @@ -425,7 +424,7 @@ impl GraphDot for DummyGraph { // Provide a basic implementation or a placeholder "digraph DummyGraph {}".to_string() } - + fn cluster_str( &self, _arena: &mut RangeArena, @@ -433,26 +432,29 @@ impl GraphDot for DummyGraph { _cluster_num: &mut usize, _is_killed: bool, _handled_nodes: std::sync::Arc>>, - _handled_edges: std::sync::Arc>>>, + _handled_edges: std::sync::Arc< + std::sync::Mutex>>, + >, _depth: usize, _as_mermaid: bool, ) -> Option where - Self: std::marker::Sized { + Self: std::marker::Sized, + { todo!() } - + fn dot_str_no_tmps(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, { todo!() } - + fn mermaid_str(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, { todo!() } -} \ No newline at end of file +} diff --git a/crates/graph/src/range/elem/elem_enum/impls.rs b/crates/graph/src/range/elem/elem_enum/impls.rs index a592d16a..8ef2a405 100644 --- a/crates/graph/src/range/elem/elem_enum/impls.rs +++ b/crates/graph/src/range/elem/elem_enum/impls.rs @@ -75,13 +75,13 @@ impl Elem { let rhs_str = range_expr.rhs.arena_graph_node_label(); let op = range_expr.op.clone(); format!("expr-{}", &self) - }, + } Elem::Arena(arena_idx) => { format!("Arena({})", arena_idx) - }, + } Elem::Null => { todo!() - }, + } } } } diff --git a/crates/pyrometer/src/analyzer.rs b/crates/pyrometer/src/analyzer.rs index 2bc1e924..943f6966 100644 --- a/crates/pyrometer/src/analyzer.rs +++ b/crates/pyrometer/src/analyzer.rs @@ -1,12 +1,12 @@ use crate::builtin_fns; +use analyzers::LocStrSpan; use graph::elem::Elem; +use graph::{nodes::*, ContextEdge, Edge, Node, VarType}; use reqwest::Client; use serde::{Deserialize, Serialize}; -use shared::{RangeArena, USE_DEBUG_SITE}; use shared::GraphDot; -use analyzers::LocStrSpan; -use graph::{nodes::*, ContextEdge, Edge, Node, VarType}; use shared::{AnalyzerLike, GraphLike, JoinStats, NodeIdx, Search}; +use shared::{RangeArena, USE_DEBUG_SITE}; use solc_expressions::{ExprErr, FnCallBuilder, IntoExprErr, StatementParser}; use tokio::runtime::Runtime; use tracing::{debug, error, info, trace, warn}; @@ -1403,7 +1403,7 @@ impl Analyzer { ty_node } - fn post_source_to_site(file_no: usize, path: &PathBuf, source: &str) + fn post_source_to_site(file_no: usize, path: &PathBuf, source: &str) where Self: std::marker::Sized, Self: AnalyzerLike, @@ -1414,7 +1414,8 @@ impl Analyzer { }); } - async fn post_source_to_site_async(file_no: usize, path: &PathBuf, source: &str) where + async fn post_source_to_site_async(file_no: usize, path: &PathBuf, source: &str) + where Self: std::marker::Sized, Self: AnalyzerLike, { @@ -1436,7 +1437,6 @@ impl Analyzer { } else { error!("Failed to post source to site: {:?}", res.status()); } - } } @@ -1504,4 +1504,4 @@ struct SourceMessage { file_number: usize, path: PathBuf, source: String, -} \ No newline at end of file +} diff --git a/crates/pyrometer/src/analyzer_backend.rs b/crates/pyrometer/src/analyzer_backend.rs index d9171ebb..b1c5b627 100644 --- a/crates/pyrometer/src/analyzer_backend.rs +++ b/crates/pyrometer/src/analyzer_backend.rs @@ -275,7 +275,7 @@ impl AnalyzerLike for Analyzer { fn handled_funcs_mut(&mut self) -> &mut Vec { &mut self.handled_funcs } - + fn file_mapping(&self) -> BTreeMap { let mut file_mapping: BTreeMap = BTreeMap::new(); for (source_path, _, o_file_no, _) in self.sources.iter() { diff --git a/crates/pyrometer/src/graph_backend.rs b/crates/pyrometer/src/graph_backend.rs index 2c6d387e..64ad9537 100644 --- a/crates/pyrometer/src/graph_backend.rs +++ b/crates/pyrometer/src/graph_backend.rs @@ -1,21 +1,21 @@ use crate::Analyzer; -use std::{collections::HashMap, fmt::Display}; -use graph::{elem::Elem, nodes::ContextVarNode, TOKYO_NIGHT_COLORS}; use graph::elem::RangeElem; use graph::nodes::Concrete; +use graph::{ + as_dot_str, nodes::ContextNode, AnalyzerBackend, AsDotStr, ContextEdge, Edge, GraphBackend, + Node, +}; +use graph::{elem::Elem, nodes::ContextVarNode, TOKYO_NIGHT_COLORS}; use reqwest::Client; use serde::{Deserialize, Serialize}; use shared::RangeArena; -use tokio::runtime::Runtime; +use shared::{GraphDot, GraphLike, NodeIdx, Search}; use std::collections::hash_map::DefaultHasher; use std::hash::Hash; use std::hash::Hasher; -use tracing::{trace, error, warn}; -use graph::{ - as_dot_str, nodes::ContextNode, AnalyzerBackend, AsDotStr, ContextEdge, Edge, GraphBackend, - Node, -}; -use shared::{GraphDot, GraphLike, NodeIdx, Search}; +use std::{collections::HashMap, fmt::Display}; +use tokio::runtime::Runtime; +use tracing::{error, trace, warn}; use petgraph::{dot::Dot, graph::EdgeIndex, visit::EdgeRef, Directed, Direction, Graph}; use std::convert::TryFrom; @@ -124,7 +124,6 @@ impl ArenaNode { } } - #[derive(Serialize, Deserialize, Debug, PartialEq, Eq, Hash)] pub enum ArenaEdge { LHS, @@ -138,7 +137,6 @@ pub enum ArenaEdge { } impl Display for ArenaEdge { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { // warning: changing these will impact the pyro-debug site when rendering edge styling let variant_name = match self { @@ -172,7 +170,7 @@ impl std::fmt::Display for ElemsError { impl std::error::Error for ElemsError {} pub struct Elems { /// arena_idx, elem - pub inner: Vec<(usize, Elem)> + pub inner: Vec<(usize, Elem)>, } impl TryFrom<&RangeArena>> for Elems { @@ -194,13 +192,16 @@ impl TryFrom<&RangeArena>> for Elems { // Search .map for any entries that werent in .ranges let inner_indices: BTreeSet<_> = inner.iter().map(|(idx, _)| *idx).collect(); - let missing_entries: Vec<_> = arena.map.iter() + let missing_entries: Vec<_> = arena + .map + .iter() .filter(|(_, &v)| !inner_indices.contains(&v)) .collect(); { // Log out missing entries - let missing_entries_str = missing_entries.iter() + let missing_entries_str = missing_entries + .iter() .map(|(idx, elem)| format!("\telem {}: {}", idx, elem)) .collect::>() .join("\n"); @@ -233,24 +234,28 @@ impl TryFrom<&RangeArena>> for Elems { } impl Elems { - /// Convert Elems into a Graph - /// - /// First pass: + /// + /// First pass: /// - create nodes for each arena index /// - this is needed separately, since when making edges some earlier nodes point to later nodes (that havent been made yet) - /// + /// /// Second pass: /// - create an edge between each arena index and the elem that it represents, make an edge between them /// - if elem is a reference, create nodes for the ContextVar that it depends on, make an edge from the elem to the ContextVar - /// + /// /// Third pass: /// - for each ContextVar node, create edges to the arena indices that it depends on - /// - pub fn to_graph(&self, graph_backend: &impl GraphBackend, arena: &mut RangeArena>) -> Graph { + /// + pub fn to_graph( + &self, + graph_backend: &impl GraphBackend, + arena: &mut RangeArena>, + ) -> Graph { let mut graph = Graph::default(); let mut arena_idx_to_node_idx = HashMap::new(); - let mut dependency_map: HashMap> = HashMap::new(); + let mut dependency_map: HashMap> = + HashMap::new(); // FIRST PASS: create nodes for each arena index self.inner.iter().for_each(|(arena_idx, _elem)| { @@ -271,33 +276,40 @@ impl Elems { let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); // attempt to add in the ContextVar node that the elem is referencing - let context_var_nodes = elem.dependent_on(graph_backend, arena).into_iter().collect::>(); + let context_var_nodes = elem + .dependent_on(graph_backend, arena) + .into_iter() + .collect::>(); context_var_nodes.into_iter().for_each(|dep_elem| { - let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { - // don't make a new ContextVar node, just use the existing one - existing_node_idx - } else { - // make a new ContextVar Node for the Arena graph - let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend, arena)))); - dependency_map.insert(dep_elem.clone(), new_node_idx); - new_node_idx - }; + let dep_node_idx = + if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { + // don't make a new ContextVar node, just use the existing one + existing_node_idx + } else { + // make a new ContextVar Node for the Arena graph + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!( + "{}", + dep_elem.as_dot_str(graph_backend, arena) + ))); + dependency_map.insert(dep_elem.clone(), new_node_idx); + new_node_idx + }; // add an edge from the node to its dependency node graph.add_edge(node_idx, dep_node_idx, ArenaEdge::VAR); }); node_idx - }, + } Elem::ConcreteDyn(_range_dyn) => { let node_str = elem.arena_graph_node_label(); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); node_idx - }, + } Elem::Concrete(_range_concrete) => { let node_str = elem.arena_graph_node_label(); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); node_idx - }, + } Elem::Expr(range_expr) => { let node_str = elem.arena_graph_node_label(); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); @@ -308,14 +320,22 @@ impl Elems { Elem::Reference(_lhs) => { // println!("LHS is a reference: {}", range_expr.lhs); // attempt to add in the ContextVar node that the elem is referencing - let context_var_nodes = elem.dependent_on(graph_backend, arena).into_iter().collect::>(); + let context_var_nodes = elem + .dependent_on(graph_backend, arena) + .into_iter() + .collect::>(); context_var_nodes.iter().for_each(|dep_elem| { - let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { + let dep_node_idx = if let Some(&existing_node_idx) = + dependency_map.get(&dep_elem) + { // don't make a new ContextVar node, just use the existing one existing_node_idx } else { // make a new ContextVar Node for the Arena graph - let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend, arena)))); + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!( + "{}", + dep_elem.as_dot_str(graph_backend, arena) + ))); dependency_map.insert(dep_elem.clone(), new_node_idx); new_node_idx }; @@ -323,25 +343,33 @@ impl Elems { graph.update_edge(node_idx, dep_node_idx, ArenaEdge::VAR); }); None - }, + } _ => None, }; let rhs_arena = match *range_expr.rhs.clone() { Elem::Arena(rhs) => { // println!("RHS is an arena index: {}", range_expr.rhs); Some(rhs) - }, + } Elem::Reference(_rhs) => { // println!("RHS is a reference: {}", range_expr.rhs); // attempt to add in the ContextVar node that the elem is referencing - let context_var_nodes = elem.dependent_on(graph_backend, arena).into_iter().collect::>(); + let context_var_nodes = elem + .dependent_on(graph_backend, arena) + .into_iter() + .collect::>(); context_var_nodes.iter().for_each(|dep_elem| { - let dep_node_idx = if let Some(&existing_node_idx) = dependency_map.get(&dep_elem) { + let dep_node_idx = if let Some(&existing_node_idx) = + dependency_map.get(&dep_elem) + { // don't make a new ContextVar node, just use the existing one existing_node_idx } else { // make a new ContextVar Node for the Arena graph - let new_node_idx = graph.add_node(ArenaNode::CVAR(format!("{}", dep_elem.as_dot_str(graph_backend, arena)))); + let new_node_idx = graph.add_node(ArenaNode::CVAR(format!( + "{}", + dep_elem.as_dot_str(graph_backend, arena) + ))); dependency_map.insert(dep_elem.clone(), new_node_idx); new_node_idx }; @@ -349,10 +377,8 @@ impl Elems { graph.update_edge(node_idx, dep_node_idx, ArenaEdge::VAR); }); None - }, - _ => { - None - }, + } + _ => None, }; // Add edges to the arena indices if they exist @@ -368,15 +394,18 @@ impl Elems { } } node_idx - }, + } Elem::Arena(range_arena_idx) => { - panic!("Arena index in elems: {:?}. This should not happen!", range_arena_idx); - }, + panic!( + "Arena index in elems: {:?}. This should not happen!", + range_arena_idx + ); + } Elem::Null => { let node_str = format!("null"); let node_idx = graph.add_node(ArenaNode::ELEM(node_str)); node_idx - }, + } }; // draw edge from the arena_node to the underlying node @@ -398,10 +427,12 @@ impl Elems { if let Some(&min_node_idx) = arena_idx_to_node_idx.get(&arena_idx) { graph.add_edge(node_idx, min_node_idx, ArenaEdge::MIN); } - }, + } _ => { // attempt to find the elem in our `inner` and get the associated arena_graph index - let min_arena_idx = self.inner.iter() + let min_arena_idx = self + .inner + .iter() .find(|(_, elem)| elem == &range_min) .map(|(idx, _)| *idx); // Add edges to the min arena indices @@ -412,7 +443,7 @@ impl Elems { graph.add_edge(node_idx, min_node_idx, ArenaEdge::MIN); } } - }, + } } } @@ -425,10 +456,12 @@ impl Elems { if let Some(&max_node_idx) = arena_idx_to_node_idx.get(&arena_idx) { graph.add_edge(node_idx, max_node_idx, ArenaEdge::MAX); } - }, + } _ => { // attempt to find the elem in our `inner` and get the associated arena_graph index - let max_arena_idx = self.inner.iter() + let max_arena_idx = self + .inner + .iter() .find(|(_, elem)| elem == &range_max) .map(|(idx, _)| *idx); // Add edges to the min arena indices @@ -439,13 +472,16 @@ impl Elems { graph.add_edge(node_idx, max_node_idx, ArenaEdge::MAX); } } - }, + } } } } // Ensure the graph does not have a cycle - debug_assert!(!petgraph::algo::is_cyclic_directed(&graph), "The graph contains a cycle!"); + debug_assert!( + !petgraph::algo::is_cyclic_directed(&graph), + "The graph contains a cycle!" + ); graph } @@ -476,27 +512,24 @@ flowchart TB "#; mermaid_str.push(raw_start_str.to_string()); - let nodes_str = graph.node_indices() + let nodes_str = graph + .node_indices() .map(|idx| { let node_str = arena_mermaid_node(graph, "\t", idx, true, true, None); node_str - }) .collect::>() .join("\n"); - let edges_str = graph.edge_indices() + let edges_str = graph + .edge_indices() .enumerate() .map(|(_i, edge)| { let (from, to) = graph.edge_endpoints(edge).unwrap(); let edge_label = format!("{}", graph[edge]); if edge_label == "" { // don't do a label - format!( - " {} --> {}", - from.index(), - to.index(), - ) + format!(" {} --> {}", from.index(), to.index(),) } else { // do an edge label format!( @@ -514,21 +547,21 @@ flowchart TB mermaid_str.push(edges_str); // Make an invisible node that holds all our edge information for coloring later on frontend - let data_str = graph.edge_indices() + let data_str = graph + .edge_indices() .enumerate() .map(|(_i, edge)| { let (from, to) = graph.edge_endpoints(edge).unwrap(); - format!( - "LS-{}_LE-{}_{}", - from.index(), - to.index(), - &graph[edge] - ) + format!("LS-{}_LE-{}_{}", from.index(), to.index(), &graph[edge]) }) .collect::>() .join(";"); - - let invis_data = format!(" {}(\"{}\"):::INVIS\n classDef INVIS display:none", graph.node_count(), data_str); + + let invis_data = format!( + " {}(\"{}\"):::INVIS\n classDef INVIS display:none", + graph.node_count(), + data_str + ); mermaid_str.push(invis_data); mermaid_str.join("\n") } @@ -541,16 +574,14 @@ pub fn arena_mermaid_node( _loc: bool, class: Option<&str>, ) -> String { - - let node = &graph[idx]; let mut node_str = match node { ArenaNode::ARENA(arena_idx) => { format!("{indent}{}{{{{\"{}\"}}}}", idx.index(), arena_idx) - }, + } ArenaNode::ELEM(label) => { format!("{indent}{}(\"{}\")", idx.index(), label.replace("\"", "'")) - }, + } ArenaNode::CVAR(label) => { format!("{indent}{}(\"{}\")", idx.index(), label.replace("\"", "'")) } @@ -565,10 +596,7 @@ pub fn arena_mermaid_node( } if let Some(class) = class { - node_str.push_str(&format!( - "\n{indent}class {} {class}", - idx.index(), - )); + node_str.push_str(&format!("\n{indent}class {} {class}", idx.index(),)); } node_str @@ -1185,28 +1213,32 @@ impl GraphDot for G<'_> { _as_mermaid: bool, ) -> Option where - Self: std::marker::Sized { + Self: std::marker::Sized, + { todo!() } fn dot_str(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, - Self: shared::AnalyzerLike { + Self: shared::AnalyzerLike, + { todo!() } fn dot_str_no_tmps(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, - Self: GraphLike + shared::AnalyzerLike { + Self: GraphLike + shared::AnalyzerLike, + { todo!() } fn mermaid_str(&self, _arena: &mut RangeArena) -> String where Self: std::marker::Sized, - Self: shared::AnalyzerLike { + Self: shared::AnalyzerLike, + { todo!() } } @@ -1249,11 +1281,11 @@ pub fn mermaid_node( )); } } - + // color the forks let ctx_node = graph::nodes::ContextVarNode::from(current_node).ctx(g); gather_context_info(g, indent, ctx_node, current_node, &mut node_str); - }, + } Node::Context(ctx) => { // highlight self if let solang_parser::pt::Loc::File(f, s, e) = ctx.loc { @@ -1266,8 +1298,8 @@ pub fn mermaid_node( // color the forks let ctx_node = graph::nodes::ContextNode::from(current_node); gather_context_info(g, indent, ctx_node, current_node, &mut node_str); - }, - _ => {}, + } + _ => {} } } @@ -1288,16 +1320,18 @@ fn gather_context_info( original_cvar_node: NodeIdx, node_str: &mut String, ) { - loop { let mut found_continue = false; let current_loc = ctx_node.underlying(g).unwrap().loc; - for edge in g.graph().edges_directed(ctx_node.into(), Direction::Outgoing) { + for edge in g + .graph() + .edges_directed(ctx_node.into(), Direction::Outgoing) + { if let Edge::Context(ContextEdge::Continue(true_or_false)) = edge.weight() { let target_node = edge.target(); if let Node::Context(_ctx) = g.node(target_node) { // error!("found continue pointing to node"); - ctx_node = target_node.into(); + ctx_node = target_node.into(); found_continue = true; // Gather the edge weight and the loc of the Context node it points to if let solang_parser::pt::Loc::File(f, s, e) = current_loc { @@ -1321,4 +1355,4 @@ fn gather_context_info( break; } } -} \ No newline at end of file +} diff --git a/crates/shared/src/graph_like.rs b/crates/shared/src/graph_like.rs index 2330042e..ae073436 100644 --- a/crates/shared/src/graph_like.rs +++ b/crates/shared/src/graph_like.rs @@ -146,7 +146,6 @@ pub trait GraphDot: GraphLike { .expect("failed to execute process"); } - /// Creates a subgraph for visually identifying contexts and subcontexts fn cluster_str( &self, @@ -186,7 +185,6 @@ struct GraphMessage { timestamp: u64, } - pub fn post_to_site(graph: &G, arena: &mut RangeArena) where G: GraphDot + AnalyzerLike, @@ -222,4 +220,4 @@ where } else { error!("Failed to post graph to site: {:?}", res.status()); } -} \ No newline at end of file +}