Class: Rex::Proto::DNS::Cache
- Inherits:
-
Object
- Object
- Rex::Proto::DNS::Cache
- Includes:
- Constants
- Defined in:
- lib/rex/proto/dns/cache.rb
Constant Summary
Constants included from Constants
Rex::Proto::DNS::Constants::MATCH_FQDN, Rex::Proto::DNS::Constants::MATCH_HOSTNAME
Instance Attribute Summary collapse
-
#lock ⇒ Object
readonly
Returns the value of attribute lock.
-
#monitor_thread ⇒ Object
readonly
Returns the value of attribute monitor_thread.
-
#records ⇒ Object
readonly
Returns the value of attribute records.
Instance Method Summary collapse
-
#add(record, expire = 0) ⇒ Object
protected
Add a record to the cache with thread safety.
-
#cache_record(record, expire: true) ⇒ Object
Add record to cache, only when “running”.
-
#delete(record) ⇒ Object
protected
Delete a record from the cache with thread safety.
-
#find(search, type = Dnsruby::Types::A) ⇒ Array
Find entries in cache, substituting names for ‘*’ in return.
-
#flush ⇒ Object
Delete all cache entries, this is different from pruning because the record’s expiration is ignored.
-
#initialize ⇒ Cache
constructor
class DNSRecordError < ::Exception.
-
#prune(before = ::Time.now.to_i) ⇒ Object
Prune cache entries.
-
#start ⇒ Object
Start the cache monitor.
-
#stop(flush = false) ⇒ Object
Stop the cache monitor.
Constructor Details
#initialize ⇒ Cache
class DNSRecordError < ::Exception
Create DNS cache
15 16 17 18 |
# File 'lib/rex/proto/dns/cache.rb', line 15 def initialize @records = {} @lock = Mutex.new end |
Instance Attribute Details
#lock ⇒ Object (readonly)
Returns the value of attribute lock.
9 10 11 |
# File 'lib/rex/proto/dns/cache.rb', line 9 def lock @lock end |
#monitor_thread ⇒ Object (readonly)
Returns the value of attribute monitor_thread.
9 10 11 |
# File 'lib/rex/proto/dns/cache.rb', line 9 def monitor_thread @monitor_thread end |
#records ⇒ Object (readonly)
Returns the value of attribute records.
9 10 11 |
# File 'lib/rex/proto/dns/cache.rb', line 9 def records @records end |
Instance Method Details
#add(record, expire = 0) ⇒ Object (protected)
Add a record to the cache with thread safety
117 118 119 120 121 |
# File 'lib/rex/proto/dns/cache.rb', line 117 def add(record, expire = 0) self.lock.synchronize do self.records[record] = expire end end |
#cache_record(record, expire: true) ⇒ Object
Add record to cache, only when “running”
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 |
# File 'lib/rex/proto/dns/cache.rb', line 48 def cache_record(record, expire: true) return unless @monitor_thread unless record.is_a?(Dnsruby::RR) raise "Invalid record for cache entry (not an RR) - #{record.inspect}" end unless (!record.respond_to?(:address) || Rex::Socket.is_ip_addr?(record.address.to_s)) raise "Invalid record for cache entry (no IP address) - #{record.inspect}" end unless record.name.to_s.match(MATCH_HOSTNAME) raise "Invalid record for cache entry (invalid hostname) - #{record.inspect}" end add(record, expire ? (::Time.now.to_i + record.ttl) : 0) end |
#delete(record) ⇒ Object (protected)
Delete a record from the cache with thread safety
127 128 129 130 131 |
# File 'lib/rex/proto/dns/cache.rb', line 127 def delete(record) self.lock.synchronize do self.records.delete(record) end end |
#find(search, type = Dnsruby::Types::A) ⇒ Array
Find entries in cache, substituting names for ‘*’ in return
27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 |
# File 'lib/rex/proto/dns/cache.rb', line 27 def find(search, type = Dnsruby::Types::A) self.records.select do |record,expire| record.type == type and (expire < 1 or expire > ::Time.now.to_i) and ( record.name == '*' or record.name.to_s == search.to_s or record.name.to_s[0..-2] == search.to_s or ( record.respond_to?(:address) and record.address.to_s == search.to_s ) ) end.keys.map do |record| if search.to_s.match(MATCH_HOSTNAME) and record.name == '*' record = Dnsruby::RR.create(name: name, type: type, address: address) else record end end end |
#flush ⇒ Object
Delete all cache entries, this is different from pruning because the record’s expiration is ignored
70 71 72 |
# File 'lib/rex/proto/dns/cache.rb', line 70 def flush self.records.each {|rec, _| delete(rec)} end |
#prune(before = ::Time.now.to_i) ⇒ Object
Prune cache entries
78 79 80 81 82 |
# File 'lib/rex/proto/dns/cache.rb', line 78 def prune(before = ::Time.now.to_i) self.records.select do |rec, expire| expire > 0 and expire < before end.each {|rec, exp| delete(rec)} end |
#start ⇒ Object
Start the cache monitor
87 88 89 90 91 92 93 94 |
# File 'lib/rex/proto/dns/cache.rb', line 87 def start @monitor_thread = Rex::ThreadFactory.spawn("DNSServerCacheMonitor", false) { while true prune Rex::ThreadSafe.sleep(0.5) end } unless @monitor_thread end |
#stop(flush = false) ⇒ Object
Stop the cache monitor
100 101 102 103 104 105 106 107 108 |
# File 'lib/rex/proto/dns/cache.rb', line 100 def stop(flush = false) self.monitor_thread.kill unless @monitor_thread.nil? @monitor_thread = nil if flush self.records.select do |rec, expire| rec.ttl > 0 end.each {|rec| delete(rec)} end end |