mirror of
https://github.com/DNSCrypt/dnscrypt-proxy.git
synced 2025-04-05 14:17:36 +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 = ["."]
|
packages = ["."]
|
||||||
revision = "6cf43fdfd7a228cf3003ae23d10ddbf65e85997b"
|
revision = "6cf43fdfd7a228cf3003ae23d10ddbf65e85997b"
|
||||||
|
|
||||||
|
[[projects]]
|
||||||
|
branch = "master"
|
||||||
|
name = "github.com/dchest/safefile"
|
||||||
|
packages = ["."]
|
||||||
|
revision = "855e8d98f1852d48dde521e0522408d1fe7e836a"
|
||||||
|
|
||||||
[[projects]]
|
[[projects]]
|
||||||
branch = "master"
|
branch = "master"
|
||||||
name = "github.com/hashicorp/golang-lru"
|
name = "github.com/hashicorp/golang-lru"
|
||||||
|
@ -46,6 +52,12 @@
|
||||||
packages = ["."]
|
packages = ["."]
|
||||||
revision = "8c253f4161c5b23a5fedd1d1ccee28d7ea312c6c"
|
revision = "8c253f4161c5b23a5fedd1d1ccee28d7ea312c6c"
|
||||||
|
|
||||||
|
[[projects]]
|
||||||
|
branch = "master"
|
||||||
|
name = "github.com/jedisct1/go-minisign"
|
||||||
|
packages = ["."]
|
||||||
|
revision = "f404c079ea5f0d4669fe617c553651f75167494e"
|
||||||
|
|
||||||
[[projects]]
|
[[projects]]
|
||||||
branch = "master"
|
branch = "master"
|
||||||
name = "github.com/jedisct1/xsecretbox"
|
name = "github.com/jedisct1/xsecretbox"
|
||||||
|
@ -87,6 +99,6 @@
|
||||||
[solve-meta]
|
[solve-meta]
|
||||||
analyzer-name = "dep"
|
analyzer-name = "dep"
|
||||||
analyzer-version = 1
|
analyzer-version = 1
|
||||||
inputs-digest = "03812a1a34033c2d39a6812a33e222c54aaa2f91d01e09072d71b7bd38dceaa3"
|
inputs-digest = "9946fe30a0b048dbe5b8a10b28e8bffd7ec6dac56380db345cbd868862fe7f08"
|
||||||
solver-name = "gps-cdcl"
|
solver-name = "gps-cdcl"
|
||||||
solver-version = 1
|
solver-version = 1
|
||||||
|
|
14
Gopkg.toml
14
Gopkg.toml
|
@ -1,15 +1,19 @@
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
name = "github.com/BurntSushi/toml"
|
name = "github.com/BurntSushi/toml"
|
||||||
version = "~0.3.0"
|
version = "0.3.0"
|
||||||
|
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
name = "github.com/VividCortex/ewma"
|
name = "github.com/VividCortex/ewma"
|
||||||
version = "~1.1.0"
|
version = "1.1.1"
|
||||||
|
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
branch = "master"
|
branch = "master"
|
||||||
name = "github.com/VividCortex/godaemon"
|
name = "github.com/VividCortex/godaemon"
|
||||||
|
|
||||||
|
[[constraint]]
|
||||||
|
branch = "master"
|
||||||
|
name = "github.com/dchest/safefile"
|
||||||
|
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
branch = "master"
|
branch = "master"
|
||||||
name = "github.com/hashicorp/golang-lru"
|
name = "github.com/hashicorp/golang-lru"
|
||||||
|
@ -18,13 +22,17 @@
|
||||||
branch = "master"
|
branch = "master"
|
||||||
name = "github.com/jedisct1/dlog"
|
name = "github.com/jedisct1/dlog"
|
||||||
|
|
||||||
|
[[constraint]]
|
||||||
|
branch = "master"
|
||||||
|
name = "github.com/jedisct1/go-minisign"
|
||||||
|
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
branch = "master"
|
branch = "master"
|
||||||
name = "github.com/jedisct1/xsecretbox"
|
name = "github.com/jedisct1/xsecretbox"
|
||||||
|
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
name = "github.com/miekg/dns"
|
name = "github.com/miekg/dns"
|
||||||
version = "~1.0.0"
|
version = "1.0.3"
|
||||||
|
|
||||||
[[constraint]]
|
[[constraint]]
|
||||||
branch = "master"
|
branch = "master"
|
||||||
|
|
|
@ -24,6 +24,7 @@ type Config struct {
|
||||||
CacheMinTTL uint32 `toml:"cache_min_ttl"`
|
CacheMinTTL uint32 `toml:"cache_min_ttl"`
|
||||||
CacheMaxTTL uint32 `toml:"cache_max_ttl"`
|
CacheMaxTTL uint32 `toml:"cache_max_ttl"`
|
||||||
ServersConfig map[string]ServerConfig `toml:"servers"`
|
ServersConfig map[string]ServerConfig `toml:"servers"`
|
||||||
|
SourcesConfig map[string]SourceConfig `toml:"sources"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func newConfig() Config {
|
func newConfig() Config {
|
||||||
|
@ -48,6 +49,14 @@ type ServerConfig struct {
|
||||||
DNSSEC bool `toml:"dnssec"`
|
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 {
|
func ConfigLoad(proxy *Proxy, config_file string) error {
|
||||||
configFile := flag.String("config", "dnscrypt-proxy.toml", "path to the configuration file")
|
configFile := flag.String("config", "dnscrypt-proxy.toml", "path to the configuration file")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
@ -77,8 +86,33 @@ func ConfigLoad(proxy *Proxy, config_file string) error {
|
||||||
config.ServerNames = append(config.ServerNames, serverName)
|
config.ServerNames = append(config.ServerNames, serverName)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if len(config.ServerNames) == 0 {
|
for sourceName, source := range config.SourcesConfig {
|
||||||
return errors.New("No servers configured")
|
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 {
|
for _, serverName := range config.ServerNames {
|
||||||
serverConfig, ok := config.ServersConfig[serverName]
|
serverConfig, ok := config.ServersConfig[serverName]
|
||||||
|
@ -98,5 +132,8 @@ func ConfigLoad(proxy *Proxy, config_file string) error {
|
||||||
proxy.registeredServers = append(proxy.registeredServers,
|
proxy.registeredServers = append(proxy.registeredServers,
|
||||||
RegisteredServer{name: serverName, stamp: stamp})
|
RegisteredServer{name: serverName, stamp: stamp})
|
||||||
}
|
}
|
||||||
|
if len(proxy.registeredServers) == 0 {
|
||||||
|
return errors.New("No servers configured")
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,7 +62,7 @@ cache_size = 256
|
||||||
|
|
||||||
## Minimum TTL for cached entries
|
## Minimum TTL for cached entries
|
||||||
|
|
||||||
cache_min_ttl = 60
|
cache_min_ttl = 600
|
||||||
|
|
||||||
|
|
||||||
## Maxmimum TTL for cached entries
|
## Maxmimum TTL for cached entries
|
||||||
|
@ -77,6 +77,15 @@ cache_neg_ttl = 60
|
||||||
|
|
||||||
############## Servers ##############
|
############## 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
|
## Static list of available servers
|
||||||
|
|
||||||
[servers]
|
[servers]
|
||||||
|
|
|
@ -2,6 +2,7 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
|
"fmt"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"net"
|
"net"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -15,6 +16,7 @@ import (
|
||||||
|
|
||||||
const (
|
const (
|
||||||
RTTEwmaDecay = 10.0
|
RTTEwmaDecay = 10.0
|
||||||
|
DefaultPort = 443
|
||||||
)
|
)
|
||||||
|
|
||||||
type ServerStamp struct {
|
type ServerStamp struct {
|
||||||
|
@ -29,6 +31,9 @@ type RegisteredServer struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewServerStampFromLegacy(serverAddrStr string, serverPkStr string, providerName string) (ServerStamp, error) {
|
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{
|
return ServerStamp{
|
||||||
serverAddrStr: serverAddrStr,
|
serverAddrStr: serverAddrStr,
|
||||||
serverPkStr: serverPkStr,
|
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