Class: Rex::Proto::DNS::Cache

Inherits:
Object
  • Object
show all
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

Instance Method Summary collapse

Constructor Details

#initializeCache

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

#lockObject (readonly)

Returns the value of attribute lock.



9
10
11
# File 'lib/rex/proto/dns/cache.rb', line 9

def lock
  @lock
end

#monitor_threadObject (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

#recordsObject (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

Parameters:

  • record (Dnsruby::RR)

    Record to add

  • expire (Fixnum) (defaults to: 0)

    Time in seconds when record becomes stale



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”

Parameters:

  • record (Dnsruby::RR)

    Record to cache



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

Parameters:

  • record (Dnsruby::RR)

    Record to delete



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

Parameters:

  • search (String)

    Name or address to search for

  • type (Dnsruby::Types) (defaults to: Dnsruby::Types::A)

    Record type to search for

Returns:

  • (Array)

    Records found



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

#flushObject

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

Parameters:

  • before (Fixnum) (defaults to: ::Time.now.to_i)

    Time in seconds before which records are evicted



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

#startObject

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

Parameters:

  • flush (TrueClass, FalseClass) (defaults to: false)

    Remove non-static entries



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