dnscrypt-proxy/dnscrypt-proxy/sources.go

167 lines
3.8 KiB
Go
Raw Normal View History

2018-01-13 23:52:44 +01:00
package main
import (
"encoding/csv"
"fmt"
"io/ioutil"
"net/http"
"os"
"strings"
"time"
"github.com/dchest/safefile"
"github.com/jedisct1/dlog"
"github.com/jedisct1/go-minisign"
)
type SourceFormat int
const (
SourceFormatV1 = iota
)
type Source struct {
url string
format SourceFormat
in string
}
func fetchFromCache(cacheFile string) ([]byte, error) {
dlog.Infof("Loading source information from cache file [%s]", cacheFile)
return ioutil.ReadFile(cacheFile)
}
func fetchWithCache(url string, cacheFile string, refreshDelay time.Duration) (in string, cached bool, err error) {
var bin []byte
cached, usableCache, hotCache := false, false, false
2018-01-13 23:52:44 +01:00
fi, err := os.Stat(cacheFile)
if err == nil {
usableCache = true
2018-01-14 23:53:17 +01:00
elapsed := time.Since(fi.ModTime())
2018-01-13 23:52:44 +01:00
if elapsed < refreshDelay && elapsed >= 0 {
hotCache = true
2018-01-13 23:52:44 +01:00
}
}
if hotCache {
2018-01-13 23:52:44 +01:00
bin, err = fetchFromCache(cacheFile)
2018-01-14 00:20:22 +01:00
if err == nil {
cached = true
2018-01-13 23:52:44 +01:00
}
2018-01-14 00:20:22 +01:00
}
if !cached {
2018-01-13 23:52:44 +01:00
var resp *http.Response
dlog.Infof("Loading source information from URL [%s]", url)
resp, err = http.Get(url)
if err != nil {
if usableCache {
bin, err = fetchFromCache(cacheFile)
}
if err != nil {
return
}
2018-01-16 00:37:04 +01:00
} else {
bin, err = ioutil.ReadAll(resp.Body)
resp.Body.Close()
2018-01-13 23:52:44 +01:00
if err != nil {
2018-01-16 00:37:04 +01:00
if usableCache {
bin, err = fetchFromCache(cacheFile)
}
if err != nil {
return
}
2018-01-13 23:52:44 +01:00
}
}
}
in = string(bin)
return
}
func AtomicFileWrite(file string, data []byte) error {
return safefile.WriteFile(file, data, 0644)
}
func NewSource(url string, minisignKeyStr string, cacheFile string, formatStr string, refreshDelay time.Duration) (Source, error) {
source := Source{url: url}
if formatStr != "v1" {
return source, fmt.Errorf("Unsupported source format: [%s]", formatStr)
}
source.format = SourceFormatV1
minisignKey, err := minisign.NewPublicKey(minisignKeyStr)
if err != nil {
return source, err
}
in, cached, err := fetchWithCache(url, cacheFile, refreshDelay)
if err != nil {
return source, err
}
sigCacheFile := cacheFile + ".minisig"
sigURL := url + ".minisig"
sigStr, sigCached, err := fetchWithCache(sigURL, sigCacheFile, refreshDelay)
if err != nil {
return source, err
}
signature, err := minisign.DecodeSignature(sigStr)
if err != nil {
return source, err
}
res, err := minisignKey.Verify([]byte(in), signature)
2018-01-18 14:28:05 +01:00
if err != nil || !res {
2018-01-13 23:52:44 +01:00
return source, err
}
2018-01-18 14:28:05 +01:00
if !cached {
2018-01-13 23:52:44 +01:00
if err = AtomicFileWrite(cacheFile, []byte(in)); err != nil {
return source, err
}
}
2018-01-18 14:28:05 +01:00
if !sigCached {
2018-01-13 23:52:44 +01:00
if err = AtomicFileWrite(sigCacheFile, []byte(sigStr)); err != nil {
return source, err
}
}
dlog.Noticef("Source [%s] loaded", url)
source.in = in
return source, nil
}
func (source *Source) Parse() ([]RegisteredServer, error) {
var registeredServers []RegisteredServer
csvReader := csv.NewReader(strings.NewReader(source.in))
records, err := csvReader.ReadAll()
if err != nil {
return registeredServers, nil
}
for lineNo, record := range records {
2018-01-13 23:53:33 +01:00
if len(record) == 0 {
continue
}
2018-01-13 23:52:44 +01:00
if len(record) < 14 {
2018-01-17 09:44:03 +01:00
return registeredServers, fmt.Errorf("Parse error at line %d", 1+lineNo)
2018-01-13 23:52:44 +01:00
}
if lineNo == 0 {
2018-01-13 23:52:44 +01:00
continue
}
name := record[0]
serverAddrStr := record[10]
providerName := record[11]
serverPkStr := record[12]
props := ServerInformalProperties(0)
if strings.EqualFold(record[7], "yes") {
props |= ServerInformalPropertyDNSSEC
}
if strings.EqualFold(record[8], "yes") {
props |= ServerInformalPropertyNoLog
}
stamp, err := NewServerStampFromLegacy(serverAddrStr, serverPkStr, providerName, props)
2018-01-13 23:52:44 +01:00
if err != nil {
return registeredServers, err
}
registeredServer := RegisteredServer{
name: name, stamp: stamp,
}
registeredServers = append(registeredServers, registeredServer)
}
return registeredServers, nil
}