mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-12 13:55:28 +00:00
337e54c672
Moves from concatted Dockerfiles to pseudo-templated Dockerfiles via a dedicated Rust program. Removes the unmaintained kubernetes, not because we shouldn't have/use it, but because it's unmaintained and needs to be reworked before it's present again. Replaces the compose with the work in the new orchestrator binary which spawns everything as expected. While this arguably re-invents the wheel, it correctly manages secrets and handles the variadic Dockerfiles. Also adds an unrelated patch for zstd and simplifies running services a bit by greater utilizing the existing infrastructure. --- * Delete all Dockerfile fragments, add new orchestator to generate Dockerfiles Enables greater templating. Also delete the unmaintained kubernetes folder *for now*. This should be restored in the future. * Use Dockerfiles from the orchestator * Ignore Dockerfiles in the git repo * Remove CI job to check Dockerfiles are as expected now that they're no longer committed * Remove old Dockerfiles from repo * Use Debian for monero-wallet-rpc * Remove replace_cmds for proper usage of entry-dev Consolidates ports a bit. Updates serai-docker-tests from "compose" to "build". * Only write a new dockerfile if it's distinct Preserves the updated time metadata. * Update serai-docker-tests * Correct the path Dockerfiles are built from * Correct inclusion of orchestration folder in Docker builds * Correct debug/release flagging in the cargo command Apparently, --debug isn't an effective NOP yet an error. * Correct path used to run the Serai node within a Dockerfile * Correct path in Monero Dockerfile * Attempt storing monerod in /usr/bin * Use sudo to move into /usr/bin in CI * Correct 18.3.0 to 18.3.1 * Escape * with quotes * Update deny.toml, ADD orchestration in runtime Dockerfile * Add --detach to the Monero GH CI * Diversify dockerfiles by network * Fixes to network-diversified orchestration * Bitcoin and Monero testnet scripts * Permissions and tweaks * Flatten scripts folders * Add missing folder specification to Monero Dockerfile * Have monero-wallet-rpc specify the monerod login * Have the Docker CMD specify env variables inserted at time of Dockerfile generation They're overrideable with the global enviornment as for tests. This enables variable generation in orchestrator and output to productionized Docker files without creating a life-long file within the Docker container. * Don't add Dockerfiles into Docker containers now that they have secrets Solely add the source code for them as needed to satisfy the workspace bounds. * Download arm64 Monero on arm64 * Ensure constant host architecture when reproducibly building the wasm Host architecture, for some reason, can effect the generated code despite the target architecture always being foreign to the host architecture. * Randomly generate infrastructure keys * Have orchestrator generate a key, be able to create/start containers * Ensure bash is used over sh * Clean dated docs * Change how quoting occurs * Standardize to sh * Have Docker test build the dev Dockerfiles * Only key_gen once * cargo update Adds a patch for zstd and reconciles the breaking nightly change which just occurred. * Use a dedicated network for Serai Also fixes SERAI_HOSTNAME passed to coordinator. * Support providing a key over the env for the Serai node * Enable and document running daemons for tests via serai-orchestrator Has running containers under the dev network port forward the RPC ports. * Use volumes for bitcoin/monero * Use bitcoin's run.sh in GH CI * Only use the volume for testnet (not dev)
293 lines
8.8 KiB
Rust
293 lines
8.8 KiB
Rust
use std::collections::HashSet;
|
|
|
|
use rand_core::{OsRng, RngCore};
|
|
|
|
use serde::Deserialize;
|
|
use serde_json::json;
|
|
|
|
use monero_serai::{
|
|
transaction::Transaction,
|
|
rpc::{EmptyResponse, HttpRpc, Rpc},
|
|
wallet::{
|
|
address::{Network, AddressSpec, SubaddressIndex, MoneroAddress},
|
|
extra::{MAX_TX_EXTRA_NONCE_SIZE, Extra},
|
|
Scanner,
|
|
},
|
|
};
|
|
|
|
mod runner;
|
|
|
|
async fn make_integrated_address(rpc: &Rpc<HttpRpc>, payment_id: [u8; 8]) -> String {
|
|
#[derive(Debug, Deserialize)]
|
|
struct IntegratedAddressResponse {
|
|
integrated_address: String,
|
|
}
|
|
|
|
let res = rpc
|
|
.json_rpc_call::<IntegratedAddressResponse>(
|
|
"make_integrated_address",
|
|
Some(json!({ "payment_id": hex::encode(payment_id) })),
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
res.integrated_address
|
|
}
|
|
|
|
async fn initialize_rpcs() -> (Rpc<HttpRpc>, Rpc<HttpRpc>, String) {
|
|
let wallet_rpc = HttpRpc::new("http://127.0.0.1:18082".to_string()).await.unwrap();
|
|
let daemon_rpc = runner::rpc().await;
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
struct AddressResponse {
|
|
address: String,
|
|
}
|
|
|
|
let mut wallet_id = [0; 8];
|
|
OsRng.fill_bytes(&mut wallet_id);
|
|
let _: EmptyResponse = wallet_rpc
|
|
.json_rpc_call(
|
|
"create_wallet",
|
|
Some(json!({ "filename": hex::encode(wallet_id), "language": "English" })),
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let address: AddressResponse =
|
|
wallet_rpc.json_rpc_call("get_address", Some(json!({ "account_index": 0 }))).await.unwrap();
|
|
|
|
// Fund the new wallet
|
|
daemon_rpc.generate_blocks(&address.address, 70).await.unwrap();
|
|
|
|
(wallet_rpc, daemon_rpc, address.address)
|
|
}
|
|
|
|
async fn from_wallet_rpc_to_self(spec: AddressSpec) {
|
|
// initialize rpc
|
|
let (wallet_rpc, daemon_rpc, wallet_rpc_addr) = initialize_rpcs().await;
|
|
|
|
// make an addr
|
|
let (_, view_pair, _) = runner::random_address();
|
|
let addr = view_pair.address(Network::Mainnet, spec);
|
|
|
|
// refresh & make a tx
|
|
let _: EmptyResponse = wallet_rpc.json_rpc_call("refresh", None).await.unwrap();
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
struct TransferResponse {
|
|
tx_hash: String,
|
|
}
|
|
let tx: TransferResponse = wallet_rpc
|
|
.json_rpc_call(
|
|
"transfer",
|
|
Some(json!({
|
|
"destinations": [{"address": addr.to_string(), "amount": 1_000_000_000_000u64 }],
|
|
})),
|
|
)
|
|
.await
|
|
.unwrap();
|
|
let tx_hash = hex::decode(tx.tx_hash).unwrap().try_into().unwrap();
|
|
|
|
// TODO: Needs https://github.com/monero-project/monero/pull/8882
|
|
// let fee_rate = daemon_rpc
|
|
// // Uses `FeePriority::Low` instead of `FeePriority::Lowest` because wallet RPC
|
|
// // adjusts `monero_rpc::TransferPriority::Default` up by 1
|
|
// .get_fee(daemon_rpc.get_protocol().await.unwrap(), FeePriority::Low)
|
|
// .await
|
|
// .unwrap();
|
|
|
|
// unlock it
|
|
runner::mine_until_unlocked(&daemon_rpc, &wallet_rpc_addr, tx_hash).await;
|
|
|
|
// Create the scanner
|
|
let mut scanner = Scanner::from_view(view_pair, Some(HashSet::new()));
|
|
if let AddressSpec::Subaddress(index) = spec {
|
|
scanner.register_subaddress(index);
|
|
}
|
|
|
|
// Retrieve it and scan it
|
|
let tx = daemon_rpc.get_transaction(tx_hash).await.unwrap();
|
|
let output = scanner.scan_transaction(&tx).not_locked().swap_remove(0);
|
|
|
|
// TODO: Needs https://github.com/monero-project/monero/pull/8882
|
|
// runner::check_weight_and_fee(&tx, fee_rate);
|
|
|
|
match spec {
|
|
AddressSpec::Subaddress(index) => assert_eq!(output.metadata.subaddress, Some(index)),
|
|
AddressSpec::Integrated(payment_id) => {
|
|
assert_eq!(output.metadata.payment_id, payment_id);
|
|
assert_eq!(output.metadata.subaddress, None);
|
|
}
|
|
AddressSpec::Standard | AddressSpec::Featured { .. } => {
|
|
assert_eq!(output.metadata.subaddress, None)
|
|
}
|
|
}
|
|
assert_eq!(output.commitment().amount, 1000000000000);
|
|
}
|
|
|
|
async_sequential!(
|
|
async fn receipt_of_wallet_rpc_tx_standard() {
|
|
from_wallet_rpc_to_self(AddressSpec::Standard).await;
|
|
}
|
|
|
|
async fn receipt_of_wallet_rpc_tx_subaddress() {
|
|
from_wallet_rpc_to_self(AddressSpec::Subaddress(SubaddressIndex::new(0, 1).unwrap())).await;
|
|
}
|
|
|
|
async fn receipt_of_wallet_rpc_tx_integrated() {
|
|
let mut payment_id = [0u8; 8];
|
|
OsRng.fill_bytes(&mut payment_id);
|
|
from_wallet_rpc_to_self(AddressSpec::Integrated(payment_id)).await;
|
|
}
|
|
);
|
|
|
|
#[derive(PartialEq, Eq, Debug, Deserialize)]
|
|
struct Index {
|
|
major: u32,
|
|
minor: u32,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
struct Transfer {
|
|
payment_id: String,
|
|
subaddr_index: Index,
|
|
amount: u64,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
struct TransfersResponse {
|
|
transfer: Transfer,
|
|
}
|
|
|
|
test!(
|
|
send_to_wallet_rpc_standard,
|
|
(
|
|
|_, mut builder: Builder, _| async move {
|
|
// initialize rpc
|
|
let (wallet_rpc, _, wallet_rpc_addr) = initialize_rpcs().await;
|
|
|
|
// add destination
|
|
builder
|
|
.add_payment(MoneroAddress::from_str(Network::Mainnet, &wallet_rpc_addr).unwrap(), 1000000);
|
|
(builder.build().unwrap(), wallet_rpc)
|
|
},
|
|
|_, tx: Transaction, _, data: Rpc<HttpRpc>| async move {
|
|
// confirm receipt
|
|
let _: EmptyResponse = data.json_rpc_call("refresh", None).await.unwrap();
|
|
let transfer: TransfersResponse = data
|
|
.json_rpc_call("get_transfer_by_txid", Some(json!({ "txid": hex::encode(tx.hash()) })))
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(transfer.transfer.subaddr_index, Index { major: 0, minor: 0 });
|
|
assert_eq!(transfer.transfer.amount, 1000000);
|
|
},
|
|
),
|
|
);
|
|
|
|
test!(
|
|
send_to_wallet_rpc_subaddress,
|
|
(
|
|
|_, mut builder: Builder, _| async move {
|
|
// initialize rpc
|
|
let (wallet_rpc, _, _) = initialize_rpcs().await;
|
|
|
|
// make the subaddress
|
|
#[derive(Debug, Deserialize)]
|
|
struct AccountResponse {
|
|
address: String,
|
|
account_index: u32,
|
|
}
|
|
let addr: AccountResponse = wallet_rpc.json_rpc_call("create_account", None).await.unwrap();
|
|
assert!(addr.account_index != 0);
|
|
|
|
builder
|
|
.add_payment(MoneroAddress::from_str(Network::Mainnet, &addr.address).unwrap(), 1000000);
|
|
(builder.build().unwrap(), (wallet_rpc, addr.account_index))
|
|
},
|
|
|_, tx: Transaction, _, data: (Rpc<HttpRpc>, u32)| async move {
|
|
// confirm receipt
|
|
let _: EmptyResponse = data.0.json_rpc_call("refresh", None).await.unwrap();
|
|
let transfer: TransfersResponse = data
|
|
.0
|
|
.json_rpc_call(
|
|
"get_transfer_by_txid",
|
|
Some(json!({ "txid": hex::encode(tx.hash()), "account_index": data.1 })),
|
|
)
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(transfer.transfer.subaddr_index, Index { major: data.1, minor: 0 });
|
|
assert_eq!(transfer.transfer.amount, 1000000);
|
|
|
|
// Make sure only one R was included in TX extra
|
|
assert!(Extra::read::<&[u8]>(&mut tx.prefix.extra.as_ref())
|
|
.unwrap()
|
|
.keys()
|
|
.unwrap()
|
|
.1
|
|
.is_none());
|
|
},
|
|
),
|
|
);
|
|
|
|
test!(
|
|
send_to_wallet_rpc_integrated,
|
|
(
|
|
|_, mut builder: Builder, _| async move {
|
|
// initialize rpc
|
|
let (wallet_rpc, _, _) = initialize_rpcs().await;
|
|
|
|
// make the addr
|
|
let mut payment_id = [0u8; 8];
|
|
OsRng.fill_bytes(&mut payment_id);
|
|
let addr = make_integrated_address(&wallet_rpc, payment_id).await;
|
|
|
|
builder.add_payment(MoneroAddress::from_str(Network::Mainnet, &addr).unwrap(), 1000000);
|
|
(builder.build().unwrap(), (wallet_rpc, payment_id))
|
|
},
|
|
|_, tx: Transaction, _, data: (Rpc<HttpRpc>, [u8; 8])| async move {
|
|
// confirm receipt
|
|
let _: EmptyResponse = data.0.json_rpc_call("refresh", None).await.unwrap();
|
|
let transfer: TransfersResponse = data
|
|
.0
|
|
.json_rpc_call("get_transfer_by_txid", Some(json!({ "txid": hex::encode(tx.hash()) })))
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(transfer.transfer.subaddr_index, Index { major: 0, minor: 0 });
|
|
assert_eq!(transfer.transfer.payment_id, hex::encode(data.1));
|
|
assert_eq!(transfer.transfer.amount, 1000000);
|
|
},
|
|
),
|
|
);
|
|
|
|
test!(
|
|
send_to_wallet_rpc_with_arb_data,
|
|
(
|
|
|_, mut builder: Builder, _| async move {
|
|
// initialize rpc
|
|
let (wallet_rpc, _, wallet_rpc_addr) = initialize_rpcs().await;
|
|
|
|
// add destination
|
|
builder
|
|
.add_payment(MoneroAddress::from_str(Network::Mainnet, &wallet_rpc_addr).unwrap(), 1000000);
|
|
|
|
// Make 2 data that is the full 255 bytes
|
|
for _ in 0 .. 2 {
|
|
// Subtract 1 since we prefix data with 127
|
|
let data = vec![b'a'; MAX_TX_EXTRA_NONCE_SIZE - 1];
|
|
builder.add_data(data).unwrap();
|
|
}
|
|
|
|
(builder.build().unwrap(), wallet_rpc)
|
|
},
|
|
|_, tx: Transaction, _, data: Rpc<HttpRpc>| async move {
|
|
// confirm receipt
|
|
let _: EmptyResponse = data.json_rpc_call("refresh", None).await.unwrap();
|
|
let transfer: TransfersResponse = data
|
|
.json_rpc_call("get_transfer_by_txid", Some(json!({ "txid": hex::encode(tx.hash()) })))
|
|
.await
|
|
.unwrap();
|
|
assert_eq!(transfer.transfer.subaddr_index, Index { major: 0, minor: 0 });
|
|
assert_eq!(transfer.transfer.amount, 1000000);
|
|
},
|
|
),
|
|
);
|