4
0

srv.rs 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. use log::*;
  2. use crate::strings::{Labels, ReadLabels};
  3. use crate::wire::*;
  4. /// A **SRV** record, which contains an IP address as well as a port number,
  5. /// for specifying the location of services more precisely.
  6. ///
  7. /// # References
  8. ///
  9. /// - [RFC 2782](https://tools.ietf.org/html/rfc2782) — A DNS RR for specifying the location of services (February 2000)
  10. #[derive(PartialEq, Debug, Clone)]
  11. pub struct SRV {
  12. /// The priority of this host among all that get returned. Lower values
  13. /// are higher priority.
  14. pub priority: u16,
  15. /// A weight to choose among results with the same priority. Higher values
  16. /// are higher priority.
  17. pub weight: u16,
  18. /// The port the service is serving on.
  19. pub port: u16,
  20. /// The hostname of the machine the service is running on.
  21. pub target: Labels,
  22. }
  23. impl Wire for SRV {
  24. const NAME: &'static str = "SRV";
  25. const RR_TYPE: u16 = 33;
  26. #[cfg_attr(all(test, feature = "with_mutagen"), ::mutagen::mutate)]
  27. fn read(len: u16, c: &mut Cursor<&[u8]>) -> Result<Self, WireError> {
  28. let priority = c.read_u16::<BigEndian>()?;
  29. trace!("Parsed priority -> {:?}", priority);
  30. let weight = c.read_u16::<BigEndian>()?;
  31. trace!("Parsed weight -> {:?}", weight);
  32. let port = c.read_u16::<BigEndian>()?;
  33. trace!("Parsed port -> {:?}", port);
  34. let (target, target_len) = c.read_labels()?;
  35. trace!("Parsed target -> {:?}", target);
  36. let got_len = 3 * 2 + target_len;
  37. if len == got_len {
  38. trace!("Length is correct");
  39. Ok(Self { priority, weight, port, target })
  40. }
  41. else {
  42. warn!("Length is incorrect (record length {:?}, fields plus target length {:?})", len, got_len);
  43. Err(WireError::WrongLabelLength { expected: len, got: got_len })
  44. }
  45. }
  46. }
  47. #[cfg(test)]
  48. mod test {
  49. use super::*;
  50. #[test]
  51. fn parses() {
  52. let buf = &[
  53. 0x00, 0x01, // priority
  54. 0x00, 0x01, // weight
  55. 0x92, 0x7c, // port
  56. 0x03, 0x61, 0x74, 0x61, 0x05, 0x6c, 0x6f, 0x63, 0x61, 0x6c, 0x04,
  57. 0x6e, 0x6f, 0x64, 0x65, 0x03, 0x64, 0x63, 0x31, 0x06, 0x63, 0x6f,
  58. 0x6e, 0x73, 0x75, 0x6c, // target
  59. 0x00, // target terminator
  60. ];
  61. assert_eq!(SRV::read(buf.len() as _, &mut Cursor::new(buf)).unwrap(),
  62. SRV {
  63. priority: 1,
  64. weight: 1,
  65. port: 37500,
  66. target: Labels::encode("ata.local.node.dc1.consul").unwrap(),
  67. });
  68. }
  69. #[test]
  70. fn record_empty() {
  71. assert_eq!(SRV::read(0, &mut Cursor::new(&[])),
  72. Err(WireError::IO));
  73. }
  74. #[test]
  75. fn buffer_ends_abruptly() {
  76. let buf = &[
  77. 0x00, // half a priority
  78. ];
  79. assert_eq!(SRV::read(23, &mut Cursor::new(buf)),
  80. Err(WireError::IO));
  81. }
  82. }