1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
|
// Copyright (C) 2010 Internet Systems Consortium, Inc. ("ISC")
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice appear in all copies.
//
// THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
// REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
// AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
// INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
// LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
// OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
// PERFORMANCE OF THIS SOFTWARE.
#ifndef MESSAGE_ENTRY_H
#define MESSAGE_ENTRY_H
#include <vector>
#include <dns/message.h>
#include <dns/rrset.h>
#include <nsas/nsas_entry.h>
#include "rrset_cache.h"
#include "rrset_entry.h"
namespace isc {
namespace cache {
class RRsetEntry;
/// \brief Message Entry
///
/// The object of MessageEntry represents one response message
/// answered to the resolver client.
class MessageEntry : public isc::nsas::NsasEntry<MessageEntry> {
// Noncopyable
private:
MessageEntry(const MessageEntry& source);
MessageEntry& operator=(const MessageEntry& source);
/// \brief Information to refer an RRset.
///
/// There is no class information here, since the rrsets are cached in
/// the class-specific rrset cache.
struct RRsetRef{
/// \brief Constructor
///
/// \param name The Name for the RRset
/// \param type The RRType for the RRrset
/// \param cache Which cache the RRset is stored in
RRsetRef(const isc::dns::Name& name, const isc::dns::RRType& type,
RRsetCache* cache):
name_(name), type_(type), cache_(cache)
{}
isc::dns::Name name_; // Name of rrset.
isc::dns::RRType type_; // Type of rrset.
RRsetCache* cache_; //Which cache the RRset is stored
};
public:
/// \brief Initialize the message entry object with one dns
/// message.
/// \param message The message used to initialize MessageEntry.
/// \param rrset_cache the pointer of RRsetCache. When one message
/// entry is created, rrset cache needs to be updated,
/// since some new rrset entries may be inserted into
/// rrset cache, or the existed rrset entries need
/// to be updated.
/// \param negative_soa_cache the pointer of RRsetCache. This
/// cache is used only for storing SOA rrset from negative
/// response (NXDOMAIN or NOERROR_NODATA)
MessageEntry(const isc::dns::Message& message,
const RRsetCachePtr& rrset_cache,
const RRsetCachePtr& negative_soa_cache);
~MessageEntry() { delete hash_key_ptr_; };
/// \brief generate one dns message according
/// the rrsets information of the message.
///
/// \param time_now set the ttl of each rrset in the message
/// as "expire_time - time_now" (expire_time is the
/// expiration time of the rrset).
/// \param response generated dns message.
/// \return return true if the response message can be generated
/// from the cached information, or else, return false.
bool genMessage(const time_t& time_now, isc::dns::Message& response);
/// \brief Get the hash key of the message entry.
///
/// \return return hash key
virtual isc::nsas::HashKey hashKey() const {
return (*hash_key_ptr_);
}
/// \brief Get expire time of the message entry.
/// \return return the expire time of message entry.
time_t getExpireTime() const {
return (expire_time_);
}
/// \short Protected memebers, so they can be accessed by tests.
//@{
protected:
/// \brief Initialize the message entry with dns message.
///
/// \param message The Message to initialize the entry with
void initMessageEntry(const isc::dns::Message& message);
/// \brief Parse the rrsets in specified section.
///
/// \param msg The message to parse the RRsets from
/// \param section The Section to parse the RRsets from
/// \param smaller_ttl Get the smallest ttl of rrsets in
/// specified section, if it's smaller than the given value.
/// \param rrset_count the rrset count of the section.
/// (TODO for Message, getRRsetCount() should be one
/// interface provided by Message.)
void parseSection(const isc::dns::Message& msg,
const isc::dns::Message::Section& section,
uint32_t& smaller_ttl,
uint16_t& rrset_count);
/// \brief Parse the RRsets in the authority section of
/// negative response. The SOA RRset need to be located and
/// stored in a seperate cache
/// \param msg The message to parse the RRsets from
/// \param min_ttl Get the minimum ttl of rrset in the authority section
/// \param rrset_count the rrset count of the authority section
void parseNegativeResponseAuthoritySection(const isc::dns::Message& msg,
uint32_t& min_ttl,
uint16_t& rrset_count);
/// \brief Get RRset Trustworthiness
/// The algorithm refers to RFC2181 section 5.4.1
/// Only the rrset can be updated by the rrsets
/// with higher trust level.
///
/// \param message Message that the rrset belongs to
/// \param rrset specified rrset which needs to get its
/// trust worthiness
/// \param section Section of the rrset
/// \return return rrset trust level.
RRsetTrustLevel getRRsetTrustLevel(const isc::dns::Message& message,
const isc::dns::RRsetPtr& rrset,
const isc::dns::Message::Section& section);
/// \brief Add rrset to one section of message.
///
/// \param message The message to add rrsets to.
/// \param rrset_entry_vec vector for rrset entries in
/// different sections.
/// \param section The section to add to
void addRRset(isc::dns::Message& message,
const std::vector<RRsetEntryPtr>& rrset_entry_vec,
const isc::dns::Message::Section& section);
/// \brief Get the all the rrset entries for the message entry.
///
/// \param rrset_entry_vec vector to add unexpired rrset entries to
/// \param time_now the time of now. Used to compare with rrset
/// entry's expire time.
/// \return return false if any rrset entry has expired, true
/// otherwise.
bool getRRsetEntries(std::vector<RRsetEntryPtr>& rrset_entry_vec,
const time_t time_now);
time_t expire_time_; // Expiration time of the message.
//@}
private:
std::string entry_name_; // The name for this entry(name + type)
isc::nsas::HashKey* hash_key_ptr_; // the key for messag entry in hash table.
std::vector<RRsetRef> rrsets_;
RRsetCachePtr rrset_cache_; //Normal rrset cache
// SOA rrset from negative response
RRsetCachePtr negative_soa_cache_;
std::string query_name_; // query name of the message.
uint16_t query_class_; // query class of the message.
uint16_t query_type_; // query type of message.
uint16_t query_count_; // query count in query section.
uint16_t answer_count_; // rrset count in answer section.
uint16_t authority_count_; // rrset count in authority section.
uint16_t additional_count_; // rrset count in addition section.
//TODO, there should be a better way to cache these header flags
bool headerflag_aa_; // Whether AA bit is set.
bool headerflag_tc_; // Whether TC bit is set.
};
typedef boost::shared_ptr<MessageEntry> MessageEntryPtr;
} // namespace cache
} // namespace isc
#endif // MESSAGE_ENTRY_H
|