mirror of
https://github.com/DNSCrypt/dnscrypt-proxy.git
synced 2025-04-04 05:37:38 +03:00
Preliminary support for remote sources
This commit is contained in:
parent
3824d0527a
commit
a361aa52f3
20 changed files with 945 additions and 7 deletions
14
Gopkg.lock
generated
14
Gopkg.lock
generated
|
@ -31,6 +31,12 @@
|
|||
packages = ["."]
|
||||
revision = "6cf43fdfd7a228cf3003ae23d10ddbf65e85997b"
|
||||
|
||||
[[projects]]
|
||||
branch = "master"
|
||||
name = "github.com/dchest/safefile"
|
||||
packages = ["."]
|
||||
revision = "855e8d98f1852d48dde521e0522408d1fe7e836a"
|
||||
|
||||
[[projects]]
|
||||
branch = "master"
|
||||
name = "github.com/hashicorp/golang-lru"
|
||||
|
@ -46,6 +52,12 @@
|
|||
packages = ["."]
|
||||
revision = "8c253f4161c5b23a5fedd1d1ccee28d7ea312c6c"
|
||||
|
||||
[[projects]]
|
||||
branch = "master"
|
||||
name = "github.com/jedisct1/go-minisign"
|
||||
packages = ["."]
|
||||
revision = "f404c079ea5f0d4669fe617c553651f75167494e"
|
||||
|
||||
[[projects]]
|
||||
branch = "master"
|
||||
name = "github.com/jedisct1/xsecretbox"
|
||||
|
@ -87,6 +99,6 @@
|
|||
[solve-meta]
|
||||
analyzer-name = "dep"
|
||||
analyzer-version = 1
|
||||
inputs-digest = "03812a1a34033c2d39a6812a33e222c54aaa2f91d01e09072d71b7bd38dceaa3"
|
||||
inputs-digest = "9946fe30a0b048dbe5b8a10b28e8bffd7ec6dac56380db345cbd868862fe7f08"
|
||||
solver-name = "gps-cdcl"
|
||||
solver-version = 1
|
||||
|
|
14
Gopkg.toml
14
Gopkg.toml
|
@ -1,15 +1,19 @@
|
|||
[[constraint]]
|
||||
name = "github.com/BurntSushi/toml"
|
||||
version = "~0.3.0"
|
||||
version = "0.3.0"
|
||||
|
||||
[[constraint]]
|
||||
name = "github.com/VividCortex/ewma"
|
||||
version = "~1.1.0"
|
||||
version = "1.1.1"
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
name = "github.com/VividCortex/godaemon"
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
name = "github.com/dchest/safefile"
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
name = "github.com/hashicorp/golang-lru"
|
||||
|
@ -18,13 +22,17 @@
|
|||
branch = "master"
|
||||
name = "github.com/jedisct1/dlog"
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
name = "github.com/jedisct1/go-minisign"
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
name = "github.com/jedisct1/xsecretbox"
|
||||
|
||||
[[constraint]]
|
||||
name = "github.com/miekg/dns"
|
||||
version = "~1.0.0"
|
||||
version = "1.0.3"
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
|
|
|
@ -24,6 +24,7 @@ type Config struct {
|
|||
CacheMinTTL uint32 `toml:"cache_min_ttl"`
|
||||
CacheMaxTTL uint32 `toml:"cache_max_ttl"`
|
||||
ServersConfig map[string]ServerConfig `toml:"servers"`
|
||||
SourcesConfig map[string]SourceConfig `toml:"sources"`
|
||||
}
|
||||
|
||||
func newConfig() Config {
|
||||
|
@ -48,6 +49,14 @@ type ServerConfig struct {
|
|||
DNSSEC bool `toml:"dnssec"`
|
||||
}
|
||||
|
||||
type SourceConfig struct {
|
||||
URL string
|
||||
MinisignKeyStr string `toml:"minisign_key"`
|
||||
CacheFile string `toml:"cache_file"`
|
||||
FormatStr string `toml:"format"`
|
||||
RefreshDelay int `toml:"refresh_delay"`
|
||||
}
|
||||
|
||||
func ConfigLoad(proxy *Proxy, config_file string) error {
|
||||
configFile := flag.String("config", "dnscrypt-proxy.toml", "path to the configuration file")
|
||||
flag.Parse()
|
||||
|
@ -77,8 +86,33 @@ func ConfigLoad(proxy *Proxy, config_file string) error {
|
|||
config.ServerNames = append(config.ServerNames, serverName)
|
||||
}
|
||||
}
|
||||
if len(config.ServerNames) == 0 {
|
||||
return errors.New("No servers configured")
|
||||
for sourceName, source := range config.SourcesConfig {
|
||||
if source.URL == "" {
|
||||
return fmt.Errorf("Missing URL for source [%s]", sourceName)
|
||||
}
|
||||
if source.MinisignKeyStr == "" {
|
||||
return fmt.Errorf("Missing Minisign key for source [%s]", sourceName)
|
||||
}
|
||||
if source.CacheFile == "" {
|
||||
return fmt.Errorf("Missing cache file for source [%s]", sourceName)
|
||||
}
|
||||
if source.FormatStr == "" {
|
||||
return fmt.Errorf("Missing format for source [%s]", sourceName)
|
||||
}
|
||||
if source.RefreshDelay <= 0 {
|
||||
source.RefreshDelay = 24
|
||||
}
|
||||
source, err := NewSource(source.URL, source.MinisignKeyStr, source.CacheFile, source.FormatStr, time.Duration(source.RefreshDelay)*time.Hour)
|
||||
if err != nil {
|
||||
dlog.Criticalf("Unable use source [%s]: [%s]", sourceName, err)
|
||||
continue
|
||||
}
|
||||
registeredServers, err := source.Parse()
|
||||
if err != nil {
|
||||
dlog.Criticalf("Unable use source [%s]: [%s]", sourceName, err)
|
||||
continue
|
||||
}
|
||||
proxy.registeredServers = append(proxy.registeredServers, registeredServers...)
|
||||
}
|
||||
for _, serverName := range config.ServerNames {
|
||||
serverConfig, ok := config.ServersConfig[serverName]
|
||||
|
@ -98,5 +132,8 @@ func ConfigLoad(proxy *Proxy, config_file string) error {
|
|||
proxy.registeredServers = append(proxy.registeredServers,
|
||||
RegisteredServer{name: serverName, stamp: stamp})
|
||||
}
|
||||
if len(proxy.registeredServers) == 0 {
|
||||
return errors.New("No servers configured")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
|
|
@ -62,7 +62,7 @@ cache_size = 256
|
|||
|
||||
## Minimum TTL for cached entries
|
||||
|
||||
cache_min_ttl = 60
|
||||
cache_min_ttl = 600
|
||||
|
||||
|
||||
## Maxmimum TTL for cached entries
|
||||
|
@ -77,6 +77,15 @@ cache_neg_ttl = 60
|
|||
|
||||
############## Servers ##############
|
||||
|
||||
## Server sources
|
||||
[sources]
|
||||
[sources."github-csv"]
|
||||
url = "https://raw.githubusercontent.com/DNSCrypt/dnscrypt-resolvers/master/v1/dnscrypt-resolvers.csv"
|
||||
minisign_key = "RWQf6LRCGA9i53mlYecO4IzT51TGPpvWucNSCh1CBM0QTaLn73Y7GFO3"
|
||||
cache_file = "/tmp/dnscrypt-resolvers.csv"
|
||||
format = "v1"
|
||||
refresh_delay = 24
|
||||
|
||||
## Static list of available servers
|
||||
|
||||
[servers]
|
||||
|
|
|
@ -2,6 +2,7 @@ package main
|
|||
|
||||
import (
|
||||
"encoding/hex"
|
||||
"fmt"
|
||||
"math/rand"
|
||||
"net"
|
||||
"strings"
|
||||
|
@ -15,6 +16,7 @@ import (
|
|||
|
||||
const (
|
||||
RTTEwmaDecay = 10.0
|
||||
DefaultPort = 443
|
||||
)
|
||||
|
||||
type ServerStamp struct {
|
||||
|
@ -29,6 +31,9 @@ type RegisteredServer struct {
|
|||
}
|
||||
|
||||
func NewServerStampFromLegacy(serverAddrStr string, serverPkStr string, providerName string) (ServerStamp, error) {
|
||||
if strings.Contains(serverAddrStr, "]") && !strings.Contains(serverAddrStr, ":") {
|
||||
serverAddrStr = fmt.Sprintf("%s:d", serverAddrStr, DefaultPort)
|
||||
}
|
||||
return ServerStamp{
|
||||
serverAddrStr: serverAddrStr,
|
||||
serverPkStr: serverPkStr,
|
||||
|
|
154
dnscrypt-proxy/sources.go
Normal file
154
dnscrypt-proxy/sources.go
Normal file
|
@ -0,0 +1,154 @@
|
|||
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 := false, false
|
||||
fi, err := os.Stat(cacheFile)
|
||||
if err == nil {
|
||||
elapsed := time.Now().Sub(fi.ModTime())
|
||||
if elapsed < refreshDelay && elapsed >= 0 {
|
||||
usableCache = true
|
||||
}
|
||||
}
|
||||
if usableCache {
|
||||
bin, err = fetchFromCache(cacheFile)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
} else {
|
||||
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
|
||||
}
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
bin, err = ioutil.ReadAll(resp.Body)
|
||||
if err != nil {
|
||||
if usableCache {
|
||||
bin, err = fetchFromCache(cacheFile)
|
||||
}
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
}
|
||||
cached = true
|
||||
}
|
||||
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)
|
||||
if err != nil || res != true {
|
||||
return source, err
|
||||
}
|
||||
if cached == false {
|
||||
if err = AtomicFileWrite(cacheFile, []byte(in)); err != nil {
|
||||
return source, err
|
||||
}
|
||||
}
|
||||
if sigCached == false {
|
||||
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 line, record := range records {
|
||||
if len(record) < 14 {
|
||||
return registeredServers, fmt.Errorf("Parse error at line %d", line)
|
||||
}
|
||||
if line == 0 {
|
||||
continue
|
||||
}
|
||||
name := record[0]
|
||||
serverAddrStr := record[10]
|
||||
providerName := record[11]
|
||||
serverPkStr := record[12]
|
||||
stamp, err := NewServerStampFromLegacy(serverAddrStr, serverPkStr, providerName)
|
||||
if err != nil {
|
||||
return registeredServers, err
|
||||
}
|
||||
registeredServer := RegisteredServer{
|
||||
name: name, stamp: stamp,
|
||||
}
|
||||
registeredServers = append(registeredServers, registeredServer)
|
||||
}
|
||||
return registeredServers, nil
|
||||
}
|
8
vendor/github.com/dchest/safefile/.travis.yml
generated
vendored
Normal file
8
vendor/github.com/dchest/safefile/.travis.yml
generated
vendored
Normal file
|
@ -0,0 +1,8 @@
|
|||
language: go
|
||||
|
||||
go:
|
||||
- 1.1
|
||||
- 1.2
|
||||
- 1.3
|
||||
- 1.4
|
||||
- tip
|
26
vendor/github.com/dchest/safefile/LICENSE
generated
vendored
Normal file
26
vendor/github.com/dchest/safefile/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,26 @@
|
|||
Copyright (c) 2013 Dmitry Chestnykh <dmitry@codingrobots.com>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
|
||||
* Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
|
||||
* Redistributions in binary form must reproduce the above
|
||||
copyright notice, this list of conditions and the following
|
||||
disclaimer in the documentation and/or other materials
|
||||
provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
44
vendor/github.com/dchest/safefile/README.md
generated
vendored
Normal file
44
vendor/github.com/dchest/safefile/README.md
generated
vendored
Normal file
|
@ -0,0 +1,44 @@
|
|||
# safefile
|
||||
|
||||
[](https://travis-ci.org/dchest/safefile) [](https://ci.appveyor.com/project/dchest/safefile)
|
||||
|
||||
Go package safefile implements safe "atomic" saving of files.
|
||||
|
||||
Instead of truncating and overwriting the destination file, it creates a
|
||||
temporary file in the same directory, writes to it, and then renames the
|
||||
temporary file to the original name when calling Commit.
|
||||
|
||||
|
||||
### Installation
|
||||
|
||||
```
|
||||
$ go get github.com/dchest/safefile
|
||||
```
|
||||
|
||||
### Documentation
|
||||
|
||||
<https://godoc.org/github.com/dchest/safefile>
|
||||
|
||||
### Example
|
||||
|
||||
```go
|
||||
f, err := safefile.Create("/home/ken/report.txt", 0644)
|
||||
if err != nil {
|
||||
// ...
|
||||
}
|
||||
// Created temporary file /home/ken/sf-ppcyksu5hyw2mfec.tmp
|
||||
|
||||
defer f.Close()
|
||||
|
||||
_, err = io.WriteString(f, "Hello world")
|
||||
if err != nil {
|
||||
// ...
|
||||
}
|
||||
// Wrote "Hello world" to /home/ken/sf-ppcyksu5hyw2mfec.tmp
|
||||
|
||||
err = f.Commit()
|
||||
if err != nil {
|
||||
// ...
|
||||
}
|
||||
// Renamed /home/ken/sf-ppcyksu5hyw2mfec.tmp to /home/ken/report.txt
|
||||
```
|
24
vendor/github.com/dchest/safefile/appveyor.yml
generated
vendored
Normal file
24
vendor/github.com/dchest/safefile/appveyor.yml
generated
vendored
Normal file
|
@ -0,0 +1,24 @@
|
|||
version: "{build}"
|
||||
|
||||
os: Windows Server 2012 R2
|
||||
|
||||
clone_folder: c:\projects\src\github.com\dchest\safefile
|
||||
|
||||
environment:
|
||||
PATH: c:\projects\bin;%PATH%
|
||||
GOPATH: c:\projects
|
||||
NOTIFY_TIMEOUT: 5s
|
||||
|
||||
install:
|
||||
- go version
|
||||
- go get golang.org/x/tools/cmd/vet
|
||||
- go get -v -t ./...
|
||||
|
||||
build_script:
|
||||
- go tool vet -all .
|
||||
- go build ./...
|
||||
- go test -v -race ./...
|
||||
|
||||
test: off
|
||||
|
||||
deploy: off
|
9
vendor/github.com/dchest/safefile/rename.go
generated
vendored
Normal file
9
vendor/github.com/dchest/safefile/rename.go
generated
vendored
Normal file
|
@ -0,0 +1,9 @@
|
|||
// +build !plan9,!windows windows,go1.5
|
||||
|
||||
package safefile
|
||||
|
||||
import "os"
|
||||
|
||||
func rename(oldname, newname string) error {
|
||||
return os.Rename(oldname, newname)
|
||||
}
|
51
vendor/github.com/dchest/safefile/rename_nonatomic.go
generated
vendored
Normal file
51
vendor/github.com/dchest/safefile/rename_nonatomic.go
generated
vendored
Normal file
|
@ -0,0 +1,51 @@
|
|||
// +build plan9 windows,!go1.5
|
||||
|
||||
// os.Rename on Windows before Go 1.5 and Plan 9 will not overwrite existing
|
||||
// files, thus we cannot guarantee atomic saving of file by doing rename.
|
||||
// We will have to do some voodoo to minimize data loss on those systems.
|
||||
|
||||
package safefile
|
||||
|
||||
import (
|
||||
"os"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
func rename(oldname, newname string) error {
|
||||
err := os.Rename(oldname, newname)
|
||||
if err != nil {
|
||||
// If newname exists ("original"), we will try renaming it to a
|
||||
// new temporary name, then renaming oldname to the newname,
|
||||
// and deleting the renamed original. If system crashes between
|
||||
// renaming and deleting, the original file will still be available
|
||||
// under the temporary name, so users can manually recover data.
|
||||
// (No automatic recovery is possible because after crash the
|
||||
// temporary name is not known.)
|
||||
var origtmp string
|
||||
for {
|
||||
origtmp, err = makeTempName(newname, filepath.Base(newname))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
_, err = os.Stat(origtmp)
|
||||
if err == nil {
|
||||
continue // most likely will never happen
|
||||
}
|
||||
break
|
||||
}
|
||||
err = os.Rename(newname, origtmp)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err = os.Rename(oldname, newname)
|
||||
if err != nil {
|
||||
// Rename still fails, try to revert original rename,
|
||||
// ignoring errors.
|
||||
os.Rename(origtmp, newname)
|
||||
return err
|
||||
}
|
||||
// Rename succeeded, now delete original file.
|
||||
os.Remove(origtmp)
|
||||
}
|
||||
return nil
|
||||
}
|
197
vendor/github.com/dchest/safefile/safefile.go
generated
vendored
Normal file
197
vendor/github.com/dchest/safefile/safefile.go
generated
vendored
Normal file
|
@ -0,0 +1,197 @@
|
|||
// Copyright 2013 Dmitry Chestnykh. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
// Package safefile implements safe "atomic" saving of files.
|
||||
//
|
||||
// Instead of truncating and overwriting the destination file, it creates a
|
||||
// temporary file in the same directory, writes to it, and then renames the
|
||||
// temporary file to the original name when calling Commit.
|
||||
//
|
||||
// Example:
|
||||
//
|
||||
// f, err := safefile.Create("/home/ken/report.txt", 0644)
|
||||
// if err != nil {
|
||||
// // ...
|
||||
// }
|
||||
// // Created temporary file /home/ken/sf-ppcyksu5hyw2mfec.tmp
|
||||
//
|
||||
// defer f.Close()
|
||||
//
|
||||
// _, err = io.WriteString(f, "Hello world")
|
||||
// if err != nil {
|
||||
// // ...
|
||||
// }
|
||||
// // Wrote "Hello world" to /home/ken/sf-ppcyksu5hyw2mfec.tmp
|
||||
//
|
||||
// err = f.Commit()
|
||||
// if err != nil {
|
||||
// // ...
|
||||
// }
|
||||
// // Renamed /home/ken/sf-ppcyksu5hyw2mfec.tmp to /home/ken/report.txt
|
||||
//
|
||||
package safefile
|
||||
|
||||
import (
|
||||
"crypto/rand"
|
||||
"encoding/base32"
|
||||
"errors"
|
||||
"io"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// ErrAlreadyCommitted error is returned when calling Commit on a file that
|
||||
// has been already successfully committed.
|
||||
var ErrAlreadyCommitted = errors.New("file already committed")
|
||||
|
||||
type File struct {
|
||||
*os.File
|
||||
origName string
|
||||
closeFunc func(*File) error
|
||||
isClosed bool // if true, temporary file has been closed, but not renamed
|
||||
isCommitted bool // if true, the file has been successfully committed
|
||||
}
|
||||
|
||||
func makeTempName(origname, prefix string) (tempname string, err error) {
|
||||
origname = filepath.Clean(origname)
|
||||
if len(origname) == 0 || origname[len(origname)-1] == filepath.Separator {
|
||||
return "", os.ErrInvalid
|
||||
}
|
||||
// Generate 10 random bytes.
|
||||
// This gives 80 bits of entropy, good enough
|
||||
// for making temporary file name unpredictable.
|
||||
var rnd [10]byte
|
||||
if _, err := rand.Read(rnd[:]); err != nil {
|
||||
return "", err
|
||||
}
|
||||
name := prefix + "-" + strings.ToLower(base32.StdEncoding.EncodeToString(rnd[:])) + ".tmp"
|
||||
return filepath.Join(filepath.Dir(origname), name), nil
|
||||
}
|
||||
|
||||
// Create creates a temporary file in the same directory as filename,
|
||||
// which will be renamed to the given filename when calling Commit.
|
||||
func Create(filename string, perm os.FileMode) (*File, error) {
|
||||
for {
|
||||
tempname, err := makeTempName(filename, "sf")
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
f, err := os.OpenFile(tempname, os.O_RDWR|os.O_CREATE|os.O_EXCL, perm)
|
||||
if err != nil {
|
||||
if os.IsExist(err) {
|
||||
continue
|
||||
}
|
||||
return nil, err
|
||||
}
|
||||
return &File{
|
||||
File: f,
|
||||
origName: filename,
|
||||
closeFunc: closeUncommitted,
|
||||
}, nil
|
||||
}
|
||||
}
|
||||
|
||||
// OrigName returns the original filename given to Create.
|
||||
func (f *File) OrigName() string {
|
||||
return f.origName
|
||||
}
|
||||
|
||||
// Close closes temporary file and removes it.
|
||||
// If the file has been committed, Close is no-op.
|
||||
func (f *File) Close() error {
|
||||
return f.closeFunc(f)
|
||||
}
|
||||
|
||||
func closeUncommitted(f *File) error {
|
||||
err0 := f.File.Close()
|
||||
err1 := os.Remove(f.Name())
|
||||
f.closeFunc = closeAgainError
|
||||
if err0 != nil {
|
||||
return err0
|
||||
}
|
||||
return err1
|
||||
}
|
||||
|
||||
func closeAfterFailedRename(f *File) error {
|
||||
// Remove temporary file.
|
||||
//
|
||||
// The note from Commit function applies here too, as we may be
|
||||
// removing a different file. However, since we rely on our temporary
|
||||
// names being unpredictable, this should not be a concern.
|
||||
f.closeFunc = closeAgainError
|
||||
return os.Remove(f.Name())
|
||||
}
|
||||
|
||||
func closeCommitted(f *File) error {
|
||||
// noop
|
||||
return nil
|
||||
}
|
||||
|
||||
func closeAgainError(f *File) error {
|
||||
return os.ErrInvalid
|
||||
}
|
||||
|
||||
// Commit safely commits data into the original file by syncing temporary
|
||||
// file to disk, closing it and renaming to the original file name.
|
||||
//
|
||||
// In case of success, the temporary file is closed and no longer exists
|
||||
// on disk. It is safe to call Close after Commit: the operation will do
|
||||
// nothing.
|
||||
//
|
||||
// In case of error, the temporary file is still opened and exists on disk;
|
||||
// it must be closed by callers by calling Close or by trying to commit again.
|
||||
|
||||
// Note that when trying to Commit again after a failed Commit when the file
|
||||
// has been closed, but not renamed to its original name (the new commit will
|
||||
// try again to rename it), safefile cannot guarantee that the temporary file
|
||||
// has not been changed, or that it is the same temporary file we were dealing
|
||||
// with. However, since the temporary name is unpredictable, it is unlikely
|
||||
// that this happened accidentally. If complete atomicity is needed, do not
|
||||
// Commit again after error, write the file again.
|
||||
func (f *File) Commit() error {
|
||||
if f.isCommitted {
|
||||
return ErrAlreadyCommitted
|
||||
}
|
||||
if !f.isClosed {
|
||||
// Sync to disk.
|
||||
err := f.Sync()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
// Close underlying os.File.
|
||||
err = f.File.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
f.isClosed = true
|
||||
}
|
||||
// Rename.
|
||||
err := rename(f.Name(), f.origName)
|
||||
if err != nil {
|
||||
f.closeFunc = closeAfterFailedRename
|
||||
return err
|
||||
}
|
||||
f.closeFunc = closeCommitted
|
||||
f.isCommitted = true
|
||||
return nil
|
||||
}
|
||||
|
||||
// WriteFile is a safe analog of ioutil.WriteFile.
|
||||
func WriteFile(filename string, data []byte, perm os.FileMode) error {
|
||||
f, err := Create(filename, perm)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer f.Close()
|
||||
n, err := f.Write(data)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err == nil && n < len(data) {
|
||||
err = io.ErrShortWrite
|
||||
return err
|
||||
}
|
||||
return f.Commit()
|
||||
}
|
155
vendor/github.com/dchest/safefile/safefile_test.go
generated
vendored
Normal file
155
vendor/github.com/dchest/safefile/safefile_test.go
generated
vendored
Normal file
|
@ -0,0 +1,155 @@
|
|||
package safefile
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func ensureFileContains(name, data string) error {
|
||||
b, err := ioutil.ReadFile(name)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if string(b) != data {
|
||||
return fmt.Errorf("wrong data in file: expected %s, got %s", data, string(b))
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func tempFileName(count int) string {
|
||||
return filepath.Join(os.TempDir(), fmt.Sprintf("safefile-test-%d-%x", count, time.Now().UnixNano()))
|
||||
}
|
||||
|
||||
var testData = "Hello, this is a test data"
|
||||
|
||||
func testInTempDir() error {
|
||||
name := tempFileName(0)
|
||||
defer os.Remove(name)
|
||||
f, err := Create(name, 0666)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if name != f.OrigName() {
|
||||
f.Close()
|
||||
return fmt.Errorf("name %q differs from OrigName: %q", name, f.OrigName())
|
||||
}
|
||||
_, err = io.WriteString(f, testData)
|
||||
if err != nil {
|
||||
f.Close()
|
||||
return err
|
||||
}
|
||||
err = f.Commit()
|
||||
if err != nil {
|
||||
f.Close()
|
||||
return err
|
||||
}
|
||||
err = f.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return ensureFileContains(name, testData)
|
||||
}
|
||||
|
||||
func TestMakeTempName(t *testing.T) {
|
||||
// Make sure temp name is random.
|
||||
m := make(map[string]bool)
|
||||
for i := 0; i < 100; i++ {
|
||||
name, err := makeTempName("/tmp", "sf")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if m[name] {
|
||||
t.Fatal("repeated file name")
|
||||
}
|
||||
m[name] = true
|
||||
}
|
||||
}
|
||||
|
||||
func TestFile(t *testing.T) {
|
||||
err := testInTempDir()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestWriteFile(t *testing.T) {
|
||||
name := tempFileName(1)
|
||||
err := WriteFile(name, []byte(testData), 0666)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
err = ensureFileContains(name, testData)
|
||||
if err != nil {
|
||||
os.Remove(name)
|
||||
t.Fatal(err)
|
||||
}
|
||||
os.Remove(name)
|
||||
}
|
||||
|
||||
func TestAbandon(t *testing.T) {
|
||||
name := tempFileName(2)
|
||||
f, err := Create(name, 0666)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
err = f.Close()
|
||||
if err != nil {
|
||||
t.Fatalf("Abandon failed: %s", err)
|
||||
}
|
||||
// Make sure temporary file doesn't exist.
|
||||
_, err = os.Stat(f.Name())
|
||||
if err != nil && !os.IsNotExist(err) {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestDoubleCommit(t *testing.T) {
|
||||
name := tempFileName(3)
|
||||
f, err := Create(name, 0666)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
err = f.Commit()
|
||||
if err != nil {
|
||||
os.Remove(name)
|
||||
t.Fatalf("First commit failed: %s", err)
|
||||
}
|
||||
err = f.Commit()
|
||||
if err != ErrAlreadyCommitted {
|
||||
os.Remove(name)
|
||||
t.Fatalf("Second commit didn't fail: %s", err)
|
||||
}
|
||||
err = f.Close()
|
||||
if err != nil {
|
||||
os.Remove(name)
|
||||
t.Fatalf("Close failed: %s", err)
|
||||
}
|
||||
os.Remove(name)
|
||||
}
|
||||
|
||||
func TestOverwriting(t *testing.T) {
|
||||
name := tempFileName(4)
|
||||
defer os.Remove(name)
|
||||
|
||||
olddata := "This is old data"
|
||||
err := ioutil.WriteFile(name, []byte(olddata), 0600)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
newdata := "This is new data"
|
||||
err = WriteFile(name, []byte(newdata), 0600)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
err = ensureFileContains(name, newdata)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
14
vendor/github.com/jedisct1/go-minisign/.gitignore
generated
vendored
Normal file
14
vendor/github.com/jedisct1/go-minisign/.gitignore
generated
vendored
Normal file
|
@ -0,0 +1,14 @@
|
|||
# Binaries for programs and plugins
|
||||
*.exe
|
||||
*.dll
|
||||
*.so
|
||||
*.dylib
|
||||
|
||||
# Test binary, build with `go test -c`
|
||||
*.test
|
||||
|
||||
# Output of the go coverage tool, specifically when used with LiteIDE
|
||||
*.out
|
||||
|
||||
# Project-local glide cache, RE: https://github.com/Masterminds/glide/issues/736
|
||||
.glide/
|
18
vendor/github.com/jedisct1/go-minisign/Gopkg.lock
generated
vendored
Normal file
18
vendor/github.com/jedisct1/go-minisign/Gopkg.lock
generated
vendored
Normal file
|
@ -0,0 +1,18 @@
|
|||
# This file is autogenerated, do not edit; changes may be undone by the next 'dep ensure'.
|
||||
|
||||
|
||||
[[projects]]
|
||||
branch = "master"
|
||||
name = "golang.org/x/crypto"
|
||||
packages = [
|
||||
"ed25519",
|
||||
"ed25519/internal/edwards25519"
|
||||
]
|
||||
revision = "13931e22f9e72ea58bb73048bc752b48c6d4d4ac"
|
||||
|
||||
[solve-meta]
|
||||
analyzer-name = "dep"
|
||||
analyzer-version = 1
|
||||
inputs-digest = "5046e265393bd5e54f570ce29ae8bc6fa3f30ef5110e922996540400f287c64a"
|
||||
solver-name = "gps-cdcl"
|
||||
solver-version = 1
|
25
vendor/github.com/jedisct1/go-minisign/Gopkg.toml
generated
vendored
Normal file
25
vendor/github.com/jedisct1/go-minisign/Gopkg.toml
generated
vendored
Normal file
|
@ -0,0 +1,25 @@
|
|||
# Gopkg.toml example
|
||||
#
|
||||
# Refer to https://github.com/golang/dep/blob/master/docs/Gopkg.toml.md
|
||||
# for detailed Gopkg.toml documentation.
|
||||
#
|
||||
# required = ["github.com/user/thing/cmd/thing"]
|
||||
# ignored = ["github.com/user/project/pkgX", "bitbucket.org/user/project/pkgA/pkgY"]
|
||||
#
|
||||
# [[constraint]]
|
||||
# name = "github.com/user/project"
|
||||
# version = "1.0.0"
|
||||
#
|
||||
# [[constraint]]
|
||||
# name = "github.com/user/project2"
|
||||
# branch = "dev"
|
||||
# source = "github.com/myfork/project2"
|
||||
#
|
||||
# [[override]]
|
||||
# name = "github.com/x/y"
|
||||
# version = "2.4.0"
|
||||
|
||||
|
||||
[[constraint]]
|
||||
branch = "master"
|
||||
name = "golang.org/x/crypto"
|
21
vendor/github.com/jedisct1/go-minisign/LICENSE
generated
vendored
Normal file
21
vendor/github.com/jedisct1/go-minisign/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
|||
MIT License
|
||||
|
||||
Copyright (c) 2018 Frank Denis
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
4
vendor/github.com/jedisct1/go-minisign/README.md
generated
vendored
Normal file
4
vendor/github.com/jedisct1/go-minisign/README.md
generated
vendored
Normal file
|
@ -0,0 +1,4 @@
|
|||
# go-minisign
|
||||
|
||||
A Golang library to verify [Minisign](https://jedisct1.github.io/minisign/) signatures.
|
||||
|
117
vendor/github.com/jedisct1/go-minisign/minisign.go
generated
vendored
Normal file
117
vendor/github.com/jedisct1/go-minisign/minisign.go
generated
vendored
Normal file
|
@ -0,0 +1,117 @@
|
|||
package minisign
|
||||
|
||||
import (
|
||||
"encoding/base64"
|
||||
"errors"
|
||||
"io/ioutil"
|
||||
"strings"
|
||||
|
||||
"golang.org/x/crypto/ed25519"
|
||||
)
|
||||
|
||||
type PublicKey struct {
|
||||
SignatureAlgorithm [2]byte
|
||||
KeyId [8]byte
|
||||
PublicKey [32]byte
|
||||
}
|
||||
|
||||
type Signature struct {
|
||||
UntrustedComment string
|
||||
SignatureAlgorithm [2]byte
|
||||
KeyId [8]byte
|
||||
Signature [64]byte
|
||||
TrustedComment string
|
||||
GlobalSignature [64]byte
|
||||
}
|
||||
|
||||
func NewPublicKey(publicKeyStr string) (PublicKey, error) {
|
||||
var publicKey PublicKey
|
||||
bin, err := base64.StdEncoding.DecodeString(publicKeyStr)
|
||||
if err != nil || len(bin) != 42 {
|
||||
return publicKey, errors.New("Invalid encoded public key")
|
||||
}
|
||||
copy(publicKey.SignatureAlgorithm[:], bin[0:2])
|
||||
copy(publicKey.KeyId[:], bin[2:10])
|
||||
copy(publicKey.PublicKey[:], bin[10:42])
|
||||
return publicKey, nil
|
||||
}
|
||||
|
||||
func DecodePublicKey(in string) (PublicKey, error) {
|
||||
var publicKey PublicKey
|
||||
lines := strings.SplitN(in, "\n", 2)
|
||||
if len(lines) < 2 {
|
||||
return publicKey, errors.New("Incomplete encoded public key")
|
||||
}
|
||||
return NewPublicKey(lines[1])
|
||||
}
|
||||
|
||||
func DecodeSignature(in string) (Signature, error) {
|
||||
var signature Signature
|
||||
lines := strings.SplitN(in, "\n", 4)
|
||||
if len(lines) < 4 {
|
||||
return signature, errors.New("Incomplete encoded signature")
|
||||
}
|
||||
signature.UntrustedComment = lines[0]
|
||||
bin1, err := base64.StdEncoding.DecodeString(lines[1])
|
||||
if err != nil || len(bin1) != 74 {
|
||||
return signature, errors.New("Invalid encoded signature")
|
||||
}
|
||||
signature.TrustedComment = lines[2]
|
||||
bin2, err := base64.StdEncoding.DecodeString(lines[3])
|
||||
if err != nil || len(bin2) != 64 {
|
||||
return signature, errors.New("Invalid encoded signature")
|
||||
}
|
||||
copy(signature.SignatureAlgorithm[:], bin1[0:2])
|
||||
copy(signature.KeyId[:], bin1[2:10])
|
||||
copy(signature.Signature[:], bin1[10:74])
|
||||
copy(signature.GlobalSignature[:], bin2)
|
||||
return signature, nil
|
||||
}
|
||||
|
||||
func NewPublicKeyFromFile(file string) (PublicKey, error) {
|
||||
var publicKey PublicKey
|
||||
bin, err := ioutil.ReadFile(file)
|
||||
if err != nil {
|
||||
return publicKey, err
|
||||
}
|
||||
return DecodePublicKey(string(bin))
|
||||
}
|
||||
|
||||
func NewSignatureFromFile(file string) (Signature, error) {
|
||||
var signature Signature
|
||||
bin, err := ioutil.ReadFile(file)
|
||||
if err != nil {
|
||||
return signature, err
|
||||
}
|
||||
return DecodeSignature(string(bin))
|
||||
}
|
||||
|
||||
func (publicKey *PublicKey) Verify(bin []byte, signature Signature) (bool, error) {
|
||||
if publicKey.SignatureAlgorithm != signature.SignatureAlgorithm {
|
||||
return false, errors.New("Incompatible signature algorithm")
|
||||
}
|
||||
if signature.SignatureAlgorithm[0] != 0x45 && signature.SignatureAlgorithm[1] != 0x64 {
|
||||
return false, errors.New("Unsupported signature algorithm")
|
||||
}
|
||||
if publicKey.KeyId != signature.KeyId {
|
||||
return false, errors.New("Incompatible key identifiers")
|
||||
}
|
||||
if !strings.HasPrefix(signature.TrustedComment, "trusted comment: ") {
|
||||
return false, errors.New("Unexpected format for the trusted comment")
|
||||
}
|
||||
if !ed25519.Verify(ed25519.PublicKey(publicKey.PublicKey[:]), bin, signature.Signature[:]) {
|
||||
return false, errors.New("Invalid signature")
|
||||
}
|
||||
if !ed25519.Verify(ed25519.PublicKey(publicKey.PublicKey[:]), append(signature.Signature[:], []byte(signature.TrustedComment)[17:]...), signature.GlobalSignature[:]) {
|
||||
return false, errors.New("Invalid global signature")
|
||||
}
|
||||
return true, nil
|
||||
}
|
||||
|
||||
func (publicKey *PublicKey) VerifyFromFile(file string, signature Signature) (bool, error) {
|
||||
bin, err := ioutil.ReadFile(file)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
return publicKey.Verify(bin, signature)
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue