main.rs 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. mod client_lib;
  2. use client_lib::*;
  3. mod client_net;
  4. use client_net::HyperNet;
  5. use curve25519_dalek::scalar::Scalar;
  6. use getopts::Options;
  7. use lox_library::bridge_table::BridgeLine;
  8. use lox_library::bridge_table::MAX_BRIDGES_PER_BUCKET;
  9. use lox_library::IssuerPubKey;
  10. use serde::Serialize;
  11. use std::env::args;
  12. use std::fs::File;
  13. use std::io::Write;
  14. use std::path::Path;
  15. use std::str::FromStr;
  16. // Prints the argument details for this program
  17. fn print_usage(program: &str, opts: Options) {
  18. let brief = format!("Usage: {} [options]", program);
  19. print!("{}", opts.usage(&brief));
  20. }
  21. // Helper function to save serializable objects to files
  22. fn save_object<T: Serialize>(obj: T, filename: &str) {
  23. let mut outfile = File::create(filename).expect(&("Failed to create ".to_string() + filename));
  24. write!(outfile, "{}", serde_json::to_string(&obj).unwrap())
  25. .expect(&("Failed to write to ".to_string() + filename));
  26. }
  27. #[tokio::main]
  28. async fn main() {
  29. let args: Vec<String> = args().collect();
  30. let mut opts = Options::new();
  31. opts.optflag("h", "help", "print this help menu");
  32. //#[cfg(test)]
  33. opts.optopt(
  34. "A",
  35. "advance-days",
  36. "increase server days by NUM_DAYS",
  37. "NUM_DAYS",
  38. );
  39. opts.optflag("L", "level-up", "increase trust level");
  40. opts.optflag("N", "new-lox-cred", "get a new Lox Credential");
  41. opts.optopt(
  42. "",
  43. "server",
  44. "Lox Auth server address [http://localhost:8001]",
  45. "ADDR",
  46. );
  47. let matches = match opts.parse(&args[1..]) {
  48. Ok(m) => m,
  49. Err(f) => {
  50. panic!("{}", f.to_string())
  51. }
  52. };
  53. if matches.opt_present("h") {
  54. print_usage(&args[0], opts);
  55. return;
  56. }
  57. let net = if matches.opt_present("server") {
  58. HyperNet {
  59. hostname: matches.opt_str("server").unwrap(),
  60. }
  61. } else {
  62. HyperNet {
  63. hostname: "http://localhost:8001".to_string(),
  64. }
  65. };
  66. // Advance days on server (TESTING ONLY)
  67. //#[cfg(test)]
  68. if matches.opt_present("A") {
  69. let days: u16 = u16::from_str(matches.opt_str("A").unwrap().as_str()).unwrap();
  70. let today: u32 = advance_days(&net, days).await;
  71. println!("Today's date according to the server: {}", today);
  72. }
  73. // Get Lox Authority public keys
  74. // TODO: Make this filename configurable
  75. let lox_auth_pubkeys_filename = "lox_auth_pubkeys.json";
  76. let lox_auth_pubkeys: Vec<IssuerPubKey> = if Path::new(lox_auth_pubkeys_filename).exists() {
  77. // read in file
  78. let lox_auth_pubkeys_infile = File::open(lox_auth_pubkeys_filename).unwrap();
  79. serde_json::from_reader(lox_auth_pubkeys_infile).unwrap()
  80. } else {
  81. // download from Lox Auth
  82. let pubkeys = get_lox_auth_keys(&net).await;
  83. // save to file for next time
  84. save_object(&pubkeys, &lox_auth_pubkeys_filename);
  85. pubkeys
  86. };
  87. // Get Lox Credential and BridgeLine
  88. let lox_cred_filename = "lox_cred.json";
  89. let bucket_filename = "bucket.json";
  90. let (lox_cred, bucket) = if matches.opt_present("N")
  91. || !Path::new(lox_cred_filename).exists()
  92. || !Path::new(bucket_filename).exists()
  93. {
  94. // get new Lox Credential
  95. let open_invite = get_open_invitation(&net).await;
  96. let (cred, bl) =
  97. get_lox_credential(&net, &open_invite, get_lox_pub(&lox_auth_pubkeys)).await;
  98. let mut bucket = [BridgeLine::default(); MAX_BRIDGES_PER_BUCKET];
  99. // note: this is a bucket with one real bridgeline and n-1
  100. // default (zeroed out) bridgelines
  101. bucket[0] = bl;
  102. // save to files for next time
  103. save_object(&cred, &lox_cred_filename);
  104. save_object(&bucket, &bucket_filename);
  105. (cred, bucket)
  106. } else {
  107. // Read existing Lox Credential and BridgeLine from files
  108. let cred = serde_json::from_reader(File::open(lox_cred_filename).unwrap()).unwrap();
  109. let bucket = serde_json::from_reader(File::open(bucket_filename).unwrap()).unwrap();
  110. (cred, bucket)
  111. };
  112. let lox_cred = if matches.opt_present("L") {
  113. let old_level = get_cred_trust_level(&lox_cred);
  114. // If trust level is 0, do trust promotion, otherwise level up.
  115. let cred = if old_level == 0 {
  116. if eligible_for_trust_promotion(&net, &lox_cred).await {
  117. let migration_cred =
  118. trust_promotion(&net, &lox_cred, get_lox_pub(&lox_auth_pubkeys)).await;
  119. let cred = trust_migration(
  120. &net,
  121. &lox_cred,
  122. &migration_cred,
  123. get_lox_pub(&lox_auth_pubkeys),
  124. get_migration_pub(&lox_auth_pubkeys),
  125. )
  126. .await;
  127. cred
  128. } else {
  129. lox_cred
  130. }
  131. } else {
  132. if eligible_for_level_up(&net, &lox_cred).await {
  133. let encbuckets = get_reachability_credential(&net).await;
  134. let cred = level_up(
  135. &net,
  136. &lox_cred,
  137. &encbuckets,
  138. get_lox_pub(&lox_auth_pubkeys),
  139. get_reachability_pub(&lox_auth_pubkeys),
  140. )
  141. .await;
  142. cred
  143. } else {
  144. lox_cred
  145. }
  146. };
  147. save_object(&cred, &lox_cred_filename);
  148. let new_level = get_cred_trust_level(&cred);
  149. if new_level > old_level {
  150. println!("Old level: {}\nNew level: {}", old_level, new_level);
  151. } else if new_level == old_level {
  152. println!("Unable to level up. Current level: {}", new_level);
  153. }
  154. cred
  155. } else {
  156. lox_cred
  157. };
  158. }