Answer.cpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. /*
  2. * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "Answer.h"
  7. #include <LibIPC/Decoder.h>
  8. #include <LibIPC/Encoder.h>
  9. #include <time.h>
  10. namespace DNS {
  11. Answer::Answer(Name const& name, RecordType type, RecordClass class_code, u32 ttl, ByteString const& record_data, bool mdns_cache_flush)
  12. : m_name(name)
  13. , m_type(type)
  14. , m_class_code(class_code)
  15. , m_ttl(ttl)
  16. , m_record_data(record_data)
  17. , m_mdns_cache_flush(mdns_cache_flush)
  18. {
  19. time(&m_received_time);
  20. }
  21. bool Answer::has_expired() const
  22. {
  23. return time(nullptr) >= static_cast<time_t>(m_received_time + m_ttl);
  24. }
  25. unsigned Answer::hash() const
  26. {
  27. auto hash = pair_int_hash(CaseInsensitiveStringTraits::hash(name().as_string()), (u32)type());
  28. hash = pair_int_hash(hash, pair_int_hash((u32)class_code(), ttl()));
  29. hash = pair_int_hash(hash, record_data().hash());
  30. hash = pair_int_hash(hash, (u32)mdns_cache_flush());
  31. return hash;
  32. }
  33. bool Answer::operator==(Answer const& other) const
  34. {
  35. if (&other == this)
  36. return true;
  37. if (!Name::Traits::equals(name(), other.name()))
  38. return false;
  39. if (type() != other.type())
  40. return false;
  41. if (class_code() != other.class_code())
  42. return false;
  43. if (ttl() != other.ttl())
  44. return false;
  45. if (record_data() != other.record_data())
  46. return false;
  47. if (mdns_cache_flush() != other.mdns_cache_flush())
  48. return false;
  49. return true;
  50. }
  51. }
  52. ErrorOr<void> AK::Formatter<DNS::RecordType>::format(AK::FormatBuilder& builder, DNS::RecordType value)
  53. {
  54. switch (value) {
  55. case DNS::RecordType::A:
  56. return builder.put_string("A"sv);
  57. case DNS::RecordType::NS:
  58. return builder.put_string("NS"sv);
  59. case DNS::RecordType::CNAME:
  60. return builder.put_string("CNAME"sv);
  61. case DNS::RecordType::SOA:
  62. return builder.put_string("SOA"sv);
  63. case DNS::RecordType::PTR:
  64. return builder.put_string("PTR"sv);
  65. case DNS::RecordType::MX:
  66. return builder.put_string("MX"sv);
  67. case DNS::RecordType::TXT:
  68. return builder.put_string("TXT"sv);
  69. case DNS::RecordType::AAAA:
  70. return builder.put_string("AAAA"sv);
  71. case DNS::RecordType::SRV:
  72. return builder.put_string("SRV"sv);
  73. }
  74. TRY(builder.put_string("DNS record type "sv));
  75. TRY(builder.put_u64((u16)value));
  76. return {};
  77. }
  78. ErrorOr<void> AK::Formatter<DNS::RecordClass>::format(AK::FormatBuilder& builder, DNS::RecordClass value)
  79. {
  80. switch (value) {
  81. case DNS::RecordClass::IN:
  82. return builder.put_string("IN"sv);
  83. }
  84. TRY(builder.put_string("DNS record class "sv));
  85. TRY(builder.put_u64((u16)value));
  86. return {};
  87. }
  88. namespace IPC {
  89. template<>
  90. ErrorOr<void> encode(Encoder& encoder, DNS::Answer const& answer)
  91. {
  92. TRY(encoder.encode(answer.name().as_string()));
  93. TRY(encoder.encode(static_cast<u16>(answer.type())));
  94. TRY(encoder.encode(static_cast<u16>(answer.class_code())));
  95. TRY(encoder.encode(answer.ttl()));
  96. TRY(encoder.encode(answer.record_data()));
  97. TRY(encoder.encode(answer.mdns_cache_flush()));
  98. return {};
  99. }
  100. template<>
  101. ErrorOr<DNS::Answer> decode(Decoder& decoder)
  102. {
  103. auto name = TRY(decoder.decode<ByteString>());
  104. auto record_type = TRY(decoder.decode<DNS::RecordType>());
  105. auto class_code = TRY(decoder.decode<DNS::RecordClass>());
  106. auto ttl = TRY(decoder.decode<u32>());
  107. auto record_data = TRY(decoder.decode<ByteString>());
  108. auto cache_flush = TRY(decoder.decode<bool>());
  109. return DNS::Answer { name, record_type, class_code, ttl, record_data, cache_flush };
  110. }
  111. }