Answer.cpp 3.6 KB

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