Merge pull request 'rust2.0' (#1) from rust2.0 into main
Reviewed-on: #1
This commit is contained in:
		| @ -13,3 +13,5 @@ indent_size = 2 | |||||||
| indent_size = 4 | indent_size = 4 | ||||||
| [*.dart] | [*.dart] | ||||||
| indent_size = 2 | indent_size = 2 | ||||||
|  | [*.rs] | ||||||
|  | indent_size = 4 | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								README.md
									
									
									
									
									
								
							| @ -6,6 +6,9 @@ Type/Service definition language and code generator for json-rpc 2.0. Currently | |||||||
| | ------- | --------------------------------- | | | ------- | --------------------------------- | | ||||||
| | ts-node | Typescript for NodeJS             | | | ts-node | Typescript for NodeJS             | | ||||||
| | ts-esm  | Typescript in ESM format for Deno | | | ts-esm  | Typescript in ESM format for Deno | | ||||||
|  | | rust    | Rust                              | | ||||||
|  | | dart    | Dart                              | | ||||||
|  | | c#      | C#                                | | ||||||
|  |  | ||||||
| ## Usage | ## Usage | ||||||
|  |  | ||||||
| @ -23,7 +26,7 @@ enum TestEnum { | |||||||
| type Test { | type Test { | ||||||
|   testen: TestEnum; |   testen: TestEnum; | ||||||
|   someString: string; |   someString: string; | ||||||
|   someNumber: number; |   someNumber?: number; | ||||||
|   array: string[]; |   array: string[]; | ||||||
|   map: {number, TestEnum}; |   map: {number, TestEnum}; | ||||||
| } | } | ||||||
| @ -44,4 +47,8 @@ Then run the generator like this `jrpc compile test.jrpc -o=ts-node:output/`. | |||||||
|  |  | ||||||
| This will generate the Client and Server code in the specified folder. | This will generate the Client and Server code in the specified folder. | ||||||
|  |  | ||||||
| //TODO: Make Documentation better | ## TODOS | ||||||
|  |  | ||||||
|  | 1. Documentation | ||||||
|  | 2. Null Checks/Enforcements in all languages | ||||||
|  | 3. More and better tests | ||||||
|  | |||||||
| @ -41,6 +41,10 @@ class TestSrvimpl : Example.TestServiceServer<int> | |||||||
|       throw new Exception("This is a remote error :)"); |       throw new Exception("This is a remote error :)"); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|  |    public override Task<double> FunctionWithKeywords(double type, double static_, double event_, int ctx) | ||||||
|  |    { | ||||||
|  |       throw new NotImplementedException(); | ||||||
|  |    } | ||||||
| } | } | ||||||
|  |  | ||||||
| class CopyTransportS2 : Example.JRpcTransport | class CopyTransportS2 : Example.JRpcTransport | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								examples/Rust/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								examples/Rust/.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,2 +1,3 @@ | |||||||
| Generated/ | Generated/ | ||||||
| Impl/target | Server/target | ||||||
|  | Client/target | ||||||
|  | |||||||
							
								
								
									
										589
									
								
								examples/Rust/Client/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										589
									
								
								examples/Rust/Client/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @ -0,0 +1,589 @@ | |||||||
