2020-10-08 22:14:07 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/gob"
|
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/miekg/dns"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
type cacheItem struct {
|
|
|
|
Query []byte
|
2021-09-16 09:40:03 +02:00
|
|
|
ExpireDate time.Time
|
2020-10-08 22:14:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//DomainCache stores a domain name inside the cache
|
|
|
|
func DomainCache(s string, resp *dns.Msg) {
|
|
|
|
|
|
|
|
var domain2cache cacheItem
|
|
|
|
var err error
|
|
|
|
var dom2 bytes.Buffer
|
|
|
|
enc := gob.NewEncoder(&dom2)
|
|
|
|
|
|
|
|
domain2cache.Query, err = resp.Pack()
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Problems packing the response: ", err.Error())
|
|
|
|
}
|
2021-09-16 09:40:03 +02:00
|
|
|
if resp.Rcode == dns.RcodeSuccess{
|
|
|
|
// on success stores response normally
|
|
|
|
domain2cache.ExpireDate = time.Now().Add((time.Duration(ZabovCacheTTL) * time.Hour))
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
// on failure stores response for a very short time
|
|
|
|
if ZabovDebug {
|
|
|
|
fmt.Println("DomainCache(): DNS error Rcode: ", resp.Rcode, s, "cache time reduced to 10 seconds...")
|
|
|
|
}
|
|
|
|
domain2cache.ExpireDate = time.Now().Add((time.Duration(10) * time.Second))
|
|
|
|
}
|
2020-10-08 22:14:07 +02:00
|
|
|
|
|
|
|
err = enc.Encode(domain2cache)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Cannot GOB the domain to cache: ", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
cacheDomain(s, dom2.Bytes())
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func cacheDomain(key string, domain []byte) {
|
|
|
|
|
|
|
|
err := MyZabovCDB.Put([]byte(key), domain, nil)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Cannot write to Cache DB: ", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//GetDomainFromCache stores a domain name inside the cache
|
|
|
|
func GetDomainFromCache(s string) *dns.Msg {
|
|
|
|
|
|
|
|
ret := new(dns.Msg)
|
|
|
|
var cache bytes.Buffer
|
|
|
|
dec := gob.NewDecoder(&cache)
|
|
|
|
var record cacheItem
|
|
|
|
var conf []byte
|
|
|
|
var errDB error
|
|
|
|
|
|
|
|
if domainInCache(s) == false {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
conf, errDB = MyZabovCDB.Get([]byte(s), nil)
|
|
|
|
if errDB != nil {
|
2021-09-16 09:40:03 +02:00
|
|
|
fmt.Println("Cant READ DB:" , errDB.Error() )
|
2020-10-08 22:14:07 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cache.Write(conf)
|
|
|
|
|
|
|
|
err := dec.Decode(&record)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Decode error :", err.Error())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-09-16 09:40:03 +02:00
|
|
|
if time.Now().After(record.ExpireDate) {
|
|
|
|
if ZabovDebug {
|
|
|
|
fmt.Println("GetDomainFromCache(): entry expired:", s)
|
|
|
|
}
|
2020-10-08 22:14:07 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ret.Unpack(record.Query)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Problem unpacking response: ", err.Error())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func domainInCache(domain string) bool {
|
|
|
|
|
|
|
|
has, err := MyZabovCDB.Has([]byte(domain), nil)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Cannot search Cache DB: ", err.Error())
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return has
|
|
|
|
|
|
|
|
}
|