main.rs 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. //! dog, the command-line DNS client.
  2. #![warn(deprecated_in_future)]
  3. #![warn(future_incompatible)]
  4. #![warn(missing_copy_implementations)]
  5. #![warn(missing_docs)]
  6. #![warn(nonstandard_style)]
  7. #![warn(rust_2018_compatibility)]
  8. #![warn(rust_2018_idioms)]
  9. #![warn(single_use_lifetimes)]
  10. #![warn(trivial_casts, trivial_numeric_casts)]
  11. #![warn(unused)]
  12. #![warn(clippy::all, clippy::pedantic)]
  13. #![allow(clippy::enum_glob_use)]
  14. #![allow(clippy::module_name_repetitions)]
  15. #![allow(clippy::unit_arg)]
  16. #![allow(clippy::useless_let_if_seq)]
  17. #![allow(clippy::wildcard_imports)]
  18. #![deny(unsafe_code)]
  19. use log::*;
  20. mod colours;
  21. mod connect;
  22. mod logger;
  23. mod output;
  24. mod requests;
  25. mod resolve;
  26. mod table;
  27. mod txid;
  28. mod options;
  29. use self::options::*;
  30. /// Configures logging, parses the command-line options, and handles any
  31. /// errors before passing control over to the Dog type.
  32. fn main() {
  33. use std::env;
  34. use std::process::exit;
  35. logger::configure(env::var_os("DOG_DEBUG"));
  36. match Options::getopts(env::args_os().skip(1)) {
  37. OptionsResult::Ok(options) => {
  38. info!("Running with options -> {:#?}", options);
  39. exit(run(options));
  40. }
  41. OptionsResult::Help(help_reason, use_colours) => {
  42. if use_colours.should_use_colours() {
  43. print!("{}", include_str!(concat!(env!("OUT_DIR"), "/usage.pretty.txt")));
  44. }
  45. else {
  46. print!("{}", include_str!(concat!(env!("OUT_DIR"), "/usage.bland.txt")));
  47. }
  48. if help_reason == HelpReason::NoDomains {
  49. exit(exits::OPTIONS_ERROR);
  50. }
  51. else {
  52. exit(exits::SUCCESS);
  53. }
  54. }
  55. OptionsResult::Version(use_colours) => {
  56. if use_colours.should_use_colours() {
  57. print!("{}", include_str!(concat!(env!("OUT_DIR"), "/version.pretty.txt")));
  58. }
  59. else {
  60. print!("{}", include_str!(concat!(env!("OUT_DIR"), "/version.bland.txt")));
  61. }
  62. exit(exits::SUCCESS);
  63. }
  64. OptionsResult::InvalidOptionsFormat(oe) => {
  65. eprintln!("Invalid options: {}", oe);
  66. exit(exits::OPTIONS_ERROR);
  67. }
  68. OptionsResult::InvalidOptions(why) => {
  69. eprintln!("Invalid options: {}", why);
  70. exit(exits::OPTIONS_ERROR);
  71. }
  72. }
  73. }
  74. fn run(Options { requests, format, measure_time }: Options) -> i32 {
  75. use std::time::Instant;
  76. let mut runtime = dns_transport::Runtime::new().expect("Failed to create runtime");
  77. let should_show_opt = requests.edns.should_show();
  78. let mut responses = Vec::new();
  79. let timer = if measure_time { Some(Instant::now()) } else { None };
  80. let mut errored = false;
  81. for (request, transport) in requests.generate() {
  82. let result = runtime.block_on(async { transport.send(&request).await });
  83. match result {
  84. Ok(mut response) => {
  85. if ! should_show_opt {
  86. response.answers.retain(dns::Answer::is_standard);
  87. response.authorities.retain(dns::Answer::is_standard);
  88. response.additionals.retain(dns::Answer::is_standard);
  89. }
  90. responses.push(response);
  91. }
  92. Err(e) => {
  93. format.print_error(e);
  94. errored = true;
  95. }
  96. }
  97. }
  98. let duration = timer.map(|t| t.elapsed());
  99. if format.print(responses, duration) {
  100. if errored {
  101. exits::NETWORK_ERROR
  102. }
  103. else {
  104. exits::SUCCESS
  105. }
  106. }
  107. else {
  108. exits::NO_SHORT_RESULTS
  109. }
  110. }
  111. mod exits {
  112. #![allow(unused)]
  113. /// Exit code for when everything turns out OK.
  114. pub const SUCCESS: i32 = 0;
  115. /// Exit code for when there was at least one network error during execution.
  116. pub const NETWORK_ERROR: i32 = 1;
  117. /// Exit code for when there is no result from the server when running in
  118. /// short mode. This can be any received server error, not just NXDOMAIN.
  119. pub const NO_SHORT_RESULTS: i32 = 2;
  120. /// Exit code for when the command-line options are invalid.
  121. pub const OPTIONS_ERROR: i32 = 3;
  122. }