|  | # This file is automatically @generated by Cargo. | ||||||
|  | # It is not intended for manual editing. | ||||||
|  | version = 3 | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "async-trait" | ||||||
|  | version = "0.1.59" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "31e6e93155431f3931513b243d371981bb2770112b370c82745a1d19d2f99364" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "atty" | ||||||
|  | version = "0.2.14" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8" | ||||||
|  | dependencies = [ | ||||||
|  |  "hermit-abi", | ||||||
|  |  "libc", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "autocfg" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "bitflags" | ||||||
|  | version = "1.3.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "bytes" | ||||||
|  | version = "1.3.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dfb24e866b15a1af2a1b663f10c6b6b8f397a84aadb828f12e5b289ec23a3a3c" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "cfg-if" | ||||||
|  | version = "1.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "colored" | ||||||
|  | version = "2.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b3616f750b84d8f0de8a58bda93e08e2a81ad3f523089b05f1dffecab48c6cbd" | ||||||
|  | dependencies = [ | ||||||
|  |  "atty", | ||||||
|  |  "lazy_static", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "example" | ||||||
|  | version = "0.1.0" | ||||||
|  | dependencies = [ | ||||||
|  |  "async-trait", | ||||||
|  |  "int-enum", | ||||||
|  |  "log", | ||||||
|  |  "nanoid", | ||||||
|  |  "serde", | ||||||
|  |  "serde_json", | ||||||
|  |  "simple_logger", | ||||||
|  |  "tokio", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "getrandom" | ||||||
|  | version = "0.2.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "4eb1a864a501629691edf6c15a593b7a51eebaa1e8468e9ddc623de7c9b58ec6" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "wasi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "hermit-abi" | ||||||
|  | version = "0.1.19" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "int-enum" | ||||||
|  | version = "0.5.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "cff87d3cc4b79b4559e3c75068d64247284aceb6a038bd4bb38387f3f164476d" | ||||||
|  | dependencies = [ | ||||||
|  |  "int-enum-impl", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "int-enum-impl" | ||||||
|  | version = "0.5.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "df1f2f068675add1a3fc77f5f5ab2e29290c841ee34d151abc007bce902e5d34" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro-crate", | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "itoa" | ||||||
|  | version = "1.0.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "1aab8fc367588b89dcee83ab0fd66b72b50b72fa1904d7095045ace2b0c81c35" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "lazy_static" | ||||||
|  | version = "1.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "libc" | ||||||
|  | version = "0.2.138" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "db6d7e329c562c5dfab7a46a2afabc8b987ab9a4834c9d1ca04dc54c1546cef8" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "lock_api" | ||||||
|  | version = "0.4.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "435011366fe56583b16cf956f9df0095b405b82d76425bc8981c0e22e60ec4df" | ||||||
|  | dependencies = [ | ||||||
|  |  "autocfg", | ||||||
|  |  "scopeguard", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "log" | ||||||
|  | version = "0.4.17" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "memchr" | ||||||
|  | version = "2.5.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "mio" | ||||||
|  | version = "0.8.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e5d732bc30207a6423068df043e3d02e0735b155ad7ce1a6f76fe2baa5b158de" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "log", | ||||||
|  |  "wasi", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "nanoid" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "3ffa00dec017b5b1a8b7cf5e2c008bfda1aa7e0697ac1508b491fdf2622fb4d8" | ||||||
|  | dependencies = [ | ||||||
|  |  "rand", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "num_cpus" | ||||||
|  | version = "1.13.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1" | ||||||
|  | dependencies = [ | ||||||
|  |  "hermit-abi", | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "num_threads" | ||||||
|  | version = "0.1.6" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2819ce041d2ee131036f4fc9d6ae7ae125a3a40e97ba64d04fe799ad9dabbb44" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "once_cell" | ||||||
|  | version = "1.16.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "86f0b0d4bf799edbc74508c1e8bf170ff5f41238e5f8225603ca7caaae2b7860" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot" | ||||||
|  | version = "0.12.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f" | ||||||
|  | dependencies = [ | ||||||
|  |  "lock_api", | ||||||
|  |  "parking_lot_core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot_core" | ||||||
|  | version = "0.9.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7ff9f3fef3968a3ec5945535ed654cb38ff72d7495a25619e2247fb15a2ed9ba" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "redox_syscall", | ||||||
|  |  "smallvec", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "pin-project-lite" | ||||||
|  | version = "0.2.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e0a7ae3ac2f1173085d398531c705756c94a4c56843785df85a60c1a0afac116" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "ppv-lite86" | ||||||
|  | version = "0.2.16" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "proc-macro-crate" | ||||||
|  | version = "1.2.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "eda0fc3b0fb7c975631757e14d9049da17374063edb6ebbcbc54d880d4fe94e9" | ||||||
|  | dependencies = [ | ||||||
|  |  "once_cell", | ||||||
|  |  "thiserror", | ||||||
|  |  "toml", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "proc-macro2" | ||||||
|  | version = "1.0.47" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5ea3d908b0e36316caf9e9e2c4625cdde190a7e6f440d794667ed17a1855e725" | ||||||
|  | dependencies = [ | ||||||
|  |  "unicode-ident", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "quote" | ||||||
|  | version = "1.0.18" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "rand" | ||||||
|  | version = "0.8.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "rand_chacha", | ||||||
|  |  "rand_core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "rand_chacha" | ||||||
|  | version = "0.3.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" | ||||||
|  | dependencies = [ | ||||||
|  |  "ppv-lite86", | ||||||
|  |  "rand_core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "rand_core" | ||||||
|  | version = "0.6.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7" | ||||||
|  | dependencies = [ | ||||||
|  |  "getrandom", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "redox_syscall" | ||||||
|  | version = "0.2.16" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" | ||||||
|  | dependencies = [ | ||||||
|  |  "bitflags", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "ryu" | ||||||
|  | version = "1.0.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "73b4b750c782965c211b42f022f59af1fbceabdd026623714f104152f1ec149f" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "scopeguard" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "serde" | ||||||
|  | version = "1.0.150" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e326c9ec8042f1b5da33252c8a37e9ffbd2c9bef0155215b6e6c80c790e05f91" | ||||||
|  | dependencies = [ | ||||||
|  |  "serde_derive", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "serde_derive" | ||||||
|  | version = "1.0.150" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "42a3df25b0713732468deadad63ab9da1f1fd75a48a15024b50363f128db627e" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "serde_json" | ||||||
|  | version = "1.0.89" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "020ff22c755c2ed3f8cf162dbb41a7268d934702f3ed3631656ea597e08fc3db" | ||||||
|  | dependencies = [ | ||||||
|  |  "itoa", | ||||||
|  |  "ryu", | ||||||
|  |  "serde", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "signal-hook-registry" | ||||||
|  | version = "1.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e51e73328dc4ac0c7ccbda3a494dfa03df1de2f46018127f60c693f2648455b0" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "simple_logger" | ||||||
|  | version = "4.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e190a521c2044948158666916d9e872cbb9984f755e9bb3b5b75a836205affcd" | ||||||
|  | dependencies = [ | ||||||
|  |  "atty", | ||||||
|  |  "colored", | ||||||
|  |  "log", | ||||||
|  |  "time", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "smallvec" | ||||||
|  | version = "1.10.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a507befe795404456341dfab10cef66ead4c041f62b8b11bbb92bffe5d0953e0" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "socket2" | ||||||
|  | version = "0.4.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "02e2d2db9033d13a1567121ddd7a095ee144db4e1ca1b1bda3419bc0da294ebd" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "syn" | ||||||
|  | version = "1.0.105" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "60b9b43d45702de4c839cb9b51d9f529c5dd26a4aff255b42b1ebc03e88ee908" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "unicode-ident", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "test-impl-client" | ||||||
|  | version = "0.1.0" | ||||||
|  | dependencies = [ | ||||||
|  |  "async-trait", | ||||||
|  |  "example", | ||||||
|  |  "serde_json", | ||||||
|  |  "tokio", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "thiserror" | ||||||
|  | version = "1.0.37" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "10deb33631e3c9018b9baf9dcbbc4f737320d2b576bac10f6aefa048fa407e3e" | ||||||
|  | dependencies = [ | ||||||
|  |  "thiserror-impl", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "thiserror-impl" | ||||||
|  | version = "1.0.37" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "982d17546b47146b28f7c22e3d08465f6b8903d0ea13c1660d9d84a6e7adcdbb" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "time" | ||||||
|  | version = "0.3.17" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a561bf4617eebd33bca6434b988f39ed798e527f51a1e797d0ee4f61c0a38376" | ||||||
|  | dependencies = [ | ||||||
|  |  "itoa", | ||||||
|  |  "libc", | ||||||
|  |  "num_threads", | ||||||
|  |  "serde", | ||||||
|  |  "time-core", | ||||||
|  |  "time-macros", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "time-core" | ||||||
|  | version = "0.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2e153e1f1acaef8acc537e68b44906d2db6436e2b35ac2c6b42640fff91f00fd" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "time-macros" | ||||||
|  | version = "0.2.6" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d967f99f534ca7e495c575c62638eebc2898a8c84c119b89e250477bc4ba16b2" | ||||||
|  | dependencies = [ | ||||||
|  |  "time-core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "tokio" | ||||||
|  | version = "1.23.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "eab6d665857cc6ca78d6e80303a02cea7a7851e85dfbd77cbdc09bd129f1ef46" | ||||||
|  | dependencies = [ | ||||||
|  |  "autocfg", | ||||||
|  |  "bytes", | ||||||
|  |  "libc", | ||||||
|  |  "memchr", | ||||||
|  |  "mio", | ||||||
|  |  "num_cpus", | ||||||
|  |  "parking_lot", | ||||||
|  |  "pin-project-lite", | ||||||
|  |  "signal-hook-registry", | ||||||
|  |  "socket2", | ||||||
|  |  "tokio-macros", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "tokio-macros" | ||||||
|  | version = "1.8.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d266c00fde287f55d3f1c3e96c500c362a2b8c695076ec180f27918820bc6df8" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "toml" | ||||||
|  | version = "0.5.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8d82e1a7758622a465f8cee077614c73484dac5b836c02ff6a40d5d1010324d7" | ||||||
|  | dependencies = [ | ||||||
|  |  "serde", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "unicode-ident" | ||||||
|  | version = "1.0.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "6ceab39d59e4c9499d4e5a8ee0e2735b891bb7308ac83dfb4e80cad195c9f6f3" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "wasi" | ||||||
|  | version = "0.11.0+wasi-snapshot-preview1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "winapi" | ||||||
|  | version = "0.3.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi-i686-pc-windows-gnu", | ||||||
|  |  "winapi-x86_64-pc-windows-gnu", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-i686-pc-windows-gnu" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-x86_64-pc-windows-gnu" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows-sys" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5a3e1820f08b8513f676f7ab6c1f99ff312fb97b553d30ff4dd86f9f15728aa7" | ||||||
|  | dependencies = [ | ||||||
|  |  "windows_aarch64_gnullvm", | ||||||
|  |  "windows_aarch64_msvc", | ||||||
|  |  "windows_i686_gnu", | ||||||
|  |  "windows_i686_msvc", | ||||||
|  |  "windows_x86_64_gnu", | ||||||
|  |  "windows_x86_64_gnullvm", | ||||||
|  |  "windows_x86_64_msvc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_aarch64_gnullvm" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "41d2aa71f6f0cbe00ae5167d90ef3cfe66527d6f613ca78ac8024c3ccab9a19e" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_aarch64_msvc" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dd0f252f5a35cac83d6311b2e795981f5ee6e67eb1f9a7f64eb4500fbc4dcdb4" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_i686_gnu" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "fbeae19f6716841636c28d695375df17562ca208b2b7d0dc47635a50ae6c5de7" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_i686_msvc" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "84c12f65daa39dd2babe6e442988fc329d6243fdce47d7d2d155b8d874862246" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_gnu" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "bf7b1b21b5362cbc318f686150e5bcea75ecedc74dd157d874d754a2ca44b0ed" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_gnullvm" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "09d525d2ba30eeb3297665bd434a54297e4170c7f1a44cad4ef58095b4cd2028" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_msvc" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f40009d85759725a34da6d89a94e63d7bdc50a862acf0dbc7c8e488f1edcb6f5" | ||||||
							
								
								
									
										10
									
								
								examples/Rust/Client/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								examples/Rust/Client/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | [package] | ||||||
|  | name = "test-impl-client" | ||||||
|  | version = "0.1.0" | ||||||
|  | edition = "2021" | ||||||
|  |  | ||||||
|  | [dependencies] | ||||||
|  | example = { path = "../Generated/" } | ||||||
|  | async-trait = "0.1.59" | ||||||
|  | tokio = { version = "1.22.0", features = ["full"] } | ||||||
|  | serde_json = "1.0.88" | ||||||
							
								
								
									
										52
									
								
								examples/Rust/Client/src/main.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								examples/Rust/Client/src/main.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,52 @@ | |||||||
|  | use std::sync::atomic::AtomicU64; | ||||||
|  | use std::sync::Arc; | ||||||
|  |  | ||||||
|  | use example::client::SimpleTestService; | ||||||
|  | use example::JRPCClient; | ||||||
|  | use tokio::io::{AsyncBufReadExt, AsyncWriteExt, BufReader}; | ||||||
|  | use tokio::net::TcpStream; | ||||||
|  | use tokio::sync::mpsc::channel; | ||||||
|  |  | ||||||
|  | #[tokio::main] | ||||||
|  | pub async fn main() { | ||||||
|  |     let (tx, mut rx) = channel(1); | ||||||
|  |     let stream = TcpStream::connect("127.0.0.1:4321").await.unwrap(); | ||||||
|  |     let client = example::JRPCClient::new(tx.clone()); | ||||||
|  |     let test_service = SimpleTestService::new(client.clone()); | ||||||
|  |  | ||||||
|  |     let (reader, mut writer) = stream.into_split(); | ||||||
|  |     let mut reader = BufReader::new(reader); | ||||||
|  |     let mut line = String::new(); | ||||||
|  |  | ||||||
|  |     tokio::spawn(async move { | ||||||
|  |         loop { | ||||||
|  |             line.clear(); | ||||||
|  |             reader.read_line(&mut line).await.unwrap(); | ||||||
|  |             client.on_result(serde_json::from_str(&line).unwrap()).await; | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     tokio::spawn(async move { | ||||||
|  |         while let Some(message) = rx.recv().await { | ||||||
|  |             writer | ||||||
|  |                 .write_all((serde_json::to_string(&message).unwrap() + "\n").as_bytes()) | ||||||
|  |                 .await | ||||||
|  |                 .unwrap(); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     let ticks = Arc::new(AtomicU64::new(0)); | ||||||
|  |     let t2 = ticks.clone(); | ||||||
|  |     tokio::spawn(async move { | ||||||
|  |         loop { | ||||||
|  |             tokio::time::sleep(tokio::time::Duration::from_secs(1)).await; | ||||||
|  |             println!("Ticks: {}", t2.load(std::sync::atomic::Ordering::Relaxed)); | ||||||
|  |             t2.store(0, std::sync::atomic::Ordering::Relaxed); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     loop { | ||||||
|  |         let _result = test_service.GetTest("Hi".to_owned(), 55).await.unwrap(); | ||||||
|  |         ticks.fetch_add(1, std::sync::atomic::Ordering::Relaxed); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										248
									
								
								examples/Rust/Impl/Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										248
									
								
								examples/Rust/Impl/Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -1,248 +0,0 @@ | |||||||
| # This file is automatically @generated by Cargo. |  | ||||||
| # It is not intended for manual editing. |  | ||||||
| version = 3 |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "cfg-if" |  | ||||||
| version = "0.1.10" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "4785bdd1c96b2a846b2bd7cc02e86b6b3dbf14e7e53446c4f54c92a361040822" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "cfg-if" |  | ||||||
| version = "1.0.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "getrandom" |  | ||||||
| version = "0.2.7" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "4eb1a864a501629691edf6c15a593b7a51eebaa1e8468e9ddc623de7c9b58ec6" |  | ||||||
| dependencies = [ |  | ||||||
|  "cfg-if 1.0.0", |  | ||||||
|  "libc", |  | ||||||
|  "wasi", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "hermit-abi" |  | ||||||
| version = "0.1.19" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" |  | ||||||
| dependencies = [ |  | ||||||
|  "libc", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "int-enum" |  | ||||||
| version = "0.4.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "2b1428b2b1abe959e6eedb0a17d0ab12f6ba20e1106cc29fc4874e3ba393c177" |  | ||||||
| dependencies = [ |  | ||||||
|  "cfg-if 0.1.10", |  | ||||||
|  "int-enum-impl", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "int-enum-impl" |  | ||||||
| version = "0.4.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "a2c3cecaad8ca1a5020843500c696de2b9a07b63b624ddeef91f85f9bafb3671" |  | ||||||
| dependencies = [ |  | ||||||
|  "cfg-if 0.1.10", |  | ||||||
|  "proc-macro-crate", |  | ||||||
|  "proc-macro2", |  | ||||||
|  "quote", |  | ||||||
|  "syn", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "itoa" |  | ||||||
| version = "1.0.1" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "1aab8fc367588b89dcee83ab0fd66b72b50b72fa1904d7095045ace2b0c81c35" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "libc" |  | ||||||
| version = "0.2.123" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "cb691a747a7ab48abc15c5b42066eaafde10dc427e3b6ee2a1cf43db04c763bd" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "nanoid" |  | ||||||
| version = "0.4.0" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "3ffa00dec017b5b1a8b7cf5e2c008bfda1aa7e0697ac1508b491fdf2622fb4d8" |  | ||||||
| dependencies = [ |  | ||||||
|  "rand", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "num_cpus" |  | ||||||
| version = "1.13.1" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1" |  | ||||||
| dependencies = [ |  | ||||||
|  "hermit-abi", |  | ||||||
|  "libc", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "ppv-lite86" |  | ||||||
| version = "0.2.16" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "proc-macro-crate" |  | ||||||
| version = "0.1.5" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "1d6ea3c4595b96363c13943497db34af4460fb474a95c43f4446ad341b8c9785" |  | ||||||
| dependencies = [ |  | ||||||
|  "toml", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "proc-macro2" |  | ||||||
| version = "1.0.37" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "ec757218438d5fda206afc041538b2f6d889286160d649a86a24d37e1235afd1" |  | ||||||
| dependencies = [ |  | ||||||
|  "unicode-xid", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "quote" |  | ||||||
| version = "1.0.18" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" |  | ||||||
| dependencies = [ |  | ||||||
|  "proc-macro2", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "rand" |  | ||||||
| version = "0.8.5" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" |  | ||||||
| dependencies = [ |  | ||||||
|  "libc", |  | ||||||
|  "rand_chacha", |  | ||||||
|  "rand_core", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "rand_chacha" |  | ||||||
| version = "0.3.1" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" |  | ||||||
| dependencies = [ |  | ||||||
|  "ppv-lite86", |  | ||||||
|  "rand_core", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "rand_core" |  | ||||||
| version = "0.6.3" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7" |  | ||||||
| dependencies = [ |  | ||||||
|  "getrandom", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "ryu" |  | ||||||
| version = "1.0.9" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "73b4b750c782965c211b42f022f59af1fbceabdd026623714f104152f1ec149f" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "serde" |  | ||||||
| version = "1.0.136" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "ce31e24b01e1e524df96f1c2fdd054405f8d7376249a5110886fb4b658484789" |  | ||||||
| dependencies = [ |  | ||||||
|  "serde_derive", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "serde_derive" |  | ||||||
| version = "1.0.136" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "08597e7152fcd306f41838ed3e37be9eaeed2b61c42e2117266a554fab4662f9" |  | ||||||
| dependencies = [ |  | ||||||
|  "proc-macro2", |  | ||||||
|  "quote", |  | ||||||
|  "syn", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "serde_json" |  | ||||||
| version = "1.0.79" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "8e8d9fa5c3b304765ce1fd9c4c8a3de2c8db365a5b91be52f186efc675681d95" |  | ||||||
| dependencies = [ |  | ||||||
|  "itoa", |  | ||||||
|  "ryu", |  | ||||||
|  "serde", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "syn" |  | ||||||
| version = "1.0.91" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "b683b2b825c8eef438b77c36a06dc262294da3d5a5813fac20da149241dcd44d" |  | ||||||
| dependencies = [ |  | ||||||
|  "proc-macro2", |  | ||||||
|  "quote", |  | ||||||
|  "unicode-xid", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "test" |  | ||||||
| version = "0.1.0" |  | ||||||
| dependencies = [ |  | ||||||
|  "int-enum", |  | ||||||
|  "nanoid", |  | ||||||
|  "serde", |  | ||||||
|  "serde_json", |  | ||||||
|  "threadpool", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "test-impl" |  | ||||||
| version = "0.1.0" |  | ||||||
| dependencies = [ |  | ||||||
|  "test", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "threadpool" |  | ||||||
| version = "1.8.1" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "d050e60b33d41c19108b32cea32164033a9013fe3b46cbd4457559bfbf77afaa" |  | ||||||
| dependencies = [ |  | ||||||
|  "num_cpus", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "toml" |  | ||||||
| version = "0.5.9" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "8d82e1a7758622a465f8cee077614c73484dac5b836c02ff6a40d5d1010324d7" |  | ||||||
| dependencies = [ |  | ||||||
|  "serde", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "unicode-xid" |  | ||||||
| version = "0.2.2" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "wasi" |  | ||||||
| version = "0.11.0+wasi-snapshot-preview1" |  | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" |  | ||||||
| checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" |  | ||||||
| @ -1,7 +0,0 @@ | |||||||
| [package] |  | ||||||
| name = "test-impl" |  | ||||||
| version = "0.1.0" |  | ||||||
| edition = "2021" |  | ||||||
|  |  | ||||||
| [dependencies] |  | ||||||
| test = { path = "../Generated/" } |  | ||||||
| @ -1,52 +0,0 @@ | |||||||
| use test::{JRPCServer,Result,Test}; |  | ||||||
| use test::server::{TestService,TestServiceHandler}; |  | ||||||
| use std::io::{BufReader,BufRead,Write}; |  | ||||||
| use std::sync::mpsc::channel; |  | ||||||
|  |  | ||||||
| #[derive(Clone, Copy)] |  | ||||||
| struct MyCtx {} |  | ||||||
|  |  | ||||||
| struct TestServiceImplementation {} |  | ||||||
|  |  | ||||||
| impl TestService<MyCtx> for TestServiceImplementation { |  | ||||||
|    fn GetTest(&self, name: String, age: i64, _context: &MyCtx) -> Result<Test> { |  | ||||||
|       return Ok(Test { name, age }); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    fn TestNot(&self, _: &MyCtx) -> Result<()> { |  | ||||||
|       return Ok(()); |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub fn main() { |  | ||||||
|    let mut srv = JRPCServer::<MyCtx>::new(); |  | ||||||
|    srv.add_service(TestServiceHandler::new(Box::from(TestServiceImplementation {}))); |  | ||||||
|  |  | ||||||
|    let listener = std::net::TcpListener::bind("127.0.0.1:4321").expect("Could not start listener!"); |  | ||||||
|  |  | ||||||
|    for stream in listener.incoming() { |  | ||||||
|       println!("Got Connection!"); |  | ||||||
|       let (stx, srx) = channel::<String>(); |  | ||||||
|       let (rtx, rrx) = channel::<String>(); |  | ||||||
|  |  | ||||||
|       let mut stream = stream.expect("Bad stream"); |  | ||||||
|       let mut r = BufReader::new(stream.try_clone().unwrap()); |  | ||||||
|       let mut line = String::new(); |  | ||||||
|  |  | ||||||
|       srv.start_session(srx, rtx, MyCtx {}); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|       r.read_line(&mut line).expect("Could not read line!"); |  | ||||||
|       println!("Got line: {}", line); |  | ||||||
|       stx.send(line).expect("Could not send packet to handler!"); |  | ||||||
|       println!("Sending to handler succeeded"); |  | ||||||
|       let response = rrx.recv().expect("Could not get reponse from handler!"); |  | ||||||
|       println!("Prepared response {}", response); |  | ||||||
|        |  | ||||||
|       stream.write((response + "\n").as_bytes()).expect("Could not send reponse!"); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    println!("Hello World"); |  | ||||||
|  |  | ||||||
|    // return Ok(()); |  | ||||||
| } |  | ||||||
							
								
								
									
										589
									
								
								examples/Rust/Server/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										589
									
								
								examples/Rust/Server/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @ -0,0 +1,589 @@ | |||||||
|  | # This file is automatically @generated by Cargo. | ||||||
|  | # It is not intended for manual editing. | ||||||
|  | version = 3 | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "async-trait" | ||||||
|  | version = "0.1.59" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "31e6e93155431f3931513b243d371981bb2770112b370c82745a1d19d2f99364" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "atty" | ||||||
|  | version = "0.2.14" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8" | ||||||
|  | dependencies = [ | ||||||
|  |  "hermit-abi", | ||||||
|  |  "libc", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "autocfg" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "bitflags" | ||||||
|  | version = "1.3.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "bytes" | ||||||
|  | version = "1.3.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dfb24e866b15a1af2a1b663f10c6b6b8f397a84aadb828f12e5b289ec23a3a3c" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "cfg-if" | ||||||
|  | version = "1.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "colored" | ||||||
|  | version = "2.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b3616f750b84d8f0de8a58bda93e08e2a81ad3f523089b05f1dffecab48c6cbd" | ||||||
|  | dependencies = [ | ||||||
|  |  "atty", | ||||||
|  |  "lazy_static", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "example" | ||||||
|  | version = "0.1.0" | ||||||
|  | dependencies = [ | ||||||
|  |  "async-trait", | ||||||
|  |  "int-enum", | ||||||
|  |  "log", | ||||||
|  |  "nanoid", | ||||||
|  |  "serde", | ||||||
|  |  "serde_json", | ||||||
|  |  "simple_logger", | ||||||
|  |  "tokio", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "getrandom" | ||||||
|  | version = "0.2.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "4eb1a864a501629691edf6c15a593b7a51eebaa1e8468e9ddc623de7c9b58ec6" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "wasi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "hermit-abi" | ||||||
|  | version = "0.1.19" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "int-enum" | ||||||
|  | version = "0.5.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "cff87d3cc4b79b4559e3c75068d64247284aceb6a038bd4bb38387f3f164476d" | ||||||
|  | dependencies = [ | ||||||
|  |  "int-enum-impl", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "int-enum-impl" | ||||||
|  | version = "0.5.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "df1f2f068675add1a3fc77f5f5ab2e29290c841ee34d151abc007bce902e5d34" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro-crate", | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "itoa" | ||||||
|  | version = "1.0.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "1aab8fc367588b89dcee83ab0fd66b72b50b72fa1904d7095045ace2b0c81c35" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "lazy_static" | ||||||
|  | version = "1.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "libc" | ||||||
|  | version = "0.2.138" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "db6d7e329c562c5dfab7a46a2afabc8b987ab9a4834c9d1ca04dc54c1546cef8" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "lock_api" | ||||||
|  | version = "0.4.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "435011366fe56583b16cf956f9df0095b405b82d76425bc8981c0e22e60ec4df" | ||||||
|  | dependencies = [ | ||||||
|  |  "autocfg", | ||||||
|  |  "scopeguard", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "log" | ||||||
|  | version = "0.4.17" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "memchr" | ||||||
|  | version = "2.5.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "mio" | ||||||
|  | version = "0.8.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e5d732bc30207a6423068df043e3d02e0735b155ad7ce1a6f76fe2baa5b158de" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "log", | ||||||
|  |  "wasi", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "nanoid" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "3ffa00dec017b5b1a8b7cf5e2c008bfda1aa7e0697ac1508b491fdf2622fb4d8" | ||||||
|  | dependencies = [ | ||||||
|  |  "rand", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "num_cpus" | ||||||
|  | version = "1.13.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1" | ||||||
|  | dependencies = [ | ||||||
|  |  "hermit-abi", | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "num_threads" | ||||||
|  | version = "0.1.6" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2819ce041d2ee131036f4fc9d6ae7ae125a3a40e97ba64d04fe799ad9dabbb44" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "once_cell" | ||||||
|  | version = "1.16.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "86f0b0d4bf799edbc74508c1e8bf170ff5f41238e5f8225603ca7caaae2b7860" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot" | ||||||
|  | version = "0.12.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f" | ||||||
|  | dependencies = [ | ||||||
|  |  "lock_api", | ||||||
|  |  "parking_lot_core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot_core" | ||||||
|  | version = "0.9.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7ff9f3fef3968a3ec5945535ed654cb38ff72d7495a25619e2247fb15a2ed9ba" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "redox_syscall", | ||||||
|  |  "smallvec", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "pin-project-lite" | ||||||
|  | version = "0.2.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e0a7ae3ac2f1173085d398531c705756c94a4c56843785df85a60c1a0afac116" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "ppv-lite86" | ||||||
|  | version = "0.2.16" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "proc-macro-crate" | ||||||
|  | version = "1.2.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "eda0fc3b0fb7c975631757e14d9049da17374063edb6ebbcbc54d880d4fe94e9" | ||||||
|  | dependencies = [ | ||||||
|  |  "once_cell", | ||||||
|  |  "thiserror", | ||||||
|  |  "toml", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "proc-macro2" | ||||||
|  | version = "1.0.47" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5ea3d908b0e36316caf9e9e2c4625cdde190a7e6f440d794667ed17a1855e725" | ||||||
|  | dependencies = [ | ||||||
|  |  "unicode-ident", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "quote" | ||||||
|  | version = "1.0.18" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "rand" | ||||||
|  | version = "0.8.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "rand_chacha", | ||||||
|  |  "rand_core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "rand_chacha" | ||||||
|  | version = "0.3.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" | ||||||
|  | dependencies = [ | ||||||
|  |  "ppv-lite86", | ||||||
|  |  "rand_core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "rand_core" | ||||||
|  | version = "0.6.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7" | ||||||
|  | dependencies = [ | ||||||
|  |  "getrandom", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "redox_syscall" | ||||||
|  | version = "0.2.16" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" | ||||||
|  | dependencies = [ | ||||||
|  |  "bitflags", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "ryu" | ||||||
|  | version = "1.0.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "73b4b750c782965c211b42f022f59af1fbceabdd026623714f104152f1ec149f" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "scopeguard" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "serde" | ||||||
|  | version = "1.0.150" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e326c9ec8042f1b5da33252c8a37e9ffbd2c9bef0155215b6e6c80c790e05f91" | ||||||
|  | dependencies = [ | ||||||
|  |  "serde_derive", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "serde_derive" | ||||||
|  | version = "1.0.150" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "42a3df25b0713732468deadad63ab9da1f1fd75a48a15024b50363f128db627e" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "serde_json" | ||||||
|  | version = "1.0.89" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "020ff22c755c2ed3f8cf162dbb41a7268d934702f3ed3631656ea597e08fc3db" | ||||||
|  | dependencies = [ | ||||||
|  |  "itoa", | ||||||
|  |  "ryu", | ||||||
|  |  "serde", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "signal-hook-registry" | ||||||
|  | version = "1.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e51e73328dc4ac0c7ccbda3a494dfa03df1de2f46018127f60c693f2648455b0" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "simple_logger" | ||||||
|  | version = "4.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e190a521c2044948158666916d9e872cbb9984f755e9bb3b5b75a836205affcd" | ||||||
|  | dependencies = [ | ||||||
|  |  "atty", | ||||||
|  |  "colored", | ||||||
|  |  "log", | ||||||
|  |  "time", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "smallvec" | ||||||
|  | version = "1.10.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a507befe795404456341dfab10cef66ead4c041f62b8b11bbb92bffe5d0953e0" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "socket2" | ||||||
|  | version = "0.4.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "02e2d2db9033d13a1567121ddd7a095ee144db4e1ca1b1bda3419bc0da294ebd" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "syn" | ||||||
|  | version = "1.0.105" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "60b9b43d45702de4c839cb9b51d9f529c5dd26a4aff255b42b1ebc03e88ee908" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "unicode-ident", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "test-impl" | ||||||
|  | version = "0.1.0" | ||||||
|  | dependencies = [ | ||||||
|  |  "async-trait", | ||||||
|  |  "example", | ||||||
|  |  "serde_json", | ||||||
|  |  "tokio", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "thiserror" | ||||||
|  | version = "1.0.37" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "10deb33631e3c9018b9baf9dcbbc4f737320d2b576bac10f6aefa048fa407e3e" | ||||||
|  | dependencies = [ | ||||||
|  |  "thiserror-impl", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "thiserror-impl" | ||||||
|  | version = "1.0.37" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "982d17546b47146b28f7c22e3d08465f6b8903d0ea13c1660d9d84a6e7adcdbb" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "time" | ||||||
|  | version = "0.3.17" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a561bf4617eebd33bca6434b988f39ed798e527f51a1e797d0ee4f61c0a38376" | ||||||
|  | dependencies = [ | ||||||
|  |  "itoa", | ||||||
|  |  "libc", | ||||||
|  |  "num_threads", | ||||||
|  |  "serde", | ||||||
|  |  "time-core", | ||||||
|  |  "time-macros", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "time-core" | ||||||
|  | version = "0.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2e153e1f1acaef8acc537e68b44906d2db6436e2b35ac2c6b42640fff91f00fd" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "time-macros" | ||||||
|  | version = "0.2.6" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d967f99f534ca7e495c575c62638eebc2898a8c84c119b89e250477bc4ba16b2" | ||||||
|  | dependencies = [ | ||||||
|  |  "time-core", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "tokio" | ||||||
|  | version = "1.23.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "eab6d665857cc6ca78d6e80303a02cea7a7851e85dfbd77cbdc09bd129f1ef46" | ||||||
|  | dependencies = [ | ||||||
|  |  "autocfg", | ||||||
|  |  "bytes", | ||||||
|  |  "libc", | ||||||
|  |  "memchr", | ||||||
|  |  "mio", | ||||||
|  |  "num_cpus", | ||||||
|  |  "parking_lot", | ||||||
|  |  "pin-project-lite", | ||||||
|  |  "signal-hook-registry", | ||||||
|  |  "socket2", | ||||||
|  |  "tokio-macros", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "tokio-macros" | ||||||
|  | version = "1.8.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d266c00fde287f55d3f1c3e96c500c362a2b8c695076ec180f27918820bc6df8" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "toml" | ||||||
|  | version = "0.5.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8d82e1a7758622a465f8cee077614c73484dac5b836c02ff6a40d5d1010324d7" | ||||||
|  | dependencies = [ | ||||||
|  |  "serde", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "unicode-ident" | ||||||
|  | version = "1.0.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "6ceab39d59e4c9499d4e5a8ee0e2735b891bb7308ac83dfb4e80cad195c9f6f3" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "wasi" | ||||||
|  | version = "0.11.0+wasi-snapshot-preview1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "winapi" | ||||||
|  | version = "0.3.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi-i686-pc-windows-gnu", | ||||||
|  |  "winapi-x86_64-pc-windows-gnu", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-i686-pc-windows-gnu" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-x86_64-pc-windows-gnu" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows-sys" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5a3e1820f08b8513f676f7ab6c1f99ff312fb97b553d30ff4dd86f9f15728aa7" | ||||||
|  | dependencies = [ | ||||||
|  |  "windows_aarch64_gnullvm", | ||||||
|  |  "windows_aarch64_msvc", | ||||||
|  |  "windows_i686_gnu", | ||||||
|  |  "windows_i686_msvc", | ||||||
|  |  "windows_x86_64_gnu", | ||||||
|  |  "windows_x86_64_gnullvm", | ||||||
|  |  "windows_x86_64_msvc", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_aarch64_gnullvm" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "41d2aa71f6f0cbe00ae5167d90ef3cfe66527d6f613ca78ac8024c3ccab9a19e" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_aarch64_msvc" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dd0f252f5a35cac83d6311b2e795981f5ee6e67eb1f9a7f64eb4500fbc4dcdb4" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_i686_gnu" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "fbeae19f6716841636c28d695375df17562ca208b2b7d0dc47635a50ae6c5de7" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_i686_msvc" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "84c12f65daa39dd2babe6e442988fc329d6243fdce47d7d2d155b8d874862246" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_gnu" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "bf7b1b21b5362cbc318f686150e5bcea75ecedc74dd157d874d754a2ca44b0ed" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_gnullvm" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "09d525d2ba30eeb3297665bd434a54297e4170c7f1a44cad4ef58095b4cd2028" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_msvc" | ||||||
|  | version = "0.42.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f40009d85759725a34da6d89a94e63d7bdc50a862acf0dbc7c8e488f1edcb6f5" | ||||||
							
								
								
									
										10
									
								
								examples/Rust/Server/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								examples/Rust/Server/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | [package] | ||||||
|  | name = "test-impl" | ||||||
|  | version = "0.1.0" | ||||||
|  | edition = "2021" | ||||||
|  |  | ||||||
|  | [dependencies] | ||||||
|  | example = { path = "../Generated/" } | ||||||
|  | async-trait = "0.1.59" | ||||||
|  | tokio = { version = "1.22.0", features = ["full"] } | ||||||
|  | serde_json = "1.0.88" | ||||||
							
								
								
									
										75
									
								
								examples/Rust/Server/src/main.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								examples/Rust/Server/src/main.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,75 @@ | |||||||
|  | use async_trait::async_trait; | ||||||
|  | use example::base_lib::JRPCResult; | ||||||
|  | use example::server::{SimpleTestService, SimpleTestServiceHandler}; | ||||||
|  | use example::{JRPCServer, Result, Test2}; | ||||||
|  | use tokio::io::{AsyncBufReadExt, AsyncWriteExt, BufReader}; | ||||||
|  | use tokio::net::TcpStream; | ||||||
|  | use tokio::sync::mpsc::channel; | ||||||
|  |  | ||||||
|  | #[derive(Clone)] | ||||||
|  | struct TestServiceImplementation {} | ||||||
|  |  | ||||||
|  | #[async_trait] | ||||||
|  | impl SimpleTestService for TestServiceImplementation { | ||||||
|  |     async fn GetTest(&self, name: String, age: i64) -> Result<Test2> { | ||||||
|  |         return Ok(Test2 { name, age }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn TestNot(&self) -> Result<()> { | ||||||
|  |         return Ok(()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn handle_connection(stream: TcpStream, server: &JRPCServer) { | ||||||
|  |     let (response_sender, mut response_receiver) = channel::<JRPCResult>(1); | ||||||
|  |  | ||||||
|  |     let session = server.get_session(response_sender); | ||||||
|  |  | ||||||
|  |     let (r, mut w) = stream.into_split(); | ||||||
|  |     tokio::spawn(async move { | ||||||
|  |         loop { | ||||||
|  |             let res = response_receiver.recv().await; | ||||||
|  |             match res { | ||||||
|  |                 None => break, | ||||||
|  |                 Some(res) => w | ||||||
|  |                     .write_all((serde_json::to_string(&res).unwrap() + "\n").as_bytes()) | ||||||
|  |                     .await | ||||||
|  |                     .unwrap(), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     tokio::spawn(async move { | ||||||
|  |         let mut reader = BufReader::new(r); | ||||||
|  |         loop { | ||||||
|  |             let mut line = String::new(); | ||||||
|  |             let res = reader.read_line(&mut line).await; | ||||||
|  |             match res { | ||||||
|  |                 Ok(size) => { | ||||||
|  |                     if size > 0 { | ||||||
|  |                         session.handle_request(serde_json::from_str(&line).unwrap()) | ||||||
|  |                     } else { | ||||||
|  |                         break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 Err(_) => break, | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[tokio::main] | ||||||
|  | pub async fn main() { | ||||||
|  |     let mut srv = JRPCServer::new(); | ||||||
|  |     srv.add_service(SimpleTestServiceHandler::new(Box::from( | ||||||
|  |         TestServiceImplementation {}, | ||||||
|  |     ))); | ||||||
|  |  | ||||||
|  |     let listener = tokio::net::TcpListener::bind("0.0.0.0:4321").await.unwrap(); | ||||||
|  |  | ||||||
|  |     loop { | ||||||
|  |         let (stream, address) = listener.accept().await.unwrap(); | ||||||
|  |         println!("Got connection from {}", address); | ||||||
|  |         handle_connection(stream, &srv); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -1,18 +1,31 @@ | |||||||
|  |  | ||||||
| import * as net from "net"; | import * as net from "net"; | ||||||
| import { Server, AddValueRequest, AddValueResponse } from "./out"; | import { Server, AddValueRequest, AddValueResponse, Test2 } from "./out"; | ||||||
| import * as readline from 'node:readline'; | import * as readline from 'node:readline'; | ||||||
|  |  | ||||||
|  |  | ||||||
| const server = new Server.ServiceProvider(); | const server = new Server.ServiceProvider(); | ||||||
|  |  | ||||||
|  | class SimpleTestService extends Server.SimpleTestService<undefined> { | ||||||
|  |    async GetTest(name: string, age: number, ctx: undefined): Promise<Test2> { | ||||||
|  |       return { | ||||||
|  |          name, | ||||||
|  |          age, | ||||||
|  |       }; | ||||||
|  |    } | ||||||
|  |    TestNot(ctx: undefined): void { | ||||||
|  |  | ||||||
|  |    } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestService extends Server.TestService<undefined> { | class TestService extends Server.TestService<undefined> { | ||||||
|    async AddValuesSingleParam( |    async AddValuesSingleParam( | ||||||
|       request: AddValueRequest, |       request: AddValueRequest, | ||||||
|       ctx: undefined |       ctx: undefined | ||||||
|    ): Promise<AddValueResponse> { |    ): Promise<AddValueResponse> { | ||||||
|       return { |       return { | ||||||
|          value: request.value1 + request!.value2, |          value: request.value1! + request!.value2!, | ||||||
|       }; |       }; | ||||||
|    } |    } | ||||||
|    async AddValuesMultipleParams( |    async AddValuesMultipleParams( | ||||||
| @ -41,6 +54,7 @@ class TestService extends Server.TestService<undefined> { | |||||||
| } | } | ||||||
|  |  | ||||||
| server.addService(new TestService()); | server.addService(new TestService()); | ||||||
|  | server.addService(new SimpleTestService()); | ||||||
|  |  | ||||||
| net.createServer((socket) => { | net.createServer((socket) => { | ||||||
|    socket.on("error", console.error); |    socket.on("error", console.error); | ||||||
| @ -57,6 +71,6 @@ net.createServer((socket) => { | |||||||
|       sess.onMessage(JSON.parse(line)); |       sess.onMessage(JSON.parse(line)); | ||||||
|    }) |    }) | ||||||
|    rl.on("error", console.error); |    rl.on("error", console.error); | ||||||
| }).listen(8859).on("listening", () => { | }).listen(4321).on("listening", () => { | ||||||
|    console.log("Is listening on :8859"); |    console.log("Is listening on :4321"); | ||||||
| }).on("error", console.error) | }).on("error", console.error) | ||||||
| @ -52,4 +52,23 @@ service TestService { | |||||||
|  |  | ||||||
|  |  | ||||||
|    FunctionWithArrayAsParamAndReturn(values1: float[], values2: float[]): float[]; |    FunctionWithArrayAsParamAndReturn(values1: float[], values2: float[]): float[]; | ||||||
|  |  | ||||||
|  |    FunctionWithKeywords(type: float, static: float, event: float): float; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type Test2 { | ||||||
|  |    name: string; | ||||||
|  |    age:  int; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type TestKeywords { | ||||||
|  |    type: float; | ||||||
|  |    static: float; | ||||||
|  |    event: float; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | service SimpleTestService { | ||||||
|  |    @Description("asdasdasd") | ||||||
|  |    GetTest(name: string, age: int): Test2; | ||||||
|  |    notification TestNot(); | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,5 +1,5 @@ | |||||||
| type TestAtom { | type TestAtom { | ||||||
|    val_number: float; |    val_number?: float; | ||||||
|    val_boolean: boolean; |    val_boolean?: boolean; | ||||||
|    val_string: string; |    val_string?: string; | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,14 +0,0 @@ | |||||||
| define rust_crate test; |  | ||||||
|  |  | ||||||
| type Test { |  | ||||||
|    name: string; |  | ||||||
|    age:  int; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| service TestService { |  | ||||||
|    @Description("asdasdasd") |  | ||||||
|    GetTest(name: string, age: int): Test; |  | ||||||
|    notification TestNot(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // { "jsonrpc": "2.0", "method": "TestService.GetTest", "params": [ "SomeName", 25 ], "id": "someid" } |  | ||||||
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|    "name": "@hibas123/jrpcgen", |    "name": "@hibas123/jrpcgen", | ||||||
|    "version": "1.1.6", |    "version": "1.2.17", | ||||||
|    "main": "lib/index.js", |    "main": "lib/index.js", | ||||||
|    "license": "MIT", |    "license": "MIT", | ||||||
|    "packageManager": "yarn@3.1.1", |    "packageManager": "yarn@3.1.1", | ||||||
|  | |||||||
| @ -4,13 +4,15 @@ import yargs from "yargs"; | |||||||
| import { hideBin } from "yargs/helpers"; | import { hideBin } from "yargs/helpers"; | ||||||
| import startCompile, { Target, Targets } from "./process"; | import startCompile, { Target, Targets } from "./process"; | ||||||
|  |  | ||||||
|  | const pkg = require("../package.json"); | ||||||
|  |  | ||||||
| import dbg from "debug"; | import dbg from "debug"; | ||||||
| const log = dbg("app"); | const log = dbg("app"); | ||||||
|  |  | ||||||
| dbg.disable(); | dbg.disable(); | ||||||
|  |  | ||||||
| yargs(hideBin(process.argv)) | yargs(hideBin(process.argv)) | ||||||
|    .version("1.0.0") |    .version(pkg.version) | ||||||
|    .command( |    .command( | ||||||
|       "compile <input>", |       "compile <input>", | ||||||
|       "Compile source", |       "Compile source", | ||||||
|  | |||||||
| @ -14,6 +14,7 @@ export interface TypeFieldDefinition { | |||||||
|    name: string; |    name: string; | ||||||
|    type: string; |    type: string; | ||||||
|    array: boolean; |    array: boolean; | ||||||
|  |    optional: boolean; | ||||||
|    map?: string; |    map?: string; | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -127,6 +128,7 @@ export default function get_ir(parsed: Parsed): IR { | |||||||
|                type: field.fieldtype, |                type: field.fieldtype, | ||||||
|                array: field.array, |                array: field.array, | ||||||
|                map: field.map, |                map: field.map, | ||||||
|  |                optional: field.optional, | ||||||
|             }; |             }; | ||||||
|          }); |          }); | ||||||
|          steps.push([ |          steps.push([ | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ export interface ImportStatement extends DefinitionNode { | |||||||
| export interface TypeFieldStatement extends DefinitionNode { | export interface TypeFieldStatement extends DefinitionNode { | ||||||
|    type: "type_field"; |    type: "type_field"; | ||||||
|    name: string; |    name: string; | ||||||
|  |    optional: boolean; | ||||||
|    fieldtype: string; |    fieldtype: string; | ||||||
|    array: boolean; |    array: boolean; | ||||||
|    map?: string; |    map?: string; | ||||||
| @ -114,8 +115,8 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|       return idx; |       return idx; | ||||||
|    }; |    }; | ||||||
|  |  | ||||||
|    const eatText = (): [string, number] => { |    const eatText = (allowKeyword?: boolean): [string, number] => { | ||||||
|       checkTypes("text"); |       checkTypes(...(allowKeyword ? ["text", "keyword"] : ["text"])); | ||||||
|       let val = currentToken.value; |       let val = currentToken.value; | ||||||
|       let idx = currentToken.startIdx; |       let idx = currentToken.startIdx; | ||||||
|       eatToken(); |       eatToken(); | ||||||
| @ -139,8 +140,7 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|    const checkTypes = (...types: string[]) => { |    const checkTypes = (...types: string[]) => { | ||||||
|       if (types.indexOf(currentToken.type) < 0) { |       if (types.indexOf(currentToken.type) < 0) { | ||||||
|          throw new ParserError( |          throw new ParserError( | ||||||
|             `Unexpected token value, expected ${types.join(" | ")}, received '${ |             `Unexpected token value, expected ${types.join(" | ")}, received '${currentToken.value | ||||||
|                currentToken.value |  | ||||||
|             }'`, |             }'`, | ||||||
|             currentToken |             currentToken | ||||||
|          ); |          ); | ||||||
| @ -170,6 +170,11 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|       const idx = currentToken.startIdx; |       const idx = currentToken.startIdx; | ||||||
|       let name = currentToken.value; |       let name = currentToken.value; | ||||||
|       eatToken(); |       eatToken(); | ||||||
|  |       let optional = false; | ||||||
|  |       if (currentToken.type === "questionmark") { | ||||||
|  |          eatToken("?"); | ||||||
|  |          optional = true; | ||||||
|  |       } | ||||||
|       eatToken(":"); |       eatToken(":"); | ||||||
|  |  | ||||||
|       let array = false; |       let array = false; | ||||||
| @ -198,6 +203,7 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|          array, |          array, | ||||||
|          map: mapKey, |          map: mapKey, | ||||||
|          location: { file, idx }, |          location: { file, idx }, | ||||||
|  |          optional | ||||||
|       }; |       }; | ||||||
|    }; |    }; | ||||||
|  |  | ||||||
| @ -303,9 +309,9 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|       eatToken("("); |       eatToken("("); | ||||||
|       let args: string[] = []; |       let args: string[] = []; | ||||||
|       let first = true; |       let first = true; | ||||||
|       while(currentToken.value !== ")") { |       while (currentToken.value !== ")") { | ||||||
|          if(first) { |          if (first) { | ||||||
|             first= false; |             first = false; | ||||||
|          } else { |          } else { | ||||||
|             eatToken(","); |             eatToken(","); | ||||||
|          } |          } | ||||||
| @ -345,7 +351,7 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|  |  | ||||||
|       if (currentToken.value !== ")") { |       if (currentToken.value !== ")") { | ||||||
|          while (true) { |          while (true) { | ||||||
|             const [name] = eatText(); |             const [name] = eatText(true); | ||||||
|             eatToken(":"); |             eatToken(":"); | ||||||
|             const [type] = eatText(); |             const [type] = eatText(); | ||||||
|             let array = false; |             let array = false; | ||||||
| @ -399,8 +405,8 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|       let functions: ServiceFunctionStatement[] = []; |       let functions: ServiceFunctionStatement[] = []; | ||||||
|  |  | ||||||
|       while (currentToken.type !== "curly_close") { |       while (currentToken.type !== "curly_close") { | ||||||
|          let decorators:Decorators = new Map; |          let decorators: Decorators = new Map; | ||||||
|          while(currentToken.type == "at") { |          while (currentToken.type == "at") { | ||||||
|             parseFunctionDecorator(decorators); |             parseFunctionDecorator(decorators); | ||||||
|          } |          } | ||||||
|  |  | ||||||
| @ -427,8 +433,8 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|  |  | ||||||
|       let [key] = eatText() |       let [key] = eatText() | ||||||
|       let value: string = undefined; |       let value: string = undefined; | ||||||
|       if(currentToken.type == "string") { |       if (currentToken.type == "string") { | ||||||
|          value = currentToken.value.slice(1,-1); |          value = currentToken.value.slice(1, -1); | ||||||
|          eatToken(); |          eatToken(); | ||||||
|       } else { |       } else { | ||||||
|          [value] = eatText() |          [value] = eatText() | ||||||
| @ -437,8 +443,8 @@ export default function parse(tokens: Token[], file: string): Parsed { | |||||||
|       eatToken(";"); |       eatToken(";"); | ||||||
|  |  | ||||||
|       return { |       return { | ||||||
|          type :"define", |          type: "define", | ||||||
|          location: {file, idx}, |          location: { file, idx }, | ||||||
|          key, |          key, | ||||||
|          value |          value | ||||||
|       } |       } | ||||||
|  | |||||||
| @ -18,7 +18,7 @@ import { ZIGTarget } from "./targets/zig"; | |||||||
| import { DartTarget } from "./targets/dart"; | import { DartTarget } from "./targets/dart"; | ||||||
| import { URL } from "url"; | import { URL } from "url"; | ||||||
|  |  | ||||||
| class CatchedError extends Error {} | class CatchedError extends Error { } | ||||||
|  |  | ||||||
| const log = dbg("app"); | const log = dbg("app"); | ||||||
|  |  | ||||||
| @ -46,15 +46,22 @@ function indexToLineAndCol(src: string, index: number) { | |||||||
|    return { line, col }; |    return { line, col }; | ||||||
| } | } | ||||||
|  |  | ||||||
| function resolve(base: string, ...parts: string[]) { | function resolve(base: string, sub?: string) { | ||||||
|    if (base.startsWith("http://") || base.startsWith("https://")) { |    if (sub && (sub.startsWith("http://") || sub.startsWith("https://"))) { | ||||||
|  |       let u = new URL(sub); | ||||||
|  |       return u.href; | ||||||
|  |    } else if (base.startsWith("http://") || base.startsWith("https://")) { | ||||||
|       let u = new URL(base); |       let u = new URL(base); | ||||||
|       for (const part of parts) { |       if (sub) { | ||||||
|          u = new URL(part, u); |          if (!sub.endsWith(".jrpc")) { | ||||||
|  |             sub += ".jrpc"; | ||||||
|  |          } | ||||||
|  |          u = new URL(sub, u); | ||||||
|       } |       } | ||||||
|       return u.href; |       return u.href; | ||||||
|    } else { |    } else { | ||||||
|       return Path.resolve(base, ...parts); |       if (!sub) return Path.resolve(base); | ||||||
|  |       else return Path.resolve(Path.dirname(base), sub + ".jrpc"); | ||||||
|    } |    } | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -151,13 +158,7 @@ async function processFile( | |||||||
|       let resolved: Parsed = []; |       let resolved: Parsed = []; | ||||||
|       for (const statement of parsed) { |       for (const statement of parsed) { | ||||||
|          if (statement.type == "import") { |          if (statement.type == "import") { | ||||||
|             let res: string; |             let res = resolve(file, statement.path); | ||||||
|             if (file.startsWith("http://") || file.startsWith("https://")) { |  | ||||||
|                res = resolve(file, statement.path + ".jrpc"); |  | ||||||
|             } else { |  | ||||||
|                const base = Path.dirname(file); |  | ||||||
|                res = resolve(base, statement.path + ".jrpc"); |  | ||||||
|             } |  | ||||||
|             resolved.push(...((await processFile(ctx, res)) || [])); |             resolved.push(...((await processFile(ctx, res)) || [])); | ||||||
|          } else { |          } else { | ||||||
|             resolved.push(statement); |             resolved.push(statement); | ||||||
|  | |||||||
| @ -8,6 +8,7 @@ import { | |||||||
|  |  | ||||||
| import { CompileTarget } from "../compile"; | import { CompileTarget } from "../compile"; | ||||||
| import { LineAppender } from "../utils"; | import { LineAppender } from "../utils"; | ||||||
|  | import chalk from "chalk"; | ||||||
|  |  | ||||||
| const conversion = { | const conversion = { | ||||||
|    boolean: "bool", |    boolean: "bool", | ||||||
| @ -22,6 +23,16 @@ function toCSharpType(type: string): string { | |||||||
|    return (conversion as any)[type] || type; |    return (conversion as any)[type] || type; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // TODO: Add other keywords as well! | ||||||
|  | const keywords = new Set(["event", "internal", "public", "private", "static"]); | ||||||
|  |  | ||||||
|  | const fixKeywordName = (name: string) => { | ||||||
|  |    if (keywords.has(name)) { | ||||||
|  |       return `${name}_`; | ||||||
|  |    } | ||||||
|  |    return name; | ||||||
|  | } | ||||||
|  |  | ||||||
| export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | ||||||
|    name: string = "c#"; |    name: string = "c#"; | ||||||
|  |  | ||||||
| @ -61,11 +72,20 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|       a(0, ``); |       a(0, ``); | ||||||
|       a(0, `public class ${definition.name} {`); |       a(0, `public class ${definition.name} {`); | ||||||
|       for (const field of definition.fields) { |       for (const field of definition.fields) { | ||||||
|  |          let fn = field.name; | ||||||
|  |          if (keywords.has(field.name)) { | ||||||
|  |             console.log( | ||||||
|  |                chalk.yellow("[RUST] WARNING:"), | ||||||
|  |                `Field name '${fn}' is not allowed in C#. Renaming to '${fn}_'` | ||||||
|  |             ); | ||||||
|  |             fn = fixKeywordName(fn); | ||||||
|  |             a(1, `[JsonPropertyName("${fn}")]`); | ||||||
|  |          } | ||||||
|  |  | ||||||
|          if (field.array) { |          if (field.array) { | ||||||
|             a( |             a( | ||||||
|                1, |                1, | ||||||
|                `public IList<${toCSharpType(field.type)}>? ${ |                `public IList<${toCSharpType(field.type)}>? ${fn | ||||||
|                   field.name |  | ||||||
|                } { get; set; }` |                } { get; set; }` | ||||||
|             ); |             ); | ||||||
|          } else if (field.map) { |          } else if (field.map) { | ||||||
| @ -73,12 +93,12 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|                1, |                1, | ||||||
|                `public Dictionary<${toCSharpType(field.map)}, ${toCSharpType( |                `public Dictionary<${toCSharpType(field.map)}, ${toCSharpType( | ||||||
|                   field.type |                   field.type | ||||||
|                )}>? ${field.name} { get; set; }` |                )}>? ${fn} { get; set; }` | ||||||
|             ); |             ); | ||||||
|          } else { |          } else { | ||||||
|             a( |             a( | ||||||
|                1, |                1, | ||||||
|                `public ${toCSharpType(field.type)}? ${field.name} { get; set; }` |                `public ${toCSharpType(field.type)}? ${fn} { get; set; }` | ||||||
|             ); |             ); | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
| @ -127,10 +147,12 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|       for (const fnc of definition.functions) { |       for (const fnc of definition.functions) { | ||||||
|          let params = fnc.inputs |          let params = fnc.inputs | ||||||
|             .map((inp) => { |             .map((inp) => { | ||||||
|  |                let name = fixKeywordName(inp.name); | ||||||
|  |  | ||||||
|                if (inp.array) { |                if (inp.array) { | ||||||
|                   return `List<${toCSharpType(inp.type)}> ${inp.name}`; |                   return `List<${toCSharpType(inp.type)}> ${name}`; | ||||||
|                } else { |                } else { | ||||||
|                   return `${toCSharpType(inp.type)} ${inp.name}`; |                   return `${toCSharpType(inp.type)} ${name}`; | ||||||
|                } |                } | ||||||
|             }) |             }) | ||||||
|             .join(", "); |             .join(", "); | ||||||
| @ -139,7 +161,7 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|             a( |             a( | ||||||
|                2, |                2, | ||||||
|                `var param = new JsonArray(${fnc.inputs |                `var param = new JsonArray(${fnc.inputs | ||||||
|                   .map((e) => `JsonSerializer.SerializeToNode(${e.name})`) |                   .map((e) => `JsonSerializer.SerializeToNode(${fixKeywordName(e.name)})`) | ||||||
|                   .join(", ")});` |                   .join(", ")});` | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
| @ -219,10 +241,11 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|       for (const fnc of definition.functions) { |       for (const fnc of definition.functions) { | ||||||
|          let params = [ |          let params = [ | ||||||
|             ...fnc.inputs.map((inp) => { |             ...fnc.inputs.map((inp) => { | ||||||
|  |                let name = fixKeywordName(inp.name) | ||||||
|                if (inp.array) { |                if (inp.array) { | ||||||
|                   return `List<${toCSharpType(inp.type)}> ${inp.name}`; |                   return `List<${toCSharpType(inp.type)}> ${name}`; | ||||||
|                } else { |                } else { | ||||||
|                   return `${toCSharpType(inp.type)} ${inp.name}`; |                   return `${toCSharpType(inp.type)} ${name}`; | ||||||
|                } |                } | ||||||
|             }), |             }), | ||||||
|             "TContext ctx", |             "TContext ctx", | ||||||
| @ -272,15 +295,15 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|          a( |          a( | ||||||
|             4, |             4, | ||||||
|             pref + |             pref + | ||||||
|                `this.${fnc.name}(${[ |             `this.${fnc.name}(${[ | ||||||
|                   ...fnc.inputs.map((inp, idx) => { |                ...fnc.inputs.map((inp, idx) => { | ||||||
|                      let type = inp.array |                   let type = inp.array | ||||||
|                         ? `List<${toCSharpType(inp.type)}>` |                      ? `List<${toCSharpType(inp.type)}>` | ||||||
|                         : `${toCSharpType(inp.type)}`; |                      : `${toCSharpType(inp.type)}`; | ||||||
|                      return `param[${idx}]!.Deserialize<${type}>()`; |                   return `param[${idx}]!.Deserialize<${type}>()`; | ||||||
|                   }), |                }), | ||||||
|                   "context", |                "context", | ||||||
|                ].join(", ")});` |             ].join(", ")});` | ||||||
|          ); |          ); | ||||||
|  |  | ||||||
|          if (fnc.return && fnc.return.type != "void") { |          if (fnc.return && fnc.return.type != "void") { | ||||||
| @ -314,5 +337,5 @@ export class CSharpTarget extends CompileTarget<{ csharp_namespace: string }> { | |||||||
|       this.generateServiceServer(definition); |       this.generateServiceServer(definition); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    finalize(steps: Step[]): void {} |    finalize(steps: Step[]): void { } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| import chalk from "chalk"; | import chalk from "chalk"; | ||||||
| import { CompileTarget } from "../compile"; | import { CompileTarget } from "../compile"; | ||||||
| import { TypeDefinition, EnumDefinition, ServiceDefinition, Step } from "../ir"; | import { TypeDefinition, EnumDefinition, ServiceDefinition, Step, IR } from "../ir"; | ||||||
| import { lineAppender, LineAppender } from "../utils"; | import { lineAppender, LineAppender } from "../utils"; | ||||||
|  |  | ||||||
| const conversion = { | const conversion = { | ||||||
| @ -23,6 +23,16 @@ function toSnake(input: string) { | |||||||
|    ); |    ); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // TODO: Add other keywords as well! | ||||||
|  | const keywords = new Set(["type", "static"]); | ||||||
|  |  | ||||||
|  | const fixKeywordName = (name: string) => { | ||||||
|  |    if (keywords.has(name)) { | ||||||
|  |       return `${name}_`; | ||||||
|  |    } | ||||||
|  |    return name; | ||||||
|  | } | ||||||
|  |  | ||||||
| export class RustTarget extends CompileTarget<{ rust_crate: string }> { | export class RustTarget extends CompileTarget<{ rust_crate: string }> { | ||||||
|    name: string = "rust"; |    name: string = "rust"; | ||||||
|  |  | ||||||
| @ -71,17 +81,36 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|       a(0, `#[derive(Clone, Debug, Serialize, Deserialize)]`); |       a(0, `#[derive(Clone, Debug, Serialize, Deserialize)]`); | ||||||
|       a(0, `pub struct ${definition.name} {`); |       a(0, `pub struct ${definition.name} {`); | ||||||
|       for (const field of definition.fields) { |       for (const field of definition.fields) { | ||||||
|  |          a(1, `#[allow(non_snake_case)]`); | ||||||
|  |  | ||||||
|  |          let fn = `pub ${field.name}:`; | ||||||
|  |          if (keywords.has(field.name)) { | ||||||
|  |             // TODO: Add other keywords as well! | ||||||
|  |             console.log( | ||||||
|  |                chalk.yellow("[RUST] WARNING:"), | ||||||
|  |                `Field name '${field.name}' is not allowed in Rust. Renaming to '${field.name}_'` | ||||||
|  |             ); | ||||||
|  |             fn = `pub ${fixKeywordName(field.name)}:`; | ||||||
|  |             a(1, `#[serde(rename = "${field.name}")]`); | ||||||
|  |          } | ||||||
|  |          let opts = ""; | ||||||
|  |          let opte = ""; | ||||||
|  |          if (field.optional) { | ||||||
|  |             opts = "Option<"; | ||||||
|  |             opte = ">"; | ||||||
|  |          } | ||||||
|  |  | ||||||
|          if (field.array) { |          if (field.array) { | ||||||
|             a(1, `pub ${field.name}: Vec<${toRustType(field.type)}>,`); |             a(1, `${fn} ${opts}Vec<${toRustType(field.type)}>${opte},`); | ||||||
|          } else if (field.map) { |          } else if (field.map) { | ||||||
|             a( |             a( | ||||||
|                1, |                1, | ||||||
|                `pub ${field.name}: HashMap<${toRustType( |                `${fn} ${opts}HashMap<${toRustType( | ||||||
|                   field.map |                   field.map | ||||||
|                )}, ${toRustType(field.type)}>,` |                )}, ${toRustType(field.type)}>${opte},` | ||||||
|             ); |             ); | ||||||
|          } else { |          } else { | ||||||
|             a(1, `pub ${field.name}: ${toRustType(field.type)},`); |             a(1, `${fn} ${opts}${toRustType(field.type)}${opte},`); | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
|       a(0, `}`); |       a(0, `}`); | ||||||
| @ -101,7 +130,7 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|       a(0, `#[repr(i64)]`); |       a(0, `#[repr(i64)]`); | ||||||
|       a( |       a( | ||||||
|          0, |          0, | ||||||
|          "#[derive(Clone, Copy, Debug, Eq, PartialEq, IntEnum, Deserialize, Serialize)]" |          "#[derive(Clone, Copy, Debug, Eq, PartialEq, IntEnum)]" | ||||||
|       ); |       ); | ||||||
|       a(0, `pub enum ${definition.name} {`); |       a(0, `pub enum ${definition.name} {`); | ||||||
|       for (const field of definition.values) { |       for (const field of definition.values) { | ||||||
| @ -122,7 +151,7 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|  |  | ||||||
|       this.addDependencies(a, definition); |       this.addDependencies(a, definition); | ||||||
|  |  | ||||||
|       a(0, `use crate::base_lib::{JRPCClient,JRPCRequest,Result};`); |       a(0, `use crate::base_lib::{JRPCClient, JRPCRequest, Result};`); | ||||||
|       a(0, `use serde_json::{json};`); |       a(0, `use serde_json::{json};`); | ||||||
|       a(0, ``); |       a(0, ``); | ||||||
|       a(0, `pub struct ${definition.name}{`); |       a(0, `pub struct ${definition.name}{`); | ||||||
| @ -137,35 +166,40 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|       a(0, ``); |       a(0, ``); | ||||||
|       for (const fnc of definition.functions) { |       for (const fnc of definition.functions) { | ||||||
|          let params = fnc.inputs |          let params = fnc.inputs | ||||||
|             .map((i) => i.name + ": " + typeToRust(i.type, i.array)) |             .map((i) => fixKeywordName(i.name) + ": " + typeToRust(i.type, i.array)) | ||||||
|             .join(", "); |             .join(", "); | ||||||
|          let ret = fnc.return |          let ret = fnc.return | ||||||
|             ? typeToRust(fnc.return.type, fnc.return.array) |             ? typeToRust(fnc.return.type, fnc.return.array) | ||||||
|             : "()"; |             : "()"; | ||||||
|          a(1, `pub fn ${fnc.name}(&self, ${params}) -> Result<${ret}> {`); |          a(1, `#[allow(non_snake_case)]`); | ||||||
|  |          a(1, `pub async fn ${fnc.name}(&self, ${params}) -> Result<${ret}> {`); | ||||||
|          a(2, `let l_req = JRPCRequest {`); |          a(2, `let l_req = JRPCRequest {`); | ||||||
|          a(3, `jsonrpc: "2.0".to_owned(),`); |          a(3, `jsonrpc: "2.0".to_owned(),`); | ||||||
|          a(3, `id: None, // 'id' will be set by the send_request function`); |          a(3, `id: None, // 'id' will be set by the send_request function`); | ||||||
|          a(3, `method: "${definition.name}.${fnc.name}".to_owned(),`); |          a(3, `method: "${definition.name}.${fnc.name}".to_owned(),`); | ||||||
|          a(3, `params: json!([${fnc.inputs.map((e) => e.name)}])`); |          a(3, `params: json!([${fnc.inputs.map((e) => fixKeywordName(e.name))}])`); | ||||||
|          a(2, `};`); |          a(2, `};`); | ||||||
|          a(2, ``); |          a(2, ``); | ||||||
|          if (fnc.return) { |          if (fnc.return) { | ||||||
|             a(2, `let l_res = self.client.send_request(l_req);`); |             a(2, `let l_res = self.client.send_request(l_req).await;`); | ||||||
|             a(2, `if let Err(e) = l_res {`); |             a(2, `match l_res {`); | ||||||
|             a(3, `return Err(e);`); |             a(3, `Err(e) => return Err(e),`); | ||||||
|             a(2, `} else if let Ok(o) = l_res {`); |  | ||||||
|             if (fnc.return.type == "void") { |             if (fnc.return.type == "void") { | ||||||
|                a(3, `return ();`); |                a(3, `Ok(_) => {`); | ||||||
|  |                a(4, `return Ok(());`); | ||||||
|  |                a(3, `}`); | ||||||
|             } else { |             } else { | ||||||
|  |                a(3, `Ok(o) => {`); | ||||||
|                a( |                a( | ||||||
|                   3, |                   4, | ||||||
|                   `return serde_json::from_value(o).map_err(|e| Box::from(e));` |                   `return serde_json::from_value(o).map_err(|e| Box::from(e));` | ||||||
|                ); |                ); | ||||||
|  |                a(3, `}`); | ||||||
|             } |             } | ||||||
|             a(2, `} else { panic!("What else cases could there be?"); }`); |             a(2, `}`); | ||||||
|  |  | ||||||
|          } else { |          } else { | ||||||
|             a(2, `self.client.send_notification(l_req);`); |             a(2, `self.client.send_notification(l_req).await;`); | ||||||
|             a(2, `return Ok(());`); |             a(2, `return Ok(());`); | ||||||
|          } |          } | ||||||
|          a(1, `}`); |          a(1, `}`); | ||||||
| @ -180,61 +214,65 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|       const { a, getResult } = LineAppender(); |       const { a, getResult } = LineAppender(); | ||||||
|  |  | ||||||
|       this.addDependencies(a, definition); |       this.addDependencies(a, definition); | ||||||
|       a(0, `use crate::base_lib::{JRPCServiceHandler,JRPCRequest,Result};`); |       a(0, `use crate::base_lib::{JRPCServerService, JRPCRequest, Result};`); | ||||||
|       a(0, `use serde_json::{Value};`); |       a(0, `use serde_json::{Value};`); | ||||||
|  |       a(0, `use std::sync::Arc;`); | ||||||
|  |       a(0, `use async_trait::async_trait;`); | ||||||
|  |  | ||||||
|       const typeToRust = (type: string, array: boolean) => { |       const typeToRust = (type: string, array: boolean) => { | ||||||
|          let rt = toRustType(type); |          let rt = toRustType(type); | ||||||
|          return array ? `Vec<${rt}>` : rt; |          return array ? `Vec<${rt}>` : rt; | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       const CTX_TYPE = "'static + Sync + Send + Copy"; |       a(0, `#[async_trait]`); | ||||||
|  |       a(0, `pub trait ${definition.name} {`); | ||||||
|       a(0, ``); |  | ||||||
|       a(0, `pub trait ${definition.name}<C: ${CTX_TYPE}> {`); |  | ||||||
|       for (const fnc of definition.functions) { |       for (const fnc of definition.functions) { | ||||||
|          let params = |          let params = | ||||||
|             fnc.inputs.length > 0 |             fnc.inputs.length > 0 | ||||||
|                ? fnc.inputs |                ? fnc.inputs | ||||||
|                     .map((i) => i.name + ": " + typeToRust(i.type, i.array)) |                   .map((i) => fixKeywordName(i.name) + ": " + typeToRust(i.type, i.array)) | ||||||
|                     .join(", ") + ", " |                   .join(", ") | ||||||
|                : ""; |                : ""; | ||||||
|          let ret = fnc.return |          let ret = fnc.return | ||||||
|             ? typeToRust(fnc.return.type, fnc.return.array) |             ? typeToRust(fnc.return.type, fnc.return.array) | ||||||
|             : "()"; |             : "()"; | ||||||
|          a(1, `fn ${fnc.name}(&self, ${params}context: &C) -> Result<${ret}>;`); |          a(1, `#[allow(non_snake_case)]`); | ||||||
|  |          a(1, `async fn ${fnc.name}(&self, ${params}) -> Result<${ret}>;`); | ||||||
|       } |       } | ||||||
|       a(0, `}`); |       a(0, `}`); | ||||||
|  |  | ||||||
|       a(0, ``); |       a(0, ``); | ||||||
|       a(0, `pub struct ${definition.name}Handler<C: ${CTX_TYPE}> {`); |       a(0, `pub struct ${definition.name}Handler {`); | ||||||
|       a(1, `implementation: Box<dyn ${definition.name}<C> + Sync + Send>,`); |       a(1, `implementation: Box<dyn ${definition.name} + Sync + Send + 'static>,`); | ||||||
|       a(0, `}`); |       a(0, `}`); | ||||||
|       a(0, ``); |       a(0, ``); | ||||||
|       a(0, `impl<C: ${CTX_TYPE}> ${definition.name}Handler<C> {`); |       a(0, `impl ${definition.name}Handler {`); | ||||||
|  |       //TODO: Maybe add a new definition like, pub fn new2<T>(implementation: T) where T: ${definition.name} + Sync + Send + 'static {} | ||||||
|       a( |       a( | ||||||
|          1, |          1, | ||||||
|          `pub fn new(implementation: Box<dyn ${definition.name}<C> + Sync + Send>) -> Box<Self> {` |          `pub fn new(implementation: Box<dyn ${definition.name} + Sync + Send + 'static>) -> Arc<Self> {` | ||||||
|       ); |       ); | ||||||
|       a(2, `return Box::from(Self { implementation });`); |       a(2, `return Arc::from(Self { implementation });`); | ||||||
|       a(1, `}`); |       a(1, `}`); | ||||||
|       a(0, `}`); |       a(0, `}`); | ||||||
|  |  | ||||||
|       a(0, ``); |       a(0, ``); | ||||||
|  |  | ||||||
|  |       a(0, `#[async_trait]`); | ||||||
|       a( |       a( | ||||||
|          0, |          0, | ||||||
|          `impl<C: ${CTX_TYPE}> JRPCServiceHandler<C> for ${definition.name}Handler<C> {` |          `impl JRPCServerService for ${definition.name}Handler {` | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       a(1, `fn get_name(&self) -> String { "${definition.name}".to_owned() }`); |       a(1, `fn get_id(&self) -> String { "${definition.name}".to_owned() }`); | ||||||
|       a(0, ``); |       a(0, ``); | ||||||
|  |  | ||||||
|  |       a(1, `#[allow(non_snake_case)]`); | ||||||
|       a( |       a( | ||||||
|          1, |          1, | ||||||
|          `fn on_message(&self, msg: JRPCRequest, function: String, ctx: &C) -> Result<(bool, Value)> {` |          `async fn handle(&self, msg: &JRPCRequest, function: &str) -> Result<(bool, Value)> {` | ||||||
|       ); |       ); | ||||||
|       a(2, `match function.as_str() {`); |       a(2, `match function {`); | ||||||
|       for (const fnc of definition.functions) { |       for (const fnc of definition.functions) { | ||||||
|          a(3, `"${fnc.name}" => {`); |          a(3, `"${fnc.name}" => {`); | ||||||
|          a(4, `if msg.params.is_array() {`); |          a(4, `if msg.params.is_array() {`); | ||||||
| @ -250,10 +288,10 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|             a(6, `serde_json::from_value(arr[${i}].clone())`); |             a(6, `serde_json::from_value(arr[${i}].clone())`); | ||||||
|             a( |             a( | ||||||
|                7, |                7, | ||||||
|                `.map_err(|_| "Parameter for field '${inp.name}' should be of type '${inp.type}'!")?,` //TODO: Array |                `.map_err(|_| "Parameter for field '${inp.name}' should be of type '${inp.type}'!")?${i == fnc.inputs.length - 1 ? "" : ","}` //TODO: Array | ||||||
|             ); |             ); | ||||||
|          } |          } | ||||||
|          a(5, `ctx)?;`); |          a(5, `).await?;`); | ||||||
|          if (fnc.return) { |          if (fnc.return) { | ||||||
|             a(5, `return Ok((true, serde_json::to_value(res)?));`); |             a(5, `return Ok((true, serde_json::to_value(res)?));`); | ||||||
|          } else { |          } else { | ||||||
| @ -296,7 +334,7 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|       const as = ls.a; |       const as = ls.a; | ||||||
|  |  | ||||||
|       a(0, `pub mod base_lib;`); |       a(0, `pub mod base_lib;`); | ||||||
|       a(0, `pub use base_lib::{JRPCServer,Result};`); |       a(0, `pub use base_lib::{JRPCServer, JRPCClient, Result};`); | ||||||
|  |  | ||||||
|       for (const [typ, def] of steps) { |       for (const [typ, def] of steps) { | ||||||
|          if (typ == "type" || typ == "enum") { |          if (typ == "type" || typ == "enum") { | ||||||
| @ -306,19 +344,18 @@ export class RustTarget extends CompileTarget<{ rust_crate: string }> { | |||||||
|             as(0, `mod ${toSnake(def.name)};`); |             as(0, `mod ${toSnake(def.name)};`); | ||||||
|             as( |             as( | ||||||
|                0, |                0, | ||||||
|                `pub use ${toSnake(def.name)}::{${def.name}, ${ |                `pub use ${toSnake(def.name)}::{${def.name}, ${def.name | ||||||
|                   def.name |  | ||||||
|                }Handler};` |                }Handler};` | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|             ac(0, `mod ${toSnake(def.name)};`); |             ac(0, `mod ${toSnake(def.name)};`); | ||||||
|             ac(0, `pub use ${toSnake(def.name)}::${def.name};`); |             ac(0, `pub use ${toSnake(def.name)}::${def.name};`); | ||||||
|  |  | ||||||
|             a(0, `pub mod server;`); |  | ||||||
|             a(0, `pub mod client;`); |  | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       a(0, `pub mod server;`); | ||||||
|  |       a(0, `pub mod client;`); | ||||||
|  |  | ||||||
|       this.writeFile(`src/lib.rs`, getResult()); |       this.writeFile(`src/lib.rs`, getResult()); | ||||||
|       this.writeFile(`src/server/mod.rs`, ls.getResult()); |       this.writeFile(`src/server/mod.rs`, ls.getResult()); | ||||||
|       this.writeFile(`src/client/mod.rs`, lc.getResult()); |       this.writeFile(`src/client/mod.rs`, lc.getResult()); | ||||||
|  | |||||||
| @ -33,9 +33,8 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|    } |    } | ||||||
|  |  | ||||||
|    private generateImport(imports: string, path: string) { |    private generateImport(imports: string, path: string) { | ||||||
|       return `import ${imports} from "${ |       return `import ${imports} from "${path + (this.flavour === "esm" ? ".js" : "") | ||||||
|          path + (this.flavour === "esm" ? ".ts" : "") |          }";\n`; | ||||||
|       }";\n`; |  | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    private generateImports( |    private generateImports( | ||||||
| @ -89,7 +88,7 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|             } else { |             } else { | ||||||
|                type = toJSType(field.type); |                type = toJSType(field.type); | ||||||
|             } |             } | ||||||
|             return `${field.name}?: ${type}; `; |             return `${field.name}${field.optional ? "?" : ""}: ${type}; `; | ||||||
|          }) |          }) | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
| @ -125,10 +124,18 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|          ); |          ); | ||||||
|          a(1, `let res = new ${def.name}() as any;`); |          a(1, `let res = new ${def.name}() as any;`); | ||||||
|          def.fields.forEach((field) => { |          def.fields.forEach((field) => { | ||||||
|             a( |             if (field.optional) { | ||||||
|                1, |                a( | ||||||
|                `if(data["${field.name}"] !== null && data["${field.name}"] !== undefined) {` |                   1, | ||||||
|             ); |                   `if(data["${field.name}"] !== null && data["${field.name}"] !== undefined) {` | ||||||
|  |                ); | ||||||
|  |             } else { | ||||||
|  |                a( | ||||||
|  |                   1, | ||||||
|  |                   `if(data["${field.name}"] === null || data["${field.name}"] === undefined) throw new VerificationError("${def.name}", "${field.name}", data["${field.name}"]);` | ||||||
|  |                ); | ||||||
|  |                a(1, `else {`); | ||||||
|  |             } | ||||||
|             if (field.array) { |             if (field.array) { | ||||||
|                a( |                a( | ||||||
|                   2, |                   2, | ||||||
| @ -200,12 +207,12 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|       this.writeFormattedFile( |       this.writeFormattedFile( | ||||||
|          "service_client.ts", |          "service_client.ts", | ||||||
|          this.generateImport( |          this.generateImport( | ||||||
|             "{ RequestObject, ResponseObject, ErrorCodes, Logging }", |             "{ type RequestObject, type ResponseObject, ErrorCodes, Logging }", | ||||||
|             "./service_base" |             "./service_base" | ||||||
|          ) + |          ) + | ||||||
|             this.generateImport(" { VerificationError }", "./ts_base") + |          this.generateImport(" { VerificationError }", "./ts_base") + | ||||||
|             "\n\n" + |          "\n\n" + | ||||||
|             this.getTemplate("ts_service_client.ts") |          this.getTemplate("ts_service_client.ts") | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       const { a, getResult } = LineAppender(); |       const { a, getResult } = LineAppender(); | ||||||
| @ -297,12 +304,12 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|       this.writeFormattedFile( |       this.writeFormattedFile( | ||||||
|          "service_server.ts", |          "service_server.ts", | ||||||
|          this.generateImport( |          this.generateImport( | ||||||
|             "{ RequestObject, ResponseObject, ErrorCodes, Logging }", |             "{ type RequestObject, type ResponseObject, ErrorCodes, Logging }", | ||||||
|             "./service_base" |             "./service_base" | ||||||
|          ) + |          ) + | ||||||
|             this.generateImport(" { VerificationError }", "./ts_base") + |          this.generateImport(" { VerificationError }", "./ts_base") + | ||||||
|             "\n\n" + |          "\n\n" + | ||||||
|             this.getTemplate("ts_service_server.ts") |          this.getTemplate("ts_service_server.ts") | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       this.generateImports(a, def); |       this.generateImports(a, def); | ||||||
| @ -335,9 +342,8 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|             `ctx: T`, |             `ctx: T`, | ||||||
|          ].join(", "); |          ].join(", "); | ||||||
|          const retVal = fnc.return |          const retVal = fnc.return | ||||||
|             ? `Promise<${ |             ? `Promise<${toJSType(fnc.return.type) + (fnc.return.array ? "[]" : "") | ||||||
|                  toJSType(fnc.return.type) + (fnc.return.array ? "[]" : "") |             }>` | ||||||
|               }>` |  | ||||||
|             : `void`; |             : `void`; | ||||||
|          a(1, `abstract ${fnc.name}(${params}): ${retVal};`); |          a(1, `abstract ${fnc.name}(${params}): ${retVal};`); | ||||||
|  |  | ||||||
| @ -380,13 +386,12 @@ export class TypescriptTarget extends CompileTarget { | |||||||
|          a( |          a( | ||||||
|             2, |             2, | ||||||
|             `return this.${fnc.name}.call(this, ...p)` + //TODO: Refactor. This line is way to compicated for anyone to understand, including me |             `return this.${fnc.name}.call(this, ...p)` + //TODO: Refactor. This line is way to compicated for anyone to understand, including me | ||||||
|                (fnc.return |             (fnc.return | ||||||
|                   ? `.then(${ |                ? `.then(${fnc.return?.array | ||||||
|                        fnc.return?.array |                   ? `res => res.map(e => apply_${fnc.return.type}(e))` | ||||||
|                           ? `res => res.map(e => apply_${fnc.return.type}(e))` |                   : `res => apply_${fnc.return.type}(res)` | ||||||
|                           : `res => apply_${fnc.return.type}(res)` |                });` | ||||||
|                     });` |                : "") | ||||||
|                   : "") |  | ||||||
|          ); |          ); | ||||||
|          a(1, `}`); |          a(1, `}`); | ||||||
|          a(0, ``); |          a(0, ``); | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								templates/Rust/.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								templates/Rust/.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | [*.rs] | ||||||
|  | indent_size = 4 | ||||||
| @ -6,10 +6,10 @@ edition = "2021" | |||||||
| # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| int-enum = "0.4.0" | int-enum = { version ="0.5.0", features = ["serde", "convert"] } | ||||||
| serde = { version = "1.0.136", features = ["derive"] } | serde = { version = "1.0.147", features = ["derive"] } | ||||||
| serde_json = "1.0.79" | serde_json = "1.0.88" | ||||||
| threadpool = "1.8.1" |  | ||||||
| nanoid = "0.4.0" | nanoid = "0.4.0" | ||||||
|  | tokio = { version = "1.22.0", features = ["full"] } | ||||||
|  | log = "0.4.17" | ||||||
|  | async-trait = "0.1.59" | ||||||
|  | |||||||
| @ -1,286 +1,226 @@ | |||||||
|  | use log::{info, trace, warn}; | ||||||
| use nanoid::nanoid; | use nanoid::nanoid; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serde_json::Value; | use serde_json::Value; | ||||||
| use std::boxed::Box; | use std::boxed::Box; | ||||||
| use std::collections::HashMap; |  | ||||||
| use std::error::Error; | use std::error::Error; | ||||||
| use std::marker::PhantomData; |  | ||||||
| use std::marker::Send; | use std::marker::Send; | ||||||
| use std::sync::mpsc::{Receiver, Sender}; | use std::{collections::HashMap, sync::Arc}; | ||||||
| use std::sync::{Arc, Mutex}; | use tokio::sync::{mpsc::Sender, Mutex}; | ||||||
| use threadpool::ThreadPool; |  | ||||||
|  |  | ||||||
| pub type Result<T> = std::result::Result<T, Box<dyn Error>>; | pub type Result<T> = std::result::Result<T, Box<dyn Error + Send + Sync>>; | ||||||
|  |  | ||||||
| // TODO: Check what happens when error code is not included | #[derive(Clone, Debug, Serialize, Deserialize)] | ||||||
| // #[repr(i64)] |  | ||||||
| // #[derive(Clone, Copy, Debug, Eq, PartialEq, IntEnum, Deserialize, Serialize)] |  | ||||||
| // pub enum ErrorCodes { |  | ||||||
| //    ParseError = -32700, |  | ||||||
| //    InvalidRequest = -32600, |  | ||||||
| //    MethodNotFound = -32601, |  | ||||||
| //    InvalidParams = -32602, |  | ||||||
| //    InternalError = -32603, |  | ||||||
| // } |  | ||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize)] |  | ||||||
| pub struct JRPCRequest { | pub struct JRPCRequest { | ||||||
|    pub jsonrpc: String, |     pub jsonrpc: String, | ||||||
|    pub id: Option<String>, |     pub id: Option<String>, | ||||||
|    pub method: String, |     pub method: String, | ||||||
|    pub params: Value, |     pub params: Value, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize)] | impl JRPCRequest { | ||||||
|  |     pub fn new_request(method: String, params: Value) -> JRPCRequest { | ||||||
|  |         JRPCRequest { | ||||||
|  |             jsonrpc: "2.0".to_string(), | ||||||
|  |             id: None, | ||||||
|  |             method, | ||||||
|  |             params, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Clone, Debug, Serialize, Deserialize)] | ||||||
| pub struct JRPCError { | pub struct JRPCError { | ||||||
|    pub code: i64, |     pub code: i64, | ||||||
|    pub message: String, |     pub message: String, | ||||||
|    pub data: Value, |     pub data: Value, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize)] | #[derive(Clone, Debug, Serialize, Deserialize)] | ||||||
| pub struct JRPCResult { | pub struct JRPCResult { | ||||||
|    pub jsonrpc: String, |     pub jsonrpc: String, | ||||||
|    pub id: String, |     pub id: String, | ||||||
|    pub result: Value, |     pub result: Option<Value>, | ||||||
|    pub error: Option<JRPCError>, |     pub error: Option<JRPCError>, | ||||||
| } | } | ||||||
|  |  | ||||||
| // ****************************************************************************** | #[derive(Debug, Clone)] | ||||||
| // * SERVER |  | ||||||
| // ****************************************************************************** |  | ||||||
|  |  | ||||||
| pub trait JRPCServiceHandler<C: Sync>: Send { |  | ||||||
|    fn get_name(&self) -> String; |  | ||||||
|    fn on_message(&self, msg: JRPCRequest, function: String, ctx: &C) -> Result<(bool, Value)>; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type Shared<T> = Arc<Mutex<T>>; |  | ||||||
| type SharedHM<K, V> = Shared<HashMap<K, V>>; |  | ||||||
| type ServiceSharedHM<C> = SharedHM<String, Box<dyn JRPCServiceHandler<C>>>; |  | ||||||
|  |  | ||||||
| type SharedThreadPool = Shared<ThreadPool>; |  | ||||||
|  |  | ||||||
| pub struct JRPCServer<CTX: 'static + Sync + Send + Copy> { |  | ||||||
|    services: ServiceSharedHM<CTX>, |  | ||||||
|    pool: SharedThreadPool, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl<CTX: 'static + Sync + Send + Copy> JRPCServer<CTX> { |  | ||||||
|    pub fn new() -> Self { |  | ||||||
|       return Self { |  | ||||||
|          services: Arc::new(Mutex::new(HashMap::new())), |  | ||||||
|          pool: Arc::new(Mutex::new(ThreadPool::new(32))), |  | ||||||
|       }; |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    pub fn add_service(&mut self, service: Box<dyn JRPCServiceHandler<CTX>>) { |  | ||||||
|       let mut services = self.services.lock().unwrap(); |  | ||||||
|       services.insert(service.get_name(), service); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    pub fn start_session( |  | ||||||
|       &mut self, |  | ||||||
|       read_ch: Receiver<String>, |  | ||||||
|       write_ch: Sender<String>, |  | ||||||
|       context: CTX, |  | ||||||
|    ) { |  | ||||||
|       let services = self.services.clone(); |  | ||||||
|       let p = self.pool.lock().unwrap(); |  | ||||||
|       let pool = self.pool.clone(); |  | ||||||
|       p.execute(move || { |  | ||||||
|          JRPCSession::start(read_ch, write_ch, context, services, pool); |  | ||||||
|       }); |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub struct JRPCSession<CTX: 'static + Sync + Send + Copy> { |  | ||||||
|    _ctx: PhantomData<CTX>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| unsafe impl<CTX: 'static + Sync + Send + Copy> Sync for JRPCSession<CTX> {} |  | ||||||
|  |  | ||||||
| impl<CTX: 'static + Sync + Send + Copy> JRPCSession<CTX> { |  | ||||||
|    fn start( |  | ||||||
|       read_ch: Receiver<String>, |  | ||||||
|       write_ch: Sender<String>, |  | ||||||
|       context: CTX, |  | ||||||
|       services: ServiceSharedHM<CTX>, |  | ||||||
|       pool: SharedThreadPool, |  | ||||||
|    ) { |  | ||||||
|       loop { |  | ||||||
|          let pkg = read_ch.recv(); |  | ||||||
|          let data = match pkg { |  | ||||||
|             Err(_) => return, |  | ||||||
|             Ok(res) => res, |  | ||||||
|          }; |  | ||||||
|          if data.len() == 0 { |  | ||||||
|             //TODO: This can be done better |  | ||||||
|             return; |  | ||||||
|          } |  | ||||||
|          let ctx = context.clone(); |  | ||||||
|          let svs = services.clone(); |  | ||||||
|          let wc = write_ch.clone(); |  | ||||||
|          pool.lock().unwrap().execute(move || { |  | ||||||
|             JRPCSession::handle_packet(data, wc, ctx, svs); |  | ||||||
|          }) |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    fn handle_packet( |  | ||||||
|       data: String, |  | ||||||
|       write_ch: Sender<String>, |  | ||||||
|       context: CTX, |  | ||||||
|       services: ServiceSharedHM<CTX>, |  | ||||||
|    ) { |  | ||||||
|       let req: Result<JRPCRequest> = |  | ||||||
|          serde_json::from_str(data.as_str()).map_err(|err| Box::from(err)); |  | ||||||
|  |  | ||||||
|       let req = match req { |  | ||||||
|          Err(_) => { |  | ||||||
|             return; |  | ||||||
|          } |  | ||||||
|          Ok(parsed) => parsed, |  | ||||||
|       }; |  | ||||||
|  |  | ||||||
|       let req_id = req.id.clone(); |  | ||||||
|  |  | ||||||
|       let mut parts: Vec<String> = req.method.splitn(2, '.').map(|e| e.to_owned()).collect(); |  | ||||||
|       if parts.len() != 2 { |  | ||||||
|          return Self::send_err_res(req_id, write_ch, Box::from("Error".to_owned())); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       let service = parts.remove(0); |  | ||||||
|       let function = parts.remove(0); |  | ||||||
|  |  | ||||||
|       let svs = services.lock().unwrap(); |  | ||||||
|       let srv = svs.get(&service); |  | ||||||
|  |  | ||||||
|       if let Some(srv) = srv { |  | ||||||
|          match srv.on_message(req, function, &context) { |  | ||||||
|             Ok((is_send, value)) => { |  | ||||||
|                if is_send { |  | ||||||
|                   if let Some(id) = req_id { |  | ||||||
|                      let r = JRPCResult { |  | ||||||
|                         jsonrpc: "2.0".to_owned(), |  | ||||||
|                         id, |  | ||||||
|                         result: value, |  | ||||||
|                         error: None, |  | ||||||
|                      }; |  | ||||||
|                      let s = serde_json::to_string(&r); |  | ||||||
|                      if s.is_ok() { |  | ||||||
|                         write_ch |  | ||||||
|                            .send(s.unwrap()) |  | ||||||
|                            .expect("Sending data into channel failed!"); |  | ||||||
|                      } |  | ||||||
|                   } |  | ||||||
|                } |  | ||||||
|             } |  | ||||||
|             Err(err) => return Self::send_err_res(req_id, write_ch, err), |  | ||||||
|          } |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    fn send_err_res(id: Option<String>, write_ch: Sender<String>, err: Box<dyn Error>) { |  | ||||||
|       if let Some(id) = id { |  | ||||||
|          let error = JRPCError { |  | ||||||
|             code: 0, //TODO: Make this better? |  | ||||||
|             message: err.to_string(), |  | ||||||
|             data: Value::Null, |  | ||||||
|          }; |  | ||||||
|  |  | ||||||
|          let r = JRPCResult { |  | ||||||
|             jsonrpc: "2.0".to_owned(), |  | ||||||
|             id: id.clone(), |  | ||||||
|             result: Value::Null, |  | ||||||
|             error: Option::from(error), |  | ||||||
|          }; |  | ||||||
|  |  | ||||||
|          let s = serde_json::to_string(&r); |  | ||||||
|  |  | ||||||
|          if s.is_ok() { |  | ||||||
|             write_ch |  | ||||||
|                .send(s.unwrap()) |  | ||||||
|                .expect("Sending data into channel failed!"); |  | ||||||
|          } |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       return (); |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ****************************************************************************** |  | ||||||
| // * CLIENT |  | ||||||
| // ****************************************************************************** |  | ||||||
|  |  | ||||||
| #[derive(Clone)] |  | ||||||
| pub struct JRPCClient { | pub struct JRPCClient { | ||||||
|    write_ch: Sender<String>, |     message_sender: Sender<JRPCRequest>, | ||||||
|    requests: SharedHM<String, Sender<Result<Value>>>, |     requests: Arc<Mutex<HashMap<String, Sender<JRPCResult>>>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| unsafe impl Send for JRPCClient {} //TODO: Is this a problem |  | ||||||
|  |  | ||||||
| impl JRPCClient { | impl JRPCClient { | ||||||
|    pub fn new(write_ch: Sender<String>, read_ch: Receiver<String>) -> Self { |     pub fn new(sender: Sender<JRPCRequest>) -> JRPCClient { | ||||||
|       let n = Self { |         JRPCClient { | ||||||
|          write_ch, |             message_sender: sender, | ||||||
|          requests: Arc::new(Mutex::new(HashMap::new())), |             requests: Arc::new(Mutex::new(HashMap::new())), | ||||||
|       }; |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|       n.start(read_ch); |     pub async fn send_request(&self, mut request: JRPCRequest) -> Result<Value> { | ||||||
|       return n; |         let (sender, mut receiver) = tokio::sync::mpsc::channel(1); | ||||||
|    } |  | ||||||
|  |  | ||||||
|    pub fn start(&self, read_ch: Receiver<String>) { |         if request.id.is_none() { | ||||||
|       let s = self.clone(); |             request.id = Some(nanoid!()); | ||||||
|       std::thread::spawn(move || { |         } | ||||||
|          s.start_reader(read_ch); |  | ||||||
|       }); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    fn start_reader(&self, read_ch: Receiver<String>) { |         { | ||||||
|       loop { |             let mut self_requests = self.requests.lock().await; | ||||||
|          let data = read_ch.recv().expect("Error receiving packet!"); |             self_requests.insert(request.id.clone().unwrap(), sender); | ||||||
|          let response: JRPCResult = |         } | ||||||
|             serde_json::from_str(data.as_str()).expect("Error decoding response!"); |         self.message_sender.send(request).await?; | ||||||
|          let id = response.id; |  | ||||||
|  |  | ||||||
|          let reqs = self.requests.lock().expect("Error locking requests map!"); |         let result = receiver.recv().await; | ||||||
|          let req = reqs.get(&id); |  | ||||||
|          if let Some(req) = req { |         if let Some(result) = result { | ||||||
|             let res = if let Some(err) = response.error { |             if let Some(error) = result.error { | ||||||
|                Err(Box::from(err.message)) |                 return Err(format!("Error while receiving result: {}", error.message).into()); | ||||||
|  |             } else if let Some(result) = result.result { | ||||||
|  |                 return Ok(result); | ||||||
|             } else { |             } else { | ||||||
|                Ok(response.result) |                 return Ok(Value::Null); | ||||||
|  |                 // return Err(format!("No result received").into()); | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             return Err("Error while receiving result".into()); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub async fn send_notification(&self, mut request: JRPCRequest) { | ||||||
|  |         request.id = None; | ||||||
|  |  | ||||||
|  |         _ = self.message_sender.send(request).await; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub async fn on_result(&self, result: JRPCResult) { | ||||||
|  |         let id = result.id.clone(); | ||||||
|  |         let mut self_requests = self.requests.lock().await; | ||||||
|  |         let sender = self_requests.get(&id); | ||||||
|  |         if let Some(sender) = sender { | ||||||
|  |             _ = sender.send(result).await; | ||||||
|  |             self_requests.remove(&id); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[async_trait::async_trait] | ||||||
|  | pub trait JRPCServerService: Send + Sync + 'static { | ||||||
|  |     fn get_id(&self) -> String; | ||||||
|  |     async fn handle(&self, request: &JRPCRequest, function: &str) -> Result<(bool, Value)>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub type JRPCServiceHandle = Arc<dyn JRPCServerService>; | ||||||
|  |  | ||||||
|  | #[derive(Clone)] | ||||||
|  | pub struct JRPCSession { | ||||||
|  |     server: JRPCServer, | ||||||
|  |     message_sender: Sender<JRPCResult>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl JRPCSession { | ||||||
|  |     pub fn new(server: JRPCServer, sender: Sender<JRPCResult>) -> JRPCSession { | ||||||
|  |         JRPCSession { | ||||||
|  |             server, | ||||||
|  |             message_sender: sender, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn send_error(&self, request: JRPCRequest, error_msg: String, error_code: i64) -> () { | ||||||
|  |         if let Some(request_id) = request.id { | ||||||
|  |             let error = JRPCError { | ||||||
|  |                 code: error_code, | ||||||
|  |                 message: error_msg, | ||||||
|  |                 data: Value::Null, | ||||||
|  |             }; | ||||||
|  |             let result = JRPCResult { | ||||||
|  |                 jsonrpc: "2.0".to_string(), | ||||||
|  |                 id: request_id, | ||||||
|  |                 result: None, | ||||||
|  |                 error: Some(error), | ||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             req.send(res).expect("Error sending reponse!"); |             // Send result | ||||||
|          } |             let result = self.message_sender.send(result).await; | ||||||
|       } |             if let Err(err) = result { | ||||||
|    } |                 warn!("Error while sending result: {}", err); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|    pub fn send_request(&self, mut req: JRPCRequest) -> Result<Value> { |     pub fn handle_request(&self, request: JRPCRequest) -> () { | ||||||
|       let mut reqs = self.requests.lock().expect("Error locking requests map!"); |         let session = self.clone(); | ||||||
|       let id = nanoid!(); |         tokio::task::spawn(async move { | ||||||
|       req.id = Some(id.clone()); |             info!("Received request: {}", request.method); | ||||||
|  |             trace!("Request data: {:?}", request); | ||||||
|  |             let method: Vec<&str> = request.method.split('.').collect(); | ||||||
|  |             if method.len() != 2 { | ||||||
|  |                 warn!("Invalid method received: {}", request.method); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             let service = method[0]; | ||||||
|  |             let function = method[1]; | ||||||
|  |  | ||||||
|       let (tx, rx) = std::sync::mpsc::channel(); |             let service = session.server.services.get(service); | ||||||
|  |             if let Some(service) = service { | ||||||
|       reqs.insert(id, tx); |                 let result = service.handle(&request, function).await; | ||||||
|       self |                 match result { | ||||||
|          .write_ch |                     Ok((is_send, result)) => { | ||||||
|          .send(serde_json::to_string(&req).expect("Error converting Request to JSON!")) |                         if is_send && request.id.is_some() { | ||||||
|          .expect("Error Sending to Channel!"); |                             let result = session | ||||||
|       return rx.recv().expect("Error getting response!"); |                                 .message_sender | ||||||
|    } |                                 .send(JRPCResult { | ||||||
|  |                                     jsonrpc: "2.0".to_string(), | ||||||
|    pub fn send_notification(&self, mut req: JRPCRequest) { |                                     id: request.id.unwrap(), | ||||||
|       req.id = None; |                                     result: Some(result), | ||||||
|  |                                     error: None, | ||||||
|       self |                                 }) | ||||||
|          .write_ch |                                 .await; | ||||||
|          .send(serde_json::to_string(&req).expect("Error converting Request to JSON!")) |                             if let Err(err) = result { | ||||||
|          .expect("Error Sending to Channel!"); |                                 warn!("Error while sending result: {}", err); | ||||||
|    } |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     Err(err) => { | ||||||
|  |                         warn!("Error while handling request: {}", err); | ||||||
|  |                         session | ||||||
|  |                             .send_error( | ||||||
|  |                                 request, | ||||||
|  |                                 format!("Error while handling request: {}", err), | ||||||
|  |                                 1, | ||||||
|  |                             ) | ||||||
|  |                             .await; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 warn!("Service not found: {}", method[0]); | ||||||
|  |                 session | ||||||
|  |                     .send_error(request, "Service not found".to_string(), 1) | ||||||
|  |                     .await; | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Clone)] | ||||||
|  | pub struct JRPCServer { | ||||||
|  |     services: HashMap<String, JRPCServiceHandle>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl JRPCServer { | ||||||
|  |     pub fn new() -> JRPCServer { | ||||||
|  |         JRPCServer { | ||||||
|  |             services: HashMap::new(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn add_service(&mut self, service: JRPCServiceHandle) -> () { | ||||||
|  |         let id = service.get_id(); | ||||||
|  |         self.services.insert(id, service); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn get_session(&self, sender: Sender<JRPCResult>) -> JRPCSession { | ||||||
|  |         JRPCSession::new(self.clone(), sender) | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,14 +1,22 @@ | |||||||
|  | function form_verficiation_error_message(type?: string, field?: string) { | ||||||
|  |    let msg = "Parameter verification failed! "; | ||||||
|  |    if (type && field) { | ||||||
|  |       msg += `At ${type}.${field}! `; | ||||||
|  |    } else if (type) { | ||||||
|  |       msg += `At type ${type}! `; | ||||||
|  |    } else if (field) { | ||||||
|  |       msg += `At field ${field}! `; | ||||||
|  |    } | ||||||
|  |    return msg; | ||||||
|  | } | ||||||
|  |  | ||||||
| export class VerificationError extends Error { | export class VerificationError extends Error { | ||||||
|    constructor( |    constructor( | ||||||
|       public readonly type?: string, |       public readonly type?: string, | ||||||
|       public readonly field?: string, |       public readonly field?: string, | ||||||
|       public readonly value?: any |       public readonly value?: any | ||||||
|    ) { |    ) { | ||||||
|       super( |       super(form_verficiation_error_message(type, field)); | ||||||
|          "Parameter verification failed! " + |  | ||||||
|             (type ? "Expected " + type + "! " : "") + |  | ||||||
|             (field ? "At: " + field + "! " : "") |  | ||||||
|       ); |  | ||||||
|    } |    } | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -33,4 +41,4 @@ export function apply_boolean(data: any) { | |||||||
|    return Boolean(data); |    return Boolean(data); | ||||||
| } | } | ||||||
|  |  | ||||||
| export function apply_void(data: any) {} | export function apply_void(data: any) { } | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| //@template-ignore | //@template-ignore | ||||||
| import { VerificationError } from "./ts_base"; | import { VerificationError } from "./ts_base"; | ||||||
| //@template-ignore | //@template-ignore | ||||||
| import { RequestObject, ResponseObject, ErrorCodes, Logging } from "./ts_service_base"; | import { type RequestObject, type ResponseObject, ErrorCodes, Logging } from "./ts_service_base"; | ||||||
|  |  | ||||||
|  |  | ||||||
| export type IMessageCallback = (data: any) => void; | export type IMessageCallback = (data: any) => void; | ||||||
|  | |||||||
| @ -1,14 +1,14 @@ | |||||||
| //@template-ignore | //@template-ignore | ||||||
| import { VerificationError } from "./ts_base"; | import { VerificationError } from "./ts_base"; | ||||||
| //@template-ignore | //@template-ignore | ||||||
| import { RequestObject, ResponseObject, ErrorCodes, Logging } from "./ts_service_base"; | import { type RequestObject, type ResponseObject, ErrorCodes, Logging } from "./ts_service_base"; | ||||||
|  |  | ||||||
|  |  | ||||||
| export class Service<T> { | export class Service<T> { | ||||||
|    public name: string = null as any; |    public name: string = null as any; | ||||||
|    public functions = new Set<string>(); |    public functions = new Set<string>(); | ||||||
|  |  | ||||||
|    constructor() {} |    constructor() { } | ||||||
| } | } | ||||||
|  |  | ||||||
| type ISendMessageCB = (data: any, catchedErr?: Error) => void; | type ISendMessageCB = (data: any, catchedErr?: Error) => void; | ||||||
| @ -37,7 +37,7 @@ class Session<T> { | |||||||
|       this.ctx = ctx || {}; |       this.ctx = ctx || {}; | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    send(data: any, catchedErr?:Error) { |    send(data: any, catchedErr?: Error) { | ||||||
|       Logging.log("SERVER: Sending Message", data) |       Logging.log("SERVER: Sending Message", data) | ||||||
|       this._send(data, catchedErr); |       this._send(data, catchedErr); | ||||||
|    } |    } | ||||||
| @ -95,7 +95,7 @@ class Session<T> { | |||||||
|          } |          } | ||||||
|  |  | ||||||
|          let result = await (service as any)["_" + fncName](data.params, this.ctx); |          let result = await (service as any)["_" + fncName](data.params, this.ctx); | ||||||
|          if(data.id) { //Request |          if (data.id) { //Request | ||||||
|             this.send({ |             this.send({ | ||||||
|                jsonrpc: "2.0", |                jsonrpc: "2.0", | ||||||
|                id: data.id, |                id: data.id, | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user