diff --git a/Gopkg.lock b/Gopkg.lock index 05eff1a..25ae11f 100644 --- a/Gopkg.lock +++ b/Gopkg.lock @@ -2,12 +2,20 @@ [[projects]] - digest = "1:b16fbfbcc20645cb419f78325bb2e85ec729b338e996a228124d68931a6f2a37" + digest = "1:9f3b30d9f8e0d7040f729b82dcbc8f0dead820a133b3147ce355fc451f32d761" name = "github.com/BurntSushi/toml" packages = ["."] pruneopts = "UT" - revision = "b26d9c308763d68093482582cea63d69be07a0f0" - version = "v0.3.0" + revision = "3012a1dbe2e4bd1391d42b32f0577cb7bbc7f005" + version = "v0.3.1" + +[[projects]] + branch = "master" + digest = "1:038001d3925d508051d48a4e5e3e097015fa627ec9e7fc0c57e6f6ca7dc33c5e" + name = "github.com/ScaleFT/sshkeys" + packages = ["."] + pruneopts = "UT" + revision = "82451a80368171b074c7129d43b47fc2773f6e9f" [[projects]] digest = "1:ffe9824d294da03b391f44e1ae8281281b4afc1bdaa9588c9097785e3af10cec" @@ -17,6 +25,14 @@ revision = "8991bc29aa16c548c550c7ff78260e27b9ab7c73" version = "v1.1.1" +[[projects]] + branch = "master" + digest = "1:75a25fddc76a45f923f04b28f07827f49aaf6d51db4a56ec89a7cfb53e1601ab" + name = "github.com/dchest/bcrypt_pbkdf" + packages = ["."] + pruneopts = "UT" + revision = "83f37f9c154a678179d11e218bff73ebe5717f99" + [[projects]] branch = "master" digest = "1:36fe9527deed01d2a317617e59304eb2c4ce9f8a24115bcc5c2e37b3aee5bae4" @@ -132,9 +148,10 @@ [[projects]] branch = "master" - digest = "1:dec72d8441e4e88bbf27cc98e28e962e065ebd11bcd42063f0444020b7249618" + digest = "1:2c0831757b96a0e62b2b0b851f139f50190c1e3efa6c6a7eac6f48e1a3d8f576" name = "golang.org/x/crypto" packages = [ + "blowfish", "curve25519", "ed25519", "ed25519/internal/edwards25519", @@ -144,13 +161,16 @@ "ssh", ] pruneopts = "UT" - revision = "182538f80094b6a8efaade63a8fd8e0d9d5843dd" + revision = "8986dd9e96cf0a6f74da406c005ba3df38527c04" [[projects]] branch = "master" - digest = "1:4b487c782bc804d994e91adbd3d2a8a77a482671efd87b2fde0805adb01a39c0" + digest = "1:0cd3b4a6aec2641ff2bf7e35d93427787c60e5d94998460aab8f54921a1bc2db" name = "golang.org/x/sys" - packages = ["unix"] + packages = [ + "cpu", + "unix", + ] pruneopts = "UT" revision = "fa5fdf94c78965f1aa8423f0cc50b8b8d728b05a" @@ -174,6 +194,7 @@ analyzer-version = 1 input-imports = [ "github.com/BurntSushi/toml", + "github.com/ScaleFT/sshkeys", "github.com/gin-gonic/gin", "github.com/jessevdk/go-flags", "github.com/jmoiron/sqlx", diff --git a/Gopkg.toml b/Gopkg.toml index ed46f1d..e644da5 100644 --- a/Gopkg.toml +++ b/Gopkg.toml @@ -68,3 +68,7 @@ [prune] go-tests = true unused-packages = true + +[[constraint]] + branch = "master" + name = "github.com/ScaleFT/sshkeys" diff --git a/pkg/api/api.go b/pkg/api/api.go index 765d8ef..92fab89 100644 --- a/pkg/api/api.go +++ b/pkg/api/api.go @@ -138,7 +138,7 @@ func Connect(c *gin.Context) { return } - opts := command.Options{Url: url} + opts := command.Options{URL: url} url, err := connection.FormatURL(opts) if err != nil { diff --git a/pkg/api/helpers.go b/pkg/api/helpers.go index 03980fd..7470402 100644 --- a/pkg/api/helpers.go +++ b/pkg/api/helpers.go @@ -115,11 +115,12 @@ func parseIntFormValue(c *gin.Context, name string, defValue int) (int, error) { func parseSshInfo(c *gin.Context) *shared.SSHInfo { info := shared.SSHInfo{ - Host: c.Request.FormValue("ssh_host"), - Port: c.Request.FormValue("ssh_port"), - User: c.Request.FormValue("ssh_user"), - Password: c.Request.FormValue("ssh_password"), - Key: c.Request.FormValue("ssh_key"), + Host: c.Request.FormValue("ssh_host"), + Port: c.Request.FormValue("ssh_port"), + User: c.Request.FormValue("ssh_user"), + Password: c.Request.FormValue("ssh_password"), + Key: c.Request.FormValue("ssh_key"), + KeyPassword: c.Request.FormValue("ssh_key_password"), } if info.Port == "" { diff --git a/pkg/bookmarks/bookmarks.go b/pkg/bookmarks/bookmarks.go index 91a85b1..2648b04 100644 --- a/pkg/bookmarks/bookmarks.go +++ b/pkg/bookmarks/bookmarks.go @@ -13,24 +13,27 @@ import ( "github.com/sosedoff/pgweb/pkg/shared" ) +// Bookmark contains information about bookmarked database connection type Bookmark struct { - Url string `json:"url"` // Postgres connection URL + URL string `json:"url"` // Postgres connection URL Host string `json:"host"` // Server hostname Port int `json:"port"` // Server port User string `json:"user"` // Database user Password string `json:"password"` // User password Database string `json:"database"` // Database name Ssl string `json:"ssl"` // Connection SSL mode - Ssh *shared.SSHInfo `json:"ssh"` // SSH tunnel config + SSH *shared.SSHInfo `json:"ssh"` // SSH tunnel config } +// SSHInfoIsEmpty returns true if ssh configration is not provided func (b Bookmark) SSHInfoIsEmpty() bool { - return b.Ssh == nil || b.Ssh.User == "" && b.Ssh.Host == "" && b.Ssh.Port == "" + return b.SSH == nil || b.SSH.User == "" && b.SSH.Host == "" && b.SSH.Port == "" } +// ConvertToOptions returns an options struct from connection details func (b Bookmark) ConvertToOptions() command.Options { return command.Options{ - Url: b.Url, + URL: b.URL, Host: b.Host, Port: b.Port, User: b.User, @@ -71,8 +74,9 @@ func readServerConfig(path string) (Bookmark, error) { bookmark.Ssl = "disable" } - if bookmark.Ssh != nil && bookmark.Ssh.Port == "" { - bookmark.Ssh.Port = "22" + // Set default SSH port if it's not provided by user + if bookmark.SSH != nil && bookmark.SSH.Port == "" { + bookmark.SSH.Port = "22" } return bookmark, err @@ -83,15 +87,16 @@ func fileBasename(path string) string { return strings.Replace(filename, filepath.Ext(path), "", 1) } +// Path returns bookmarks storage path func Path(overrideDir string) string { if overrideDir == "" { path, _ := homedir.Dir() return fmt.Sprintf("%s/.pgweb/bookmarks", path) } - return overrideDir } +// ReadAll returns all available bookmarks func ReadAll(path string) (map[string]Bookmark, error) { results := map[string]Bookmark{} @@ -105,7 +110,7 @@ func ReadAll(path string) (map[string]Bookmark, error) { continue } - fullPath := path + "/" + file.Name() + fullPath := filepath.Join(path, file.Name()) key := fileBasename(file.Name()) config, err := readServerConfig(fullPath) @@ -120,6 +125,7 @@ func ReadAll(path string) (map[string]Bookmark, error) { return results, nil } +// GetBookmark reads an existing bookmark func GetBookmark(bookmarkPath string, bookmarkName string) (Bookmark, error) { bookmarks, err := ReadAll(bookmarkPath) if err != nil { diff --git a/pkg/bookmarks/bookmarks_test.go b/pkg/bookmarks/bookmarks_test.go index 5c02949..0b43497 100644 --- a/pkg/bookmarks/bookmarks_test.go +++ b/pkg/bookmarks/bookmarks_test.go @@ -26,7 +26,7 @@ func Test_Bookmark(t *testing.T) { assert.Equal(t, "mydatabase", bookmark.Database) assert.Equal(t, "disable", bookmark.Ssl) assert.Equal(t, "", bookmark.Password) - assert.Equal(t, "", bookmark.Url) + assert.Equal(t, "", bookmark.URL) bookmark, err = readServerConfig("../../data/bookmark_invalid_ssl.toml") assert.Equal(t, nil, err) @@ -37,7 +37,7 @@ func Test_Bookmark_URL(t *testing.T) { bookmark, err := readServerConfig("../../data/bookmark_url.toml") assert.Equal(t, nil, err) - assert.Equal(t, "postgres://username:password@host:port/database?sslmode=disable", bookmark.Url) + assert.Equal(t, "postgres://username:password@host:port/database?sslmode=disable", bookmark.URL) assert.Equal(t, "", bookmark.Host) assert.Equal(t, 5432, bookmark.Port) assert.Equal(t, "", bookmark.User) @@ -106,19 +106,19 @@ func Test_Bookmark_SSHInfoIsEmpty(t *testing.T) { User: "postgres", } - b := Bookmark{Ssh: nil} + b := Bookmark{SSH: nil} assert.True(t, b.SSHInfoIsEmpty()) - b = Bookmark{Ssh: emptySSH} + b = Bookmark{SSH: emptySSH} assert.True(t, b.SSHInfoIsEmpty()) - b.Ssh = populatedSSH + b.SSH = populatedSSH assert.False(t, b.SSHInfoIsEmpty()) } func Test_ConvertToOptions(t *testing.T) { b := Bookmark{ - Url: "postgres://username:password@host:port/database?sslmode=disable", + URL: "postgres://username:password@host:port/database?sslmode=disable", Host: "localhost", Port: 5432, User: "postgres", @@ -128,7 +128,7 @@ func Test_ConvertToOptions(t *testing.T) { } expOpt := command.Options{ - Url: "postgres://username:password@host:port/database?sslmode=disable", + URL: "postgres://username:password@host:port/database?sslmode=disable", Host: "localhost", Port: 5432, User: "postgres", diff --git a/pkg/cli/cli.go b/pkg/cli/cli.go index 99b26f2..a30dfd7 100644 --- a/pkg/cli/cli.go +++ b/pkg/cli/cli.go @@ -44,8 +44,8 @@ func initClientUsingBookmark(bookmarkPath, bookmarkName string) (*client.Client, opt := bookmark.ConvertToOptions() var connStr string - if opt.Url != "" { // if the bookmark has url set, use it - connStr = opt.Url + if opt.URL != "" { // if the bookmark has url set, use it + connStr = opt.URL } else { connStr, err = connection.BuildStringFromOptions(opt) if err != nil { @@ -55,7 +55,7 @@ func initClientUsingBookmark(bookmarkPath, bookmarkName string) (*client.Client, var ssh *shared.SSHInfo if !bookmark.SSHInfoIsEmpty() { - ssh = bookmark.Ssh + ssh = bookmark.SSH } return client.NewFromUrl(connStr, ssh) @@ -87,7 +87,7 @@ func initClient() { msg := err.Error() // Check if we're trying to connect to the default database. - if command.Opts.DbName == "" && command.Opts.Url == "" { + if command.Opts.DbName == "" && command.Opts.URL == "" { // If database does not exist, allow user to connect from the UI. if strings.Contains(msg, "database") && strings.Contains(msg, "does not exist") { fmt.Println("Error:", msg) @@ -164,7 +164,7 @@ func startServer() { fmt.Println("Starting server...") go func() { - err := router.Run(fmt.Sprintf("%v:%v", options.HttpHost, options.HttpPort)) + err := router.Run(fmt.Sprintf("%v:%v", options.HTTPHost, options.HTTPPort)) if err != nil { fmt.Println("Cant start server:", err) if strings.Contains(err.Error(), "address already in use") { @@ -182,7 +182,7 @@ func handleSignals() { } func openPage() { - url := fmt.Sprintf("http://%v:%v/%s", options.HttpHost, options.HttpPort, options.Prefix) + url := fmt.Sprintf("http://%v:%v/%s", options.HTTPHost, options.HTTPPort, options.Prefix) fmt.Println("To view database open", url, "in browser") if options.SkipOpen { diff --git a/pkg/client/client_test.go b/pkg/client/client_test.go index 1e5967f..4d3f59d 100644 --- a/pkg/client/client_test.go +++ b/pkg/client/client_test.go @@ -78,8 +78,8 @@ func onWindows() bool { } func setup() { - // No pretty JSON for testsm - command.Opts.DisablePrettyJson = true + // No pretty JSON for tests + command.Opts.DisablePrettyJSON = true out, err := exec.Command( testCommands["createdb"], diff --git a/pkg/client/result.go b/pkg/client/result.go index 96c9255..80cfb5f 100644 --- a/pkg/client/result.go +++ b/pkg/client/result.go @@ -121,7 +121,7 @@ func (res *Result) CSV() []byte { func (res *Result) JSON() []byte { var data []byte - if command.Opts.DisablePrettyJson { + if command.Opts.DisablePrettyJSON { data, _ = json.Marshal(res.Format()) } else { data, _ = json.MarshalIndent(res.Format(), "", " ") diff --git a/pkg/client/tunnel.go b/pkg/client/tunnel.go index 94a937d..784c01e 100644 --- a/pkg/client/tunnel.go +++ b/pkg/client/tunnel.go @@ -1,6 +1,7 @@ package client import ( + "errors" "fmt" "io" "io/ioutil" @@ -12,6 +13,7 @@ import ( "sync" "time" + "github.com/ScaleFT/sshkeys" "golang.org/x/crypto/ssh" "github.com/sosedoff/pgweb/pkg/connection" @@ -51,13 +53,22 @@ func fileExists(path string) bool { return err == nil } -func parsePrivateKey(keyPath string) (ssh.Signer, error) { +func parsePrivateKey(keyPath string, keyPass string) (ssh.Signer, error) { buff, err := ioutil.ReadFile(keyPath) if err != nil { return nil, err } - return ssh.ParsePrivateKey(buff) + signer, err := ssh.ParsePrivateKey(buff) + if err != nil { + if strings.Contains(err.Error(), "cannot decode encrypted private keys") { + if keyPass == "" { + return nil, errors.New("SSH key password is not provided") + } + return sshkeys.ParseEncryptedPrivateKey(buff, []byte(keyPass)) + } + } + return signer, err } func makeConfig(info *shared.SSHInfo) (*ssh.ClientConfig, error) { @@ -71,16 +82,21 @@ func makeConfig(info *shared.SSHInfo) (*ssh.ClientConfig, error) { keyPath = expandKeyPath(keyPath) } - if fileExists(keyPath) { - key, err := parsePrivateKey(keyPath) - if err != nil { - return nil, err - } - - methods = append(methods, ssh.PublicKeys(key)) + if !fileExists(keyPath) { + return nil, errors.New("ssh public key not found at " + keyPath) } - methods = append(methods, ssh.Password(info.Password)) + // Appen public key authentication method + key, err := parsePrivateKey(keyPath, info.KeyPassword) + if err != nil { + return nil, err + } + methods = append(methods, ssh.PublicKeys(key)) + + // Append password authentication method + if info.Password != "" { + methods = append(methods, ssh.Password(info.Password)) + } cfg := &ssh.ClientConfig{ User: info.User, diff --git a/pkg/command/options.go b/pkg/command/options.go index 47b4340..7eeaf83 100644 --- a/pkg/command/options.go +++ b/pkg/command/options.go @@ -12,15 +12,15 @@ import ( type Options struct { Version bool `short:"v" long:"version" description:"Print version"` Debug bool `short:"d" long:"debug" description:"Enable debugging mode"` - Url string `long:"url" description:"Database connection string"` + URL string `long:"url" description:"Database connection string"` Host string `long:"host" description:"Server hostname or IP" default:"localhost"` Port int `long:"port" description:"Server port" default:"5432"` User string `long:"user" description:"Database user"` Pass string `long:"pass" description:"Password for user"` DbName string `long:"db" description:"Database name"` Ssl string `long:"ssl" description:"SSL option"` - HttpHost string `long:"bind" description:"HTTP server host" default:"localhost"` - HttpPort uint `long:"listen" description:"HTTP server listen port" default:"8081"` + HTTPHost string `long:"bind" description:"HTTP server host" default:"localhost"` + HTTPPort uint `long:"listen" description:"HTTP server listen port" default:"8081"` AuthUser string `long:"auth-user" description:"HTTP basic auth user"` AuthPass string `long:"auth-pass" description:"HTTP basic auth password"` SkipOpen bool `short:"s" long:"skip-open" description:"Skip browser open on start"` @@ -30,7 +30,7 @@ type Options struct { LockSession bool `long:"lock-session" description:"Lock session to a single database connection"` Bookmark string `short:"b" long:"bookmark" description:"Bookmark to use for connection. Bookmark files are stored under $HOME/.pgweb/bookmarks/*.toml" default:""` BookmarksDir string `long:"bookmarks-dir" description:"Overrides default directory for bookmark files to search" default:""` - DisablePrettyJson bool `long:"no-pretty-json" description:"Disable JSON formatting feature for result export"` + DisablePrettyJSON bool `long:"no-pretty-json" description:"Disable JSON formatting feature for result export"` DisableSSH bool `long:"no-ssh" description:"Disable database connections via SSH"` ConnectBackend string `long:"connect-backend" description:"Enable database authentication through a third party backend"` ConnectToken string `long:"connect-token" description:"Authentication token for the third-party connect backend"` @@ -51,8 +51,8 @@ func ParseOptions(args []string) (Options, error) { return opts, err } - if opts.Url == "" { - opts.Url = os.Getenv("DATABASE_URL") + if opts.URL == "" { + opts.URL = os.Getenv("DATABASE_URL") } if opts.Prefix == "" { @@ -80,7 +80,7 @@ func ParseOptions(args []string) (Options, error) { if opts.Sessions || opts.ConnectBackend != "" { opts.Bookmark = "" - opts.Url = "" + opts.URL = "" opts.Host = "" opts.User = "" opts.Pass = "" diff --git a/pkg/command/options_test.go b/pkg/command/options_test.go index 0d2d1b8..961370c 100644 --- a/pkg/command/options_test.go +++ b/pkg/command/options_test.go @@ -15,7 +15,7 @@ func TestParseOptions(t *testing.T) { assert.Equal(t, "", opts.ConnectToken) assert.Equal(t, "", opts.ConnectHeaders) assert.Equal(t, false, opts.DisableSSH) - assert.Equal(t, false, opts.DisablePrettyJson) + assert.Equal(t, false, opts.DisablePrettyJSON) assert.Equal(t, false, opts.DisableConnectionIdleTimeout) assert.Equal(t, 180, opts.ConnectionIdleTimeout) assert.Equal(t, false, opts.Cors) diff --git a/pkg/connection/connection_string.go b/pkg/connection/connection_string.go index 5a06f80..4ea141b 100644 --- a/pkg/connection/connection_string.go +++ b/pkg/connection/connection_string.go @@ -48,7 +48,7 @@ func valsFromQuery(vals neturl.Values) map[string]string { // FormatURL reformats the existing connection string func FormatURL(opts command.Options) (string, error) { - url := opts.Url + url := opts.URL // Validate connection string prefix if !hasValidPrefix(url) { @@ -88,13 +88,13 @@ func FormatURL(opts command.Options) (string, error) { // IsBlank returns true if command options do not contain connection details func IsBlank(opts command.Options) bool { - return opts.Host == "" && opts.User == "" && opts.DbName == "" && opts.Url == "" + return opts.Host == "" && opts.User == "" && opts.DbName == "" && opts.URL == "" } // BuildStringFromOptions returns a new connection string built from options func BuildStringFromOptions(opts command.Options) (string, error) { // If connection string is provided we just use that - if opts.Url != "" { + if opts.URL != "" { return FormatURL(opts) } diff --git a/pkg/connection/connection_string_test.go b/pkg/connection/connection_string_test.go index de109ea..5b4b004 100644 --- a/pkg/connection/connection_string_test.go +++ b/pkg/connection/connection_string_test.go @@ -19,7 +19,7 @@ func Test_Invalid_Url(t *testing.T) { } for _, val := range examples { - opts.Url = val + opts.URL = val str, err := BuildStringFromOptions(opts) assert.Equal(t, "", str) @@ -30,7 +30,7 @@ func Test_Invalid_Url(t *testing.T) { func Test_Valid_Url(t *testing.T) { url := "postgres://myhost/database" - str, err := BuildStringFromOptions(command.Options{Url: url}) + str, err := BuildStringFromOptions(command.Options{URL: url}) assert.Equal(t, nil, err) assert.Equal(t, url, str) @@ -38,7 +38,7 @@ func Test_Valid_Url(t *testing.T) { func Test_Url_And_Ssl_Flag(t *testing.T) { str, err := BuildStringFromOptions(command.Options{ - Url: "postgres://myhost/database", + URL: "postgres://myhost/database", Ssl: "disable", }) @@ -48,13 +48,13 @@ func Test_Url_And_Ssl_Flag(t *testing.T) { func Test_Localhost_Url_And_No_Ssl_Flag(t *testing.T) { str, err := BuildStringFromOptions(command.Options{ - Url: "postgres://localhost/database", + URL: "postgres://localhost/database", }) assert.Equal(t, nil, err) assert.Equal(t, "postgres://localhost/database?sslmode=disable", str) str, err = BuildStringFromOptions(command.Options{ - Url: "postgres://127.0.0.1/database", + URL: "postgres://127.0.0.1/database", }) assert.Equal(t, nil, err) assert.Equal(t, "postgres://127.0.0.1/database?sslmode=disable", str) @@ -62,14 +62,14 @@ func Test_Localhost_Url_And_No_Ssl_Flag(t *testing.T) { func Test_Localhost_Url_And_Ssl_Flag(t *testing.T) { str, err := BuildStringFromOptions(command.Options{ - Url: "postgres://localhost/database", + URL: "postgres://localhost/database", Ssl: "require", }) assert.Equal(t, nil, err) assert.Equal(t, "postgres://localhost/database?sslmode=require", str) str, err = BuildStringFromOptions(command.Options{ - Url: "postgres://127.0.0.1/database", + URL: "postgres://127.0.0.1/database", Ssl: "require", }) assert.Equal(t, nil, err) @@ -78,13 +78,13 @@ func Test_Localhost_Url_And_Ssl_Flag(t *testing.T) { func Test_Localhost_Url_And_Ssl_Arg(t *testing.T) { str, err := BuildStringFromOptions(command.Options{ - Url: "postgres://localhost/database?sslmode=require", + URL: "postgres://localhost/database?sslmode=require", }) assert.Equal(t, nil, err) assert.Equal(t, "postgres://localhost/database?sslmode=require", str) str, err = BuildStringFromOptions(command.Options{ - Url: "postgres://127.0.0.1/database?sslmode=require", + URL: "postgres://127.0.0.1/database?sslmode=require", }) assert.Equal(t, nil, err) assert.Equal(t, "postgres://127.0.0.1/database?sslmode=require", str) @@ -159,5 +159,5 @@ func Test_Blank(t *testing.T) { assert.Equal(t, true, IsBlank(command.Options{})) assert.Equal(t, false, IsBlank(command.Options{Host: "host", User: "user"})) assert.Equal(t, false, IsBlank(command.Options{Host: "host", User: "user", DbName: "db"})) - assert.Equal(t, false, IsBlank(command.Options{Url: "url"})) + assert.Equal(t, false, IsBlank(command.Options{URL: "url"})) } diff --git a/pkg/data/bindata.go b/pkg/data/bindata.go index e955f89..fb4892e 100644 --- a/pkg/data/bindata.go +++ b/pkg/data/bindata.go @@ -1,32 +1,30 @@ -// Code generated by go-bindata. DO NOT EDIT. +// Package data Code generated by go-bindata. (@generated) DO NOT EDIT. // sources: -// static/css/app.css (12.874kB) -// static/css/bootstrap.css (109.518kB) -// static/css/font-awesome.css (21.984kB) -// static/fonts/FontAwesome.otf (85.908kB) -// static/fonts/fontawesome-webfont.eot (56.006kB) -// static/fonts/fontawesome-webfont.svg (287.007kB) -// static/fonts/fontawesome-webfont.ttf (112.16kB) -// static/fonts/fontawesome-webfont.woff (65.452kB) -// static/img/icon.ico (104.736kB) -// static/img/icon.png (7.945kB) -// static/index.html (13.106kB) -// static/js/ace-pgsql.js (53.342kB) -// static/js/ace.js (327.872kB) -// static/js/app.js (38.933kB) -// static/js/base64.js (3.517kB) -// static/js/bootstrap-contextmenu.js (5.3kB) -// static/js/bootstrap3-typeahead.min.js (9.335kB) -// static/js/jquery.js (84.245kB) -// static/js/theme-tomorrow.js (2.556kB) -// static/js/utils.js (831B) - +// static/css/app.css +// static/css/bootstrap.css +// static/css/font-awesome.css +// static/fonts/FontAwesome.otf +// static/fonts/fontawesome-webfont.eot +// static/fonts/fontawesome-webfont.svg +// static/fonts/fontawesome-webfont.ttf +// static/fonts/fontawesome-webfont.woff +// static/img/icon.ico +// static/img/icon.png +// static/index.html +// static/js/ace-pgsql.js +// static/js/ace.js +// static/js/app.js +// static/js/base64.js +// static/js/bootstrap-contextmenu.js +// static/js/bootstrap3-typeahead.min.js +// static/js/jquery.js +// static/js/theme-tomorrow.js +// static/js/utils.js package data import ( "bytes" "compress/gzip" - "crypto/sha256" "fmt" "io" "io/ioutil" @@ -57,9 +55,8 @@ func bindataRead(data []byte, name string) ([]byte, error) { } type asset struct { - bytes []byte - info os.FileInfo - digest [sha256.Size]byte + bytes []byte + info os.FileInfo } type bindataFileInfo struct { @@ -69,26 +66,37 @@ type bindataFileInfo struct { modTime time.Time } +// Name return file name func (fi bindataFileInfo) Name() string { return fi.name } + +// Size return file size func (fi bindataFileInfo) Size() int64 { return fi.size } + +// Mode return file mode func (fi bindataFileInfo) Mode() os.FileMode { return fi.mode } + +// ModTime return file modify time func (fi bindataFileInfo) ModTime() time.Time { return fi.modTime } + +// IsDir return file whether a directory func (fi bindataFileInfo) IsDir() bool { - return false + return fi.mode&os.ModeDir != 0 } + +// Sys return file is sys mode func (fi bindataFileInfo) Sys() interface{} { return nil } -var _staticCssAppCss = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xcc\x5a\x5b\x8f\xe4\xa8\x15\x7e\xef\x5f\x41\xa6\x14\x6d\x32\x1a\x6a\xea\xda\x5d\xed\xd1\x46\x8a\xa2\xec\x6b\x1e\xf6\x71\xb5\x1a\x51\x06\xb7\x49\xbb\xc0\x0b\xb8\xab\x7a\x46\xfb\xdf\x23\x30\xb6\x01\x83\xed\x9e\x9d\x44\x99\x92\x76\xdb\x17\xce\x39\x9c\xcb\x77\x2e\x78\x75\x41\x94\x81\xaf\x77\x00\x60\x2a\xeb\x0a\xbd\x66\x80\x71\x46\x3e\xdd\xfd\x7e\x77\xb7\x62\xe8\xc5\x3c\xaa\xb9\xa4\x8a\x72\x96\x81\x82\xde\x08\xfe\x74\x07\x80\xe2\x75\x06\x36\xfa\xaf\x8a\x14\x2a\x03\xbb\xe3\xa6\xbe\xe9\x4b\x41\x9f\x4a\x65\x1f\x95\xa4\xbd\xe8\x9e\x9d\x51\xfe\xfc\x24\x78\xc3\x70\x06\x56\x0f\x8f\xc7\xd3\x63\x31\x30\x6a\x2a\xc3\xeb\x82\xc4\x13\x65\x19\xb0\x4b\x6a\x84\x31\x65\x4f\xfd\x75\x48\xb2\x97\xfa\x5c\xf1\xfc\xd9\xa3\x56\xd1\x25\x04\x8b\x8a\x23\x95\x99\x5d\xb8\xf4\x0f\x8f\xed\xe3\x8a\x32\x02\xc3\x9b\x05\x67\x0a\x4a\xfa\x85\x64\x60\xbb\x1f\x93\x05\xdb\x43\x7b\x33\xe7\x15\x17\x19\x58\xe1\xfb\x3c\x27\xbb\x7e\xe5\xb5\xdf\x83\xd1\x52\x2f\x20\xd8\xda\x65\x8d\x90\x7a\x5d\xcd\x29\x53\x44\xc4\xb6\xa9\x05\x93\x0a\x4a\xf5\x5a\x91\xce\x62\xee\x3d\xa8\x5e\x6b\x12\x9a\xd2\xe8\x24\x2b\xa8\x90\x0a\xe6\x25\xad\xb0\xa3\x1f\xd8\x9a\xd1\x28\xc5\x7f\xbf\xe4\x2f\x44\x98\x37\xbb\xed\x14\x45\x11\xbc\xb4\x96\xa4\x22\xb9\x22\x38\x70\x17\x41\x2a\xa4\xe8\x0b\x19\x19\xdf\x90\x18\x28\x6e\x02\x4d\xa4\xc8\x67\x67\x52\x70\x41\xac\x34\x4c\x11\xa6\x32\xf0\x03\xf8\xe1\x93\xc7\x15\x9d\x25\xaf\x1a\xd5\x72\xe5\x4a\xf1\x4b\x06\xa0\xd5\xad\xdd\xe6\xa7\xa8\x9f\xda\x77\xae\x14\xf3\xab\xcc\xc0\x76\xb3\xf9\x73\x5c\xf2\x84\x74\xa1\x6a\xcf\x5c\x60\x22\xac\x6a\x07\x5b\x48\x8a\xc9\x19\x89\x54\x68\x5d\x29\x56\xa5\x13\x51\x8e\x65\xba\xb8\xb3\xe1\x64\xf7\xd6\x5f\xde\xb4\x53\x1a\x2f\xb4\x9c\xcf\x7c\x1c\x77\xc5\xa3\xfe\xf9\x92\xac\xf3\x46\x08\xc2\x14\xc4\x48\xa1\x33\x92\x24\x25\x5b\x52\x96\x50\xe8\x30\x4e\xbd\x38\xea\x6e\x7a\xd1\x70\xe6\x15\x0e\x83\x2b\x88\x23\xeb\x35\x51\x18\x01\xe0\x0b\xa4\x0c\x93\x5b\x06\x1e\x37\x9b\xb9\xed\xad\xaf\x02\xd5\x13\xce\xea\x4b\x3a\x4d\x4b\xd6\x88\x8d\xee\x42\x86\x2e\x64\x1c\x5f\xfb\x10\xb6\x28\x33\x9a\xe9\xc3\xba\xd3\xe3\x26\xa1\xc7\x6b\x49\x15\x81\xb2\x46\xb9\x09\x6f\xbd\x0d\x7d\x5b\xc7\x68\x51\xf1\x6b\x06\x4a\x8a\x31\x61\xc6\x3c\xe4\xa6\xe0\xf0\x80\x54\x15\xad\x25\x95\x51\x84\x99\xde\x21\x5d\x17\x28\xe5\x1a\x6e\xb0\xb5\x7b\x3c\x39\xde\xb1\x3d\xcd\xeb\x8f\xb2\xba\x51\x6b\x0d\x58\xa8\x24\x28\x02\x4a\xff\x17\x4a\x7b\xfb\x1e\x82\x84\x9a\xf4\xdb\x94\xf4\xbb\x5d\xd4\xfb\xbd\xa0\x61\x5c\x5c\x50\x65\x36\xd3\x28\xad\x14\x87\x9b\x81\x80\x36\x21\x8d\xb1\x27\xe1\xc6\x82\x14\x82\xc8\x72\xc2\xc6\x6e\xd0\x5b\x0c\xbd\x4f\xa6\xad\x40\xf4\x48\x89\x91\x94\xc7\xe6\x9d\x91\x54\x09\x27\x78\x93\x79\xb2\x0c\x5e\xc9\xf9\x99\x2a\x68\x1e\xc0\xba\x42\x39\x29\x79\x85\x53\x89\xae\xa3\xdc\x54\x81\x8d\xe3\x3a\x8a\x7a\x6f\xe7\x54\xf0\x35\x03\xa8\x51\xbc\x7d\xf1\xd6\x83\xe2\xbe\xb3\xff\x80\x63\x8f\x8f\x13\xec\x91\x11\x60\xaa\x14\xd9\xeb\x52\x64\x0c\x60\xfc\xfc\x6f\x92\x2b\x09\x25\x41\x22\x0f\x4d\xed\x17\x78\xd1\x14\x94\x82\xfa\x70\xa3\x43\x58\x4d\xb1\x9f\x43\x62\x37\x68\x94\x40\x4c\xd6\x48\xdb\x36\xc6\xb8\xdf\xf8\x61\x36\x21\x76\x57\x6d\x02\xdd\xd6\x37\x20\x79\x45\x31\x58\x11\x12\x3a\x66\x20\xaf\xc1\xc2\xa8\xea\xa6\x91\xf0\xc1\x0f\x66\xb4\xd5\xbf\x05\xac\x14\xbd\x10\x09\x73\x2a\xf2\x6a\x0a\x7a\x6d\x28\x2e\xe0\x18\x01\xa6\xe9\x74\x30\x27\x55\xa4\x42\x3c\x9d\x4e\x33\x44\x74\xdc\xfd\xa2\x9d\xf9\xc7\x77\x1a\x72\xdf\xfd\xda\x16\x4c\x69\x63\x8f\xe1\xcc\x77\xfe\x00\x2c\x6d\x45\x19\x62\xa9\x57\x86\x74\x37\xc3\x96\xc0\x0b\xdf\xdd\xd1\xf7\xfa\x2e\x4a\x03\xbc\x75\x37\x8b\xe9\xcb\x5a\xa1\x73\x45\x24\xd4\x15\xf9\xd2\x3a\xcf\x18\xed\x14\x8b\xb8\xde\x4f\xc7\x21\xd6\x22\xc9\x14\xf7\x21\xc0\xfa\x32\xd7\xd6\xb5\x11\x2a\x69\x32\xab\xf6\xe2\x03\x58\x49\xf2\x5b\x43\x58\x6e\xfe\xb6\x66\x1d\x41\xd1\x2e\x44\x9d\x9e\x1c\xa4\xac\xd0\x49\x4b\x2b\xe3\x9b\x2a\xe0\x7e\x17\xfb\xb7\x96\xc1\x9d\x9b\x63\x52\xa0\xa6\x52\x4b\x04\x9c\x57\xde\x00\xe7\x0b\xa8\x29\xaa\x6c\x14\x47\x5c\xd7\x94\x1e\xc6\xeb\xf5\x8a\x0c\x34\x75\x4d\x44\x8e\x24\xf1\xa2\x18\xa1\x91\x1f\x8f\xf0\x4f\xc7\xc8\xa9\x76\x91\xae\x2d\xc6\x26\x60\x2e\x21\xb0\xe6\x23\x53\xe3\x81\x45\x6b\xcd\xff\x0c\x05\x3f\xf6\x0e\x41\xb1\x73\x08\xf7\xa0\x77\xd0\xfe\x77\x26\xd4\x1f\x7e\x7a\xf8\xe7\xc3\x4f\xe0\x4f\xf4\x52\x73\xa1\x10\x5b\x66\x58\x57\x3a\x5d\x64\x78\xf8\x75\xf8\x49\xff\x0c\x16\x50\x06\xad\x3b\x6e\x1c\xf9\xc2\x76\xf9\xcc\xf1\xeb\x1b\x32\x69\xef\xd9\x81\xef\x76\x4d\x69\x3a\xca\x0d\x70\x7a\x0e\xb9\x3b\x1e\xa7\xf2\x6e\xbb\xc0\xf8\x71\x4d\xc2\x9e\x73\x48\xb4\xce\xab\x28\xd7\x0f\xe5\x08\x90\xbb\x2a\xae\x37\xd1\x36\x51\x6e\x07\xd9\xb5\x75\x39\xed\x7c\x2b\x8c\x71\x94\xd5\x4a\x10\xd9\x54\x0a\x0a\x7e\x95\x30\xe7\x0d\x53\xa9\x20\x49\xf6\x00\x61\x30\x44\x23\x24\x18\xf6\x58\xa0\x17\x5d\x64\xfb\x7e\xd5\x97\x5f\x81\xb0\xeb\xb3\x62\x63\x87\xde\x27\xaa\x21\x7f\x30\xb4\x77\x33\xcc\x30\x83\x1a\x17\x71\x9d\x10\xc7\xe3\xd1\xcd\x7e\x43\x71\x7f\x83\xb2\x44\x58\x5b\x3b\xd6\x5e\x24\x92\xa7\x83\x00\x66\x77\x8b\x34\x92\x50\x01\xac\x05\xbd\x20\xf1\xea\x4e\x3a\xfa\x90\x24\xe7\xe3\xc1\xc3\xad\xee\x4e\x82\x98\x53\x43\x24\xb6\xe1\xc5\xf7\x88\xe1\xe1\x70\x18\x65\xff\x19\xb9\x23\x65\xcb\xfd\xe1\x71\xb3\x27\x11\xf2\xdd\x83\x98\xeb\xfe\xd6\x10\xf1\xfa\xb9\x16\xfc\x49\x10\x19\xc5\xca\x91\x96\x17\x39\x8e\x87\xf8\x29\xb5\x15\x3c\x6f\x5a\x9e\x7d\x3d\xb2\x89\x3a\x49\x5a\x1f\x09\xd5\x6b\x8b\x6d\x8f\x87\xb4\x1a\x6d\x16\x4d\x2d\x3f\x9f\xcf\x69\x96\x15\x8a\x8d\x20\x5d\xf4\xd3\x2b\x87\xe2\xc2\x5c\x35\xaa\x83\xbe\xa9\x9a\xdb\x2d\xa6\x76\xc7\xfb\x05\x30\x3b\x33\x20\x8e\xa1\x70\x8d\x9e\x28\x1b\x4a\x98\x91\xbd\xe3\x02\xda\x3c\xd2\x55\x10\x21\x6c\x8e\x70\xd5\x6d\xb5\xfd\x7a\x30\x82\xca\x33\x2d\x4d\xe0\x0d\xe6\xb9\xee\x0f\x35\x2a\xad\x8a\xad\xfe\x8d\xb6\xb6\xae\xd1\x93\x09\x02\xa6\x10\x65\xd6\xc8\xd6\x95\x8d\x0a\xc7\x2b\x0a\x5a\x29\x22\xa4\xfb\x66\xe7\xf4\xd1\xf2\x30\xb6\xb6\xcf\xc6\x49\xac\x8f\x51\x0e\xc7\x88\x7e\x8c\xed\x82\x18\xdb\x8d\x4c\xdf\x15\x1a\x23\xf4\x8f\x8a\x68\x26\xbe\xa9\x1c\x35\xd8\x39\x9a\x75\x16\x45\xbf\x1f\x12\xa7\x64\xcf\x91\x96\x6e\x9d\xf3\xaa\xb9\xb4\x9a\xec\x24\x1a\xc6\x99\x13\xeb\xda\x4b\x6f\xdd\x66\x72\xdd\x50\x93\x04\x3b\x4d\x4d\xb6\x26\x73\x74\x74\xe7\x8b\x9d\xe7\xdb\x33\x53\x3c\xac\x46\x63\xb5\xc9\x84\x79\x9a\x15\xcd\xa2\x66\x4c\x5b\x33\xd6\xd5\xab\xb5\x74\xf5\x82\x30\xec\x06\x60\x3a\x80\x23\xaf\x47\xbd\x36\x48\xfd\xbb\x48\x38\x2c\x8a\x2a\x2d\x4b\x5b\xce\x25\xda\xc2\x19\xc4\x75\xd6\xaf\xc9\xa5\x56\xaf\x40\xb9\xa7\x29\xe1\x40\xb3\x63\xa5\x44\xc6\x54\xd9\x26\x95\xbf\x90\x17\xc2\xfe\x0a\xfe\x16\xae\x8c\x0e\x62\x0b\xf3\x6f\x82\x18\xc7\x78\x31\x2d\xe7\x6c\xa5\xa7\x55\xba\x7e\x68\x30\xdd\x1f\xd1\x46\x31\x1b\x91\x47\xf2\xf8\x29\x9c\xe6\xc5\x0e\x25\x77\xc1\x21\xc9\x7e\xbf\x4f\x82\xe2\x68\xd0\x73\x07\x40\x37\x0a\x6d\xed\x74\xe1\x5c\x95\x86\x1f\x62\x8a\xa2\x8a\x22\xd9\x76\x31\xf0\xc2\xbf\xc0\x46\x12\x01\x5b\x98\x18\x76\x01\x9f\x4b\x75\xa9\x12\xcf\x2c\xf1\xe8\xc3\xc8\x4d\x5f\x71\xa9\xda\xa2\x38\xe9\x5f\xf8\xb6\x29\x31\x5e\x48\xe4\xf5\x7d\x81\x02\x0b\xff\x82\x91\x42\xf0\xc2\x31\xf9\xf1\xdd\x59\x77\x1d\xe4\xdd\xaf\x3e\xc3\xc8\x34\xdb\x27\x8a\xf4\x2f\x90\xc1\xb4\x81\x4a\x58\x32\x9d\xc7\xf8\xde\x16\xee\xd2\xae\x71\x9d\x24\xe2\x10\xc5\x4e\xff\x42\x2f\x1d\xce\x5d\x63\xbc\xf6\xa7\x87\x03\x7e\x08\x0a\x67\x67\xb6\x3d\x6a\x99\x7b\xc2\xd8\x2b\x8b\xf7\x68\x7f\xdf\x3a\xd5\x0b\x11\x8a\xe6\xa8\x82\xa8\xa2\x4f\x2c\x03\x17\x8a\x71\x35\x71\xda\xe0\x10\xc4\xf4\xc5\xd6\x7a\xb7\xae\xa1\xde\x1f\x7b\xc8\x1f\xc6\xe0\xc7\x6d\xaa\x9b\x9d\x3c\xd1\xf2\x8e\x78\x6a\x41\x8c\x6b\x7f\x7c\x3f\xe7\x82\x13\x5e\x7d\x91\xf1\x07\xef\x3f\xc6\xb6\x66\xce\x29\x58\x53\x55\x9e\xe6\x4c\x11\x3c\x39\xd9\x09\x48\xe9\x37\x91\x20\xc8\xa3\xb2\xd9\x6c\x02\x23\x46\xd3\xa1\x13\x37\xce\x41\xf4\x07\x97\xfe\xe8\x84\xda\x1f\x67\x6c\x8f\x21\x00\x33\x0e\x73\xc1\xeb\x84\x01\x7b\x43\xaf\x0b\xbd\x73\xb7\x44\xef\x6a\xd7\xd0\xc7\xd6\x57\xaa\x4a\xe8\xe4\xaa\xd1\xa2\xe3\xa2\x55\xa9\x0a\x7c\x38\x0e\xfa\xf8\x1e\xc0\xef\xf6\x4f\x5b\xfd\x6e\x95\x37\x52\xf1\xcb\x67\xd3\xf1\xf9\x43\xc1\x47\xaf\xb1\xb7\xe3\xb6\xde\x32\x39\x67\x8c\x98\xfe\xe7\xf3\x95\x32\xcc\xaf\x66\x71\x7f\xce\xb3\xff\x14\x1f\x1c\xc5\xea\x91\x74\x4f\x30\xd5\xe2\x8c\x87\x77\xeb\x41\x26\xe8\x0e\x7c\x22\x62\x44\x8e\x16\xb6\x49\xba\x2b\x82\xa9\xfa\x3c\x10\xff\x00\x56\x79\xc5\x25\x71\x6e\x45\x5a\xcd\xfe\x2c\x34\x3c\xe9\xf7\x8b\xb7\x3e\x4f\x07\x4c\x00\x8d\xb4\x91\x87\x5e\xfb\x01\x7f\xd7\x06\x91\x0d\x38\x2f\x12\x21\xb8\x48\xbc\xb7\x3e\x73\xfe\x7c\x41\xe2\x39\x35\x1e\x75\x35\x2c\x89\x52\x94\x3d\x49\xb7\xa8\xbe\xdf\x8c\xfb\x4f\xe7\x60\x70\xf0\xa3\xbe\x68\x8f\x52\x5c\x97\x04\x75\xe7\x96\x76\x55\xe7\x0a\xbb\xe3\x92\x85\xe5\xb6\x8d\x3d\x8d\x51\x16\xd6\x73\xd2\xa5\x3c\x73\xb7\xef\x1a\xeb\x9b\x49\x49\x6d\xcf\xe8\x7f\xc1\xd3\x8d\x93\x62\x87\xd3\x73\x12\xac\x5f\x88\x90\x9d\x63\x4c\x0c\x7a\xed\xec\x3b\x2e\x69\x04\x01\xa6\x99\x36\x35\x46\x6a\x7a\x18\x3f\x62\x12\x9e\x30\x76\xa6\xd3\xab\x96\xc5\x5a\x2f\x87\xce\x04\x91\xf2\xe0\x5e\xff\x3c\x56\xdd\x01\x94\x8d\x05\x81\x30\x6d\x64\x2f\x81\x49\x60\x89\x47\x36\xf1\xc5\x9e\x4e\x4a\x55\xaf\x4b\x52\xd5\x6d\xbb\x9d\x3c\xd9\x49\x2f\xd7\x49\x43\x43\x13\xb4\x5b\x88\x24\x9a\xcd\x98\x44\x5e\x92\x0b\x71\x1a\x9c\x19\x35\xca\x32\xfd\xae\x03\xad\xdb\x6e\x12\xf8\xdd\xb3\xc1\xc7\xf7\xe0\x67\x7b\xc2\xf0\xb3\x96\x1d\x81\x7f\xd9\x71\x55\xfb\xf0\xfb\x72\xbb\x5b\xcb\x96\xc9\xd7\xdf\xfb\x3f\x29\x48\xff\xfb\x9a\x18\x9e\x44\x30\x12\x38\x14\xd7\x05\x82\x85\xf9\x0e\x02\xf2\xa5\x14\x13\xcb\x6b\xc2\x5c\x1a\x5f\x03\x23\x0d\xcb\xd6\xe4\x56\x23\x86\x09\x8e\xad\x9f\xe7\x9a\x58\x9e\xe4\xda\x4b\x6b\xff\x3f\x7c\xa8\x35\xfb\x35\xda\x3e\x08\x70\xe7\x5b\xcc\xb7\x1e\x04\x9c\x12\x9f\xcc\x44\xe4\xf3\xc7\x6f\x91\xb0\x08\xf5\x30\xb3\xd0\xc1\xc7\x24\xa7\xee\x4e\x1b\x62\x5a\xaf\x79\x49\x5e\x04\x67\x10\xf3\x6b\xea\x73\xe1\xc5\xe4\xdc\x2b\x38\x1c\x89\x8e\xf5\x1a\xe9\xc1\xbe\x59\xd5\xd1\xd3\xb1\x6f\x13\x79\x38\xa0\x0a\xb2\xdf\xdc\xc0\xb2\x1f\xbf\xc4\xa7\x46\x4b\xa5\xb1\x5f\x4a\xc7\x3f\x5b\x98\xaa\xf9\x96\xd3\xb7\xdf\x4e\xbf\xe1\x23\xe3\x11\xff\x65\x87\x1c\x63\x0b\xc7\x8c\x17\xb4\x29\x76\xaa\x3f\xfb\x59\xdf\x0d\xca\x5c\xf0\xaa\xfa\x86\xdd\xd3\xb1\x7d\xdf\x4a\x23\x3d\x8f\xb0\xe3\xf6\x25\x33\x9b\xb7\x32\x4d\x8e\x4c\x86\x11\xff\x42\x7a\x0e\xa0\x4c\xc6\xff\xe8\x3b\xbc\x3f\x44\xdf\x84\xc4\x1f\x03\x98\x81\xb0\x0d\x94\xff\x45\x73\xf8\xf1\x3d\xf8\x7b\x4e\xc0\x3f\x4c\x87\x48\xbf\xa0\xb6\xa3\xfa\xef\x94\x02\x28\x27\x9f\x9f\x1a\xa5\x88\xf8\x00\x9c\x0b\x98\x13\x3b\x7e\x08\x1b\xc7\xc4\x08\xc8\x4e\x27\xa2\xd5\x9d\x26\xdb\xfa\x2f\xec\x3f\xe0\x70\xc9\x6a\xa3\xf8\x5d\xfa\x7f\x02\x00\x00\xff\xff\xa3\xfa\x84\xc0\x4a\x32\x00\x00") +var _staticCssAppCss = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xcc\x5a\x5b\x8f\xdc\xb6\x15\x7e\xdf\x5f\xc1\x7a\x50\xa4\x35\xcc\xf1\x5c\x77\x67\x65\xa4\x40\x51\x34\xaf\x7d\xc8\x63\x10\x18\x1c\x91\x5a\xb1\xab\x21\x15\x92\xda\x99\xb5\x91\xff\x5e\x90\xa2\x24\x92\x22\x25\xad\xe3\x16\xf5\x20\xb1\x75\xe1\x39\x87\xe7\xf2\x9d\x0b\xb5\xba\x20\xca\xc0\xd7\x3b\x00\x30\x95\x75\x85\x5e\x33\xc0\x38\x23\x9f\xee\x7e\xbf\xbb\x5b\x31\xf4\x62\x1e\xd5\x5c\x52\x45\x39\xcb\x40\x41\x6f\x04\x7f\xba\x03\x40\xf1\x3a\x03\x1b\xfd\xaf\x8a\x14\x2a\x03\xbb\xe3\xa6\xbe\xe9\x4b\x41\x9f\x4a\x65\x1f\x95\xa4\xbd\xe8\x9e\x9d\x51\xfe\xfc\x24\x78\xc3\x70\x06\x56\x0f\x8f\xc7\xd3\x63\x31\x30\x6a\x2a\xc3\xeb\x82\xc4\x13\x65\x19\xb0\x4b\x6a\x84\x31\x65\x4f\xfd\x75\x48\xb2\x97\xfa\x5c\xf1\xfc\xd9\xa3\x56\xd1\x25\x04\x8b\x8a\x23\x95\x99\x5d\xb8\xf4\x0f\x8f\xed\xe3\x8a\x32\x02\xc3\x9b\x05\x67\x0a\x4a\xfa\x85\x64\x60\xbb\x1f\x93\x05\xdb\x43\x7b\x33\xe7\x15\x17\x19\x58\xe1\xfb\x3c\x27\xbb\x7e\xe5\xb5\xdf\x83\xd1\x52\x2f\x20\xd8\xda\x65\x8d\x90\x7a\x5d\xcd\x29\x53\x44\xc4\xb6\xa9\x05\x93\x0a\x4a\xf5\x5a\x91\xce\x62\xee\x3d\xa8\x5e\x6b\x12\x9a\xd2\xe8\x24\x2b\xa8\x90\x0a\xe6\x25\xad\xb0\xa3\x1f\xd8\x9a\xd1\x28\xc5\x7f\xbf\xe4\x2f\x44\x98\x37\xbb\xed\x14\x45\x11\xbc\xb4\x96\xa4\x22\xb9\x22\x38\x70\x17\x41\x2a\xa4\xe8\x0b\x19\x19\xdf\x90\x18\x28\x6e\x02\x4d\xa4\xc8\x67\x67\x52\x70\x41\xac\x34\x4c\x11\xa6\x32\xf0\x03\xf8\xe1\x93\xc7\x15\x9d\x25\xaf\x1a\xd5\x72\xe5\x4a\xf1\x4b\x06\xa0\xd5\xad\xdd\xe6\xa7\xa8\x9f\xda\x77\xae\x14\xf3\xab\xcc\xc0\x76\xb3\xf9\x73\x5c\xf2\x84\x74\xa1\x6a\xcf\x5c\x60\x22\xac\x6a\x07\x5b\x48\x8a\xc9\x19\x89\x54\x68\x5d\x29\x56\xa5\x13\x51\x8e\x65\xba\xb8\xb3\xe1\x64\xf7\xd6\x5f\xde\xb4\x53\x1a\x2f\xb4\x9c\xcf\x7c\x1c\x77\xc5\xa3\xfe\xf9\x92\xac\xf3\x46\x08\xc2\x14\xc4\x48\xa1\x33\x92\x24\x25\x5b\x52\x96\x50\xe8\x30\x4e\xbd\x38\xea\x6e\x7a\xd1\x70\xe6\x15\x0e\x83\x2b\x88\x23\xeb\x35\x51\x18\x01\xe0\x0b\xa4\x0c\x93\x5b\x06\x1e\x37\x9b\xb9\xed\xad\xaf\x02\xd5\x13\xce\xea\x4b\x3a\x4d\x4b\xd6\x88\x8d\xee\x42\x86\x2e\x64\x1c\x5f\xfb\x10\xb6\x28\x33\x9a\xe9\xc3\xba\xd3\xe3\x26\xa1\xc7\x6b\x49\x15\x81\xb2\x46\xb9\x09\x6f\xbd\x0d\x7d\x5b\xc7\x68\x51\xf1\x6b\x06\x4a\x8a\x31\x61\xc6\x3c\xe4\xa6\xe0\xf0\x80\x54\x15\xad\x25\x95\x51\x84\x99\xde\x21\x5d\x17\x28\xe5\x1a\x6e\xb0\xb5\x7b\x3c\x39\xde\xb1\x3d\xcd\xeb\x8f\xb2\xba\x51\x6b\x0d\x58\xa8\x24\x28\x02\x4a\xff\x17\x4a\x7b\xfb\x1e\x82\x84\x9a\xf4\xdb\x94\xf4\xbb\x5d\xd4\xfb\xbd\xa0\x61\x5c\x5c\x50\x65\x36\xd3\x28\xad\x14\x87\x9b\x81\x80\x36\x21\x8d\xb1\x27\xe1\xc6\x82\x14\x82\xc8\x72\xc2\xc6\x6e\xd0\x5b\x0c\xbd\x4f\xa6\xad\x40\xf4\x48\x89\x91\x94\xc7\xe6\x9d\x91\x54\x09\x27\x78\x93\x79\xb2\x0c\x5e\xc9\xf9\x99\x2a\x68\x1e\xc0\xba\x42\x39\x29\x79\x85\x53\x89\xae\xa3\xdc\x54\x81\x8d\xe3\x3a\x8a\x7a\x6f\xe7\x54\xf0\x35\x03\xa8\x51\xbc\x7d\xf1\xd6\x83\xe2\xbe\xb3\xff\x80\x63\x8f\x8f\x13\xec\x91\x11\x60\xaa\x14\xd9\xeb\x52\x64\x0c\x60\xfc\xfc\x6f\x92\x2b\x09\x25\x41\x22\x0f\x4d\xed\x17\x78\xd1\x14\x94\x82\xfa\x70\xa3\x43\x58\x4d\xb1\x9f\x43\x62\x37\x68\x94\x40\x4c\xd6\x48\xdb\x36\xc6\xb8\xdf\xf8\x61\x36\x21\x76\x57\x6d\x02\xdd\xd6\x37\x20\x79\x45\x31\x58\x11\x12\x3a\x66\x20\xaf\xc1\xc2\xa8\xea\xa6\x91\xf0\xc1\x0f\x66\xb4\xd5\xbf\x05\xac\x14\xbd\x10\x09\x73\x2a\xf2\x6a\x0a\x7a\x6d\x28\x2e\xe0\x18\x01\xa6\xe9\x74\x30\x27\x55\xa4\x42\x3c\x9d\x4e\x33\x44\x74\xdc\xfd\xa2\x9d\xf9\xc7\x77\x1a\x72\xdf\xfd\xda\x16\x4c\x69\x63\x8f\xe1\xcc\x77\xfe\x00\x2c\x6d\x45\x19\x62\xa9\x57\x86\x74\x37\xc3\x96\xc0\x0b\xdf\xdd\xd1\xf7\xfa\x2e\x4a\x03\xbc\x75\x37\x8b\xe9\xcb\x5a\xa1\x73\x45\x24\xd4\x15\xf9\xd2\x3a\xcf\x18\xed\x14\x8b\xb8\xde\x4f\xc7\x21\xd6\x22\xc9\x14\xf7\x21\xc0\xfa\x32\xd7\xd6\xb5\x11\x2a\x69\x32\xab\xf6\xe2\x03\x58\x49\xf2\x5b\x43\x58\x6e\xfe\x6d\xcd\x3a\x82\xa2\x5d\x88\x3a\x3d\x39\x48\x59\xa1\x93\x96\x56\xc6\x37\x55\xc0\xfd\x2e\xf6\x6f\x2d\x83\x3b\x37\xc7\xa4\x40\x4d\xa5\x96\x08\x38\xaf\xbc\x01\xce\x17\x50\x53\x54\xd9\x28\x8e\xb8\xae\x29\x3d\x8c\xd7\xeb\x15\x19\x68\xea\x9a\x88\x1c\x49\xe2\x45\x31\x42\x23\x3f\x1e\xe1\x9f\x8e\x91\x53\xed\x22\x5d\x5b\x8c\x4d\xc0\x5c\x42\x60\xcd\x47\xa6\xc6\x03\x8b\xd6\x9a\xbf\x0c\x05\x3f\xf6\x0e\x41\xb1\x73\x08\xf7\xa0\x77\xd0\xfe\x7f\x26\xd4\x1f\x7e\x7a\xf8\xe7\xc3\x4f\xe0\x4f\xf4\x52\x73\xa1\x10\x5b\x66\x58\x57\x3a\x5d\x64\x78\xf8\x75\xf8\x49\xff\x0c\x16\x50\x06\xad\x3b\x6e\x1c\xf9\xc2\x76\xf9\xcc\xf1\xeb\x1b\x32\x69\xef\xd9\x81\xef\x76\x4d\x69\x3a\xca\x0d\x70\x7a\x0e\xb9\x3b\x1e\xa7\xf2\x6e\xbb\xc0\xf8\x71\x4d\xc2\x9e\x73\x48\xb4\xce\xab\x28\xd7\x0f\xe5\x08\x90\xbb\x2a\xae\x37\xd1\x36\x51\x6e\x07\xd9\xb5\x75\x39\xed\x7c\x2b\x8c\x71\x94\xd5\x4a\x10\xd9\x54\x0a\x0a\x7e\x95\x30\xe7\x0d\x53\xa9\x20\x49\xf6\x00\x61\x30\x44\x23\x24\x18\xf6\x58\xa0\x17\x5d\x64\xfb\x7e\xd5\x97\x5f\x81\xb0\xeb\xb3\x62\x63\x87\xde\x27\xaa\x21\x7f\x30\xb4\x77\x33\xcc\x30\x83\x1a\x17\x71\x9d\x10\xc7\xe3\xd1\xcd\x7e\x43\x71\x7f\x83\xb2\x44\x58\x5b\x3b\xd6\x5e\x24\x92\xa7\x83\x00\x66\x77\x8b\x34\x92\x50\x01\xac\x05\xbd\x20\xf1\xea\x4e\x3a\xfa\x90\x24\xe7\xe3\xc1\xc3\xad\xee\x4e\x82\x98\x53\x43\x24\xb6\xe1\xc5\xf7\x88\xe1\xe1\x70\x18\x65\xff\x19\xb9\x23\x65\xcb\xfd\xe1\x71\xb3\x27\x11\xf2\xdd\x83\x98\xeb\xfe\xd6\x10\xf1\xfa\xb9\x16\xfc\x49\x10\x19\xc5\xca\x91\x96\x17\x39\x8e\x87\xf8\x29\xb5\x15\x3c\x6f\x5a\x9e\x7d\x3d\xb2\x89\x3a\x49\x5a\x1f\x09\xd5\x6b\x8b\x6d\x8f\x87\xb4\x1a\x6d\x16\x4d\x2d\x3f\x9f\xcf\x69\x96\x15\x8a\x8d\x20\x5d\xf4\xd3\x2b\x87\xe2\xc2\x5c\x35\xaa\x83\xbe\xa9\x9a\xdb\x2d\xa6\x76\xc7\xfb\x05\x30\x3b\x33\x20\x8e\xa1\x70\x8d\x9e\x28\x1b\x4a\x98\x91\xbd\xe3\x02\xda\x3c\xd2\x55\x10\x21\x6c\x8e\x70\xd5\x6d\xb5\xfd\x7a\x30\x82\xca\x33\x2d\x4d\xe0\x0d\xe6\xb9\xee\x0f\x35\x2a\xad\x8a\xad\xfe\x8d\xb6\xb6\xae\xd1\x93\x09\x02\xa6\x10\x65\xd6\xc8\xd6\x95\x8d\x0a\xc7\x2b\x0a\x5a\x29\x22\xa4\xfb\x66\xe7\xf4\xd1\xf2\x30\xb6\xb6\xcf\xc6\x49\xac\x8f\x51\x0e\xc7\x88\x7e\x8c\xed\x82\x18\xdb\x8d\x4c\xdf\x15\x1a\x23\xf4\x8f\x8a\x68\x26\xbe\xa9\x1c\x35\xd8\x39\x9a\x75\x16\x45\xbf\x1f\x12\xa7\x64\xcf\x91\x96\x6e\x9d\xf3\xaa\xb9\xb4\x9a\xec\x24\x1a\xc6\x99\x13\xeb\xda\x4b\x6f\xdd\x66\x72\xdd\x50\x93\x04\x3b\x4d\x4d\xb6\x26\x73\x74\x74\xe7\x8b\x9d\xe7\xdb\x33\x53\x3c\xac\x46\x63\xb5\xc9\x84\x79\x9a\x15\xcd\xa2\x66\x4c\x5b\x33\xd6\xd5\xab\xb5\x74\xf5\x82\x30\xec\x06\x60\x3a\x80\x23\xaf\x47\xbd\x36\x48\xfd\xbb\x48\x38\x2c\x8a\x2a\x2d\x4b\x5b\xce\x25\xda\xc2\x19\xc4\x75\xd6\xaf\xc9\xa5\x56\xaf\x40\xb9\xa7\x29\xe1\x40\xb3\x63\xa5\x44\xc6\x54\xd9\x26\x95\xbf\x90\x17\xc2\xfe\x0a\xfe\x16\xae\x8c\x0e\x62\x0b\xf3\x67\x82\x18\xc7\x78\x31\x2d\xe7\x6c\xa5\xa7\x55\xba\x7e\x68\x30\xdd\x1f\xd1\x46\x31\x1b\x91\x47\xf2\xf8\x29\x9c\xe6\xc5\x0e\x25\x77\xc1\x21\xc9\x7e\xbf\x4f\x82\xe2\x68\xd0\x73\x07\x40\x37\x0a\x6d\xed\x74\xe1\x5c\x95\x86\x1f\x62\x8a\xa2\x8a\x22\xd9\x76\x31\xf0\xc2\xbf\xc0\x46\x12\x01\x5b\x98\x18\x76\x01\x9f\x4b\x75\xa9\x12\xcf\x2c\xf1\xe8\xc3\xc8\x4d\x5f\x71\xa9\xda\xa2\x38\xe9\x5f\xf8\xb6\x29\x31\x5e\x48\xe4\xf5\x7d\x81\x02\x0b\xff\x82\x91\x42\xf0\xc2\x31\xf9\xf1\xdd\x59\x77\x1d\xe4\xdd\xaf\x3e\xc3\xc8\x34\xdb\x27\x8a\xf4\x2f\x90\xc1\xb4\x81\x4a\x58\x32\x9d\xc7\xf8\xde\x16\xee\xd2\xae\x71\x9d\x24\xe2\x10\xc5\x4e\xff\x42\x2f\x1d\xce\x5d\x63\xbc\xf6\xa7\x87\x03\x7e\x08\x0a\x67\x67\xb6\x3d\x6a\x99\x7b\xc2\xd8\x2b\x8b\xf7\x68\x7f\xdf\x3a\xd5\x0b\x11\x8a\xe6\xa8\x82\xa8\xa2\x4f\x2c\x03\x17\x8a\x71\x35\x71\xda\xe0\x10\xc4\xf4\xc5\xd6\x7a\xb7\xae\xa1\xde\x1f\x7b\xc8\x1f\xc6\xe0\xc7\x6d\xaa\x9b\x9d\x3c\xd1\xf2\x8e\x78\x6a\x41\x8c\x6b\x7f\x7c\x3f\xe7\x82\x13\x5e\x7d\x91\xf1\x07\xef\x3f\xc6\xb6\x66\xce\x29\x58\x53\x55\x9e\xe6\x4c\x11\x3c\x39\xd9\x09\x48\xe9\x37\x91\x20\xc8\xa3\xb2\xd9\x6c\x02\x23\x46\xd3\xa1\x13\x37\xce\x41\xf4\x07\x97\xfe\xe8\x84\xda\x1f\x67\x6c\x8f\x21\x00\x33\x0e\x73\xc1\xeb\x84\x01\x7b\x43\xaf\x0b\xbd\x73\xb7\x44\xef\x6a\xd7\xd0\xc7\xd6\x57\xaa\x4a\xe8\xe4\xaa\xd1\xa2\xe3\xa2\x55\xa9\x0a\x7c\x38\x0e\xfa\xf8\x1e\xc0\xef\xf6\x47\x5b\xfd\x6e\x95\x37\x52\xf1\xcb\x67\xd3\xf1\xf9\x43\xc1\x47\xaf\xb1\xb7\xe3\xb6\xde\x32\x39\x67\x8c\x98\xfe\xe7\xf3\x95\x32\xcc\xaf\x66\x71\x7f\xce\xb3\xff\x14\x1f\x1c\xc5\xea\x91\x74\x4f\x30\xd5\xe2\x8c\x1a\x94\xe8\xf4\x72\x3d\xc8\x09\xdd\x21\x50\x44\xb4\xc8\x71\xc3\x36\xce\x4b\xef\x9f\x60\xaa\x3e\x0f\xc4\x3f\x80\x55\x5e\x71\x49\x9c\x5b\x91\xf6\xb3\x3f\x1f\x0d\x4f\xff\xfd\x82\xae\xcf\xdd\x01\x13\x40\x23\xad\xe5\xa1\xb7\x48\xc0\xdf\xb5\x4b\x64\x03\xce\x8b\x44\x08\x2e\x12\xef\xad\xcf\x9c\x3f\x5f\x90\x78\x4e\x8d\x4c\x5d\x0d\x4b\xa2\x14\x65\x4f\xd2\x2d\xb4\xef\x37\xe3\x9e\xd4\x39\x2c\x1c\x7c\xab\x2f\xe4\xa3\x14\xd7\x25\x41\xdd\x59\xa6\x5d\xd5\xb9\xc7\xee\xb8\x64\x61\xb9\x6d\xe3\x51\xe3\x96\x85\xfa\x9c\x74\x69\xd0\xdc\xed\x3b\xc9\xfa\x66\xd2\x54\xdb\x47\xfa\x5f\xf5\x74\x23\xa6\xd8\x81\xf5\x9c\x04\xeb\x17\x22\x64\xe7\x18\x13\xc3\x5f\x3b\x0f\x8f\x4b\x1a\x41\x85\x69\xa6\x4d\x8d\x91\x9a\x1e\xd0\x8f\x98\x84\xa7\x8e\x9d\xe9\xf4\xaa\x44\xfc\xa5\xe4\xd0\xd9\x21\x52\x32\xdc\xeb\x9f\xc7\xaa\x3b\x94\xb2\xb1\x20\x10\xa6\x8d\xec\x25\x30\x49\x2d\xf1\xc8\x26\xc3\xd8\xd3\x49\xa9\xea\x75\x49\xaa\xba\x6d\xc1\x93\xa7\x3d\xe9\xe5\x3a\x91\x68\xb8\x82\x76\x0b\x91\xe4\xb3\x19\x93\xc8\x4b\x72\x21\x4e\xd3\x33\xa3\x46\x59\xa6\xdf\x75\xe0\x76\xdb\x4d\x07\xe3\x8b\xcb\xfd\x68\x7f\xa7\x71\x50\xea\xff\x76\x9b\xc1\xc2\xe1\xb4\xee\xbb\x67\x9f\x8f\xef\xc1\xcf\xf6\x44\xe3\x67\xad\x17\x04\xfe\x65\xc7\x63\xed\xc3\xef\xcb\xed\x6e\x2d\x5b\x26\x5f\x7f\xef\xff\x49\x41\xfa\xcf\xd7\xc4\xb0\x26\x82\xbf\xc0\xa1\xb8\x2e\x10\x2c\xcc\x77\x17\x90\x2f\xa5\x98\x58\x5e\x13\xe6\xd2\xf8\x1a\x38\xc0\xb0\x6c\x4d\x6e\x35\x62\x98\xe0\xd8\xfa\x79\xae\x89\xe5\x49\xae\xbd\xb4\xf6\xef\xe1\xc3\xb0\xd9\xaf\xdf\xf6\x01\x78\x38\xdf\x7e\xbe\xf5\xe0\xe1\x94\xf8\x44\x27\x22\x9f\x3f\xee\x8b\x84\x5c\xa8\x87\x99\x85\x0e\xf6\x26\x39\x75\x77\xda\x08\xd4\x7a\xcd\x4b\xf2\x22\x38\x83\x98\x5f\x53\x9f\x27\x2f\x26\xe7\x5e\xc1\xe1\x08\x76\xac\xd7\x48\xcf\xf7\xcd\xaa\x8e\x9e\xc6\x7d\x9b\xc8\xc3\x81\x58\x90\x59\xe7\x06\xa4\xfd\xb8\x27\x3e\xa5\x5a\x2a\x8d\xfd\x32\x3b\xfe\x99\x44\xaa\xc6\x7c\x1b\x7d\xfb\xad\xf6\x1b\x3e\x6a\x1e\xf1\x5f\x76\xa8\x32\xb6\x70\xcc\x78\x41\x5b\x64\x4f\x11\x66\x3f\x23\xbc\x41\x99\x0b\x5e\x55\xdf\xb0\x7b\x3a\xb6\xef\x5b\x69\xa4\xe7\x1f\x76\xbc\xbf\x64\x46\xf4\x56\xa6\xc9\x11\xcd\x70\xa4\xb0\x90\x9e\x03\x28\x93\xf1\x3f\xfa\xee\xef\x0f\xd1\x37\x21\xf1\xc7\x00\x66\x20\x6c\x03\xe5\x7f\xd1\x8c\x7e\x7c\x0f\xfe\x9e\x13\xf0\x0f\xd3\x91\xd2\x2f\xa8\xed\xd6\xfe\x3b\xa5\x00\xca\xc9\xe7\xa7\x46\x29\x22\x3e\x00\xe7\x02\xe6\xc4\x8e\x3b\xc2\x46\x35\x31\x72\xb2\xd3\x90\x68\xe5\xa8\xc9\xb6\xfe\x0b\xfb\x0f\x46\x5c\xb2\xda\x28\xfe\x54\xe0\x3f\x01\x00\x00\xff\xff\xef\xc3\x8e\x96\xba\x32\x00\x00") func staticCssAppCssBytes() ([]byte, error) { return bindataRead( @@ -103,8 +111,8 @@ func staticCssAppCss() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/css/app.css", size: 12874, mode: os.FileMode(0644), modTime: time.Unix(1562365712, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xcc, 0x1d, 0x41, 0xd9, 0x30, 0x75, 0x7, 0xd8, 0xa7, 0x9c, 0xc3, 0xe2, 0x85, 0xe2, 0xad, 0x0, 0xd3, 0x93, 0xd9, 0x61, 0xc3, 0x81, 0xc, 0x78, 0xa9, 0x80, 0x28, 0x52, 0xd, 0x0, 0x7d, 0xc}} + info := bindataFileInfo{name: "static/css/app.css", size: 12986, mode: os.FileMode(420), modTime: time.Unix(1572714887, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -123,8 +131,8 @@ func staticCssBootstrapCss() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/css/bootstrap.css", size: 109518, mode: os.FileMode(0644), modTime: time.Unix(1467073917, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xb5, 0xfd, 0x72, 0x37, 0x50, 0x76, 0x3e, 0xbb, 0x73, 0x1f, 0x92, 0x21, 0xe4, 0x13, 0xe7, 0xd6, 0x4d, 0x58, 0xd5, 0x19, 0x2d, 0xc0, 0x40, 0xe4, 0x22, 0x92, 0xed, 0x3d, 0xcc, 0xcc, 0xa7, 0x32}} + info := bindataFileInfo{name: "static/css/bootstrap.css", size: 109518, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -143,8 +151,8 @@ func staticCssFontAwesomeCss() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/css/font-awesome.css", size: 21984, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xf, 0xb1, 0xbb, 0xca, 0x73, 0x64, 0x6e, 0x8e, 0x2b, 0x93, 0xc8, 0x2e, 0x8d, 0x8b, 0x21, 0x96, 0x47, 0xb1, 0x3d, 0x4b, 0x44, 0xc, 0x48, 0xe3, 0x38, 0x29, 0xb, 0x9a, 0x68, 0x5b, 0x8d, 0xe1}} + info := bindataFileInfo{name: "static/css/font-awesome.css", size: 21984, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -163,8 +171,8 @@ func staticFontsFontawesomeOtf() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/fonts/FontAwesome.otf", size: 85908, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xd, 0x34, 0xda, 0x7, 0xe5, 0xcb, 0xcb, 0x4f, 0xf0, 0x87, 0xb3, 0x97, 0x96, 0x6a, 0x9f, 0x9f, 0xfb, 0x4d, 0x6e, 0xd7, 0xc7, 0x6, 0x5, 0x68, 0x56, 0x52, 0x3, 0xc0, 0x2f, 0x3f, 0xee, 0x11}} + info := bindataFileInfo{name: "static/fonts/FontAwesome.otf", size: 85908, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -183,8 +191,8 @@ func staticFontsFontawesomeWebfontEot() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.eot", size: 56006, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xe5, 0x11, 0x89, 0x1d, 0x3e, 0x1, 0xb0, 0xb2, 0x7a, 0xed, 0x51, 0xa2, 0x19, 0xce, 0xd5, 0x11, 0x9e, 0x2c, 0x3d, 0x4, 0x60, 0x46, 0x5a, 0xf8, 0x24, 0x2e, 0x9b, 0xff, 0x4c, 0xb6, 0x1b, 0x77}} + info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.eot", size: 56006, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -203,8 +211,8 @@ func staticFontsFontawesomeWebfontSvg() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.svg", size: 287007, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xd5, 0xb5, 0x63, 0x6e, 0xbb, 0x2e, 0x12, 0x48, 0x10, 0x43, 0x62, 0x0, 0x8, 0x6b, 0x74, 0xa6, 0xd, 0xff, 0x9e, 0x8a, 0x8b, 0xe7, 0xf4, 0xa1, 0x8, 0x8b, 0xf5, 0xd3, 0x45, 0x8b, 0xc3, 0xc8}} + info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.svg", size: 287007, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -223,8 +231,8 @@ func staticFontsFontawesomeWebfontTtf() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.ttf", size: 112160, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x4d, 0x6e, 0xb9, 0xe9, 0xd8, 0x52, 0xa2, 0xa6, 0xf7, 0x4e, 0x7c, 0x42, 0x84, 0x56, 0xa2, 0xf0, 0x7f, 0xc6, 0x3a, 0x16, 0x13, 0xd1, 0x1, 0x92, 0xd8, 0xed, 0x34, 0x1, 0xd9, 0xda, 0x5f, 0xfa}} + info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.ttf", size: 112160, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -243,8 +251,8 @@ func staticFontsFontawesomeWebfontWoff() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.woff", size: 65452, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x19, 0x94, 0x11, 0xf6, 0x59, 0xf4, 0x1a, 0xac, 0xcb, 0x95, 0x9b, 0xac, 0xb1, 0xb0, 0xde, 0x30, 0xe5, 0x4f, 0x24, 0x43, 0x52, 0xa4, 0x8c, 0x6f, 0x98, 0x94, 0xe6, 0x5a, 0xe0, 0xf8, 0xa9, 0xa1}} + info := bindataFileInfo{name: "static/fonts/fontawesome-webfont.woff", size: 65452, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -263,8 +271,8 @@ func staticImgIconIco() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/img/icon.ico", size: 104736, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xe9, 0xf0, 0xf1, 0xdd, 0x9b, 0xb0, 0xc0, 0xdc, 0x4, 0xb1, 0x63, 0x10, 0x28, 0x0, 0x39, 0xc9, 0x76, 0x8e, 0xfd, 0x37, 0x77, 0x94, 0x18, 0xc6, 0xcc, 0x1c, 0xb3, 0x9e, 0x8a, 0x4c, 0x81, 0x82}} + info := bindataFileInfo{name: "static/img/icon.ico", size: 104736, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -283,12 +291,12 @@ func staticImgIconPng() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/img/icon.png", size: 7945, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x39, 0x37, 0xea, 0x6e, 0x72, 0xc7, 0x82, 0x3a, 0x90, 0x14, 0x40, 0x81, 0x62, 0xa4, 0xb, 0xa5, 0x2b, 0xf8, 0x1b, 0xa3, 0x7a, 0x5e, 0x87, 0xbd, 0x24, 0x1e, 0x69, 0x88, 0x51, 0xb3, 0x9d, 0xa4}} + info := bindataFileInfo{name: "static/img/icon.png", size: 7945, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } -var _staticIndexHtml = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\x1b\x69\x73\xdb\xb8\xf5\xfb\xfe\x0a\x14\x3b\xb3\xd3\x4e\x4b\x73\x92\x6c\x8f\xb8\x12\xdb\x5d\xc7\x6d\xbc\xeb\x38\xb1\x95\x64\xd2\x4f\x1a\x88\x7c\xa2\x60\x83\x00\x0d\x80\x3a\xf2\xeb\x3b\x38\x28\xf1\x94\x29\xd9\xde\xe6\x43\xbf\x58\x20\x80\x77\xe0\xe1\x5d\x78\x80\x47\xbf\x7b\xf3\xfe\xec\xe3\x7f\x3e\x9c\xa3\x85\xce\x58\xf4\xdd\xc8\xfc\x20\x46\x78\x3a\xc6\xc0\x31\x5a\x67\xec\xb4\xf6\xc5\xd5\x18\x2f\xb4\xce\x4f\xc3\x70\xb5\x5a\x9d\xac\x5e\x9d\x08\x99\x86\x2f\x5e\xbf\x7e\x1d\xae\x0d\x2c\x36\x38\x80\x24\xd1\x77\x08\x8d\x34\xd5\x0c\xa2\x3c\x5d\xc1\x6c\x14\xba\x0f\xd3\x9d\x81\x26\x28\x5e\x10\xa9\x40\x8f\x71\xa1\xe7\xc1\xdf\xf0\x6e\xc0\x60\x0f\xe0\xbe\xa0\xcb\x31\xfe\x12\x7c\xfa\x29\x38\x13\x59\x4e\x34\x9d\x31\xc0\x28\x16\x5c\x03\xd7\x63\x7c\x71\x3e\x86\x24\x85\x1e\xb8\x33\x37\x2d\xb8\x24\x3c\x2d\x48\x5a\x05\x04\xee\x60\x18\xe5\x77\x48\x02\x1b\x63\xa5\x37\x0c\xd4\x02\x40\x63\xb4\x90\x30\x37\x3d\x44\xd3\x38\x8c\x95\x0a\x67\x42\x68\xa5\x25\xc9\x4f\x62\xa5\x70\x34\x0a\x0d\xdc\x21\x08\xe6\x82\xeb\x80\xac\x40\x89\x0c\x8e\xc5\x41\xf2\x7d\xe4\x69\x2c\x38\x46\x7a\x93\xc3\x18\xd3\x8c\xa4\x10\xae\x03\xd7\x57\x43\x44\xb3\x34\x34\xdd\x27\x34\x16\x18\x85\x16\x89\x8a\x25\xcd\xb5\x87\xd5\xb0\xd6\xe1\x2d\x59\x12\xd7\x8b\x91\x92\xf1\x16\xfa\x56\x85\xb7\xf7\x05\xc8\xcd\xc9\xad\xe5\xc3\xcd\x39\x1c\x09\x89\xe1\xf1\x18\x82\x3c\x55\xf7\xec\x91\x78\xb6\x3b\x1b\x58\xdd\x58\xeb\x0c\x78\xf1\x48\x9c\x85\xa6\x4c\x3d\x15\x5f\xaf\x02\x03\x40\x8c\x35\x9d\x64\x94\x3f\x56\x6c\x79\x5e\xc7\x80\x0e\xe7\x8c\x28\xf8\xcb\x8f\x0d\x3e\x46\xa1\x33\xf7\xd1\x4c\x24\x1b\xcb\x56\x42\x97\x88\x26\x63\x9c\x11\xea\x6c\xad\xd2\xc7\xc9\xd2\x77\x21\x34\x2a\x58\xd9\xb4\xfa\x6c\x27\x68\x32\x63\x30\xf5\xd6\x8a\xa3\x1b\xb1\x52\x46\xe9\xfb\x26\x2a\x2d\x8b\x58\x17\x12\x70\x34\x29\x9b\xfb\xe6\x53\x9e\xc0\x1a\x14\x8e\x2e\x5c\x63\xdf\xdc\x58\x70\xb3\x0f\x94\x6b\x85\xa3\xb3\xdd\xc7\x3e\x18\x6b\x22\x18\xc5\x8c\x28\x35\xc6\x0a\x18\xc4\x1a\x12\x1c\x5d\x9b\xfe\x7d\x80\x0b\xaa\xb4\x90\x1b\x1c\xbd\x75\x8d\x7d\x73\x49\xac\xe9\x92\xea\x0d\x8e\x7e\xf2\xad\x07\x96\xc1\x21\xd6\x54\x70\xbb\x0a\xdf\xae\x42\x8c\x42\xb3\x13\xe5\x87\xd9\x2a\xcf\xff\x0e\x34\x20\xf6\x47\xe1\x0a\x15\xe2\xfd\xcb\xf7\xd8\x52\x83\x84\xea\x2a\xb1\x12\xc9\x4c\x73\x34\xd3\x3c\x48\x60\x4e\x0a\xa6\x6d\x5b\x65\x38\x1a\xd1\x72\xc6\x9c\xa0\x39\x09\x12\xa2\x89\xd1\x30\xa3\x5c\x34\x42\x9e\xd5\x51\x48\xfa\x49\xc6\x4c\x28\x38\x84\xe6\x1b\xaa\xe2\x16\xde\x51\x98\xd0\xa5\xd7\xd3\x4a\xb3\x54\x59\x45\x13\x98\x11\xb9\x53\xdb\xaa\x80\x0a\x29\x4d\x9c\xd9\xf1\xbe\xe3\xb5\x32\x6d\x25\x49\x5e\x19\x42\xe8\x81\xb5\x8f\x54\x4e\x78\x1f\x8d\x80\x93\x0c\xfc\xfa\xdd\xd0\xb4\x0a\x6e\x40\xeb\xa4\x78\x5e\xe8\x12\xd9\xd6\xa3\x38\x04\x25\xe0\x54\x01\x91\xf1\x02\x57\x1c\x01\x46\x39\x23\x31\x2c\x04\x4b\x40\x8e\xf1\xc4\x4e\x40\x5b\x4a\x2e\x80\x6c\x89\x54\x19\x96\x30\x97\xa0\x16\x8e\x82\xff\x98\x5a\x5d\x54\x18\xd9\x2c\x60\x8c\x6f\x5c\x37\x72\xdd\x88\x51\xa5\xdb\x1a\x51\x22\xb2\x42\x69\xae\xac\xb2\x55\xcd\x8f\x8a\xe4\xc5\xec\x16\x62\xad\x02\xbf\xbe\x23\xb7\xa7\x84\x36\x7c\xec\x99\xa6\x69\x06\x2a\x88\xa9\x8c\x19\xa0\x98\x01\x91\x41\x49\x7f\x4e\x99\x06\xd9\x81\xc1\x6e\x4e\xaf\xd4\xff\x65\xc1\xb6\x52\x47\x1e\x9d\x13\xad\xc3\xb9\xdd\xfc\xe9\x76\x30\x3c\x54\x4a\x6e\x17\x02\xb7\x0b\x03\x45\x54\x9a\x47\x49\x35\xaa\x61\x3f\x88\x72\x40\xf9\x5c\xc8\x8c\x38\x1f\x75\x00\xfd\x12\x89\xd1\x29\x1c\x7d\x34\xb8\xd0\xc5\x0e\x57\x83\xa3\x3a\x10\xa3\x1c\x54\x0d\x65\x7b\x1c\x47\x13\xfa\x15\x4e\xbd\x7a\xef\x5c\xaa\x16\x9a\xb0\xa9\xa2\x5f\x77\x16\xd7\xa2\xd5\x85\xed\x0d\xd1\x04\xa9\x4e\x94\x66\x17\x8f\xc0\x68\xc3\x58\x0f\x4a\x1b\xeb\x8e\xc0\x79\xae\x34\xcd\x88\x86\x04\x49\xb1\x52\x6d\xbc\xa6\x77\x1a\x8b\xc2\xc4\xe7\x3e\xbc\x03\x95\xa1\xcb\xdf\x9a\x3c\xa2\xee\x6c\x4d\xaf\x35\x94\x3d\xca\x91\x83\xec\xd6\xcf\xb8\x50\x5a\x64\x3e\x30\x77\x30\x5a\xc1\xd3\x8e\x71\xa8\x69\xa3\xb3\x42\x6b\x13\x67\xac\x6b\x2b\x38\x46\x4b\xc2\x0a\xe3\xcf\x0a\x8e\xae\x6b\xb1\xbf\x0c\x41\x2a\xb3\x3f\xb9\xa4\x19\x31\xc3\xe1\x40\xec\xb0\xce\x99\x49\xa0\x4a\x0a\xe7\xee\x7b\x3f\x15\x1f\xef\xda\x54\x4a\x69\x58\x31\x4c\x73\x29\x52\x09\xe6\x40\xf1\x81\x81\xf1\x2b\x2b\x42\xf5\x9f\x90\x1d\x44\x54\x21\x58\x43\x5c\x68\xca\xd3\x93\x93\x93\x07\x54\x26\x2f\x18\x0b\x24\x4d\x17\xba\x21\x36\x54\xd1\x1b\x09\xaa\x60\x3a\x30\x8a\x13\xd4\x15\xa7\x09\xd2\x27\x8d\x5b\x25\x76\xa2\xf8\x65\xf2\xfe\xea\x50\x09\xec\xc1\x1d\xab\xe5\x16\xf5\xd9\xe4\xf3\x13\x62\x5e\x67\x6c\x8b\xf9\xcb\xbb\xcb\x83\x77\xed\x48\xab\xaa\x7a\xe7\x42\x1f\x68\x37\xd6\xc6\x2b\xbb\xa6\x70\xcd\x57\x37\x8d\x43\x9b\x64\xa2\x3a\x7d\x6a\x3a\x5c\xa8\xd3\x65\x21\xa0\x3a\xdf\x98\x77\x6d\xbe\xb3\xf7\x51\xa8\xcb\x03\xc4\x6e\x79\x96\xe2\x41\xeb\xcd\x49\x4a\x79\x2b\x90\x98\x88\xb0\x0d\xd5\x36\x6a\x2a\x8c\x9c\xb5\x6f\x73\x49\xeb\xd5\xca\x30\xdd\x4c\x6d\x22\x97\xfe\x74\x64\x57\x2e\xc5\xdf\x25\xcc\xac\xc8\x38\x32\xf4\xec\xa9\x52\x0a\x66\x55\xdd\x4e\xda\x03\xe7\xe8\x36\xe0\xea\x82\x13\xb9\x61\xb7\xd4\x26\x1c\x4d\x1c\x02\x07\x39\x0a\xdd\xf0\x5e\x18\xb8\x2f\x08\xc3\xd1\x78\xd0\x64\x2e\xf4\xd4\x03\xfc\xc0\xe1\xef\x83\x60\x52\x09\xc4\x8a\xef\x87\x54\x1f\x04\x31\x85\x7b\x03\x34\x90\x0c\xb3\x9e\xeb\x07\x36\x90\x86\x99\xee\x08\xb0\xa1\x04\xe8\x1d\xe0\xe8\xf2\xe2\xd7\xf3\x41\xd3\xa9\x9b\x7f\x31\x18\x80\x17\x8c\xe1\xe8\x62\x82\xae\x3e\x5d\x5e\x0e\xde\x0d\x07\x75\xf5\xfe\x63\x2f\x58\xa7\xa2\xb5\x33\xcc\x52\xe7\xaa\xca\xd6\x99\x76\x5a\xe2\xb8\x12\xf9\x7d\xc6\xe9\xfb\xeb\x27\x00\xe7\xf7\x9a\x1e\xce\x87\xbd\xd2\xdb\x91\x3c\x67\x9b\x60\x6b\x82\x8e\x2b\x55\xcc\x32\xaa\x71\xf4\x93\x19\x1c\x85\x0e\xd1\x00\xdc\xf5\xc3\x1d\x92\xa0\x40\x6f\x71\xb7\x0e\x13\x36\x39\x2f\x8f\x12\x4d\x1a\xa3\xd0\x48\xa3\xdb\x4b\x1a\xf4\xa9\x14\x45\x23\xff\xf4\x3c\xd5\x9d\xfe\x7e\x0e\x73\x09\xcb\x20\xb7\xe5\xc8\x84\x2a\x23\x51\x73\x04\xf3\xad\x36\xc7\x84\xa7\x0c\x02\x06\x73\xdd\xc7\xf6\x91\x6c\x90\x14\x50\xbc\x20\x3c\x85\x80\x51\x23\xfa\xf2\x54\x76\xc6\x68\x7c\x87\xb4\xf0\xa3\x26\xff\x43\x6e\xc6\x93\x11\xe7\xb0\xd6\x4e\x06\x3d\xeb\xf5\xb9\x44\xdf\x3e\xd5\x03\x60\xc7\x51\xdc\xcb\x97\x68\x62\x9b\x63\xfc\xa2\xf2\xa5\xcc\x67\xfb\xe0\x6a\x35\xdc\xe6\xf4\x12\x62\x21\x13\xb5\xcd\x4f\x6c\x0a\x3c\x3c\x8d\xf5\x8d\x6a\x15\x6c\x57\x9d\x98\xae\x28\x4f\xc4\xaa\x5a\x12\x6b\xd7\x59\x14\x68\x93\x78\xa9\xce\x72\x43\x19\x5a\x77\x0c\x2d\x5e\x94\xe5\xf5\xc5\x8b\x6e\xb9\x2c\x41\x2a\x1b\x0e\xf7\xc8\xae\xc8\x13\xa2\x01\x11\x06\x52\xbb\xbf\xc1\x8a\x48\x4e\x79\xda\x00\xdb\xad\x0f\x6d\xe3\xaa\x14\x46\x75\x4c\xb3\xee\x5a\x16\x42\xd2\xaf\x82\x6b\xc2\x70\x53\x10\x76\x72\x37\x2b\xc6\x45\x05\x31\xf0\x56\x18\xee\xb2\x48\xb4\x6d\x19\xd5\xaa\x48\xd1\xf7\xad\xa8\xae\x1d\xfa\xfb\x95\xd6\x68\xc8\x18\xab\x78\x01\x19\xf4\xa9\x70\x6b\x19\x7e\x7a\x34\xb1\xbf\x5d\x16\xf2\x00\x39\x4d\x78\x42\x64\xd2\x6b\x33\xb6\xd0\x07\x6d\xba\x25\x5c\x34\xf1\xad\x23\x68\xab\xc5\xf0\x75\xaa\x05\x8e\x26\x93\xb7\x9d\x3e\xa0\x33\x25\xad\x28\xa8\x0c\xab\x9f\x3d\x4a\x6f\x05\xd8\xe5\x65\x2b\xf3\xad\x52\xb5\xa7\x34\x91\xb2\x40\x65\xc1\x8b\x97\xed\xd3\x08\x23\x33\x60\xd1\xb9\xd1\x2c\xa4\x40\x2e\x41\xa2\x4f\x37\x97\x48\xf9\xcd\x73\xc3\xfb\x32\xfb\x3d\xd1\xb3\x21\xb0\x42\x32\x8c\x38\xc9\x60\x8c\x6d\x93\x14\x5a\xc4\x22\xcb\x19\x68\x18\x63\x31\x9f\xb7\xb9\xcb\x77\x36\xce\xf2\x60\xc6\x44\x7c\xd7\x9a\x84\x2c\xc3\xae\xbc\x71\x8a\x72\xa1\xb4\x39\xc5\x9d\x86\x61\xa1\x40\x9e\xe6\x44\xa9\x95\x90\xc9\x3f\x17\x42\xe9\xd3\x5c\x48\x1d\x26\xb3\x7f\x28\xc5\x32\x91\xc0\xd8\xfc\x19\xcd\x64\xeb\xec\x82\xd0\x8d\x49\xe0\x33\x21\x01\x09\x8e\x3e\x38\xa4\x93\xeb\xcb\x5d\xb1\x75\xa1\x75\xae\xfc\x7d\x5f\x49\xf4\x9e\xd9\x7b\xbf\x44\xc4\x2a\xf4\x9e\x37\xf4\x17\x05\x8c\xce\xf2\xfb\xc0\x4b\xe3\x64\xa1\x33\xf6\xfd\xe5\xc5\xcf\x1f\xae\x83\xb3\xf7\x57\x57\x93\x8f\x37\x17\x57\xff\xc6\x48\x13\x99\x82\x1e\xe3\xe9\x8c\x11\x7e\x87\xa3\x9d\xf0\x90\xd2\x92\xf2\xd4\x2f\x73\x14\x92\xe8\xa4\x29\xab\x30\x3f\xea\x9c\xf4\xa0\x16\x7a\x63\x1a\xac\x87\x68\x26\xc4\x5d\x46\xe4\x5d\xab\x64\x60\x95\xa9\xa1\x93\xaf\x90\x57\x97\xc0\x8e\xe2\xe8\x67\x0f\xdd\xa9\x7b\x1d\x4a\xfd\xba\xe3\x84\x5d\x3f\x45\xec\xd3\xc9\x0a\xaf\x5d\x79\x62\xbf\x14\x0f\x36\xc7\x41\x8b\x7f\x2b\x94\x1e\xba\xf0\xbf\xb6\x17\xde\xb6\x4b\x7b\xfa\x4b\xa7\x46\xf9\x7b\x8c\xf4\xa1\xd3\x75\x27\xed\x97\x88\x0b\x9b\x83\x05\x39\x49\x12\x1b\x11\x07\xf3\x62\x6c\x70\x48\xba\xfd\xe7\x1f\x5f\xbd\x1c\x7c\xf8\x7f\x9e\xed\xf8\xa4\x40\x1a\x87\xf5\x08\x5d\xec\x15\x83\xf1\x4e\x47\x6e\xc9\xb3\xae\xf9\x83\x77\x98\x4f\xb4\xe6\xd2\xff\xee\xb6\x7f\xdb\xf1\xed\xad\xfd\x8d\xbf\x34\x78\x8e\xfd\x4e\x66\xdf\xe2\x8a\x27\x93\x4b\xf4\xce\x04\xc1\xdf\xc4\xdb\x2a\xd5\x2c\xe0\xa0\xf6\xa9\xde\x1f\xfd\x70\xe4\x1b\xdd\xa5\x80\x0e\x40\x09\xf7\x05\x95\x80\x51\x79\xc1\x5c\xbd\x6a\xf6\x83\x83\x91\x2d\x41\xd2\xf9\x26\x98\xdb\xf2\x42\xe5\xa3\x0f\xc1\x61\xd1\xa3\x6f\x6f\xfb\x62\xb0\x5a\x74\x85\xdf\x7a\x16\xf9\xc4\x7a\xf1\x16\x3d\x4b\x30\x52\x6a\xf1\xed\x44\x23\xc3\xcc\xd0\x70\xf4\xf2\x7f\x1d\x8c\xcc\x96\x98\x80\xf4\xe4\xce\xc9\x48\xe1\x5b\x8d\x46\x66\xd1\xcf\x1a\x91\xac\x0a\xec\x0f\x49\x35\x35\x70\xd6\x4f\xbe\x05\xb9\xfc\x0a\x9b\x67\xd1\x85\x3b\xd8\x3c\x8b\x24\xfa\x04\xd3\x55\x14\x02\x29\xc5\x4e\x21\xab\x25\x98\x84\xf0\xd4\x5d\xa4\xf4\xbb\xce\x1e\xe1\x0e\x39\x12\xd7\xab\x03\xbe\x08\xdb\x2c\x09\xd8\x83\x68\xad\x92\x2b\x72\xe0\x41\xc7\x9b\xa2\x43\x4a\x10\x5d\x6f\x77\xca\xea\x58\x3f\x07\x65\x79\x22\x3a\x23\x3c\x06\x36\xbc\x1a\x51\x7e\xec\x6a\xbc\xcd\x82\x5d\x65\x6b\xdc\x5b\x88\xa9\x7f\x13\x38\xcd\x80\x17\x65\xc5\xae\xd8\x2a\x6b\x22\x45\x9e\x88\x15\x0f\xec\xb0\x2f\x81\x55\xa6\xda\x37\x58\x51\xf5\xb9\x92\x2d\x41\x96\x57\x4e\xb1\xc8\x37\x46\x70\xf9\x06\xb9\x47\x0b\x57\x36\xf5\x26\x51\xed\x55\x16\xdb\xd6\x47\x13\xba\xa4\xfe\x4e\xad\x36\xde\x4f\x01\xd6\xce\xdb\xdb\x4e\x77\x94\xf6\x97\xa8\xd1\xb9\x1d\x42\x5a\xa0\x5f\x26\xef\xaf\x3a\xa8\x1e\x88\x35\x56\xcb\x2a\xd2\xb3\xc9\xe7\xc7\xe3\x5c\x67\xac\x8a\xf3\xcb\xbb\xcb\x83\x70\x26\x45\x96\x57\xe1\x27\xd7\x5d\xf0\xc7\x4b\x57\xcb\x82\xc7\x44\x03\x8e\x3e\xfa\x96\xdb\xc7\xc3\x98\x04\x06\x06\xc5\x1b\xfb\xdb\x85\xc0\xbd\xcb\xeb\x52\xd2\x25\x85\xd5\x6f\xa6\xa2\x9f\x29\xac\xfe\xaf\xa1\xc7\x68\xe8\xf1\xd2\x69\x28\x87\xd9\x82\xa1\xba\xd1\x7c\x14\xf8\xbc\x7a\xe2\x85\x53\x8a\x62\x72\x7d\x89\x8c\xf5\x0d\x65\xb6\xfe\x68\xe0\x79\x58\x2c\x38\xbd\x2f\xc0\xdd\x5a\x2a\x3f\x66\x5f\xa1\xd8\x9b\x27\xa6\x00\x47\x9f\xec\x14\xf4\xd9\x4e\x39\x48\x39\xf6\x21\xd7\xb2\x68\xe2\x46\x7f\x44\x67\x76\xf4\x20\x22\xbc\xc8\xa6\x4a\x13\xad\x70\x74\x55\x64\x20\x69\x8c\xec\x27\xfa\x7d\x46\x79\x98\x91\x75\x48\x96\xe9\x1f\x0e\x42\x69\xec\x7b\x6a\x5f\x92\x3a\x23\x3f\x73\x2f\x17\xda\x66\xfe\xf0\xde\x49\xb1\x7a\x3e\x1f\xe4\x2f\x9b\xbd\x27\x32\xed\x83\x96\xe9\x6f\xac\x67\x5b\x3c\xfe\x6a\xfb\x46\xac\x14\xfa\xb9\x13\x63\x63\xc1\xa3\xd0\xbd\x4b\x19\x85\xee\x3f\x64\xfe\x1b\x00\x00\xff\xff\xdb\x2d\x6c\xcd\x32\x33\x00\x00") +var _staticIndexHtml = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\x3b\x5b\x73\xe3\xb6\xd5\xef\xf9\x15\xf8\x90\x99\xcc\xd7\x69\x69\xce\xee\x26\x6d\xd7\x95\xd8\x6e\xbc\x6e\xd7\x89\xe3\xbd\x68\x77\x27\x7d\xd2\x40\xe4\x91\x08\x1b\x04\x68\x00\xd4\x25\xbf\xbe\x83\x0b\x25\x5e\x65\x52\x5e\x27\xe9\x4c\x5f\x2c\x10\xc0\xb9\xe0\x5c\x81\x03\x78\xf2\x7f\xaf\xdf\x5e\x7c\xfc\xf7\xbb\x4b\x94\xea\x8c\x45\x5f\x4d\xcc\x0f\x62\x84\xaf\xa6\x18\x38\x46\xdb\x8c\x9d\xd7\xbe\xb8\x9a\xe2\x54\xeb\xfc\x3c\x0c\x37\x9b\xcd\xd9\xe6\xc5\x99\x90\xab\xf0\xd9\xcb\x97\x2f\xc3\xad\x81\xc5\x06\x07\x90\x24\xfa\x0a\xa1\x89\xa6\x9a\x41\x94\xaf\x36\xb0\x98\x84\xee\xc3\x74\x67\xa0\x09\x8a\x53\x22\x15\xe8\x29\x2e\xf4\x32\xf8\x2b\x3e\x0c\x18\xec\x01\xdc\x17\x74\x3d\xc5\x3f\x07\x9f\x5e\x05\x17\x22\xcb\x89\xa6\x0b\x06\x18\xc5\x82\x6b\xe0\x7a\x8a\xaf\x2e\xa7\x90\xac\xa0\x07\xee\xc2\x4d\x0b\xae\x09\x5f\x15\x64\x55\x05\x04\xee\x60\x18\xe5\x77\x48\x02\x9b\x62\xa5\x77\x0c\x54\x0a\xa0\x31\x4a\x25\x2c\x4d\x0f\xd1\x34\x0e\x63\xa5\xc2\x85\x10\x5a\x69\x49\xf2\xb3\x58\x29\x1c\x4d\x42\x03\x37\x06\xc1\x52\x70\x1d\x90\x0d\x28\x91\xc1\xa9\x38\x48\x7e\x8c\x3c\x8d\x05\xc7\x48\xef\x72\x98\x62\x9a\x91\x15\x84\xdb\xc0\xf5\xd5\x10\xd1\x6c\x15\x9a\xee\x33\x1a\x0b\x8c\x42\x8b\x44\xc5\x92\xe6\xda\xc3\x6a\xd8\xea\xf0\x96\xac\x89\xeb\xc5\x48\xc9\x78\x0f\x7d\xab\xc2\xdb\xfb\x02\xe4\xee\xec\xd6\xf2\xe1\xe6\x8c\x47\x42\x62\x78\x3c\x86\x20\x5f\xa9\x7b\xf6\x48\x3c\x7b\xcd\x06\xd6\x36\xb6\x3a\x03\x5e\x3c\x12\x67\xa1\x29\x53\x5f\x8a\xaf\x17\x81\x01\x20\xc6\x9b\xce\x32\xca\x1f\x2b\xb6\x3c\xaf\x63\x40\xe3\x39\x23\x0a\xfe\xfc\x6d\x83\x8f\x49\xe8\xdc\x7d\xb2\x10\xc9\xce\xb2\x95\xd0\x35\xa2\xc9\x14\x67\x84\x3a\x5f\xab\xf4\x71\xb2\xf6\x5d\x08\x4d\x0a\x56\x36\xad\x3d\xdb\x09\x9a\x2c\x18\xcc\xbd\xb7\xe2\xe8\x83\xd8\x28\x63\xf4\x7d\x13\x95\x96\x45\xac\x0b\x09\x38\x9a\x95\xcd\x63\xf3\x29\x4f\x60\x0b\x0a\x47\x57\xae\x71\x6c\x6e\x2c\xb8\xd1\x03\xe5\x5a\xe1\xe8\xe2\xf0\x71\x0c\xc6\xba\x08\x46\x31\x23\x4a\x4d\xb1\x02\x06\xb1\x86\x04\x47\xef\x4d\xff\x31\xc0\x94\x2a\x2d\xe4\x0e\x47\x6f\x5c\xe3\xd8\x5c\x12\x6b\xba\xa6\x7a\x87\xa3\x57\xbe\xf5\xc0\x32\x38\xc4\x9a\x0a\x6e\x57\xe1\xdb\x55\x88\x49\x68\x34\x51\x7e\x18\x55\x79\xfe\x0f\xa0\x01\xb1\x3f\x0a\x57\xa8\x10\x1f\x5f\xbe\xc6\x96\x1a\x24\x54\x57\x89\x95\x48\x16\x9a\xa3\x85\xe6\x41\x02\x4b\x52\x30\x6d\xdb\x2a\xc3\xd1\x84\x96\x33\x96\x04\x2d\x49\x90\x10\x4d\x8c\x85\x19\xe3\xa2\x11\xf2\xac\x4e\x42\xd2\x4f\x32\x66\x42\xc1\x18\x9a\xaf\xa9\x8a\x5b\x78\x27\x61\x42\xd7\xde\x4e\x2b\xcd\xd2\x64\x15\x4d\x60\x41\xe4\xc1\x6c\xab\x02\x2a\xa4\x34\x79\xe6\xc0\xfb\x81\xd7\xca\xb4\x8d\x24\x79\x65\x08\xa1\x07\xd6\x3e\x51\x39\xe1\x7d\x34\x02\x4e\x32\xf0\xeb\x77\x43\xf3\x2a\xb8\x01\xad\x93\xe2\x79\xa1\x4b\x64\xfb\x88\xe2\x10\x94\x80\x73\x05\x44\xc6\x29\xae\x04\x02\x8c\x72\x46\x62\x48\x05\x4b\x40\x4e\xf1\xcc\x4e\x40\x7b\x4a\x2e\x81\xec\x89\x54\x19\x96\xb0\x94\xa0\x52\x47\xc1\x7f\xcc\xad\x2d\x2a\x8c\xec\x2e\x60\x8a\x3f\xb8\x6e\xe4\xba\x11\xa3\x4a\xb7\x2d\xa2\x44\x64\x85\xd2\x5c\x59\x45\x55\xcd\x8f\x8a\xe4\xc5\xe2\x16\x62\xad\x02\xbf\xbe\x13\xd5\x53\x42\x1b\x3e\x8e\x4c\xd3\x34\x03\x15\xc4\x54\xc6\x0c\x50\xcc\x80\xc8\xa0\xa4\xbf\xa4\x4c\x83\xec\xc0\x60\x95\xd3\x2b\xf5\x7f\x5a\xb0\xbd\xd4\x91\x47\xe7\x44\xeb\x70\xee\x95\x3f\xdf\x0f\x86\x63\xa5\xe4\xb4\x10\x38\x2d\x0c\x14\x51\xe9\x1e\x25\xd5\xa8\x86\x7d\x14\xe5\x80\xf2\xa5\x90\x19\x71\x31\x6a\x04\xfd\x12\x89\xb1\x29\x1c\x7d\x34\xb8\xd0\xd5\x01\x57\x83\xa3\x3a\x10\xa3\x1c\x54\x0d\x65\x7b\x1c\x47\x33\xfa\x0b\x9c\x7b\xf3\x3e\x84\x54\x2d\x34\x61\x73\x45\x7f\x39\x78\x5c\x8b\x56\x17\xb6\xd7\x44\x13\xa4\x3a\x51\x1a\x2d\x9e\x80\xd1\xa6\xb1\x1e\x94\x36\xd7\x9d\x80\xf3\x52\x69\x9a\x11\x0d\x09\x92\x62\xa3\xda\x78\x4d\xef\x3c\x16\x85\xc9\xcf\x7d\x78\x07\x1a\x43\x57\xbc\x35\xfb\x88\x7a\xb0\x35\xbd\xd6\x51\x8e\x18\x47\x0e\xb2\xdb\x3e\xe3\x42\x69\x91\xf9\xc4\xdc\xc1\x68\x05\x4f\x3b\xc7\xa1\xa6\x8f\x2e\x0a\xad\x4d\x9e\xb1\xa1\xad\xe0\x18\xad\x09\x2b\x4c\x3c\x2b\x38\x7a\x5f\xcb\xfd\x65\x0a\x52\x99\xfd\xc9\x25\xcd\x88\x19\x0e\x07\x62\x87\x6d\xce\xcc\x06\xaa\xa4\x70\xe9\xbe\x8f\x53\xf1\xf9\xae\x4d\xa5\x94\x86\x15\xc3\x3c\x97\x62\x25\xc1\x1c\x28\xde\x31\x30\x71\x65\x43\xa8\xfe\x13\xb2\x83\x88\x2a\x04\x5b\x88\x0b\x4d\xf9\xea\xec\xec\xec\x01\x93\xc9\x0b\xc6\x02\x49\x57\xa9\x6e\x88\x0d\x55\xec\x46\x82\x2a\x98\x0e\x8c\xe1\x04\x75\xc3\x69\x82\xf4\x49\xe3\x56\x89\x83\x28\x7e\x98\xbd\xbd\x19\x2b\x81\x23\xb8\x63\xb5\xde\xa3\xbe\x98\x7d\xfe\x82\x98\xb7\x19\xdb\x63\xfe\xf9\xa7\xeb\xd1\x5a\x3b\xd1\xab\xaa\xd1\xb9\xd0\x23\xfd\xc6\xfa\x78\x45\x6b\x0a\xd7\x62\x75\xd3\x39\xb4\xd9\x4c\x54\xa7\xcf\x4d\x87\x4b\x75\xba\x2c\x04\x54\xe7\x1b\xf7\xae\xcd\x77\xfe\x3e\x09\x75\x79\x80\x38\x2c\xcf\x52\x1c\xb5\xde\x9c\xac\x28\x6f\x25\x12\x93\x11\xf6\xa9\xda\x66\x4d\x85\x91\xf3\xf6\xfd\x5e\xd2\x46\xb5\x32\x4d\x37\xb7\x36\x91\xdb\xfe\x74\xec\xae\xdc\x16\xff\xb0\x61\x66\x45\xc6\x91\xa1\x67\x4f\x95\x52\x30\x6b\xea\x76\xd2\x11\x38\x47\xb7\x01\x57\x17\x9c\xc8\x0d\xbb\xa5\x35\xe1\x68\xe6\x10\x38\xc8\x49\xe8\x86\x8f\xc2\xc0\x7d\x41\x18\x8e\xa6\x83\x26\x73\xa1\xe7\x1e\xe0\x1b\x0e\x7f\x1b\x04\xb3\x92\x40\xac\xf8\xbe\x59\xe9\x51\x10\x73\xb8\x37\x40\x03\xc9\x30\x1b\xb9\xbe\x61\x03\x69\x98\xe9\x8e\x00\x1b\x4a\x80\xde\x01\x8e\xae\xaf\x7e\xbc\x1c\x34\x9d\xba\xf9\x57\x83\x01\x78\xc1\x18\x8e\xae\x66\xe8\xe6\xd3\xf5\xf5\x60\x6d\x38\xa8\x9b\xb7\x1f\x7b\xc1\x3a\x0d\xad\xbd\xc3\x2c\x6d\xae\x6a\x6c\x9d\xdb\x4e\x4b\x1c\x57\x32\xbf\xdf\x71\xfa\xfe\xfa\x09\xc0\xc5\xbd\x66\x84\xf3\x69\xaf\x8c\x76\x24\xcf\xd9\x2e\xd8\xbb\xa0\xe3\x4a\x15\x8b\x8c\x6a\x1c\xbd\x32\x83\x93\xd0\x21\x1a\x80\xbb\x7e\xb8\x43\x12\x14\xe8\x3d\xee\xd6\x61\xc2\x6e\xce\xcb\xa3\x44\x93\xc6\x24\x34\xd2\xe8\x8e\x92\x06\xfd\x4a\x8a\xa2\xb1\xff\xf4\x3c\xd5\x83\xfe\x71\x0e\x73\x09\xeb\x20\xb7\xe5\xc8\x84\x2a\x23\x51\x73\x04\xf3\xad\x36\xc7\x84\xaf\x18\x04\x0c\x96\xba\x8f\xed\x13\xd9\x20\x2b\x40\x71\x4a\xf8\x0a\x02\x46\x8d\xe8\xcb\x53\xd9\x05\xa3\xf1\x1d\xd2\xc2\x8f\x9a\xfd\x1f\x72\x33\xbe\x18\x71\x0e\x5b\xed\x64\xd0\xb3\x5e\xbf\x97\xe8\xd3\x53\x3d\x01\x76\x1c\xc5\xbd\x7c\x89\x26\xb6\x39\xc5\xcf\x2a\x5f\xca\x7c\xb6\x0f\xae\xd6\xc2\xed\x9e\x5e\x42\x2c\x64\xa2\xf6\xfb\x13\xbb\x05\x1e\xbe\x8d\xf5\x8d\x6a\x15\xec\x50\x9d\x98\x6f\x28\x4f\xc4\xa6\x5a\x12\x6b\xd7\x59\x14\x68\xb3\xf1\x52\x9d\xe5\x86\x32\xb5\x1e\x18\x4a\x9f\x95\xe5\xf5\xf4\x59\xb7\x5c\xd6\x20\x95\x4d\x87\x47\x64\x57\xe4\x09\xd1\x80\x08\x03\xa9\xdd\xdf\x60\x43\x24\xa7\x7c\xd5\x00\x3b\xac\x0f\xed\xf3\xaa\x14\xc6\x74\x4c\xb3\x1e\x5a\x52\x21\xe9\x2f\x82\x6b\xc2\x70\x53\x10\x76\x72\x37\x2b\x26\x44\x05\x31\xf0\x56\x1a\xee\xf2\x48\xb4\x6f\x19\xd3\xaa\x48\xd1\xf7\x6d\xa8\xae\x1d\xfa\xfb\x8d\xd6\x58\xc8\x14\xab\x38\x85\x0c\xfa\x4c\xb8\xb5\x0c\x3f\x3d\x9a\xd9\xdf\x2e\x0f\x79\x80\x9c\x26\x3c\x21\x32\xe9\xf5\x19\x5b\xe8\x83\x36\xdd\x12\x2e\x9a\xf9\xd6\x09\xb4\x55\x3a\x7c\x9d\x2a\xc5\xd1\x6c\xf6\xa6\x33\x06\x74\x6e\x49\x2b\x06\x2a\xc3\xea\x67\x8f\xd1\x5b\x01\x76\x45\xd9\xca\x7c\x6b\x54\xed\x29\x4d\xa4\x2c\x50\x59\xf0\xec\x79\xfb\x34\xc2\xc8\x02\x58\x74\x69\x2c\x0b\x29\x90\x6b\x90\xe8\xd3\x87\x6b\xa4\xbc\xf2\xdc\xf0\xb1\x9d\xfd\x91\xec\xd9\x10\x58\x21\x19\x46\x9c\x64\x30\xc5\xb6\x49\x0a\x2d\x62\x91\xe5\x0c\x34\x4c\xb1\x58\x2e\xdb\xdc\xe5\x07\x1f\x67\x79\xb0\x60\x22\xbe\x6b\x4d\x42\x96\x61\x57\xde\x38\x47\xb9\x50\xda\x9c\xe2\xce\xc3\xb0\x50\x20\xcf\x73\xa2\xd4\x46\xc8\xe4\x1f\xa9\x50\xfa\x3c\x17\x52\x87\xc9\xe2\xef\x4a\xb1\x4c\x24\x30\x35\x7f\x26\x0b\xd9\x3a\xbb\x20\xf4\xc1\x6c\xe0\x33\x21\x01\x09\x8e\xde\x39\xa4\xb3\xf7\xd7\x87\x62\x6b\xaa\x75\xae\xfc\x7d\x5f\x49\xf4\x9e\xd9\x7b\xbf\x44\xc4\x2a\xf4\x91\x37\xf4\x17\x05\x8c\x2e\xf2\xfb\xc0\x4b\xe3\x2c\xd5\x19\xfb\xfa\xfa\xea\xfb\x77\xef\x83\x8b\xb7\x37\x37\xb3\x8f\x1f\xae\x6e\xfe\x85\x91\x26\x72\x05\x7a\x8a\xe7\x0b\x46\xf8\x1d\x8e\x0e\xc2\x43\x4a\x4b\xca\x57\x7e\x99\x93\x90\x44\x67\x4d\x59\x85\xf9\x49\xe7\xa4\x07\xad\xd0\x3b\xd3\x60\x3b\x44\x0b\x21\xee\x32\x22\xef\x5a\x25\x03\x6b\x4c\x0d\x9b\x7c\x81\xbc\xb9\x04\x76\x14\x47\xdf\x7b\xe8\x4e\xdb\xeb\x30\xea\x97\x1d\x27\xec\xfa\x29\xe2\x98\x4d\x56\x78\xed\xda\x27\xf6\x4b\x71\xb4\x3b\x0e\x5a\xfc\x1b\xa1\xf4\xd0\x85\xff\xa5\xbd\xf0\xb6\x5f\xda\xd3\xdf\x6a\x6e\x8c\xbf\xc7\x49\x1f\x3a\x5d\x77\xd2\x7e\x8e\xb8\xb0\x7b\xb0\x20\x27\x49\x62\x33\xe2\x60\x5e\x8c\x0f\x0e\xd9\x6e\x7f\xf7\xed\x8b\xe7\x83\x0f\xff\x4f\xa3\x8e\x4f\x0a\xa4\x09\x58\x8f\xb0\xc5\x5e\x31\x98\xe8\x74\xa2\x4a\x9e\x74\xcd\xef\x7c\xc0\xfc\x42\x6b\x2e\xe3\xef\x41\xfd\xfb\x8e\xdf\xdf\xda\x5f\xfb\x4b\x83\xa7\xd0\x77\xb2\xf8\x3d\xae\x78\x36\xbb\x46\x3f\x99\x24\xf8\xab\x44\x5b\xa5\x9a\x05\x1c\xd4\x3e\xd5\xfb\xa3\x1f\x8e\x7c\xa3\xbb\x14\xd0\x01\x28\xe1\xbe\xa0\x12\x30\x2a\x2f\x98\xab\x57\xcd\x7e\x70\x30\xb2\x35\x48\xba\xdc\x05\x4b\x5b\x5e\xa8\x7c\xf4\x21\x18\x97\x3d\xfa\x74\xdb\x97\x83\x55\xda\x95\x7e\xeb\xbb\x48\xd3\xf1\xa2\xf3\xc0\x30\x9b\xbd\x41\xd5\x3b\xee\xf4\xc5\x7f\x55\xfe\x52\x2a\xfd\xfd\x24\x30\xc3\xcc\xd0\x0c\xf6\xfc\xb7\xce\x5f\x17\x12\x12\xe0\x9a\x12\xa6\x86\x6a\xe5\xbb\x31\x82\x38\x92\xc3\x6a\x82\x28\xf3\xe8\x89\xca\xfa\x76\x9c\xb2\xea\x29\xc7\x2a\xec\x78\xce\xa9\xf1\xfa\x6e\x3f\xf7\x37\x55\xdd\xab\x42\xa7\xe8\x47\xd8\x3d\x89\xde\xee\x60\x37\x44\x14\x3f\xc2\x0e\xe5\x44\xa7\xbf\x95\xda\xee\x60\x37\x4a\x75\x8e\xdf\x47\xa8\x2f\x95\xed\x47\x02\xf5\xd8\xdc\x48\x68\x20\xa5\x38\x78\x40\xb5\x52\x94\x10\xbe\x72\xf7\x3d\xfd\x11\xbe\xc7\x44\x86\x9c\xdc\xeb\x45\x0c\x5f\x2b\x6e\x56\x2e\xec\x79\xb9\x56\x70\x16\x39\xf0\xa0\xe3\xe9\xd3\x98\x4a\x49\xd7\x13\xa3\xb2\x88\xd7\xcf\x41\x59\x45\x89\x2e\x08\x8f\x81\x0d\x2f\x9a\x94\x1f\x87\x52\x74\xb3\xae\x58\x51\x8d\x7b\xb2\x31\xf7\x4f\x17\xe7\x19\xf0\xa2\x2c\x2c\x16\x7b\x97\x4b\xa4\xc8\x13\xb1\xe1\x81\x1d\xf6\x95\xba\xca\x54\xfb\x54\x2c\xaa\xbe\xaa\xb2\x95\xd2\xf2\x66\x2c\x16\xf9\xce\x08\x2e\xdf\x21\xf7\xb6\xe2\xc6\x9e\x10\x48\x54\x7b\x3c\xc6\xf6\x65\xdc\x84\xae\xa9\xbf\xfa\xab\x8d\xf7\x53\x80\xad\xcb\x30\xb6\xd3\x9d\xf8\xfd\x5d\x6f\x74\x69\x87\x90\x16\xe8\x87\xd9\xdb\x9b\x0e\xaa\x23\xb1\xc6\x6a\x5d\x45\x7a\x31\xfb\xfc\x78\x9c\xdb\x8c\x55\x71\xfe\xfc\xd3\xf5\x28\x9c\x49\x91\xe5\x55\xf8\xd9\xfb\x2e\xf8\xd3\xa5\xab\x65\xc1\x63\xa2\x01\x47\x1f\x7d\xcb\xe9\x71\x1c\x93\xc0\xc0\xa0\x78\x6d\x7f\xbb\x10\xb8\xe7\x83\x5d\x46\xba\xa6\xb0\xf9\xd5\x4c\xf4\x33\x85\xcd\xff\x2c\xf4\x14\x0b\x3d\x5d\x3a\x0d\xe3\x30\x2a\x18\x6a\x1b\xcd\xb7\x8b\x4f\x6b\x27\x5e\x38\xa5\x28\x66\xef\xaf\x91\xf1\xbe\xa1\xcc\xd6\xdf\x36\x3c\x0d\x8b\x05\xa7\xf7\x05\xb8\xcb\x55\xe5\xc7\xec\x63\x19\x7b\x41\xc6\x14\xe0\xe8\x93\x9d\x82\x3e\xdb\x29\xa3\x8c\xe3\x18\x72\x2d\x8b\x26\x6e\xf4\x47\x74\x61\x47\x47\x11\xe1\x45\x36\x57\x9a\x68\x85\xa3\x9b\x22\x03\x49\x63\x64\x3f\xd1\xff\x67\x94\x87\x19\xd9\x86\x64\xbd\xfa\xc3\x28\x94\xc6\xbf\xe7\x76\x43\xed\x9c\xfc\xc2\x3d\xb0\x68\xbb\xf9\xc3\xba\x93\x62\xf3\x74\x31\xc8\xdf\x89\xfb\x48\x64\xda\xa3\x96\xe9\x2f\xd6\x17\x7b\x3c\xfe\x06\xfe\x83\xd8\x28\xf4\x7d\x27\xc6\xc6\x82\x27\xa1\x7b\x3e\x33\x09\xdd\x3f\xf2\xfc\x27\x00\x00\xff\xff\x64\x7e\xef\x71\xd9\x33\x00\x00") func staticIndexHtmlBytes() ([]byte, error) { return bindataRead( @@ -303,8 +311,8 @@ func staticIndexHtml() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/index.html", size: 13106, mode: os.FileMode(0644), modTime: time.Unix(1562362183, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xa0, 0x67, 0x2e, 0x1, 0x53, 0x8f, 0xe5, 0xf6, 0x54, 0x43, 0x5, 0x8e, 0xb9, 0x75, 0xd8, 0x37, 0x1b, 0x81, 0xf8, 0xd9, 0x8f, 0x35, 0xee, 0x37, 0xb, 0xe, 0x11, 0xaa, 0xf9, 0xbb, 0xf6, 0x1f}} + info := bindataFileInfo{name: "static/index.html", size: 13273, mode: os.FileMode(420), modTime: time.Unix(1572714934, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -323,8 +331,8 @@ func staticJsAcePgsqlJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/ace-pgsql.js", size: 53342, mode: os.FileMode(0644), modTime: time.Unix(1544759791, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x4e, 0x5d, 0x6a, 0x52, 0xd2, 0x68, 0x62, 0x20, 0x7c, 0xb, 0x94, 0x4a, 0x4d, 0xb, 0x7e, 0xb3, 0xdb, 0x71, 0x87, 0xda, 0xa0, 0xa1, 0xfa, 0xc3, 0xbc, 0xae, 0x80, 0xf9, 0x89, 0xce, 0x25, 0xf9}} + info := bindataFileInfo{name: "static/js/ace-pgsql.js", size: 53342, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -343,12 +351,12 @@ func staticJsAceJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/ace.js", size: 327872, mode: os.FileMode(0644), modTime: time.Unix(1544759791, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xc7, 0xd9, 0xf6, 0xd5, 0x54, 0x74, 0x96, 0x5c, 0xa4, 0x23, 0xf9, 0xf7, 0x1b, 0x43, 0x3a, 0x8e, 0xb0, 0x74, 0x31, 0x55, 0xab, 0x58, 0xe0, 0x69, 0xf6, 0x47, 0xd4, 0x3d, 0x17, 0x20, 0x5f, 0x93}} + info := bindataFileInfo{name: "static/js/ace.js", size: 327872, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } -var _staticJsAppJs = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\xbd\xef\x76\x1b\x37\xb2\x20\xfe\x5d\x4f\x01\x77\x72\xac\xe6\x35\x45\xda\x99\x3b\xbf\x7b\x0e\x25\x2a\x3f\x47\x76\x66\xbc\xd7\xb1\x73\x2d\x79\x76\xf7\x68\x74\x74\x9a\xdd\x20\x89\xa8\xd9\xe8\x34\xd0\xfa\x33\xb1\xde\x60\x1f\x60\x9f\x6f\x9f\x64\x0f\xaa\xf0\xb7\x1b\x4d\xd2\x9e\x64\x33\x3b\x7b\xfd\x21\x11\x81\x42\xa1\x00\x14\x80\xaa\x42\x55\xf5\x6d\xd6\x10\x5a\x30\xc9\x1b\xe2\xff\x9b\x93\xaa\x2d\xcb\xe3\x03\x55\x9d\xf3\xaa\xa2\xb9\xa4\x85\x5f\xbd\xcc\x4a\x41\xb1\x7e\xc1\xf9\xcd\x26\x6b\x6e\x84\x5f\xff\xcb\x23\x56\x16\x74\x99\xb5\xa5\xbc\x6e\xf8\x9d\xb8\x2e\xd9\x86\x49\x32\x27\x2f\x9e\x3f\xd7\xa8\xdb\xa6\xa1\x95\x7c\xbf\xf8\x89\xe6\x32\xec\x19\xea\x97\xac\x94\xb4\x79\x5f\x4b\xc6\x2b\xa1\xb0\x1e\x10\x92\xd0\x9f\xdb\xac\x4c\x66\x08\x9e\xcc\xc9\xe1\xab\x97\x17\x2f\x0f\x93\xb1\xaa\xab\xb8\xbc\xb6\xf5\xc9\x93\xb0\x72\xd5\xd0\x4c\xd2\x06\x9b\x26\xa7\xa6\x8e\xf8\x95\xd7\xf4\xe7\x64\x46\x92\xd3\xb0\x65\x49\x85\x30\x3d\x92\xe4\xa4\x57\x87\xad\xa0\xae\xd3\x90\xdd\x50\xd7\xf0\xed\x9b\x7f\x7f\x1d\x54\x33\xbf\x3e\x79\xd3\xab\x57\x53\xe1\x9a\xbf\x39\x27\xef\x3e\xbe\x7d\xeb\x46\x6a\xab\xa1\xea\xfd\x05\x56\x1f\x3c\x1e\x1f\x1c\x2c\xdb\x2a\x57\xb3\x46\x56\x54\x9e\x53\x21\x18\xaf\xde\x14\xe9\x08\x66\x50\xcd\x2c\x2b\xc8\x9c\x08\xac\x38\x97\xbc\xc9\x56\x74\xb2\xa2\xf2\x8d\xa4\x9b\x34\xd1\xe5\xd7\xac\x48\x46\xc7\x07\x07\x84\xb0\x25\x49\x9f\xb0\x02\x9b\x13\x6c\xbc\x6a\x59\x91\x8e\x8e\xa1\xa0\x83\x48\x44\x10\x8d\x09\x2b\x00\xfa\x51\x21\x6c\xa8\x6c\x9b\x8a\xb0\xe2\xf8\xe0\xd1\x23\x56\x50\xf9\x81\xdf\x89\xb7\x8a\x4f\xd2\xaa\xdd\x60\x87\x25\xcf\xb3\xb2\x87\xdb\x71\x54\x32\x26\x0a\x36\x44\xb5\xf2\x51\x21\x1e\xdd\x69\x9d\x35\x82\xbe\xa9\x64\x1a\xe0\x5d\x45\xf0\x8e\xc8\xa7\x4f\x11\x0e\xee\xf7\xf4\x63\xb6\x62\x55\xa6\x7e\xbd\x5f\x2e\x05\x95\xde\x44\xd7\xd9\x8a\x2a\xae\xfe\x3a\x4d\x26\x9a\xdd\x8f\x54\x59\x32\x9a\x14\x99\xcc\xd2\x04\x7f\x1c\x6b\x70\xb3\x45\x42\xf2\x8f\x1d\xf5\x29\x20\x3c\x22\x2f\x46\xe4\x5f\x10\x3a\x46\x0d\x15\x67\xbc\xad\x64\xaa\x88\x86\xbf\x1c\x41\xc3\x3d\xa8\xda\xaa\xdd\x90\xb9\x9b\x22\xdb\x9e\x4c\x89\x19\xbb\xe6\x07\xb5\x3e\x86\x84\x11\x39\x21\x9d\xae\x88\x42\xf5\xec\x59\x77\xc5\xab\x76\x13\xd2\x9b\xd5\xec\x2c\x2b\xcb\x74\x43\xe5\x9a\x17\x63\x52\x67\x72\xad\xfe\xdb\x64\x1b\x31\x26\xf9\xc2\x11\x2e\xd9\x86\xf2\x56\x91\xfe\x87\xe7\xea\xdf\x31\x99\x4e\xc9\x1f\xc9\x86\x55\x82\x30\x41\x68\xc5\xdb\xd5\x5a\x75\xf5\xf5\x24\xfb\x29\xbb\x4f\x91\x0a\xdd\x6a\x66\xfe\x18\x43\x69\xdb\x94\x33\x92\x64\x35\x4b\xc8\x33\xec\x12\x8a\x91\x88\x99\xfe\x3f\x96\xe5\x59\xbe\xa6\x33\x3c\xf0\xb0\x44\xad\xdb\xcc\x90\x08\x25\x6b\x9a\x15\xb4\x11\x33\x3d\x72\x42\x92\xfb\x23\xcd\xfb\x47\xac\x48\x66\x9d\x5d\x08\x40\x8f\xd8\x54\xb4\x79\x4e\x85\x98\x11\x33\x23\xa9\x42\x3f\xb2\x98\xf2\x05\x16\x1c\xfb\x8d\x68\xd3\xf0\xc6\x6b\x72\xbf\x6e\xc6\x44\xc8\x4c\xb6\x62\x4c\xc2\xf6\x6a\xa9\xb0\x86\xcc\xe7\x24\xd1\xb3\x90\x38\x00\xbb\x34\xf9\x22\xfd\xc5\x60\x4e\xfe\xa3\xa5\xcd\x83\x9d\xf2\x6c\x29\x69\x43\xd4\x5c\xa5\xa6\x68\xaa\x4e\xf1\xe7\x23\xf2\x8c\x24\x22\x21\x8f\x9a\x3e\xbd\xd8\x9a\xf0\x9f\x00\xcb\x04\xb8\xe9\xbf\x9c\xbf\x7f\xa7\xe8\x9c\x34\x54\xd4\xbc\x12\xf4\x82\xde\xcb\x91\x19\x96\xe2\x92\xfe\xb6\x7a\x53\x2d\x79\xaa\x58\x60\xe0\xdf\x2f\x96\x7b\x92\x15\x55\xc7\x40\x32\x65\xd5\x92\x27\x63\xf2\xcb\x23\xf0\xce\x31\x79\x0c\x10\xe2\x45\x23\x06\x71\x46\x10\x72\x6c\x32\x8c\xf3\x22\x5b\x94\x74\x18\x65\x0c\xa7\x84\x26\x3b\x50\xaa\xcd\x99\x02\xe4\x98\xf0\x5a\xea\xcd\xb0\x1d\xe5\x54\x2d\x11\xfc\xa9\xd6\x65\xaa\xb6\x64\xe2\xb5\x8e\xf6\x73\x2e\x9b\x36\x97\x6d\x43\xfb\x9d\xed\xd1\xcf\x2e\xec\x6f\xaa\x82\xde\x53\x3b\x10\x7f\x96\xf6\x1d\x05\x43\x14\x3b\xa6\xeb\x8c\x57\x42\x36\x19\xab\x64\xaf\xb3\x7d\x3b\xca\x1d\x8a\xe1\xce\xfe\xcc\x84\xe4\xcd\xc3\xe7\xb0\xd0\x1a\x9b\x0c\xe3\xfc\xce\x48\x50\x03\x58\x23\x38\xad\xd0\x35\x80\x95\xde\xd3\xbc\x95\x14\x36\x60\xfa\xb3\xfa\x6f\x38\xf9\x21\xd6\x9a\x0b\x44\x0b\x90\x0a\x25\x81\xbf\x66\xf8\x3f\x12\xed\xa0\x2e\x33\x56\x7d\x6e\x07\xba\xd9\x5e\x5d\x14\x4c\x68\xd9\x73\x9f\xd9\xb6\x5d\xb8\x66\xe1\xdc\x78\xb4\x57\x39\x2f\xfc\xb9\x09\xe4\x83\xef\x32\x41\xff\xbf\x7f\x9d\x20\x90\xae\x9f\x34\xb4\x2e\xb3\x9c\xa6\xd3\xbf\x3e\x9b\xae\xc6\x24\x39\x4a\xfc\xb2\x29\x94\x5d\xfb\x65\x73\x28\x9a\x24\x9d\x53\x6d\xd1\xb2\xb2\x38\xcf\xd7\x74\x93\x9d\x53\x3c\xbc\xab\x6c\xa3\xb6\x11\x9e\x33\xee\xba\x13\x58\x4d\xe6\x24\x49\xe0\xd2\xc5\x3b\x50\x96\xd4\x48\xc2\x84\x24\xc0\xbc\x56\x42\xc4\x7f\xc9\x85\x3e\x5e\x10\xe4\x96\xd1\xbb\x0e\x04\x49\xfe\xc2\xe8\x9d\x85\xd8\x28\xc1\x97\x65\x25\xfb\x1b\x2d\xae\x35\x78\xf2\x83\x57\x48\x02\x70\x41\x7f\x6e\x69\x95\x07\xdd\x26\xe7\xba\x50\x24\xea\x28\xb7\xf4\xb2\xdc\x09\xee\x03\xe4\x1e\x9e\x30\x92\x97\x99\x10\xf3\x64\x99\x91\x65\x76\x84\x40\xa7\x27\x53\x76\x7a\xb8\x65\x0c\xfb\xb4\x8b\x8d\x6c\x9f\x76\xb1\x21\xf6\xda\xe5\xac\xc9\x4b\x7a\xc4\x75\xd3\x60\xdc\x37\x0a\x50\x2f\x1d\xde\xc1\x6a\x95\xe1\x06\xae\xdb\x45\xc9\xf2\x64\xe4\x60\xe8\x7d\x9d\x55\x05\x2d\x70\x99\xcd\xba\x3f\x9b\x93\xe4\xa4\x60\xb7\xba\xd3\x43\x01\x4c\x03\x97\x30\xb6\x7c\x46\x92\xc3\x53\xc0\xbf\xbd\xc9\x91\xea\xfa\xf0\xd4\x92\x7f\x88\xe4\x2f\x79\x59\xd0\xe6\x88\x1f\x02\xf9\x43\xb5\x35\xad\x0c\x08\x74\x0d\xc3\x78\x46\x92\x93\x69\xc1\x6e\xf7\xea\x3d\xe7\x95\xcc\x58\x45\x1b\x20\xf6\x80\x90\x4b\xcd\x06\x63\xbd\xae\xe3\xd8\x3a\x8d\xbd\x45\xb8\x9a\x2c\x79\xf3\x3a\xcb\xd7\xa9\x15\x78\x56\x0d\x6f\x6b\x23\xc3\xc0\x8f\xeb\x70\xca\xe3\x93\x4e\x9e\x3e\x45\x68\x94\x85\x80\x8c\x51\xb7\x7d\xb8\x1c\x3b\xc7\x87\xf8\xd4\xdc\xf8\x78\xdc\xe2\xec\x87\xe0\x08\x76\x76\x7f\x95\xf2\x35\xbd\x6d\x78\x75\xd4\xb0\xd5\x5a\xc6\x97\xca\x80\x14\xfc\xae\xf2\x56\x0a\x8f\x8a\x4b\xc0\x7e\xa5\xe8\x21\x27\xa2\xce\xaa\x68\xe7\xb9\x92\xdb\x0f\x4f\x55\x33\x7d\x0e\xe9\x76\x93\x92\x56\x2b\xb9\xc6\x15\x57\xcd\x4f\xbd\x85\xef\x8c\xac\x2d\x41\xf4\x44\x94\xf3\x43\x3b\x23\x76\x2e\xec\xb2\x84\x7d\x38\x51\xb4\xd3\x77\x6f\xd5\x99\xa4\x1b\x5f\x70\xb5\x8a\xac\xe1\xca\x89\xea\x54\x41\x1d\x5b\x98\x80\xc4\x92\x75\x86\xaf\x60\x89\xfe\x3b\x24\x18\xc7\x22\x1f\x6a\xda\x1d\x0a\xd6\xb0\x02\xcb\x59\xe1\x15\x2a\x3a\x74\xb1\x42\x8c\x03\x57\xbf\xd4\x31\xe8\x2d\xc5\xd3\x6a\x21\xea\x63\x1f\xee\x64\x5a\x32\x33\xab\xc4\x13\xa7\x03\xf2\xa7\x6d\x19\xce\xbf\x91\x9b\xbb\xbb\x10\x60\x2c\xa4\xbb\xda\x34\x50\x78\x25\x95\x3c\xd3\x37\x92\xd0\x7a\xeb\xd7\x69\xf2\x95\x91\x7b\x47\x93\xb5\xdc\x94\x69\xa2\x2d\x01\x9e\x10\x1d\x55\x58\xd4\xf2\x22\xc0\xe4\x86\x3e\x08\xac\x33\x5c\x34\x9f\x93\xe7\x6e\xfd\x54\xd5\xa5\xd9\x99\x57\xf6\x92\x50\xff\x60\x67\xce\xc8\xe5\xd5\xd8\xad\x35\xa3\x77\x61\x49\xef\xd0\x08\xab\xcd\xf1\xa1\x4a\xcd\xb4\x9a\x59\x83\xff\x2d\x79\x43\x52\x7d\xaa\xb2\xaa\xa3\x36\x7d\x9d\x46\x6e\x6a\x04\x46\x0d\xeb\x12\x7f\x5c\x8d\x46\x93\xac\xae\x69\x55\x5c\x70\x7f\xd6\x82\x9e\x76\x4d\xca\x0b\xbf\xdf\x64\x82\x98\x93\xd1\x24\x2b\x8a\x33\xc5\xae\xa9\x3b\x93\x42\xc4\x0b\x56\x15\x67\xbc\x92\xf4\x5e\xfe\x40\xab\x56\xa0\x1e\xdf\xd5\xa4\xa8\xc8\xb3\x9a\xfe\x59\x2d\xa3\x90\x0d\xf6\x85\x6a\x61\x43\x9e\xa0\xbd\x8d\x7c\xfa\x44\xf4\xcf\xb6\x2a\xe8\x92\x55\xd4\x5a\x7b\x34\xef\xa0\x32\x97\xc2\x09\x36\x3d\x4d\x46\x13\xd5\x2b\x20\x44\x0e\xe9\x19\x77\x92\xe0\xbc\x51\xbd\x1c\x9e\xaa\xff\xea\x83\x24\x09\x89\x6c\xab\x0e\x99\x46\x0e\xa2\x64\x4e\x0a\x9e\xb7\x1b\x5a\xc9\x49\x0e\x56\xba\xd7\x25\x55\xbf\xd2\xa4\x60\xb7\x38\x23\x74\xc2\xaa\x8a\x36\x7f\xbe\xf8\xe1\x2d\x99\xab\x91\x78\x16\x13\x3a\xc9\xd7\xac\x2c\xde\xf1\x82\x0a\x37\xe9\x73\xf2\x9c\x7c\x4b\x92\x84\xcc\x02\x80\xcb\xe7\x57\x93\x8a\x17\xf4\x2f\x59\xd9\xd2\x9e\x42\x7a\xe6\xdb\x2b\x43\xab\x52\x68\xca\xfc\xf4\x89\xfc\x02\xa7\xd2\x8c\x24\xc9\x98\xa8\x63\x44\xfd\xa5\x38\xd0\x47\xd9\x50\xa1\xd5\x17\x6f\xe7\x35\x54\xb4\x25\xec\x3c\x6b\x6d\x48\x93\x0d\x2f\xe0\xc6\x34\xa5\x0d\xdd\xf0\x5b\x6a\xb8\x63\x53\xcb\x87\x68\x4d\xc5\x8f\xf2\x86\xd7\x38\x49\x21\xfe\x6b\x34\x5a\x04\x1b\x3c\x04\x58\xf0\xe2\x21\xa8\xf6\x49\xaf\x69\xb3\xe4\xcd\x06\x88\x7f\x89\xdb\x43\xab\x5d\x19\xfc\x1a\x13\x5a\x3a\x56\xcb\xb4\x2c\xab\x6e\xdd\xa6\xad\xf2\x4c\xd2\x44\x4d\x92\x57\x5e\xd0\x92\x4a\x6a\x0d\x13\x6a\xe9\x37\x54\x88\x6c\xa5\x18\x20\x79\xd9\x50\xf2\xc0\x5b\x22\x5a\xfd\xc7\x5d\x56\x49\x22\x39\x5c\x75\x1a\x8b\xba\xe3\x50\x9b\x0b\xf4\x3a\xf2\xad\x27\x11\x3c\xc9\x79\xb5\x64\xcd\x26\xd5\xb8\x47\x23\xbd\x80\x96\x7b\xc5\x1d\x93\xf9\x5a\x53\x6c\xa8\xc9\x33\x41\x3d\xd2\x67\x7a\xb7\x06\x5a\x56\x72\xf1\xe1\xe3\xbb\xb3\x97\x17\xaf\xc9\xc5\xcb\xef\xde\xbe\x26\x9e\x8e\x3c\x60\xdb\x41\x92\x54\xd9\x04\xec\x2e\x23\x92\x95\xb4\x91\x7e\xc9\xb1\x67\xa5\x71\xbc\xd2\xbf\x27\x16\x0d\xcd\x6e\x8e\x3d\x5a\xf5\x74\xc6\x29\x7d\xf5\xe1\xfd\x8f\xbf\x0d\x95\xc1\x65\xf2\xc5\xc4\xd3\xfb\x9a\x37\xd2\x12\x0f\x4f\x01\xbc\xd9\x64\x92\xcc\x09\x2d\xb5\xe1\x14\x4b\x12\x8b\x05\x9e\x1b\x16\x68\x6a\xfd\x4a\x6f\xc7\x6b\x05\xba\xc8\x04\x35\xe7\x55\x00\xbd\x64\x25\x45\x11\x51\x35\x34\xf2\x83\x65\x1c\xf8\x85\xbd\xf8\xad\x50\x5f\x9d\x93\x3b\x56\x15\xfc\x4e\xeb\x89\x1f\x3f\xbc\x49\x93\xf3\xd7\x6f\x5f\x9f\x5d\x90\x7f\x21\xdf\x7f\x78\xff\x83\x9b\xd9\xa0\xcf\xb6\x29\x5d\xdb\x92\xe7\x60\x35\x9e\xac\x1b\xba\x9c\x88\xba\x64\x32\x4d\xbe\x4a\x46\x97\xcf\x41\x4c\xc8\x6a\x86\x1a\xf9\xb7\x48\xc5\xdc\x11\x04\x52\x84\xa1\x1f\xcb\xcd\x60\x54\x0d\xb4\x82\x62\xa4\x56\x95\x5d\x3b\x6b\x3c\xd4\x84\x36\x49\x9f\xc6\x3b\x56\x11\x47\xa4\xd2\x01\xd2\xb6\x29\x95\x86\xbb\x28\xb3\xea\xc6\x4d\xf9\x1d\xab\x26\x4b\x9e\xb7\xde\x5a\xf7\x19\xb1\xdd\xd4\xc1\x4a\x7e\xee\x0c\x20\x33\x7c\x0b\x53\x39\x0f\x16\xe8\xf7\x1a\x52\xce\xeb\x07\x3b\x24\xf5\xe3\x82\x9f\x95\xac\x5e\xf0\xac\x29\xf0\x14\xd4\x23\x39\x9c\x1c\x8e\x2e\x5f\x5c\xf5\x31\x3d\xc6\xce\x52\xa5\x65\xeb\xa3\x54\x89\x33\x3b\x4f\xd2\x5f\xe7\xc4\x54\x5d\x41\x19\xfc\xf1\xab\x9e\x97\xbb\xcf\xa0\xbf\xbc\x79\xfd\x5f\x6d\xe7\xff\x79\x02\xe9\x33\xc7\xac\xc4\xaf\x70\x00\x29\x54\xff\x6f\x9f\x3f\x5b\x37\xab\x9a\x9e\x2f\xda\xab\x1f\xb8\xd9\xaa\x66\x93\xde\x2a\x89\x31\xba\x4f\x85\xe4\xf5\x35\x5a\x56\x7d\x35\xc9\x6e\xad\xe1\xad\xaa\x5a\x12\xb9\xa6\x38\x8b\xdf\x26\xc1\xee\xeb\x6e\x28\xbd\xf8\xf5\xea\x3a\xcf\xaa\x9c\x96\xd7\x8b\x2c\xbf\xa1\x55\x91\x02\x1f\x28\xea\xd4\x32\x8c\x8e\x93\xc1\x7d\xb6\xf7\x2e\x53\x5b\x09\xdf\x81\x52\xb1\xc6\x99\xb8\x65\xf2\xe1\xc7\xac\xa2\xe5\x18\x1f\x86\xb4\x9e\x32\xea\xcf\xa0\xe0\x8d\x7c\xd9\x34\xfc\x2e\x2d\x58\x43\xbd\x63\x03\x4f\x12\xd2\x2d\x36\x0b\xf9\xf2\xfc\xcc\xae\xa3\x51\x31\x9e\x7e\x75\xff\xcd\x1f\xbf\xfb\xe6\x38\xf1\x57\xfc\xd5\xeb\x41\xc8\x33\x03\xa9\x5f\x75\xbb\x50\x49\x9f\x5c\xd0\x03\xf1\xe0\xd0\x82\xf1\x18\x86\x70\xc6\xcb\x76\x53\xe1\xdf\xef\x9b\x82\x36\xf0\x14\xc2\x78\x25\x1c\x13\x3c\xb1\x25\xdc\xf9\x30\x3c\x9a\xb7\x56\xc3\x22\xa6\x72\x92\x69\xed\xfc\xa0\x7b\x5c\x69\x74\xba\x7f\xb3\x3c\x38\x39\x3b\x45\xfa\x2d\x42\xfd\x89\x6c\x4e\x4f\x64\x71\xfa\xfa\xc3\x87\xf7\x1f\x66\x70\x60\x04\x7d\xa0\x65\x42\x01\xa9\xff\x18\x74\x9d\x0b\xc0\xa7\xac\xe1\x77\x22\xa6\xed\xff\x1a\x44\xbe\xe3\xa4\xa1\x39\x6f\x0a\x41\x96\xbc\xad\x8a\x93\xa9\x2c\x42\xc2\x1c\x8a\x23\x45\x08\x9a\xb5\xb6\x75\x14\xea\xd8\xa8\x45\x45\xc7\x88\x7e\x30\xa5\xb3\x2f\xaa\x02\xd5\x87\x33\xcf\x9b\x29\xc8\x81\x2f\x44\xdf\x7a\x95\xf3\xd2\xdf\xfe\x39\x2f\x41\x0d\x75\xbc\xe4\x3d\xfa\xaa\xae\xc0\x92\x23\xd7\x46\x81\x06\xb9\xe2\x08\x27\xef\x48\x35\x56\xdc\x72\x4b\x7b\x96\x27\x55\xf5\x8c\x24\xf0\x77\x02\x75\x5c\x71\x27\x1c\xb9\x96\x57\x55\xdd\xa9\x07\xed\x0c\x52\x6e\x77\x5a\xe0\x91\x66\x83\xb5\x6f\x76\x22\x84\x96\x82\x7e\x16\xc5\x83\xa4\xfa\x94\x74\xfb\xd1\xe6\xad\xe9\x94\xbc\xe3\xfa\x9d\x5d\x9d\x8c\x9b\xec\x86\xc2\xc9\x88\xd3\x4d\x2a\x5e\x1d\x29\x82\x55\x97\xc1\x09\x6c\xcf\x10\x8f\xc0\x53\xd3\x09\xd4\x4c\xa7\xe4\x15\x95\xb4\xd9\xb0\x8a\x92\xbb\x35\xcb\xd7\x06\xa9\x36\x5d\x0b\xe8\x48\xd1\x4e\x32\x29\x1b\xb6\x68\x25\x85\x96\xd8\x03\x08\x03\xb8\x82\x64\xde\x63\x03\x78\xbd\x7c\xbf\x4c\x3d\x58\xdf\x52\xe2\x6d\x9b\x1e\xc3\x34\xfc\xce\x97\xed\x1a\xc7\x6b\x40\xf4\xcb\xa2\x20\x59\x09\x7c\xd0\x66\xa5\xe2\x46\xa4\x71\x4d\x1b\xea\x0c\x5c\x8c\xb0\x8a\x78\x98\x08\x69\xf4\x34\x14\xb8\x1c\x39\x2f\xb5\xc9\x12\xd7\xe2\x04\xac\x86\xe4\x99\x6f\x36\x6a\xf8\xdd\x25\xbb\x1a\x39\xc3\x3f\xec\xbd\x24\xb0\x45\x69\x8a\x14\x19\x99\x39\x39\xa5\xe4\x95\x65\xf8\x70\x3d\x3c\x3a\x4e\x4f\x32\xc3\x32\x0b\x59\x91\x85\xac\x8e\xee\x05\xfc\xcf\x89\xaa\x13\x21\x1f\xb4\x3a\xdf\xf0\xbb\x23\x2c\xd4\x0c\x85\x3f\x70\x10\x1a\xda\x08\x22\x1a\x02\x6e\x3f\x04\x50\x43\xe9\xad\xdb\x15\xc2\x2a\x29\x68\x7e\xf8\x15\xf2\xa3\x06\x02\x33\x3a\x0e\x3c\x8b\x0d\x1b\x0e\x01\x1c\x48\x03\xed\xdc\xa1\x99\x1c\x3b\xfe\x1d\x3e\x00\x15\x63\x6e\x35\xbd\xa8\x1e\xec\x26\x38\x5f\xf3\x3b\x52\xb5\x9b\x05\x6d\x08\x5f\x62\xef\x0d\xad\x0a\xda\xd0\x82\xf0\x0a\x38\xb5\xce\x56\xf4\x60\xc7\x69\x18\x3b\xb0\xf5\xe4\x8a\xae\xa9\x47\x58\xc3\xd7\x45\xb6\x48\x8d\x1b\xd8\x74\x4a\x9c\xdb\x13\x11\x6b\xde\x96\x8a\x82\xf2\x81\x2c\x28\xb9\x65\x82\x29\x25\x8d\x57\x48\xa2\xcc\x16\x7a\x57\xb2\x82\x3c\x31\x6f\x2f\xd7\x6a\x83\xd1\xca\x79\xa3\x28\x92\xf5\xd0\x03\x33\xd6\x1d\x93\xeb\xa3\xda\xf6\x96\xb8\x1d\xa4\x5a\x54\xd9\x2d\x69\x4b\x52\xb2\x89\xa0\x25\x78\x2a\x76\xdb\xbb\x72\x33\xd1\x68\xba\xf7\x8f\x7f\x1f\x46\x0f\x8f\x36\x82\x09\xa9\x88\x27\x58\xab\x46\xca\x2a\xc9\x61\x9a\xb5\x38\xab\x84\xb4\x06\x67\x7c\xc8\x0d\x4e\x66\x0b\xe3\xff\x16\xcc\xeb\x9a\xdf\x81\xec\x66\x7c\x0a\x70\x1a\x3c\x27\x83\xa8\x9c\xe6\x5d\x3d\x97\x57\xc7\xd6\x96\x8d\x3b\x3d\x14\xe8\x60\x75\xdb\x4a\xac\xd9\x52\xa6\x97\xd6\xab\x8b\xa9\x9d\xfc\x42\xdb\xb1\xd9\xd5\x44\x3f\xdf\x9b\x9f\x92\x6d\xa8\x90\xd9\xa6\xbe\xea\x58\x9b\x9d\x24\xf4\x8b\x3e\x22\xc5\x8c\x5c\x26\xe0\xdd\x97\x18\xd7\x81\xc4\x36\x4f\xae\xc6\x40\xc1\x0c\xc9\xd5\x3b\x81\x74\xf8\x49\xb3\x82\x71\x92\xf0\x2e\x69\x56\xd5\x2d\x30\x2c\x2b\xac\xd2\xe6\x71\x48\xdd\xf0\x3a\x4d\xe0\xe0\x50\xdd\x2e\xdb\xb2\xdc\x75\xc5\x07\xe6\xd0\xae\x99\x5c\x2d\x47\xe0\xb3\xe2\xde\xe2\xb5\x8a\xd6\xb7\x00\xc3\x41\x63\xe5\x33\xf5\x23\x26\xfd\xa0\x1c\x9d\xfc\x58\x52\x25\x9f\x22\x2b\x91\x0c\xed\x18\x4f\x06\x24\x8e\xae\x03\x0d\x7a\x09\x44\x19\x22\x3a\x9f\xc6\x6b\x46\x63\xf7\xd6\x4e\x5f\x41\xbf\xfb\x34\xfb\x1e\x3b\xff\x10\x53\xed\x13\xf4\xb9\xd3\xed\xfb\x0e\xfd\xe3\x4e\x39\x38\xd4\xfd\xae\x73\xbd\xcd\x13\xcb\x5c\xda\xbe\x13\x5f\x74\x05\xbe\x4e\x93\x09\x0a\x97\x0a\xb2\xd9\xe0\x15\x34\x29\x59\xa5\x18\x7e\xa2\x86\xeb\xcf\x2a\xae\x90\xe4\x32\x2b\xaf\x05\xfb\x9b\xb5\xc4\x80\x62\xeb\x8a\x7b\x2d\x54\x7d\xbf\x81\x2d\xed\xc1\xc3\x96\xeb\x37\x70\xc5\xbd\x16\xe0\x53\x6c\xee\x65\xd7\xc2\x15\xf7\x5a\x80\xb1\x87\x55\x2b\x03\x9f\x7c\xac\x6e\x2a\x7e\x57\xb9\x85\x3f\xf0\x39\xef\x7b\x70\xa0\x37\xec\x1c\x59\x68\xf9\x50\xd3\x0e\xb7\xb4\x75\x91\x49\xaa\xef\x76\xde\xa4\xee\xde\xf5\x14\xdb\x6e\x21\xd9\xe1\xe0\x3c\x26\x2f\xfc\xdf\x02\x0a\xec\xe0\x50\x52\x9c\xe8\x66\x38\xb0\x17\x4a\xbe\x79\xe1\xf1\xfd\xa4\x6e\xe8\x2d\xa0\x1e\x93\x49\x45\xef\x6d\x37\xb8\x5d\x0a\x26\xd4\x88\xe1\x2e\xb2\x7f\xc7\x99\x30\x42\x2a\x1a\x05\x7c\x7a\xdd\x08\x81\xae\x1e\x88\xe8\xc3\xb8\x45\xd3\xd3\xd4\xa9\x27\xa7\xee\xa9\x37\x18\x50\x74\x10\x46\xca\x09\x34\xad\xdd\xcd\xc2\xb1\x3f\x0e\x90\xa2\x49\x55\x14\x91\xa7\x4f\xbb\x03\x21\x27\x43\x43\xf3\xc8\xd8\xbe\x04\x83\xd4\x7f\xce\xca\x59\x11\x0f\x77\xa9\x56\xff\x0d\x8b\x78\x24\x76\x76\xcc\x96\x01\xe3\x01\x36\x54\x49\x5e\x68\x01\x31\xc2\x91\xdd\x39\x52\xe2\x32\x5f\x12\xf4\x20\x1f\xe0\x83\xa1\x8b\x4f\x49\xbd\x69\xd4\x8a\xf4\xbb\x9e\xce\xaa\x5b\x5e\x4b\xe7\xb1\x07\xde\xfd\xc6\x09\x2e\x8c\x18\x40\x4f\x0b\x0e\xf1\x0e\x33\x0b\xd0\x8f\x84\xd0\x6e\xee\xbc\x91\xd7\x28\x33\xce\x7c\xf3\x99\xab\x04\x2b\xc5\x8c\xb8\x99\x08\x1c\xea\x30\x12\xc8\xd2\x65\x30\x29\x86\xc2\x2a\xa1\x87\xa8\x15\xef\x64\x34\xb9\xcd\x4a\x4b\x65\x8d\x14\x46\xc0\xf1\x67\x08\x0e\xd7\xf3\x2c\x04\x37\x57\x36\x80\x59\xd2\x94\xd2\x5b\xd7\xe5\x83\xa6\x8f\x55\x2b\x54\x81\xd8\xd2\xd8\x10\x98\xe9\x89\x16\x7a\xc1\x10\x54\xd3\xa9\x36\x9f\x2e\xe0\xb5\x2f\xe1\xdf\x29\x35\x5e\x89\xf8\x5a\x94\x3f\x4c\x0e\xd1\xc8\xee\xb5\x7d\xa6\x4a\x8d\xc7\x4b\x10\x2b\x75\x69\x71\x5e\x59\x97\xd4\xe4\xd5\xcb\x8b\x97\xc9\xd8\xe0\x80\x01\x61\x28\xc2\xd5\xe4\x27\xce\xaa\x34\x21\x89\x11\x52\x14\x13\x5c\x26\x40\x03\xb8\xe6\xc0\x5f\x3d\x81\x09\x5c\xd4\xb5\xfb\x2a\x78\x81\x0f\xdd\xd6\x9f\x29\xf0\x44\xd4\xbd\x98\x44\x35\x60\x87\xb5\x31\x49\x71\x31\x0d\x15\x4e\x84\xe9\x5e\x75\x70\xfb\x7a\x57\x9b\x27\xb2\x39\xa1\x2b\x74\xc8\x58\xa8\xa3\xc7\x5d\x76\xc6\xb5\x51\xcd\xca\xa0\x2c\xa6\x3b\xa4\x45\x70\x1c\xb8\x9d\x8f\x96\x3d\x6b\x52\xf2\xba\x27\x72\x3d\xc1\x5a\x17\x32\xe4\x26\xc1\x06\xcf\xb9\x0a\xb4\xf6\xd9\xa0\x3a\x67\x89\x31\xf8\xf5\xd1\x61\x85\x5b\x1f\x59\x00\x87\xe3\x53\xe3\x32\xb3\xe7\xe3\x8f\x80\xc2\x86\xf6\xce\xf1\x3d\xcf\xbf\x81\x43\xd3\x85\x29\xfc\xae\x27\x64\x94\xab\x84\xa1\x2d\x71\xc6\x9e\x3e\xc7\xef\x16\xf0\xbd\xad\xe5\x86\x8b\xfb\xeb\x17\xed\x43\x34\x20\xc2\x91\x21\x51\xb9\xaf\x84\xfc\x5d\x5a\x04\x98\x2b\xe0\x99\x27\xf5\x84\x41\x27\x9e\xea\x57\xae\xc9\x3a\x13\x3d\xb3\x8a\xf3\x27\x0b\x5f\x62\x87\xe7\x55\x63\x03\x07\x2f\x88\x51\x75\x8f\x7d\x9d\x59\x76\x22\xff\xf0\x2c\x27\xa3\xde\x70\xce\x30\x4e\x80\xf1\xca\x1f\xd3\xd0\xc9\xa1\x41\xf5\x4a\xf5\x34\x19\x0f\x62\x8b\xee\x32\x64\xbd\xb9\xa1\x0f\x03\xf6\x9b\xba\x15\xeb\xf4\xf2\x86\x1a\x0b\xcd\x0d\x7d\xb8\xda\x62\x99\x71\x06\x79\x63\x9f\xb1\x46\x6b\x70\xb9\xce\xca\x96\x26\xd6\x4f\xd2\xd9\x67\xec\x7b\xdd\xbe\x87\xb6\x63\x1b\xa7\xa0\xc6\x78\x26\x78\x1d\xf4\x76\x2f\x0f\xc2\x7a\x8d\x35\xdd\xc5\xd0\x69\xf7\x39\xef\xf5\xd4\x10\x0d\x36\x59\x5d\x65\x0b\x31\x2a\x2f\xa9\x59\x61\x8b\xc0\x66\x3c\x23\x49\x91\x55\x2b\xda\x24\xee\x4d\x61\x90\xe3\x32\x4d\x2b\x8e\xa6\xb7\xc4\xb6\x7a\xcb\x02\xf7\x2e\x29\x75\xf4\x9a\xff\x9a\xf7\xc0\xbd\x8d\x5c\xfb\x4c\x72\xd3\xea\x30\x9c\x2d\xec\x6b\xb7\x92\x31\x0c\xb7\xd5\x98\x7c\xa5\xa3\x71\xc6\xe4\xab\x5c\xdc\x8e\xc9\x57\x3f\x09\xae\x7e\xdc\x6f\xca\x98\x70\x2e\x9b\x96\xda\x3d\x06\x08\xaf\xeb\x86\xaf\x1a\x2a\x7c\xad\xfb\x20\x74\x52\xf8\x7a\x22\x1b\xb6\x49\xf5\xfe\x85\x37\x7f\x3c\x10\x2e\xc0\x19\x82\x7c\xfa\x44\x5c\x1d\x78\x5d\xa6\x23\xa7\x40\x01\x96\xc1\x07\xc5\x2f\x18\x03\x84\x71\x7a\xd3\xdc\x1b\x86\xbf\x0e\x9d\xb3\x3f\x16\x53\xb5\xe7\xa1\xfb\x7f\x8e\xe6\xf8\xa9\x38\x6c\x5a\x1f\x36\x2e\x85\x72\x8e\xcf\x40\xfe\xda\x48\xfe\x96\xdf\xd1\xe6\x2c\x13\x34\x1d\xd9\xd7\xae\xc4\xc4\x79\x8d\xc8\x93\x39\x39\x0a\x1d\x9c\x77\x5f\x3d\xfe\xa3\xd2\x07\x5a\xf2\xac\x30\xb1\x01\xa4\x64\x42\xaa\xee\xb3\xea\x41\xae\x95\xbc\x7d\x97\x09\x82\x6e\xc1\xc5\x14\x1d\x8e\x8a\x0e\x89\x9b\x4c\xe6\xeb\x74\x9a\x22\xd4\xa7\xa2\xe1\xf5\xe8\xaf\x62\xca\x46\x8e\xaa\x88\xd3\x50\x2c\x26\xb5\x69\xab\xd7\x38\xb0\xff\xdc\x6b\xbf\xf5\x5e\xeb\x87\x17\xfe\x13\xef\xb5\x3d\x45\x30\xf4\x4a\xbb\xe0\x29\x5a\x3d\xdd\x5d\xfa\x1b\x73\x40\xc4\x4c\xf0\x6b\xfa\x91\x39\x6f\xb1\x48\x64\xe6\x7e\xae\x63\xc6\x6d\x2c\xe2\x35\x76\x88\x5e\x63\x87\x26\x4e\x66\xbb\xa0\x19\xb8\x94\x3d\x1e\x1c\x4c\xa7\xe4\x7b\x2a\xf3\x35\x3c\xb5\xb7\x15\xfb\xb9\xa5\xe8\x4c\x25\xe0\x7d\x1d\x1f\x02\x71\x8e\xad\xca\x8f\xcf\xb0\xe8\x8b\x10\x88\x42\x1f\xa1\x3d\xea\x3c\x02\xa6\xdf\x05\x2b\x1b\x45\x08\xc4\xd2\xb6\xf2\x43\x41\xcd\xea\x1e\x6a\x17\xaf\x57\x6f\xce\x2f\xde\xbc\x3b\xbb\x20\x60\x13\xf0\x8c\x01\xe8\xf7\x77\x68\x7c\x64\x8d\x89\xe2\x15\x13\x75\x99\x3d\x18\x77\x03\x02\x9a\x19\x2d\xc8\xe2\x81\x80\xa1\x4a\x4c\x10\xee\x62\xcd\x84\x2a\x29\x0b\xb2\xa0\x44\x94\xfc\x8e\xf0\x8a\x94\x59\xb3\x52\x83\x94\x82\x28\xc5\x1c\xf2\x01\xa8\x41\x16\xf4\x7e\x62\x22\x49\x3a\x44\x93\xfd\x49\x1e\x93\xb3\xf7\x1f\xdf\x5d\xa4\x2f\x46\xe4\xe5\x39\x41\x3b\x1f\x1a\xe3\xc2\xb1\x28\x60\xf2\xa7\x0f\xef\x3f\xfe\x48\xbe\xfb\xef\xfd\x81\xbf\xff\xf0\xea\xf5\x07\x55\xe3\x63\x78\xf5\xfa\xfc\xec\xf0\xf3\xaf\xef\x5f\xf3\x85\x66\xe0\x12\x1d\x52\xcd\x1e\x2d\xd3\x99\x87\x7e\xda\xb0\x1c\x52\x3b\x08\xf3\xbc\xbf\x64\xb4\x2c\x42\xbe\xfa\x5e\x15\xbd\x6b\x37\xe7\x0a\x2e\xe4\xa8\x61\x2e\x82\x39\x4a\x5f\x8c\xc6\x64\xc3\xaa\x34\x9c\x50\x55\x98\xdd\xf7\x0b\xb3\xdb\x55\xb7\xb0\xc7\x73\x10\x68\xf2\x7f\xd1\x64\x47\x3c\xfd\xc2\x67\x13\x78\x23\x31\x4f\xf3\x5b\xb5\xf3\xad\xaa\xb8\xf5\xae\x1c\x72\x92\xc3\xe1\x38\xf3\x13\x31\xc6\xc7\xce\xd4\x44\xdc\xaf\x86\x1b\xfa\xf7\xd2\xa3\xbb\x1b\x3d\xe0\xae\xdd\x54\x7b\xdd\xa1\xaa\x42\xb4\xeb\xcc\xa1\x3d\xe5\x4e\xf1\x4a\xd1\x2c\x70\x32\x45\xb8\x53\x2b\x1a\x82\xd7\x11\x44\x66\x92\x39\x79\x7e\x4c\x18\x39\x21\xdd\x61\x1f\x13\xf6\xec\x99\x1b\xb6\xd6\x89\xc9\xdc\x01\x5e\x32\xa3\x1c\xeb\x28\xb0\x12\x0d\x61\x9a\x2c\x08\x40\x53\x7a\x6d\xaa\x15\x5a\xf0\x34\xb8\x7c\x7e\xa5\x6d\xf5\xfa\xc7\x71\x7c\x7a\xba\x23\xc6\x00\xbe\x94\x96\x5b\x44\x3f\x56\x31\xf9\x1a\xae\x4e\x4f\x85\xbd\x6b\x98\xe6\xf1\x0b\x9b\x9c\xc6\x58\xe0\x74\x56\xad\x39\xc9\x72\x3a\x51\x3f\xd2\x24\x6f\x85\xe4\x9b\x40\x3e\xd4\xb7\xb1\xa0\xf2\x7b\x5e\xc9\x73\xf6\x37\x9a\xbe\xf8\x83\x6f\xf1\x10\x54\x5e\xac\xe9\x86\xa6\x49\x96\xd3\xa9\x54\x7f\x4e\x25\xdf\xf0\xa6\xe1\x77\x49\x07\x50\x9d\x19\x3f\x36\xac\x92\x3f\x64\xcd\x8a\x55\xa9\x13\x66\x7c\x81\x00\xee\xcd\x74\xa4\x1a\xfc\xc0\x0b\x8d\x58\x6d\x9a\x69\xbd\x12\x3f\x97\xc9\xb6\x06\x17\xd9\x02\x68\xfc\x66\x1b\xd0\x47\x41\xcf\xf9\x52\xc1\x8a\x54\xcb\xb1\x0e\x38\xe7\x9b\x4d\x56\x15\x02\x44\x1e\xfd\x77\x7a\xa9\x33\x06\xa1\xc6\xdf\xb4\x55\xa0\xf0\x2f\x58\x55\xfc\x3b\x7d\x70\x96\x81\x3b\x56\xcd\x48\x72\x26\x9b\xf2\xe8\x75\x25\x69\x63\x15\xfe\x4d\x96\xab\x0a\x44\xab\xeb\x70\x45\x75\xde\x9c\x7b\x9a\x7b\x69\x73\x90\x24\xcf\xdc\x62\x15\x69\x8f\x0f\xc6\x24\x20\x4e\x8b\x93\xfb\x13\x38\x44\xdc\x67\x12\x66\xf5\x0e\x8f\xb4\xab\x60\x66\x79\x95\x26\xf9\x3a\xab\xe0\xf5\xd2\xa2\xf2\x4f\x9f\x1e\xbb\x7a\x1e\xa7\x25\xcd\x1a\xe3\xd3\xdd\x87\x0b\xce\x8f\x18\xd7\x7b\x1e\xe1\xbd\xae\x87\x32\x78\xd5\xab\x3b\xba\x30\x13\x19\x95\x4b\x71\x86\x9c\x43\xf9\x63\x5f\xe7\x89\x27\xf1\xf2\x51\xdb\x17\x41\x6c\xf2\xf4\x29\x09\xe4\xdc\x53\x77\x00\xbb\x9d\x84\x24\xa0\x04\x7a\xec\xd7\xc1\x3c\x9d\x1b\x67\xb0\x34\xe2\xde\x9e\x15\xc5\xf9\x9a\x37\x32\x6f\xe5\x05\xe7\xa5\x64\xb5\xf0\x6c\xa5\x55\x76\xcb\x56\x99\xc2\xd4\x0a\xda\xbc\x5c\xd1\x4a\x3a\xa5\xf9\xfd\x39\xf9\x6f\xc9\xc8\xa7\x48\xab\x31\xf6\xb0\x03\xcb\x97\xba\xdc\x4c\x17\x33\xf2\xbf\xfe\xc7\xff\x7c\x86\x9c\xee\xdd\x89\x56\xfd\xde\xd1\x6e\xe8\x95\x76\x57\xa7\x8a\xb5\xbf\xa8\x57\x6c\x98\xb8\x79\x9b\x4e\xc9\x9f\xa8\x04\xc1\xa6\xcc\x24\x15\x92\x34\x14\x0d\xf2\xcb\x86\x6f\xc8\x9f\x98\x5c\xb7\x0b\xf2\xf2\xc7\x37\x41\x20\xee\x5b\x00\xfd\x80\x90\x98\x26\x0a\x85\x7a\x9c\x38\xd9\x3c\x98\xb1\x28\x9e\x48\x93\xb5\x94\xb5\x98\x4d\xa7\x59\xcd\x26\x2b\x40\xa9\x0e\xa2\x69\x43\x6b\x2e\xa6\x82\x0b\x5a\xf0\xe5\x72\x0a\x5c\x33\xd5\xfd\x8b\x29\xd2\xe3\x6f\x26\x5d\x15\x46\x4c\xe8\x42\xf4\x39\x7d\x32\x37\x91\xc0\x93\x5b\xda\x88\xc0\xe1\xb5\x17\xa2\xf5\x11\x9e\x9f\x48\x76\x9b\xb1\x12\xa2\xc6\xc8\xd9\x9a\xe6\x37\x44\x6d\x29\xf4\xc5\x47\xcc\x32\x5b\x5d\x1b\xe7\x18\x25\x05\x9e\x64\x44\x2a\xa1\x5c\xce\x8d\x6e\xa3\x5d\x57\xfd\x46\xea\x1a\xbf\x56\x5a\x1a\x78\xf5\xe2\x3c\x9e\x4c\x33\x97\x64\x40\xfb\x41\x58\xab\xf5\x91\xa0\x52\xb2\x6a\x25\xc8\x04\xdf\xc5\x8c\x2c\x60\x42\xbf\x26\xcb\xac\xa0\x6f\x2a\x2f\x7e\xca\xda\x8c\xf1\xef\x1c\x0c\x2d\x41\x74\x42\xce\x2b\xc1\x4b\x3a\x29\xf9\x2a\x4d\xce\xb2\x4a\xaa\xd5\x23\x65\xd6\x5d\x67\x5c\x94\x99\x3a\x07\x6c\xa4\xc9\xe3\x16\x7b\xfd\xb9\xa6\x35\xb5\xce\xa8\xa8\xa5\x09\xda\xdc\xd2\x86\xb0\x6a\xc9\x51\x5e\x03\xe6\xd8\x2e\x8f\x69\x82\xfd\xc8\x1a\x78\xd6\x80\x4a\xbb\x8a\xa6\xda\xd8\xa5\x40\x40\x07\x0f\x74\x5c\x6f\x52\x73\x21\x57\x0d\x15\x44\x37\x39\xd8\x3a\xc5\x1a\xc8\xba\xbb\xdc\xaa\xb5\x0b\x7a\xf4\xad\x3c\xd0\x23\x32\x87\x92\xaf\x70\x7d\x84\x22\xb3\x69\xab\x8a\x55\x2b\xa0\xc4\x78\x82\xf7\x37\x50\x38\xe0\x15\x93\xd7\x62\x9d\x41\x70\x92\x97\x09\x2e\xba\xaf\xfc\xe4\x73\xee\x04\x0e\x92\x56\x45\x67\x57\xa9\x9d\x9c\x54\x5c\xaa\x03\x91\x64\xd5\x83\x97\x27\x94\x2d\xc9\x1d\x3d\xbc\xa5\x64\xc5\x41\x97\x84\x05\x88\xae\x89\xbd\x00\xbd\xb5\xd9\x27\xa9\xc2\xa9\x2f\x55\xab\xc5\xa2\xd2\xef\xbf\x22\xab\x92\x2f\xb2\x52\x6d\x48\x0d\xe4\x6a\x51\x0a\xb5\xd2\x27\x98\x20\x37\xfc\x96\x82\x05\x80\xde\x33\xa1\x56\xd0\xc2\x1b\x93\xbe\x27\x69\xba\xf5\xbe\x76\x69\xba\x3a\x79\x34\x88\xf3\x98\x87\x1d\xac\xd1\x38\x2c\x5d\xf9\xfb\xd4\x93\xb4\xfd\x8c\x13\xf1\xce\xba\x7d\x40\x9c\x80\x39\x66\xdc\x58\x35\x10\x88\xec\xd1\x07\xa8\x18\x29\x90\x81\x88\x42\x4c\xdf\x21\xba\xca\xeb\x5f\x9f\x49\xa1\x39\x43\xdc\x98\x27\xfe\x74\x85\x8a\x0b\x89\x68\x3c\x01\x78\x57\x41\xf2\xfc\xee\x3d\x02\xd0\x38\xe4\x61\xef\x26\x78\x70\x07\x8c\x6c\x58\xb5\xf2\x64\x7d\x75\x94\x3a\x0b\x5b\x07\x6f\xdb\x94\xc6\x37\xc4\x9a\xa3\x94\x48\x6d\x12\x7f\xba\x13\x00\x13\xee\xe8\x50\x37\xed\x65\x64\x5e\xf4\xf5\xb5\xa9\x96\xc0\x7b\xde\x17\x25\x31\x21\xa5\xae\x43\x21\x6c\x87\xd6\x7c\x87\x1d\x42\xc0\x61\x56\x15\x59\x53\x40\x8a\x05\x57\x2a\xd6\x41\xac\xf0\x9a\x0b\xa9\x11\xd7\xab\x6b\xf5\xcb\x61\x34\x30\x35\x6f\x3c\x18\x08\x89\xed\xc1\x28\x41\xc6\xc1\xa8\x5f\x11\x3c\x98\x5b\xc9\x86\xaa\x9e\xf1\x4d\xcd\x2b\x5a\xc9\xd4\xa0\xce\x84\xb8\xe3\x4d\x11\xcc\xa2\x8d\xa8\x25\x0e\x7f\xb1\x08\xc6\xad\x7d\xbd\x78\x23\xe3\x1a\xb5\x1e\x40\xf2\xc7\x7f\xfd\xc3\x37\x61\xe0\x06\x1a\x30\x13\x73\x6a\xcf\xa6\xe0\x83\x0a\x83\x79\x46\x92\x19\xba\x75\x61\x26\xa7\xff\x5f\xfd\x80\xe9\xb2\x35\x0a\xed\x33\x92\x40\x1b\x8c\xdc\xfd\x56\x88\x52\xcd\x35\x06\x5a\x89\x32\x22\x58\x41\xfa\x55\x25\xaf\x92\xb9\xea\xde\x09\x7e\x50\x88\x2b\x80\xef\x25\x6a\xe1\x02\x88\x17\xdf\xfc\xdb\xe4\xf9\xe4\xf9\xe4\x85\x81\xf0\x46\x8f\x28\x9f\x3e\x0d\x5b\x68\x72\x92\x91\x9a\x11\xff\x05\x06\x64\x82\x79\x9c\x60\xef\x7d\x54\x87\x44\xb6\x4d\x69\x4c\x52\x70\x90\x40\x50\x14\xbd\x97\x64\x43\xab\x96\xe8\x50\x08\xeb\x9d\x02\xc7\x8b\x8e\xd2\xd2\x2f\xd0\x9e\x7d\x85\x55\x68\x5e\xf9\x33\x00\xfc\x40\xab\xb6\x9f\xf6\xc4\x45\xc7\xe8\x8e\x54\x3f\xfa\x55\x5b\xe4\xbc\xa6\x62\x46\x12\xb9\xd6\x5a\x15\x8a\x41\x33\xd2\x6d\x7e\xad\x5a\x19\xed\x8c\x2e\x79\x43\x7d\xed\x6a\x4c\x28\x66\x90\x19\x6b\x04\xc1\x5d\xf1\xba\x82\x51\xc0\xf8\xd4\xc9\x08\x1e\x3e\x70\xc5\x42\x39\x3c\xde\x43\x84\x36\xaf\xca\x87\x89\x27\x08\x0e\x1a\xa3\x60\xc9\xac\xa7\x90\x77\xb8\x52\xf0\xe5\xa4\x95\x7c\x85\xe1\xa8\x7e\xc0\xba\x5c\x33\x31\xc9\x4b\x2e\x28\xcc\x80\x57\xa3\x57\x46\xa7\xe0\x0e\x44\x31\xed\xee\x56\x29\x2d\xc8\x1b\xb1\x9e\xca\x31\xa1\xa1\xd1\x45\x61\x56\xa0\xb0\xbd\x94\x9c\x09\x93\x61\x2f\x0f\x1d\xda\x6f\xa0\xf4\x90\x32\xed\x05\xe1\x0f\xc4\x85\x5a\x83\xa0\x6a\xa3\x6f\x49\x24\xe6\xfa\x6b\x43\xce\x28\xf0\x29\xf2\x06\x68\x83\xaf\x3b\xf8\xd1\xfc\x7e\x8d\xe6\xf7\xa0\x8f\x21\xe3\xba\x07\x12\x37\x16\xea\x1c\x70\x63\xb5\xf0\xb8\xbe\x8a\x9a\x4e\xb3\x28\xbd\x63\xc3\x2f\x26\x02\xb1\xdb\xae\x33\x6d\x68\x69\x4f\x46\x96\xcf\x0a\x6d\x95\xc7\x0a\xaf\xe9\x1e\x33\x51\xb5\x9b\x6b\x30\x0a\xf7\x66\x21\x34\x05\xff\x86\xe3\x27\xdb\xc7\x1f\x1b\x04\xfe\x78\x8c\xdd\xd1\x9d\xc0\xb7\x6d\x7b\xbf\x18\xdc\xfb\x0d\xbf\xfb\xe2\x8d\x0f\x16\x4b\xf1\x7a\x53\xcb\x87\xd0\xe5\x2e\xf0\x66\x0a\x62\x84\x4d\xa3\xef\xcc\x01\xd1\x6d\x17\x1c\x01\x73\x77\x04\xd8\xe5\x84\x80\x38\xdd\xe9\xa7\x4f\xe4\x89\xc3\xf5\xcf\x7e\x4a\xa0\xd5\x76\xcf\x63\x02\xf3\x66\x0c\x73\x94\xc1\x8c\x56\xef\xeb\x45\x0c\xbd\x8e\xe0\x7e\x53\xdc\x6b\x69\xa2\xc3\xd4\x39\x2f\x93\xa0\x07\xdd\x00\x0e\x92\xb0\x41\x98\xc6\xc3\x83\x7d\xa7\x74\xf4\x90\x0b\xae\x4d\xc8\xf2\x3a\x19\x4d\xe8\xcf\x29\x92\x30\xea\x78\x53\x1e\x04\x9b\x34\xe6\xc8\xac\xb1\x07\xbd\x3a\xd0\xc0\x89\x59\x7f\x20\xa1\x0b\xaa\x3d\xf4\xf4\x1c\xe1\x04\x59\xdc\x30\xca\x5e\x0b\xf0\xaa\xb7\xb8\x45\xbb\xb0\xa9\xe2\xfb\xfb\x38\x78\x48\x61\x55\xa1\x1f\x59\x5f\xe9\xec\x27\x9d\xcb\x3e\x92\x1c\xa5\xbf\xe5\xdd\x06\xef\x82\x5f\x6b\x60\x7f\xb7\xff\x5e\xec\xdc\xc9\x14\x43\xfe\xbe\x2c\x47\xfb\x25\x49\x21\x9f\x9b\x28\x85\xc4\x93\xa5\xf4\x76\xd2\xf6\x35\x35\x8b\xa9\x33\x40\xe2\x2b\x99\xa7\x22\x21\xab\x74\x9f\x42\xf4\x92\x6b\xc6\xb3\x4b\xe8\x12\x4c\xf2\x2a\x4d\x6e\xe8\x43\x5b\x7b\xa6\x36\x92\xda\x55\x0b\x4c\xd3\x41\x17\x66\xd1\x54\xdf\xb7\x19\xbe\x10\xa9\x03\x51\x2b\x08\xa8\xa9\x79\x06\x94\x0f\x54\x50\x49\x04\xcd\x9a\x7c\x4d\x78\x45\x5e\x9f\x9f\x59\x21\x9a\x4e\x6e\xe8\xc3\x99\xd6\x95\xbe\xf9\x37\x75\x1c\x03\xca\xd0\x44\x02\xbe\xab\x9d\xe1\x1f\x0f\xdb\x28\x40\xf7\x53\xe4\x1f\xe9\xd1\x1e\xb9\xdd\x14\xfa\x8b\x4c\xfc\x7c\xac\x83\xa9\x1f\x01\xba\x3b\xcb\x11\xd3\x3b\xf1\x6c\xef\x3a\x2e\x00\xfb\xff\xee\x41\x9d\x22\xe9\x6d\x56\x9a\x8f\x10\x90\x6f\x42\xdb\xfa\x16\x92\xe1\x81\xa1\x64\xf9\x8d\x6f\x12\xa5\x81\x5b\x6f\x64\x6a\x7a\x3e\x54\x11\x38\x7b\x2e\x0c\x33\x09\x9c\x6c\x2e\x7e\xdb\xb9\x83\x4d\xbc\x54\xae\x5d\x4f\xe0\xa1\x91\x18\x63\x81\x4f\x57\x6c\x9e\xbc\xe4\xe0\x5b\x3b\xa5\x7e\xde\x06\x92\xb2\xe2\xde\x25\xed\xd2\x52\x82\x3d\x70\x4a\x4f\xb5\xd5\x0e\xec\x50\x1a\xbb\x12\xac\xf7\xba\xd1\xea\x34\x41\x27\xbe\x8a\xab\x50\x87\xe6\x8f\xd0\x52\x02\xf5\x5d\x6b\x4a\xe4\x6b\x0b\xff\xd1\x72\x49\xb5\x17\x1c\xa8\x69\x30\x05\x98\xc1\xce\xbe\x5f\xc8\x87\x9a\xf2\xa5\x16\x15\xe7\x68\x6c\x68\x58\xb5\x82\xd4\xca\x98\xfa\xcc\x6a\xa0\x13\x78\xcb\x70\xaa\x27\x58\x33\x1c\x7a\x6b\x03\x10\x64\x4e\xfc\xac\x69\x98\x35\xdd\x6d\x29\x72\x79\x98\x1c\x8e\xe3\x4d\x2f\x9f\x5f\x8d\xc9\x61\x32\x19\x06\x78\x01\x00\x87\x3a\xc6\xe5\xf0\xd0\x5a\xab\x35\x72\xed\xb5\xd0\xbb\xbf\x82\xb4\xaa\x30\x82\xa8\xfe\x7a\xac\x2b\xa2\xf7\x9d\xdd\x51\x41\xae\xe8\xb6\xec\xb2\x4c\xca\x8a\x2e\xc3\xe8\x34\xd5\x01\x6f\xe8\xc3\xc1\x63\x8e\x7e\x32\x6b\x67\x1f\x53\xed\xfa\x97\x6a\x78\xb1\x62\xf4\x6d\xec\x3a\x25\xbe\xcc\xed\x18\x5e\xc7\xb7\x58\x90\xbd\x6e\x5d\xe2\x1e\xf1\xe1\x5f\xe7\xe2\x0d\x81\x90\xb5\x08\xc6\x75\xc4\x79\xd2\xca\x62\xf0\xe0\x8f\x73\xc3\x8a\x8e\x0a\xe9\x65\x58\xf2\x73\xb6\x65\xb9\xcb\x20\x61\xfe\xed\x95\xcb\xd4\x35\x78\x34\x97\xe5\xa8\x67\x90\x76\xeb\x01\x59\xcc\x3f\x77\x39\x54\xa3\xcf\x58\x0c\xcc\x94\xfe\x4f\xb9\x16\x5e\x2e\xc4\xcf\x5b\x8a\xfe\xd1\x26\xf9\x6a\x55\x52\xb3\x31\xcf\xe1\xf6\xdf\x2e\x8a\x62\x0b\x17\x4f\x62\xaf\x23\x14\x1d\x7c\x08\x42\x42\x07\x4d\xb0\x1d\x75\xfa\x72\x26\x75\xb8\x09\xaa\xba\x35\x36\xd5\x1e\x62\xb4\xe5\x2a\x08\x08\xeb\xc9\x94\xf6\x90\x26\x05\x15\xb2\xe1\x0f\x03\xd5\xbf\x10\x54\x90\x79\xdb\xe4\x74\x46\x30\xe8\x00\x8a\x36\xac\x7a\x0b\x46\xd1\x19\x79\xae\x8b\xd4\x65\xa0\x98\x28\x2b\xcb\x44\x17\x65\xad\xe4\xf8\xa2\x1d\x7c\xe5\x68\xc9\xe4\x05\xd7\x99\x96\x67\xe0\x8f\xec\x84\x84\x1e\x85\x25\xe7\x37\x20\xb7\xf8\x01\x3a\x08\xa5\xa4\x06\x28\xe5\xad\x0c\x05\x07\x2d\xe5\x47\x97\x47\x3f\xc3\x21\x86\xe5\xd2\x43\xe1\x0b\x15\x5f\xa7\x26\x33\xf4\x68\xd2\xd0\xac\x78\xe8\x3a\x1f\x38\xbd\xc7\x46\xfe\x05\x62\x0c\x0c\xd4\x6b\xd2\x0f\x52\x53\x4b\x6c\x19\xcc\xa1\xf3\x42\xbe\x3a\x08\xe3\xe8\xbc\xf0\xb5\xe3\x08\x3a\x9b\xad\x63\x4f\xea\x6c\x8a\x92\x38\x75\x41\x36\x8a\xb8\xdc\xd6\x1d\xac\x4b\xc5\x71\xdc\x43\x67\x73\xb3\xec\xa6\x2e\xcc\x67\x13\xa7\xce\x04\x89\x75\x91\x0d\xa1\xd3\x81\x43\xc7\x84\x0c\x0d\xd6\x46\x66\x85\x38\x7b\xe8\x7a\xf1\x5e\xc7\x7d\x74\x2e\x04\xa8\x4b\x60\x0f\x5d\x27\xb2\xe9\x98\x90\xd0\xa6\xd5\x76\x29\xea\xb9\xc6\x84\x1e\x47\x5e\x5b\xe7\x4a\xb1\xab\x7d\xe0\x18\xe4\xbf\x7a\x89\xdb\x5d\xad\xad\x3b\x79\x02\xc0\x1d\x04\x3f\x89\xde\x8c\x6e\xc1\x80\xd0\x1d\x14\xe8\x64\xbf\x27\x06\x00\x3e\x8e\x9a\x05\x3b\x48\x12\xd9\x44\x55\x90\x20\x50\xb6\x19\xca\x06\x65\x95\x2e\x54\x15\xe3\x19\xa0\x02\x2a\x42\x55\xd5\x52\x31\xe9\x7f\xc0\x63\x80\x2c\xec\xa9\xce\x1a\x38\x51\xf4\xed\x11\x4d\xed\xbf\x7f\xaf\xa0\x28\xfc\xf6\xdd\x95\x2c\xda\x49\x98\xee\x7e\xde\x09\xda\x33\x24\x38\x71\xc0\x86\xee\x81\x9f\x6c\x58\xaf\xca\x12\xad\x8f\x7a\xe1\x1b\x4e\xdb\xea\xc6\x51\xf8\x11\xd0\xb1\x85\xec\xd6\xef\x4a\x4c\xe2\xe0\xc2\xe4\x00\x63\x12\x0d\xff\x77\x6f\xf8\x9d\xc4\x3a\xa6\x14\x2d\x48\x43\x5f\xd7\x94\xd9\xc2\xb7\x23\xe9\xd4\xf2\xc1\x25\xe5\x8c\x49\x91\xfb\xc8\xd6\x05\x96\x1b\x1f\x8d\xbb\x9c\x22\x88\xfc\x9b\x68\x37\x2a\xff\x26\x89\x53\xe5\x5d\x1c\xbb\xd1\x99\x7b\x2e\x82\xca\x5d\x69\x71\x34\x9d\x8c\xad\xc3\x53\x13\x7f\x59\xe8\xf2\x35\xb8\x19\xeb\xd4\x81\x51\x0e\x8f\x59\xdd\xad\x76\x66\x53\xb7\x86\x9c\x1c\x0a\xb4\xda\x2c\xd5\x6a\x9b\xb0\x0f\xa8\xcd\xaf\x1a\xe3\x8e\x9c\xc2\xb8\x51\x77\x1d\x89\xeb\xe8\x28\x3a\xf1\xd9\x4e\x0a\x8a\x47\x68\x07\x4e\x4e\xbd\xe4\x02\xe1\x18\xfc\x54\x00\x61\xba\x81\x1e\xa8\x4e\x05\x80\x06\x83\x97\xe7\x67\x09\xf9\x56\xa7\xe9\x25\x33\x2c\xb0\xfb\xbe\xdf\x2c\x92\xda\x61\xdf\x4c\x02\x5b\x18\xa1\x58\x94\x6e\xee\x0a\x72\x4a\x0a\x76\x3b\xe4\x37\x6b\xa8\x5a\x67\x22\x4d\x94\xae\x93\x35\x34\x4b\xe2\xce\x46\x11\x2b\xa0\xe3\x83\x79\xf8\x99\x13\x8b\x17\xbe\x9f\x32\xf2\x5c\xcf\x4c\x32\x69\x83\xce\x47\x65\x08\xd0\x0e\xf1\xf6\x27\xf8\xc4\x9b\x63\x96\xde\xcb\x14\x91\x98\x22\x2f\x92\x98\x37\x1b\xf8\x2c\x57\xc3\x4b\xd7\x24\xc7\x44\x8b\x05\x30\x92\xa1\xcc\x2b\x1c\xf9\x76\x07\x40\x6d\xcc\x35\x7f\xad\xbc\xb9\x98\x93\x3f\xb8\xc9\x30\xb4\x21\x9e\x35\x65\xab\x35\x04\x52\x7f\xf3\xfc\x79\x7d\x9f\x74\xa3\x12\xbc\xc9\x30\x0d\x35\x05\x9b\xec\xfe\x28\xda\x3a\x58\xde\x65\x43\xc5\xfa\x5a\x7f\x5c\x73\x87\xd8\xd1\x0b\xf8\xf4\x31\x05\xcf\x1e\x0a\x0d\x3e\x7d\xec\x7d\x4c\xec\x7b\xeb\xd8\x25\xcd\xbb\x3b\x6c\xc9\xaa\x22\xfe\x10\xe4\x6d\x38\x9b\x13\x27\xda\x2a\x4c\x6c\xe3\x5a\xa1\x17\xb4\xef\xc9\xe4\x37\x0e\xd2\xdc\xf8\x2b\xae\x3b\xb3\xe9\x6a\xc2\x44\x33\x57\xce\xea\x07\x59\x66\xc0\x89\xd9\xba\x5a\x77\xad\xe8\x9d\xc4\x1b\x35\xcd\xd9\xd2\xa4\xd0\x21\x41\xd8\x8e\xd9\x4f\x3e\xa3\xc4\xcf\x7d\xdf\x7c\x0d\x4e\xf0\x47\xfa\x1b\xce\xdb\xf9\xc0\xff\x22\x72\xdd\xf0\x4d\xdd\xa7\x2b\x23\x15\xbd\x43\xb7\x0f\xf3\xb9\xe9\x30\x13\x92\x3f\x4d\x88\xeb\xe9\x53\x8d\xf4\xb4\xf7\x71\xa7\x7d\x24\x11\x12\x7e\x08\xdb\x7c\x78\xf9\x73\xae\xbd\x2e\x13\xc4\x83\x03\xa8\x3f\x0d\x91\x17\x14\x6f\x60\x97\xf8\x31\xf2\xb1\xf7\xe5\x71\xb7\xe8\xb7\x59\x39\x52\x63\x0d\x43\x99\x7a\xc2\x13\x1a\xa9\x7d\x53\xfe\xa0\x6b\x5f\xb7\x69\xcc\x7e\x1d\x64\xea\x82\x67\x85\x23\x17\xf7\xb4\x75\xd5\x3f\x4f\xd4\xdb\xc9\x74\xd3\x29\x79\xd9\x4a\xbe\xc9\x24\xcb\xb3\xb2\x7c\x20\x75\x43\x97\xac\x2c\x75\xb8\x1e\xb0\x35\x5b\x12\x26\x0f\x05\x78\xc4\x0a\x2a\xc9\x03\x95\x07\xd1\xd7\xde\xdd\x61\x1c\x03\xef\xbe\xa3\xee\x36\xdb\xf3\x7d\x78\xf0\x6d\x38\x78\x2c\xec\x48\x55\x7e\xb0\x99\x9f\xd1\x6d\xe7\x66\x33\xde\xd2\x7b\x7d\x11\x5d\x5f\x78\x5c\x66\x25\xd9\xf9\x11\x75\x11\x58\xc6\xb1\xd1\x29\x09\xe2\x02\xf6\xd9\x84\x86\xc0\x67\xfe\x76\x1c\x4e\xda\x14\xf8\x72\x78\x6d\xd5\x62\x00\x0d\xa1\x33\xad\x56\xcd\x76\x66\x2c\x74\x4f\xb1\x07\x5d\xe4\xa7\xdd\x23\x65\x9f\x34\x82\x5b\x17\xd0\x47\xf0\x6b\x2e\xe0\x6e\xd2\xf7\x59\x88\x23\x6f\x21\xf6\xcd\x3b\xb8\x73\xd1\x06\xe6\xb6\xfb\x31\xbe\x7d\x97\xab\xeb\x71\xdc\xb7\x29\x6f\x7f\x3c\xed\x65\x9a\x31\x2d\x45\x37\xd5\x4c\x43\x45\xed\x09\x58\x5b\x8c\xa7\x24\x6e\x8d\x06\x04\x9a\x6a\x7b\x88\xd9\x8f\xc3\xf5\x4d\xdd\x78\x16\xa5\xf1\x3b\x23\x60\x86\x98\x9d\xdc\x99\x88\x57\x54\xbe\x0c\xf4\x72\x7f\xde\x9f\x3e\x25\xde\x12\xc4\x71\xa1\x48\xe2\x84\x88\xde\x87\x9f\x51\xef\x2e\x16\xf0\x71\xe9\x62\x31\xb3\x28\x87\xe7\xcf\x7e\xbd\xa2\xee\x46\x1c\x38\x39\xc5\xab\x3d\x26\xde\x3f\x0f\xd0\x17\xfa\x89\xfb\xd0\x24\x41\xaf\x8b\xc0\x03\xa4\x81\x74\x26\xfd\x6f\x10\x3d\x76\x0d\x80\x05\x93\xc3\xa6\xcd\xe8\xbd\xa0\xa1\xdd\x97\x1b\x35\x33\x96\x5c\xf8\x56\xd2\xbe\x13\x3c\x09\x85\xab\x58\x94\x4d\xd7\xb8\xd8\xc1\xb9\x0f\x79\x7b\x7c\xef\xc6\x7d\xba\xfb\xdb\x9e\x36\xea\x7d\x0d\x3c\x12\x61\xb8\x8d\x6e\x32\xf0\xf9\xa7\xed\x93\x13\xbc\xa8\xef\x1a\xbf\x6b\xb6\x4b\xee\x88\x46\x23\xb8\xbe\xe2\x61\x0b\x18\x5e\xb0\x8f\x64\xe0\xbb\xbf\x0c\xbb\x97\x77\x18\x24\x16\x56\x60\x8e\xb8\xa1\xeb\xc3\x46\x0c\x6c\xa7\xca\xd7\x7d\x87\x44\x4b\x54\x26\xb7\x3b\xc3\x77\x60\x7a\xee\xf0\x7f\xf7\x88\x76\xc4\x68\xec\x23\x51\xee\xc4\xf0\x85\x46\x4c\xdf\xa8\x68\xe1\xbc\x48\x91\xae\x2d\x11\x0c\xc4\xbe\xed\xaf\x1b\x84\x06\xf5\xd6\xd5\xc8\x3f\x05\x62\xd0\x3a\x98\xc4\xc2\xfb\x1b\x23\x06\x2f\xd6\x83\xa0\xe1\x39\xa9\xa9\x35\xc1\x2a\xfb\xd2\xbb\xab\xff\x2e\xbd\xbb\xc6\xf7\xb9\xf4\x8a\xf5\x3f\x02\xa9\x5d\xd0\xd0\xee\x34\x74\x90\xf8\x11\x53\x7b\xa8\x84\xc6\x3f\x29\xb4\x14\xf8\x01\x3a\xc1\x57\xd6\x93\x88\x01\x51\x3b\x3e\xd9\x8e\x2f\x15\xf4\x55\x37\xa8\x91\x2d\xbd\x80\xba\xaa\x7c\x20\xeb\x4c\x80\x0b\xa5\x00\x35\x49\x7b\x4a\x4b\xba\x99\xa8\xc2\xa7\x4f\x89\xfd\xfb\x49\x4f\xe7\x89\xc7\x65\x99\x06\xe1\xe9\xef\x9d\x13\xb8\x69\x46\x13\xd8\xe2\x5b\xbd\xfa\xa6\x53\xf2\xbd\xd2\xee\x58\x65\x89\x86\xe4\x3f\x06\x17\x31\x51\x9e\xf6\xcc\x0f\xe3\xab\xd0\x61\x8b\x0b\x3f\xbb\x7b\x18\x5d\x05\x10\xea\x77\x08\xe1\xc5\x56\xe1\x70\x84\x35\x89\xc6\xc3\xa8\x10\x8f\x2e\x0b\x21\x6d\x1c\x15\x33\x4e\xae\x4a\xda\xf2\x60\xa2\x67\x28\xfa\x92\x89\xd2\x3f\xbe\x75\xd9\x5a\x2d\x8b\x1f\x90\x69\xca\xe3\x76\x52\xd5\x85\x10\xeb\xde\xbc\x08\xb1\xf6\xe7\xc6\x01\x76\xa7\x47\x01\x7a\x53\xe4\x00\xbb\xb3\xa4\x00\xbd\x99\xf2\x30\x46\x26\x0b\xb0\x86\x13\xe6\x1a\xdc\xd0\x87\x2e\xec\x0d\x7d\x18\xe4\x28\xb1\xee\xb2\xd3\x40\xe6\x96\xce\x3c\x24\xc9\xb6\xc1\x47\x6a\xbd\x11\xc7\xda\x86\xc3\x8c\x40\xb8\x71\x85\x95\xfb\x1d\x93\xdd\x61\x9b\xf3\x3c\x32\xf6\xf8\xa1\xb4\xe4\xcd\xe6\xcb\x0c\xad\xf6\x94\xc1\x2b\xb6\x67\x0a\xd5\xb6\x21\x5e\xd3\xea\x28\x4c\xe0\x4a\x6c\x60\x62\x93\x6d\x84\xf7\x80\xd9\x36\xe5\x2c\x1e\x09\x6a\xc4\x74\xcb\xf8\xd8\x56\x1d\x2a\xf1\xd8\xc3\x81\xa0\xe5\x2f\x0d\x08\xed\x05\x71\x12\x4d\xfd\x25\x94\x5e\x11\xfb\x6f\x4e\x5e\xf4\x01\x90\xc3\xae\x34\x40\x84\xed\xec\x82\xfa\x8d\x80\xf1\xba\x8d\xfa\x71\xa0\x61\x23\xe0\xc7\x6e\xa3\x7e\x60\x68\xa7\x27\xc3\xa6\x57\x7e\x4f\x9d\xb0\xd0\x58\x43\xc5\xbd\x57\xa4\xd3\x9b\x63\xe9\xee\x2b\x43\xc0\x7a\xa0\xd6\x75\x38\xda\xa4\xfe\x8f\xa6\x64\xd4\xd1\xfa\xda\x44\x7c\x97\x31\x39\x99\x4c\xac\x90\xd6\xd7\x49\x75\x5f\xf0\xad\x08\x45\xf2\xa0\x26\x3a\xd4\x2b\xe6\xf0\xd1\xdd\x6a\xa6\x4c\x42\xab\x53\x5c\x7d\xb5\xda\x20\x99\x87\x41\x40\x83\x93\x80\x99\x93\x1c\xb2\x8e\x8c\x61\x7c\xed\x82\xe3\x2b\xec\x47\x4d\x91\xeb\xa6\xf3\xc6\x32\xd4\x7d\x54\xe3\xb2\x70\x03\x5f\xd1\x05\x32\xbb\x95\x9d\xb6\x1b\x74\xa0\x89\x8d\x22\x50\xe8\xfc\xd4\x4e\xaa\x38\x9a\xc1\xc5\x7e\x80\x0d\x22\x08\xf0\xcb\x5f\x90\x60\x41\xae\xa9\x3a\x34\xcc\x37\x21\xe9\xcf\x1f\x21\xf2\xa3\xa2\x77\xe4\xe3\x87\xb7\x69\x47\xff\x87\x0e\xe0\xb5\xd4\x44\x75\xc0\x37\x03\x55\xa3\x09\x5a\x3a\x7e\xc4\x53\x05\xd2\x96\x68\xa0\xc4\xc5\x78\xbb\x66\x4f\x9f\x7a\x38\x02\x29\x68\xe8\x33\x64\x2e\xb4\x24\x19\xbb\xb6\x7a\x66\x34\x9d\xda\xd5\x0b\x92\x4e\x9f\xcb\x4c\xd2\xf4\x97\xc7\x31\x31\x5e\x76\xf8\x35\xbc\x71\xcf\xa8\x51\x67\x72\xed\x32\xde\x1f\x7c\x5e\x76\x6c\x7f\x1f\x0c\xb1\xf3\x20\x33\xef\xd2\xfb\x3b\x87\x2d\x3e\xac\xef\x91\x5e\x60\x88\xab\xe3\x3c\xfd\xe5\x9c\x1a\xe1\x53\x6f\x67\x3f\x81\xc6\x66\xdd\x4a\x9e\xdf\x74\x52\x33\xec\x1e\x5b\x24\xde\x72\x4b\x28\xcf\xf1\x81\x82\xf9\xdf\x01\x00\x00\xff\xff\x6f\x94\xa8\x32\x15\x98\x00\x00") +var _staticJsAppJs = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\xbd\xef\x76\x1b\x37\xb2\x20\xfe\x5d\x4f\x01\x77\x72\xac\xe6\x35\x45\xda\x99\x3b\xbf\x7b\x0e\x25\x2a\x3f\x47\x76\x66\xbc\xd7\xb1\x73\x2d\x79\x76\xf7\x68\x74\x74\x9a\xdd\x20\x89\xa8\xd9\xe8\x34\xd0\xfa\x33\xb1\xde\x60\x1f\x60\x9f\x6f\x9f\x64\x0f\xaa\xf0\xb7\x1b\x4d\xd2\x9e\x64\x33\x3b\x7b\xfd\x21\x11\x81\x42\xa1\x50\x28\x00\x55\x85\x2a\xf4\x6d\xd6\x10\x5a\x30\xc9\x1b\xe2\xff\x9b\x93\xaa\x2d\xcb\xe3\x03\x55\x9d\xf3\xaa\xa2\xb9\xa4\x85\x5f\xbd\xcc\x4a\x41\xb1\x7e\xc1\xf9\xcd\x26\x6b\x6e\x84\x5f\xff\xcb\x23\x56\x16\x74\x99\xb5\xa5\xbc\x6e\xf8\x9d\xb8\x2e\xd9\x86\x49\x32\x27\x2f\x9e\x3f\xd7\xa8\xdb\xa6\xa1\x95\x7c\xbf\xf8\x89\xe6\x32\xec\x19\xea\x97\xac\x94\xb4\x79\x5f\x4b\xc6\x2b\xa1\xb0\x1e\x10\x92\xd0\x9f\xdb\xac\x4c\x66\x08\x9e\xcc\xc9\xe1\xab\x97\x17\x2f\x0f\x93\xb1\xaa\xab\xb8\xbc\xb6\xf5\xc9\x93\xb0\x72\xd5\xd0\x4c\xd2\x06\x9b\x26\xa7\xa6\x8e\xf8\x95\xd7\xf4\xe7\x64\x46\x92\xd3\xb0\x65\x49\x85\x30\x3d\x92\xe4\xa4\x57\x87\xad\xa0\xae\xd3\x90\xdd\x50\xd7\xf0\xed\x9b\x7f\x7f\x1d\x54\x33\xbf\x3e\x79\xd3\xab\x57\xac\x70\xcd\xdf\x9c\x93\x77\x1f\xdf\xbe\x75\x23\xb5\xd5\x50\xf5\xfe\x02\xab\x0f\x1e\x8f\x0f\x0e\x96\x6d\x95\x2b\xae\x91\x15\x95\xe7\x54\x08\xc6\xab\x37\x45\x3a\x02\x0e\x2a\xce\xb2\x82\xcc\x89\xc0\x8a\x73\xc9\x9b\x6c\x45\x27\x2b\x2a\xdf\x48\xba\x49\x13\x5d\x7e\xcd\x8a\x64\x74\x7c\x70\x40\x08\x5b\x92\xf4\x09\x2b\xb0\x39\xc1\xc6\xab\x96\x15\xe9\xe8\x18\x0a\x3a\x88\x44\x04\xd1\x98\xb0\x02\xa0\x1f\x15\xc2\x86\xca\xb6\xa9\x08\x2b\x8e\x0f\x1e\x3d\x62\x05\x95\x1f\xf8\x9d\x78\xab\xe4\x24\xad\xda\x0d\x76\x58\xf2\x3c\x2b\x7b\xb8\x9d\x44\x25\x63\xa2\x60\x43\x54\x2b\x1f\x15\xe2\xd1\x9d\xd6\x59\x23\xe8\x9b\x4a\xa6\x01\xde\x55\x04\xef\x88\x7c\xfa\x14\x91\xe0\x7e\x4f\x3f\x66\x2b\x56\x65\xea\xd7\xfb\xe5\x52\x50\xe9\x31\xba\xce\x56\x54\x49\xf5\xd7\x69\x32\xd1\xe2\x7e\xa4\xca\x92\xd1\xa4\xc8\x64\x96\x26\xf8\xe3\x58\x83\x9b\x25\x12\x92\x7f\xec\xa8\x4f\x01\xe1\x11\x79\x31\x22\xff\x82\xd0\x31\x6a\xa8\x38\xe3\x6d\x25\x53\x45\x34\xfc\xe5\x08\x1a\xee\x41\xd5\x56\xed\x86\xcc\x1d\x8b\x6c\x7b\x32\x25\x66\xec\x5a\x1e\xd4\xfc\x18\x12\x46\xe4\x84\x74\xba\x22\x0a\xd5\xb3\x67\xdd\x19\xaf\xda\x4d\x48\x6f\x56\xb3\xb3\xac\x2c\xd3\x0d\x95\x6b\x5e\x8c\x49\x9d\xc9\xb5\xfa\x6f\x93\x6d\xc4\x98\xe4\x0b\x47\xb8\x64\x1b\xca\x5b\x45\xfa\x1f\x9e\xab\x7f\xc7\x64\x3a\x25\x7f\x24\x1b\x56\x09\xc2\x04\xa1\x15\x6f\x57\x6b\xd5\xd5\xd7\x93\xec\xa7\xec\x3e\x45\x2a\x74\xab\x99\xf9\x63\x0c\xa5\x6d\x53\xce\x48\x92\xd5\x2c\x21\xcf\xb0\x4b\x28\x46\x22\x66\xfa\xff\x58\x96\x67\xf9\x9a\xce\x70\xc3\xc3\x12\x35\x6f\x33\x43\x22\x94\xac\x69\x56\xd0\x46\xcc\xf4\xc8\x09\x49\xee\x8f\xb4\xec\x1f\xb1\x22\x99\x75\x56\x21\x00\x3d\x62\x53\xd1\xe6\x39\x15\x62\x46\x0c\x47\x52\x85\x7e\x64\x31\xe5\x0b\x2c\x38\xf6\x1b\xd1\xa6\xe1\x8d\xd7\xe4\x7e\xdd\x8c\x89\x90\x99\x6c\xc5\x98\x84\xed\xd5\x54\x61\x0d\x99\xcf\x49\xa2\xb9\x90\x38\x00\x3b\x35\xf9\x22\xfd\xc5\x60\x4e\xfe\xa3\xa5\xcd\x83\x65\x79\xb6\x94\xb4\x21\x8a\x57\xa9\x29\x9a\xaa\x5d\xfc\xf9\x88\x3c\x23\x89\x48\xc8\xa3\xa6\x4f\x4f\xb6\x26\xfc\x27\xc0\x32\x01\x69\xfa\x2f\xe7\xef\xdf\x29\x3a\x27\x0d\x15\x35\xaf\x04\xbd\xa0\xf7\x72\x64\x86\xa5\xa4\xa4\xbf\xac\xde\x54\x4b\x9e\x2a\x11\x18\xf8\xf7\x8b\x95\x9e\x64\x45\xd5\x36\x90\x4c\x59\xb5\xe4\xc9\x98\xfc\xf2\x08\xb2\x73\x4c\x1e\x03\x84\x78\xd0\x88\x41\x9c\x11\x84\x1c\x9b\x0c\xe3\xbc\xc8\x16\x25\x1d\x46\x19\xc3\x29\xa1\xc9\x0e\x94\x6a\x71\xa6\x00\x39\x26\xbc\x96\x7a\x31\x6c\x47\x39\x55\x53\x04\x7f\xaa\x79\x99\xaa\x25\x99\x78\xad\xa3\xfd\x9c\xcb\xa6\xcd\x65\xdb\xd0\x7e\x67\x7b\xf4\xb3\x0b\xfb\x9b\xaa\xa0\xf7\xd4\x0e\xc4\xe7\xd2\xbe\xa3\x60\x88\x62\x07\xbb\xce\x78\x25\x64\x93\xb1\x4a\xf6\x3a\xdb\xb7\xa3\xdc\xa1\x18\xee\xec\xcf\x4c\x48\xde\x3c\x7c\x8e\x08\xad\xb1\xc9\x30\xce\xef\x8c\x06\x35\x80\x35\x82\xd3\x2a\x5d\x03\x58\xe9\x3d\xcd\x5b\x49\x61\x01\xa6\x3f\xab\xff\x86\xcc\x0f\xb1\xd6\x5c\x20\x5a\x80\x54\x28\x09\xfc\x35\xc3\xff\x91\x68\x07\x75\x99\xb1\xea\x73\x3b\xd0\xcd\xf6\xea\xa2\x60\x42\xeb\x9e\xfb\x70\xdb\x76\xe1\x9a\x85\xbc\xf1\x68\xaf\x72\x5e\xf8\xbc\x09\xf4\x83\xef\x32\x41\xff\xbf\x7f\x9d\x20\x90\xae\x9f\x34\xb4\x2e\xb3\x9c\xa6\xd3\xbf\x3e\x9b\xae\xc6\x24\x39\x4a\xfc\xb2\x29\x94\x5d\xfb\x65\x73\x28\x9a\x24\x9d\x5d\x6d\xd1\xb2\xb2\x38\xcf\xd7\x74\x93\x9d\x53\xdc\xbc\xab\x6c\xa3\x96\x11\xee\x33\xee\xb8\x13\x58\x4d\xe6\x24\x49\xe0\xd0\xc5\x33\x50\x96\xd4\x68\xc2\x84\x24\x20\xbc\x56\x43\xc4\x7f\xc9\x85\xde\x5e\x10\xe4\x96\xd1\xbb\x0e\x04\x49\xfe\xc2\xe8\x9d\x85\xd8\x28\xc5\x97\x65\x25\xfb\x1b\x2d\xae\x35\x78\xf2\x83\x57\x48\x02\x70\x41\x7f\x6e\x69\x95\x07\xdd\x26\xe7\xba\x50\x24\x6a\x2b\xb7\xf4\xb2\xdc\x29\xee\x03\xe4\x1e\x9e\x30\x92\x97\x99\x10\xf3\x64\x99\x91\x65\x76\x84\x40\xa7\x27\x53\x76\x7a\xb8\x65\x0c\xfb\xb4\x8b\x8d\x6c\x9f\x76\xb1\x21\xf6\xda\xe5\xac\xc9\x4b\x7a\xc4\x75\xd3\x60\xdc\x37\x0a\x50\x4f\x1d\x9e\xc1\x6a\x96\xe1\x04\xae\xdb\x45\xc9\xf2\x64\xe4\x60\xe8\x7d\x9d\x55\x05\x2d\x70\x9a\xcd\xbc\x3f\x9b\x93\xe4\xa4\x60\xb7\xba\xd3\x43\x01\x42\x03\x87\x30\xb6\x7c\x46\x92\xc3\x53\xc0\xbf\xbd\xc9\x91\xea\xfa\xf0\xd4\x92\x7f\x88\xe4\x2f\x79\x59\xd0\xe6\x88\x1f\x02\xf9\x43\xb5\x35\xad\x0c\x08\x74\x0d\xc3\x78\x46\x92\x93\x69\xc1\x6e\xf7\xea\x3d\xe7\x95\xcc\x58\x45\x1b\x20\xf6\x80\x90\x4b\x2d\x06\x63\x3d\xaf\xe3\xd8\x3c\x8d\xbd\x49\xb8\x9a\x2c\x79\xf3\x3a\xcb\xd7\xa9\x55\x78\x56\x0d\x6f\x6b\xa3\xc3\xc0\x8f\xeb\x90\xe5\x71\xa6\x93\xa7\x4f\x11\x1a\x75\x21\x20\x63\xd4\x6d\x1f\x4e\xc7\xce\xf1\x21\x3e\xc5\x1b\x1f\x8f\x9b\x9c\xfd\x10\x1c\xc1\xca\xee\xcf\x52\xbe\xa6\xb7\x0d\xaf\x8e\x1a\xb6\x5a\xcb\xf8\x54\x19\x90\x82\xdf\x55\xde\x4c\xe1\x56\x71\x09\xd8\xaf\x14\x3d\xe4\x44\xd4\x59\x15\xed\x3c\x57\x7a\xfb\xe1\xa9\x6a\xa6\xf7\x21\xdd\x6e\x52\xd2\x6a\x25\xd7\x38\xe3\xaa\xf9\xa9\x37\xf1\x9d\x91\xb5\x25\xa8\x9e\x88\x72\x7e\x68\x39\x62\x79\x61\xa7\x25\xec\xc3\xa9\xa2\x9d\xbe\x7b\xb3\xce\x24\xdd\xf8\x8a\xab\x35\x64\x8d\x54\x4e\x54\xa7\x0a\xea\xd8\xc2\x04\x24\x96\xac\x33\x7c\x05\x4b\xf4\xdf\x21\xc1\x38\x16\xf9\x50\xd3\xee\x50\xb0\x86\x15\x58\xce\x0a\xaf\x50\xd1\xa1\x8b\x15\x62\x1c\xb8\xfa\xa5\xb6\x41\x6f\x2a\x9e\x56\x0b\x51\x1f\xfb\x70\x27\xd3\x92\x19\xae\x12\x4f\x9d\x0e\xc8\x9f\xb6\x65\xc8\x7f\xa3\x37\x77\x57\x21\xc0\x58\x48\x77\xb4\x69\xa0\xf0\x48\x2a\x79\xa6\x4f\x24\xa1\xed\xd6\xaf\xd3\xe4\x2b\xa3\xf7\x8e\x26\x6b\xb9\x29\xd3\x44\x7b\x02\x3c\x25\x3a\x6a\xb0\xa8\xe9\x45\x80\xc9\x0d\x7d\x10\x58\x67\xa4\x68\x3e\x27\xcf\xdd\xfc\xa9\xaa\x4b\xb3\x32\xaf\xec\x21\xa1\xfe\xc1\xca\x9c\x91\xcb\xab\xb1\x9b\x6b\x46\xef\xc2\x92\xde\xa6\x11\x56\x9b\xed\x43\x95\x1a\xb6\x1a\xae\xc1\xff\x96\xbc\x21\xa9\xde\x55\x59\xd5\x31\x9b\xbe\x4e\x23\x27\x35\x02\xa3\x85\x75\x89\x3f\xae\x46\xa3\x49\x56\xd7\xb4\x2a\x2e\xb8\xcf\xb5\xa0\xa7\x5d\x4c\x79\xe1\xf7\x9b\x4c\x10\x73\x32\x9a\x64\x45\x71\xa6\xc4\x35\x75\x7b\x52\x88\x78\xc1\xaa\xe2\x8c\x57\x92\xde\xcb\x1f\x68\xd5\x0a\xb4\xe3\xbb\x96\x14\x15\x79\x56\xd3\x3f\xab\x69\x14\xb2\xc1\xbe\xd0\x2c\x6c\xc8\x13\xf4\xb7\x91\x4f\x9f\x88\xfe\xd9\x56\x05\x5d\xb2\x8a\x5a\x6f\x8f\x96\x1d\x34\xe6\x52\xd8\xc1\xa6\xa7\xc9\x68\xa2\x7a\x05\x84\x28\x21\x3d\xe7\x4e\x12\xec\x37\xaa\x97\xc3\x53\xf5\x5f\xbd\x91\x24\x21\x91\x6d\xd5\x21\xd3\xe8\x41\x94\xcc\x49\xc1\xf3\x76\x43\x2b\x39\xc9\xc1\x4b\xf7\xba\xa4\xea\x57\x9a\x14\xec\x16\x39\x42\x27\xac\xaa\x68\xf3\xe7\x8b\x1f\xde\x92\xb9\x1a\x89\xe7\x31\xa1\x93\x7c\xcd\xca\xe2\x1d\x2f\xa8\x70\x4c\x9f\x93\xe7\xe4\x5b\x92\x24\x64\x16\x00\x5c\x3e\xbf\x9a\x54\xbc\xa0\x7f\xc9\xca\x96\xf6\x0c\xd2\x33\xdf\x5f\x19\x7a\x95\x42\x57\xe6\xa7\x4f\xe4\x17\xd8\x95\x66\x24\x49\xc6\x44\x6d\x23\xea\x2f\x25\x81\x3e\xca\x86\x0a\x6d\xbe\x78\x2b\xaf\xa1\xa2\x2d\x61\xe5\x59\x6f\x43\x9a\x6c\x78\x01\x27\xa6\x29\x6d\xe8\x86\xdf\x52\x23\x1d\x9b\x5a\x3e\x44\x6b\x2a\x7e\x94\x37\xbc\x46\x26\x85\xf8\xaf\xd1\x69\x11\x2c\xf0\x10\x60\xc1\x8b\x87\xa0\xda\x27\xbd\xa6\xcd\x92\x37\x1b\x20\xfe\x25\x2e\x0f\x6d\x76\x65\xf0\x6b\x4c\x68\xe9\x44\x2d\xd3\xba\xac\x3a\x75\x9b\xb6\xca\x33\x49\x13\xc5\x24\xaf\xbc\xa0\x25\x95\xd4\x3a\x26\xd4\xd4\x6f\xa8\x10\xd9\x4a\x09\x40\xf2\xb2\xa1\xe4\x81\xb7\x44\xb4\xfa\x8f\xbb\xac\x92\x44\x72\x38\xea\x34\x16\x75\xc6\xa1\x35\x17\xd8\x75\xe4\x5b\x4f\x23\x78\x92\xf3\x6a\xc9\x9a\x4d\xaa\x71\x8f\x46\x7a\x02\xad\xf4\x8a\x3b\x26\xf3\xb5\xa6\xd8\x50\x93\x67\x82\x7a\xa4\xcf\xf4\x6a\x0d\xac\xac\xe4\xe2\xc3\xc7\x77\x67\x2f\x2f\x5e\x93\x8b\x97\xdf\xbd\x7d\x4d\x3c\x1b\x79\xc0\xb7\x83\x24\xa9\xb2\x09\xf8\x5d\x46\x24\x2b\x69\x23\xfd\x92\x63\xcf\x4b\xe3\x64\xa5\x7f\x4e\x2c\x1a\x9a\xdd\x1c\x7b\xb4\x6a\x76\xc6\x29\x7d\xf5\xe1\xfd\x8f\xbf\x0d\x95\xc1\x61\xf2\xc5\xc4\xd3\xfb\x9a\x37\xd2\x12\x0f\x57\x01\xbc\xd9\x64\x92\xcc\x09\x2d\xb5\xe3\x14\x4b\x12\x8b\x05\xae\x1b\x16\xe8\x6a\xfd\x4a\x2f\xc7\x6b\x05\xba\xc8\x04\x35\xfb\x55\x00\xbd\x64\x25\x45\x15\x51\x35\x34\xfa\x83\x15\x1c\xf8\x85\xbd\xf8\xad\xd0\x5e\x9d\x93\x3b\x56\x15\xfc\x4e\xdb\x89\x1f\x3f\xbc\x49\x93\xf3\xd7\x6f\x5f\x9f\x5d\x90\x7f\x21\xdf\x7f\x78\xff\x83\xe3\x6c\xd0\x67\xdb\x94\xae\x6d\xc9\x73\xf0\x1a\x4f\xd6\x0d\x5d\x4e\x44\x5d\x32\x99\x26\x5f\x25\xa3\xcb\xe7\xa0\x26\x64\x35\x43\x8b\xfc\x5b\xa4\x62\xee\x08\x02\x2d\xc2\xd0\x8f\xe5\x66\x30\xaa\x06\x5a\x41\x31\x52\xab\xca\xae\x9d\x37\x1e\x6a\x42\x9f\xa4\x4f\xe3\x1d\xab\x88\x23\x52\xd9\x00\x69\xdb\x94\xca\xc2\x5d\x94\x59\x75\xe3\x58\x7e\xc7\xaa\xc9\x92\xe7\xad\x37\xd7\x7d\x41\x6c\x37\x75\x30\x93\x9f\xcb\x01\x14\x86\x6f\x81\x95\xf3\x60\x82\x7e\xaf\x21\xe5\xbc\x7e\xb0\x43\x52\x3f\x2e\xf8\x59\xc9\xea\x05\xcf\x9a\x02\x77\x41\x3d\x92\xc3\xc9\xe1\xe8\xf2\xc5\x55\x1f\xd3\x63\x6c\x2f\x55\x56\xb6\xde\x4a\x95\x3a\xb3\x73\x27\xfd\x75\x76\x4c\xd5\x15\x94\xc1\x1f\xbf\xea\x7e\xb9\x7b\x0f\xfa\xcb\x9b\xd7\xff\xd5\x76\xfe\x9f\x3b\x90\xde\x73\xcc\x4c\xfc\x0a\x1b\x90\x42\xf5\xff\xf6\xfe\xb3\x75\xb1\x2a\xf6\x7c\xd1\x5a\xfd\xc0\xcd\x52\x35\x8b\xf4\x56\x69\x8c\xd1\x75\x2a\x24\xaf\xaf\xd1\xb3\xea\x9b\x49\x76\x69\x0d\x2f\x55\xd5\x92\xc8\x35\x45\x2e\x7e\x9b\x04\xab\xaf\xbb\xa0\xf4\xe4\xd7\xab\xeb\x3c\xab\x72\x5a\x5e\x2f\xb2\xfc\x86\x56\x45\x0a\x72\xa0\xa8\x53\xd3\x30\x3a\x4e\x06\xd7\xd9\xde\xab\x4c\x2d\x25\xbc\x07\x4a\xc5\x1a\x39\x71\xcb\xe4\xc3\x8f\x59\x45\xcb\x31\x5e\x0c\x69\x3b\x65\xd4\xe7\xa0\xe0\x8d\x7c\xd9\x34\xfc\x2e\x2d\x58\x43\xbd\x6d\x03\x77\x12\xd2\x2d\x36\x13\xf9\xf2\xfc\xcc\xce\xa3\x31\x31\x9e\x7e\x75\xff\xcd\x1f\xbf\xfb\xe6\x38\xf1\x67\xfc\xd5\xeb\x41\xc8\x33\x03\xa9\x6f\x75\xbb\x50\x49\x9f\x5c\xb0\x03\x71\xe3\xd0\x8a\xf1\x18\x86\x70\xc6\xcb\x76\x53\xe1\xdf\xef\x9b\x82\x36\x70\x15\xc2\x78\x25\x9c\x10\x3c\xb1\x25\xdc\xc5\x30\x3c\x9a\xbb\x56\x23\x22\xa6\x72\x92\x69\xeb\xfc\xa0\xbb\x5d\x69\x74\xba\x7f\x33\x3d\xc8\x9c\x9d\x2a\xfd\x16\xa5\xfe\x44\x36\xa7\x27\xb2\x38\x7d\xfd\xe1\xc3\xfb\x0f\x33\xd8\x30\x82\x3e\xd0\x33\xa1\x80\xd4\x7f\x0c\xba\xce\x01\xe0\x53\xd6\xf0\x3b\x11\xb3\xf6\x7f\x0d\x22\xdf\x71\xd2\xd0\x9c\x37\x85\x20\x4b\xde\x56\xc5\xc9\x54\x16\x21\x61\x0e\xc5\x91\x22\x04\xdd\x5a\xdb\x3a\x0a\x6d\x6c\xb4\xa2\xa2\x63\xc4\x38\x98\xd2\xf9\x17\x55\x81\xea\xc3\xb9\xe7\x0d\x0b\x72\x90\x0b\xd1\xf7\x5e\xe5\xbc\xf4\x97\x7f\xce\x4b\x30\x43\x9d\x2c\x79\x97\xbe\xaa\x2b\xf0\xe4\xc8\xb5\x31\xa0\x41\xaf\x38\x42\xe6\x1d\xa9\xc6\x4a\x5a\x6e\x69\xcf\xf3\xa4\xaa\x9e\x91\x04\xfe\x4e\xa0\x8e\x2b\xe9\x84\x2d\xd7\xca\xaa\xaa\x3b\xf5\xa0\x9d\x43\xca\xad\x4e\x0b\x3c\xd2\x62\xb0\xf6\xdd\x4e\x84\xd0\x52\xd0\xcf\xa2\x78\x90\x54\x9f\x92\x6e\x3f\xda\xbd\x35\x9d\x92\x77\x5c\xdf\xb3\xab\x9d\x71\x93\xdd\x50\xd8\x19\x91\xdd\xa4\xe2\xd5\x91\x22\x58\x75\x19\xec\xc0\x76\x0f\xf1\x08\x3c\x35\x9d\x40\xcd\x74\x4a\x5e\x51\x49\x9b\x0d\xab\x28\xb9\x5b\xb3\x7c\x6d\x90\x6a\xd7\xb5\x80\x8e\x14\xed\x24\x93\xb2\x61\x8b\x56\x52\x68\x89\x3d\x80\x32\x80\x33\x48\xe6\x3d\x31\x80\xdb\xcb\xf7\xcb\xd4\x83\xf5\x3d\x25\xde\xb2\xe9\x09\x4c\xc3\xef\x7c\xdd\xae\x71\xb2\x06\x44\xbf\x2c\x0a\x92\x95\x20\x07\x6d\x56\x2a\x69\x44\x1a\xd7\xb4\xa1\xce\xc1\xc5\x08\xab\x88\x87\x89\x90\x46\xb3\xa1\xc0\xe9\xc8\x79\xa9\x5d\x96\x38\x17\x27\xe0\x35\x24\xcf\x7c\xb7\x51\xc3\xef\x2e\xd9\xd5\xc8\x39\xfe\x61\xed\x25\x81\x2f\x4a\x53\xa4\xc8\xc8\xcc\xce\x29\x25\xaf\xac\xc0\x87\xf3\xe1\xd1\x71\x7a\x92\x19\x91\x59\xc8\x8a\x2c\x64\x75\x74\x2f\xe0\x7f\x4e\x55\x9d\x08\xf9\xa0\xcd\xf9\x86\xdf\x1d\x61\xa1\x16\x28\xfc\x81\x83\xd0\xd0\x46\x11\xd1\x10\x70\xfa\x21\x80\x1a\x4a\x6f\xde\xae\x10\x56\x69\x41\xf3\xc3\xaf\x50\x1e\x35\x10\xb8\xd1\x71\xe0\x59\x6c\xd8\xb0\x09\xe0\x40\x1a\x68\xe7\x36\xcd\xe4\xd8\xc9\xef\xf0\x06\xa8\x04\x73\xab\xeb\x45\xf5\x60\x17\xc1\xf9\x9a\xdf\x91\xaa\xdd\x2c\x68\x43\xf8\x12\x7b\x6f\x68\x55\xd0\x86\x16\x84\x57\x20\xa9\x75\xb6\xa2\x07\x3b\x76\xc3\xd8\x86\xad\x99\x2b\xba\xae\x1e\x61\x1d\x5f\x17\xd9\x22\x35\x61\x60\xd3\x29\x71\x61\x4f\x44\xac\x79\x5b\x2a\x0a\xca\x07\xb2\xa0\xe4\x96\x09\xa6\x8c\x34\x5e\x21\x89\x32\x5b\xe8\x55\xc9\x0a\xf2\xc4\xdc\xbd\x5c\xab\x05\x46\x2b\x17\x8d\xa2\x48\xd6\x43\x0f\xdc\x58\x77\x4c\xae\x8f\x6a\xdb\x5b\xe2\x56\x90\x6a\x51\x65\xb7\xa4\x2d\x49\xc9\x26\x82\x96\x10\xa9\xd8\x6d\xef\xca\x0d\xa3\xd1\x75\xef\x6f\xff\x3e\x8c\x1e\x1e\x6d\x04\x13\x52\x11\x4f\xb0\x56\x8d\x94\x55\x92\x03\x9b\xb5\x3a\xab\x94\xb4\x06\x39\x3e\x14\x06\x27\xb3\x85\x89\x7f\x0b\xf8\xba\xe6\x77\xa0\xbb\x99\x98\x02\x64\x83\x17\x64\x10\xd5\xd3\xbc\xa3\xe7\xf2\xea\xd8\xfa\xb2\x71\xa5\x87\x0a\x1d\xcc\x6e\x5b\x89\x35\x5b\xca\xf4\xd2\x46\x75\x31\xb5\x92\x5f\x68\x3f\x36\xbb\x9a\xe8\xeb\x7b\xf3\x53\xb2\x0d\x15\x32\xdb\xd4\x57\x1d\x6f\xb3\xd3\x84\x7e\xd1\x5b\xa4\x98\x91\xcb\x04\xa2\xfb\x12\x13\x3a\x90\xd8\xe6\xc9\xd5\x18\x28\x98\x21\xb9\x7a\x25\x90\x8e\x3c\x69\x51\x30\x41\x12\xde\x21\xcd\xaa\xba\x05\x81\x65\x85\x35\xda\x3c\x09\xa9\x1b\x5e\xa7\x09\x6c\x1c\xaa\xdb\x65\x5b\x96\xbb\x8e\xf8\xc0\x1d\xda\x75\x93\xab\xe9\x08\x62\x56\xdc\x5d\xbc\x36\xd1\xfa\x1e\x60\xd8\x68\xac\x7e\xa6\x7e\xc4\xb4\x1f\xd4\xa3\x93\x1f\x4b\xaa\xf4\x53\x14\x25\x92\xa1\x1f\xe3\xc9\x80\xc6\xd1\x0d\xa0\xc1\x28\x81\xa8\x40\x44\xf9\x69\xa2\x66\x34\x76\x6f\xee\xf4\x11\xf4\xbb\xb3\xd9\x8f\xd8\xf9\x87\x60\xb5\x4f\xd0\xe7\xb2\xdb\x8f\x1d\xfa\xc7\x65\x39\x04\xd4\xfd\xae\xbc\xde\x16\x89\x65\x0e\x6d\x3f\x88\x2f\x3a\x03\x5f\xa7\xc9\x04\x95\x4b\x05\xd9\x6c\xf0\x08\x9a\x94\xac\x52\x02\x3f\x51\xc3\xf5\xb9\x8a\x33\x24\xb9\xcc\xca\x6b\xc1\xfe\x66\x3d\x31\x60\xd8\xba\xe2\x5e\x0b\x55\xdf\x6f\x60\x4b\x7b\xf0\xb0\xe4\xfa\x0d\x5c\x71\xaf\x05\xc4\x14\x9b\x73\xd9\xb5\x70\xc5\xbd\x16\xe0\xec\x61\xd5\xca\xc0\x27\x1f\xab\x9b\x8a\xdf\x55\x6e\xe2\x0f\x7c\xc9\xfb\x1e\x02\xe8\x8d\x38\x47\x26\x5a\x3e\xd4\xb4\x23\x2d\x6d\x5d\x64\x92\xea\xb3\x9d\x37\xa9\x3b\x77\x3d\xc3\xb6\x5b\x48\x76\x04\x38\x8f\xc9\x0b\xff\xb7\x80\x02\x3b\x38\xd4\x14\x27\xba\x19\x0e\xec\x85\xd2\x6f\x5e\x78\x72\x3f\xa9\x1b\x7a\x0b\xa8\xc7\x64\x52\xd1\x7b\xdb\x0d\x2e\x97\x82\x09\x35\x62\x38\x8b\xec\xdf\x71\x21\x8c\x90\x8a\x4e\x01\x9f\x5e\x37\x42\xa0\xab\x07\x22\xfa\x30\x6e\xd2\x34\x9b\x3a\xf5\xe4\xd4\x5d\xf5\x06\x03\x8a\x0e\xc2\x68\x39\x81\xa5\xb5\xbb\x59\x38\xf6\xc7\x01\x52\x34\xa9\x8a\x22\xf2\xf4\x69\x77\x20\xe4\x64\x68\x68\x1e\x19\xdb\xa7\x60\x90\xfa\xcf\x99\x39\xab\xe2\xe1\x2a\xd5\xe6\xbf\x11\x11\x8f\xc4\xce\x8a\xd9\x32\x60\xdc\xc0\x86\x2a\xc9\x0b\xad\x20\x46\x24\xb2\xcb\x23\xa5\x2e\xf3\x25\xc1\x08\xf2\x01\x39\x18\x3a\xf8\x94\xd6\x9b\x46\xbd\x48\xbf\xeb\xee\xac\xba\xe5\xb5\x74\x11\x7b\x10\xdd\x6f\x82\xe0\xc2\x8c\x01\x8c\xb4\xe0\x90\xef\x30\xb3\x00\xfd\x4c\x08\x1d\xe6\xce\x1b\x79\x8d\x3a\xe3\xcc\x77\x9f\xb9\x4a\xf0\x52\xcc\x88\xe3\x44\x10\x50\x87\x99\x40\x96\x2e\x83\x49\x09\x14\x56\x09\x3d\x44\x6d\x78\x27\xa3\xc9\x6d\x56\x5a\x2a\x6b\xa4\x30\x02\x8e\x3f\x43\x70\x38\x9e\x67\x21\xb8\x39\xb2\x01\xcc\x92\xa6\x8c\xde\xba\x2e\x1f\x34\x7d\xac\x5a\xa1\x09\xc4\x96\xc6\x87\xc0\x4c\x4f\xb4\xd0\x13\x86\xa0\x9a\x4e\xb5\xf8\x74\x01\xaf\x7d\x0d\xff\x4e\x99\xf1\x4a\xc5\xd7\xaa\xfc\x61\x72\x88\x4e\x76\xaf\xed\x33\x55\x6a\x22\x5e\x82\x5c\xa9\x4b\x8b\xf3\xca\x86\xa4\x26\xaf\x5e\x5e\xbc\x4c\xc6\x06\x07\x0c\x08\x53\x11\xae\x26\x3f\x71\x56\xa5\x09\x49\x8c\x92\xa2\x84\xe0\x32\x01\x1a\x20\x34\x07\xfe\xea\x29\x4c\x10\xa2\xae\xc3\x57\x21\x0a\x7c\xe8\xb4\xfe\x4c\x85\x27\x62\xee\xc5\x34\xaa\x01\x3f\xac\xcd\x49\x8a\xab\x69\x68\x70\x22\x4c\xf7\xa8\x83\xd3\xd7\x3b\xda\x3c\x95\xcd\x29\x5d\x61\x40\xc6\x42\x6d\x3d\xee\xb0\x33\xa1\x8d\x8a\x2b\x83\xba\x98\xee\x90\x16\xc1\x76\xe0\x56\x3e\x7a\xf6\xac\x4b\xc9\xeb\x9e\xc8\xf5\x04\x6b\x5d\xca\x90\x63\x82\x4d\x9e\x73\x15\xe8\xed\xb3\x49\x75\xce\x13\x63\xf0\xeb\xad\xc3\x2a\xb7\x3e\xb2\x00\x0e\xc7\xa7\xc6\x65\xb8\xe7\xe3\x8f\x80\xc2\x82\xf6\xf6\xf1\x3d\xf7\xbf\x81\x4d\xd3\xa5\x29\xfc\xae\x3b\x64\x54\xaa\x84\xa1\x2d\x71\xce\x9e\xbe\xc4\xef\x56\xf0\xbd\xa5\xe5\x86\x8b\xeb\xeb\x17\x1d\x43\x34\xa0\xc2\x91\x21\x55\xb9\x6f\x84\xfc\x5d\x56\x04\xb8\x2b\xe0\x9a\x27\xf5\x94\x41\xa7\x9e\xea\x5b\xae\xc9\x3a\x13\x3d\xb7\x8a\x8b\x27\x0b\x6f\x62\x87\xf9\xaa\xb1\x41\x80\x17\xe4\xa8\xba\xcb\xbe\x0e\x97\x9d\xca\x3f\xcc\xe5\x64\xd4\x1b\xce\x19\xe6\x09\x30\x5e\xf9\x63\x1a\xda\x39\x34\xa8\x9e\xa9\x9e\x25\xe3\x41\x6c\xb1\x5d\x86\xbc\x37\x37\xf4\x61\xc0\x7f\x53\xb7\x62\x9d\x5e\xde\x50\xe3\xa1\xb9\xa1\x0f\x57\x5b\x3c\x33\xce\x21\x6f\xfc\x33\xd6\x69\x0d\x21\xd7\x59\xd9\xd2\xc4\xc6\x49\x3a\xff\x8c\xbd\xaf\xdb\x77\xd3\x76\x62\xe3\x0c\xd4\x98\xcc\x04\xb7\x83\xde\xea\xe5\x41\x5a\xaf\xf1\xa6\xbb\x1c\x3a\x1d\x3e\xe7\xdd\x9e\x1a\xa2\xc1\x27\xab\xab\x6c\x21\x66\xe5\x25\x35\x2b\x6c\x11\xf8\x8c\x67\x24\x29\xb2\x6a\x45\x9b\xc4\xdd\x29\x0c\x4a\x5c\xa6\x69\xc5\xd1\xf4\xa6\xd8\x56\x6f\x99\xe0\xde\x21\xa5\xb6\x5e\xf3\x5f\x73\x1f\xb8\xb7\x93\x6b\x1f\x26\x37\xad\x4e\xc3\xd9\x22\xbe\x76\x29\x19\xc7\x70\x5b\x8d\xc9\x57\x3a\x1b\x67\x4c\xbe\xca\xc5\xed\x98\x7c\xf5\x93\xe0\xea\xc7\xfd\xa6\x8c\x29\xe7\xb2\x69\xa9\x5d\x63\x80\xf0\xba\x6e\xf8\xaa\xa1\xc2\xb7\xba\x0f\xc2\x20\x85\xaf\x27\xb2\x61\x9b\x54\xaf\x5f\xb8\xf3\xc7\x0d\xe1\x02\x82\x21\xc8\xa7\x4f\xc4\xd5\x41\xd4\x65\x3a\x72\x06\x14\x60\x19\xbc\x50\xfc\x82\x31\x40\x1a\xa7\xc7\xe6\xde\x30\xfc\x79\xe8\xec\xfd\xb1\x9c\xaa\x3d\x37\xdd\xff\x73\x34\xc7\x77\xc5\x61\xd7\xfa\xb0\x73\x29\xd4\x73\x7c\x01\xf2\xe7\x46\xf2\xb7\xfc\x8e\x36\x67\x99\xa0\xe9\xc8\xde\x76\x25\x26\xcf\x6b\x44\x9e\xcc\xc9\x51\x18\xe0\xbc\xfb\xe8\xf1\x2f\x95\x3e\xd0\x92\x67\x85\xc9\x0d\x20\x25\x13\x52\x75\x9f\x55\x0f\x72\xad\xf4\xed\xbb\x4c\x10\x0c\x0b\x2e\xa6\x18\x70\x54\x74\x48\xdc\x64\x32\x5f\xa7\xd3\x14\xa1\x3e\x15\x0d\xaf\x47\x7f\x15\x53\x36\x72\x54\x45\x82\x86\x62\x39\xa9\x4d\x5b\xbd\xc6\x81\xfd\xe7\x5a\xfb\xad\xd7\x5a\x3f\xbd\xf0\x9f\x78\xad\xed\xa9\x82\x61\x54\xda\x05\x4f\xd1\xeb\xe9\xce\xd2\xdf\x58\x02\x22\x6e\x82\x5f\x33\x8e\xcc\x45\x8b\x45\x32\x33\xf7\x0b\x1d\x33\x61\x63\x91\xa8\xb1\x43\x8c\x1a\x3b\x34\x79\x32\xdb\x15\xcd\x20\xa4\xec\xf1\xe0\x60\x3a\x25\xdf\x53\x99\xaf\xe1\xaa\xbd\xad\xd8\xcf\x2d\xc5\x60\x2a\x01\xf7\xeb\x78\x11\x88\x3c\xb6\x26\x3f\x5e\xc3\x62\x2c\x42\xa0\x0a\x7d\x84\xf6\x68\xf3\x08\x60\xbf\x4b\x56\x36\x86\x10\xa8\xa5\x6d\xe5\xa7\x82\x9a\xd9\x3d\xd4\x21\x5e\xaf\xde\x9c\x5f\xbc\x79\x77\x76\x41\xc0\x27\xe0\x39\x03\x30\xee\xef\xd0\xc4\xc8\x1a\x17\xc5\x2b\x26\xea\x32\x7b\x30\xe1\x06\x04\x2c\x33\x5a\x90\xc5\x03\x01\x47\x95\x98\x20\xdc\xc5\x9a\x09\x55\x52\x16\x64\x41\x89\x28\xf9\x1d\xe1\x15\x29\xb3\x66\xa5\x06\x29\x05\x51\x86\x39\xbc\x07\xa0\x06\x59\xd0\xfb\x89\xc9\x24\xe9\x10\x4d\xf6\x27\x79\x4c\xce\xde\x7f\x7c\x77\x91\xbe\x18\x91\x97\xe7\x04\xfd\x7c\xe8\x8c\x0b\xc7\xa2\x80\xc9\x9f\x3e\xbc\xff\xf8\x23\xf9\xee\xbf\xf7\x07\xfe\xfe\xc3\xab\xd7\x1f\x54\x8d\x8f\xe1\xd5\xeb\xf3\xb3\xc3\xcf\x3f\xbe\x7f\xcd\x1b\x9a\x81\x43\x74\xc8\x34\x7b\xb4\x42\x67\x2e\xfa\x69\xc3\x72\x78\xda\x41\x98\xeb\xfd\x25\xa3\x65\x11\xca\xd5\xf7\xaa\xe8\x5d\xbb\x39\x57\x70\xa1\x44\x0d\x4b\x11\xf0\x28\x7d\x31\x1a\x93\x0d\xab\xd2\x90\xa1\xaa\x30\xbb\xef\x17\x66\xb7\xab\x6e\x61\x4f\xe6\x20\xd1\xe4\xff\x22\x66\x47\x22\xfd\xc2\x6b\x13\xb8\x23\x31\x57\xf3\x5b\xad\xf3\xad\xa6\xb8\x8d\xae\x1c\x0a\x92\xc3\xe1\x38\xf7\x13\x31\xce\xc7\x0e\x6b\x22\xe1\x57\xc3\x0d\xfd\x73\xe9\xd1\x9d\x8d\x1e\x70\xd7\x6f\xaa\xa3\xee\xd0\x54\x21\x3a\x74\xe6\xd0\xee\x72\xa7\x78\xa4\x68\x11\x38\x99\x22\xdc\xa9\x55\x0d\x21\xea\x08\x32\x33\xc9\x9c\x3c\x3f\x26\x8c\x9c\x90\xee\xb0\x8f\x09\x7b\xf6\xcc\x0d\x5b\xdb\xc4\x64\xee\x00\x2f\x99\x31\x8e\x75\x16\x58\x89\x8e\x30\x4d\x16\x24\xa0\x29\xbb\x36\xd5\x06\x2d\x44\x1a\x5c\x3e\xbf\xd2\xbe\x7a\xfd\xe3\x38\xce\x9e\xee\x88\x31\x81\x2f\xa5\xe5\x16\xd5\x8f\x55\x4c\xbe\x86\xa3\xd3\x33\x61\xef\x1a\xa6\x65\xfc\xc2\x3e\x4e\x63\x3c\x70\xfa\x55\xad\x39\xc9\x72\x3a\x51\x3f\xd2\x24\x6f\x85\xe4\x9b\x40\x3f\xd4\xa7\xb1\xa0\xf2\x7b\x5e\xc9\x73\xf6\x37\x9a\xbe\xf8\x83\xef\xf1\x10\x54\x5e\xac\xe9\x86\xa6\x49\x96\xd3\xa9\x54\x7f\x4e\x25\xdf\xf0\xa6\xe1\x77\x49\x07\x50\xed\x19\x3f\x36\xac\x92\x3f\x64\xcd\x8a\x55\xa9\x53\x66\x7c\x85\x00\xce\xcd\x74\xa4\x1a\xfc\xc0\x0b\x8d\x58\x2d\x9a\x69\xbd\x12\x3f\x97\xc9\xb6\x06\x17\xd9\x02\x68\xfc\x66\x1b\xd0\x47\x41\xcf\xf9\x52\xc1\x8a\x54\xeb\xb1\x0e\x38\xe7\x9b\x4d\x56\x15\x02\x54\x1e\xfd\x77\x7a\xa9\x5f\x0c\x42\x8b\xbf\x69\xab\xc0\xe0\x5f\xb0\xaa\xf8\x77\xfa\xe0\x3c\x03\x77\xac\x9a\x91\xe4\x4c\x36\xe5\xd1\xeb\x4a\xd2\xc6\x1a\xfc\x9b\x2c\x57\x15\x88\x56\xd7\xe1\x8c\xea\x77\x73\xee\x69\xee\x3d\x9b\x83\x24\x79\xee\x16\x6b\x48\x7b\x72\x30\x26\x01\x71\x5a\x9d\xdc\x9f\xc0\x21\xe2\x3e\x93\x30\x6b\x77\x78\xa4\x5d\x05\x9c\xe5\x55\x9a\xe4\xeb\xac\x82\xdb\x4b\x8b\xca\xdf\x7d\x7a\xe2\xea\x45\x9c\x96\x34\x6b\x4c\x4c\x77\x1f\x2e\xd8\x3f\x62\x52\xef\x45\x84\xf7\xba\x1e\x7a\xc1\xab\x5e\xdd\xd1\x85\x61\x64\x54\x2f\x45\x0e\xb9\x80\xf2\xc7\xbe\xcd\x13\x7f\xc4\xcb\x47\x6d\x6f\x04\xb1\xc9\xd3\xa7\x24\xd0\x73\x4f\xdd\x06\xec\x56\x12\x92\x80\x1a\xe8\xb1\x5f\x07\x7c\x3a\x37\xc1\x60\x69\x24\xbc\x3d\x2b\x8a\xf3\x35\x6f\x64\xde\xca\x0b\xce\x4b\xc9\x6a\xe1\xf9\x4a\xab\xec\x96\xad\x32\x85\xa9\x15\xb4\x79\xb9\xa2\x95\x74\x46\xf3\xfb\x73\xf2\xdf\x92\x91\x4f\x91\x36\x63\xec\x66\x07\x9e\x2f\x75\xb8\x99\x2e\x66\xe4\x7f\xfd\x8f\xff\xf9\x0c\x25\xdd\x3b\x13\xad\xf9\xbd\xa3\xdd\xd0\x2d\xed\xae\x4e\x95\x68\x7f\x51\xaf\xd8\x30\x71\x7c\x9b\x4e\xc9\x9f\xa8\x04\xc5\xa6\xcc\x24\x15\x92\x34\x14\x1d\xf2\xcb\x86\x6f\xc8\x9f\x98\x5c\xb7\x0b\xf2\xf2\xc7\x37\x41\x22\xee\x5b\x00\xfd\x80\x90\xf8\x4c\x14\x2a\xf5\xc8\x38\xd9\x3c\x98\xb1\x28\x99\x48\x93\xb5\x94\xb5\x98\x4d\xa7\x59\xcd\x26\x2b\x40\xa9\x36\xa2\x69\x43\x6b\x2e\xa6\x82\x0b\x5a\xf0\xe5\x72\x0a\x52\x33\xd5\xfd\x8b\x29\xd2\xe3\x2f\x26\x5d\x15\x66\x4c\xe8\x42\x8c\x39\x7d\x32\x37\x99\xc0\x93\x5b\xda\x88\x20\xe0\xb5\x97\xa2\xf5\x11\xae\x9f\x48\x76\x9b\xb1\x12\xb2\xc6\xc8\xd9\x9a\xe6\x37\x44\x2d\x29\x8c\xc5\x47\xcc\x32\x5b\x5d\x9b\xe0\x18\xa5\x05\x9e\x64\x44\x2a\xa5\x5c\xce\x8d\x6d\xa3\x43\x57\xfd\x46\xea\x18\xbf\x56\x56\x1a\x44\xf5\x22\x1f\x4f\xa6\x99\x7b\x64\x40\xc7\x41\x58\xaf\xf5\x91\xa0\x52\xb2\x6a\x25\xc8\x04\xef\xc5\x8c\x2e\x60\x52\xbf\x26\xcb\xac\xa0\x6f\x2a\x2f\x7f\xca\xfa\x8c\xf1\xef\x1c\x1c\x2d\x41\x76\x42\xce\x2b\xc1\x4b\x3a\x29\xf9\x2a\x4d\xce\xb2\x4a\xaa\xd9\x23\x65\xd6\x9d\x67\x9c\x94\x99\xda\x07\x6c\xa6\xc9\xe3\x16\x7f\xfd\xb9\xa6\x35\xb5\xc1\xa8\x68\xa5\x09\xda\xdc\xd2\x86\xb0\x6a\xc9\x51\x5f\x03\xe1\xd8\xae\x8f\x69\x82\xfd\xcc\x1a\xb8\xd6\x80\x4a\x3b\x8b\xa6\xda\xf8\xa5\x40\x41\x87\x08\x74\x9c\x6f\x52\x73\x21\x57\x0d\x15\x44\x37\x39\xd8\xca\x62\x0d\x64\xc3\x5d\x6e\xd5\xdc\x05\x3d\xfa\x5e\x1e\xe8\x11\x85\x43\xe9\x57\x38\x3f\x42\x91\xd9\xb4\x55\xc5\xaa\x15\x50\x62\x22\xc1\xfb\x0b\x28\x1c\xf0\x8a\xc9\x6b\xb1\xce\x20\x39\xc9\x7b\x09\x2e\xba\xae\xfc\xc7\xe7\xdc\x0e\x1c\x3c\x5a\x15\xe5\xae\x32\x3b\x39\xa9\xb8\x54\x1b\x22\xc9\xaa\x07\xef\x9d\x50\xb6\x24\x77\xf4\xf0\x96\x92\x15\x07\x5b\x12\x26\x20\x3a\x27\xf6\x00\xf4\xe6\x66\x9f\x47\x15\x4e\x7d\xad\x5a\x4d\x16\x95\x7e\xff\x15\x59\x95\x7c\x91\x95\x6a\x41\x6a\x20\x57\x8b\x5a\xa8\xd5\x3e\xc1\x05\xb9\xe1\xb7\x14\x3c\x00\xf4\x9e\x09\x35\x83\x16\xde\xb8\xf4\x3d\x4d\xd3\xcd\xf7\xb5\x7b\xa6\xab\xf3\x8e\x06\x71\x11\xf3\xb0\x82\x35\x1a\x87\xa5\xab\x7f\x9f\x7a\x9a\xb6\xff\xe2\x44\xbc\xb3\x6e\x1f\x90\x27\x60\xb6\x19\x37\x56\x0d\x04\x2a\x7b\xf4\x02\x2a\x46\x0a\xbc\x40\x44\x21\xa7\xef\x10\x43\xe5\xf5\xaf\xcf\xa4\xd0\xec\x21\x6e\xcc\x13\x9f\x5d\xa1\xe1\x42\x22\x16\x4f\x00\xde\x35\x90\xbc\xb8\x7b\x8f\x00\x74\x0e\x79\xd8\xbb\x0f\x3c\xb8\x0d\x46\x36\xac\x5a\x79\xba\xbe\xda\x4a\x9d\x87\xad\x83\xb7\x6d\x4a\x13\x1b\x62\xdd\x51\x4a\xa5\x36\x0f\x7f\xba\x1d\x00\x1f\xdc\xd1\xa9\x6e\x3a\xca\xc8\xdc\xe8\xeb\x63\x53\x4d\x81\x77\xbd\x2f\x4a\x62\x52\x4a\x5d\x87\x42\xd8\x0e\xad\xfb\x0e\x3b\x84\x84\xc3\xac\x2a\xb2\xa6\x80\x27\x16\x5c\xa9\x58\x07\xb9\xc2\x6b\x2e\xa4\x46\x5c\xaf\xae\xd5\x2f\x87\xd1\xc0\xd4\xbc\xf1\x60\x20\x25\xb6\x07\xa3\x14\x19\x07\xa3\x7e\x45\xf0\xe0\xdb\x4a\x36\x55\xf5\x8c\x6f\x6a\x5e\xd1\x4a\xa6\x06\x75\x26\xc4\x1d\x6f\x8a\x80\x8b\x36\xa3\x96\x38\xfc\xc5\x22\x18\xb7\x8e\xf5\xe2\x8d\x8c\x5b\xd4\x7a\x00\xc9\x1f\xff\xf5\x0f\xdf\x84\x89\x1b\xe8\xc0\x4c\xcc\xae\x3d\x9b\x42\x0c\x2a\x0c\xe6\x19\x49\x66\x18\xd6\x85\x2f\x39\xfd\xff\xea\x07\xb0\xcb\xd6\x28\xb4\xcf\x48\x02\x6d\x30\x73\xf7\x5b\x21\x4a\xc5\x6b\x4c\xb4\x12\x65\x44\xb1\x82\xe7\x57\x95\xbe\x4a\xe6\xaa\x7b\xa7\xf8\x41\x21\xce\x00\xde\x97\xa8\x89\x0b\x20\x5e\x7c\xf3\x6f\x93\xe7\x93\xe7\x93\x17\x06\xc2\x1b\x3d\xa2\x7c\xfa\x34\x6c\xa1\xc9\x49\x46\x8a\x23\xfe\x0d\x0c\xe8\x04\xf3\x38\xc1\xde\xfd\xa8\x4e\x89\x6c\x9b\xd2\xb8\xa4\x60\x23\x81\xa4\x28\x7a\x2f\xc9\x86\x56\x2d\xd1\xa9\x10\x36\x3a\x05\xb6\x17\x9d\xa5\xa5\x6f\xa0\x3d\xff\x0a\xab\xd0\xbd\xf2\x67\x00\xf8\x81\x56\x6d\xff\xd9\x13\x97\x1d\xa3\x3b\x52\xfd\xe8\x5b\x6d\x91\xf3\x9a\x8a\x19\x49\xe4\x5a\x5b\x55\xa8\x06\xcd\x48\xb7\xf9\xb5\x6a\x65\xac\x33\xba\xe4\x0d\xf5\xad\xab\x31\xa1\xf8\x82\xcc\x58\x23\x08\xce\x8a\xd7\x15\x8c\x02\xc6\xa7\x76\x46\x88\xf0\x81\x23\x16\xca\xe1\xf2\x1e\x32\xb4\x79\x55\x3e\x4c\x3c\x45\x70\xd0\x19\x05\x53\x66\x23\x85\xbc\xcd\x95\x42\x2c\x27\xad\xe4\x2b\x4c\x47\xf5\x13\xd6\xe5\x9a\x89\x49\x5e\x72\x41\x81\x03\x5e\x8d\x9e\x19\xfd\x04\x77\xa0\x8a\xe9\x70\xb7\x4a\x59\x41\xde\x88\x35\x2b\xc7\x84\x86\x4e\x17\x85\x59\x81\xc2\xf2\x52\x7a\x26\x30\xc3\x1e\x1e\x3a\xb5\xdf\x40\xe9\x21\x65\x3a\x0a\xc2\x1f\x88\x4b\xb5\x06\x45\xd5\x66\xdf\x92\x48\xce\xf5\xd7\x86\x9c\x51\x10\x53\xe4\x0d\xd0\x26\x5f\x77\xf0\xa3\xfb\xfd\x1a\xdd\xef\x41\x1f\x43\xce\x75\x0f\x24\xee\x2c\xd4\x6f\xc0\x8d\xd5\xc4\xe3\xfc\x2a\x6a\x3a\xcd\xa2\xf4\x8e\x8d\xbc\x98\x0c\xc4\x6e\xbb\x0e\xdb\xd0\xd3\x9e\x8c\xac\x9c\x15\xda\x2b\x8f\x15\x5e\xd3\x3d\x38\x51\xb5\x9b\x6b\x70\x0a\xf7\xb8\x10\xba\x82\x7f\xc3\xf1\x93\xed\xe3\x8f\x0d\x02\x7f\x3c\xc6\xce\xe8\x4e\xe2\xdb\xb6\xb5\x5f\x0c\xae\xfd\x86\xdf\x7d\xf1\xc2\x07\x8f\xa5\x78\xbd\xa9\xe5\x43\x18\x72\x17\x44\x33\x05\x39\xc2\xa6\xd1\x77\x66\x83\xe8\xb6\x0b\xb6\x80\xb9\xdb\x02\xec\x74\x42\x42\x9c\xee\xf4\xd3\x27\xf2\xc4\xe1\xfa\x67\xdf\x25\xd0\x6b\xbb\xe7\x36\x81\xef\x66\x0c\x4b\x94\xc1\x8c\x5e\xef\xeb\x45\x0c\xbd\xce\xe0\x7e\x53\xdc\x6b\x6d\xa2\x23\xd4\x39\x2f\x93\xa0\x07\xdd\x00\x36\x92\xb0\x41\xf8\x8c\x87\x07\xfb\x4e\xd9\xe8\xa1\x14\x5c\x9b\x94\xe5\x75\x32\x9a\xd0\x9f\x53\x24\x61\xd4\x89\xa6\x3c\x08\x16\x69\x2c\x90\x59\x63\x0f\x7a\x75\xa0\x41\x10\xb3\xfe\x40\x42\x17\x54\x47\xe8\x69\x1e\x21\x83\x2c\x6e\x18\x65\xaf\x05\x44\xd5\x5b\xdc\xa2\x5d\xd8\xa7\xe2\xfb\xeb\x38\xb8\x48\x61\x55\xa1\x2f\x59\x5f\xe9\xd7\x4f\x3a\x87\x7d\xe4\x71\x94\xfe\x92\x77\x0b\xbc\x0b\x7e\xad\x81\xfd\xd5\xfe\x7b\x89\x73\xe7\xa5\x18\xf2\xf7\xbd\x72\xb4\xdf\x23\x29\xe4\x73\x1f\x4a\x21\xf1\xc7\x52\x7a\x2b\x69\xfb\x9c\x9a\xc9\xd4\x2f\x40\xe2\x2d\x99\x67\x22\xa1\xa8\x74\xaf\x42\xf4\x94\x6b\xc1\xb3\x53\xe8\x1e\x98\xe4\x55\x9a\xdc\xd0\x87\xb6\xf6\x5c\x6d\x24\xb5\xb3\x16\xb8\xa6\x83\x2e\xcc\xa4\xa9\xbe\x6f\x33\xbc\x21\x52\x1b\xa2\x36\x10\xd0\x52\xf3\x1c\x28\x1f\xa8\xa0\x92\x08\x9a\x35\xf9\x9a\xf0\x8a\xbc\x3e\x3f\xb3\x4a\x34\x9d\xdc\xd0\x87\x33\x6d\x2b\x7d\xf3\x6f\x6a\x3b\x06\x94\xa1\x8b\x04\x62\x57\x3b\xc3\x3f\x1e\xf6\x51\x80\xed\xa7\xc8\x3f\xd2\xa3\x3d\x72\xab\x29\x8c\x17\x99\xf8\xef\xb1\x0e\x3e\xfd\x08\xd0\x5d\x2e\x47\x5c\xef\xc4\xf3\xbd\xeb\xbc\x00\xec\xff\xbb\x07\xb5\x8b\xa4\xb7\x59\x69\x3e\x42\x40\xbe\x09\x7d\xeb\x5b\x48\x86\x0b\x86\x92\xe5\x37\xbe\x4b\x94\x06\x61\xbd\x11\xd6\xf4\x62\xa8\x22\x70\x76\x5f\x18\x16\x12\xd8\xd9\x5c\xfe\xb6\x0b\x07\x9b\x78\x4f\xb9\x76\x23\x81\x87\x46\x62\x9c\x05\x3e\x5d\x31\x3e\x79\x8f\x83\x6f\xed\x94\xfa\xef\x36\x90\x94\x15\xf7\xee\xd1\x2e\xad\x25\xd8\x0d\xa7\xf4\x4c\x5b\x1d\xc0\x0e\xa5\xb1\x23\xc1\x46\xaf\x1b\xab\x4e\x13\x74\xe2\x9b\xb8\x0a\x75\xe8\xfe\x08\x3d\x25\x50\xdf\xf5\xa6\x44\xbe\xb6\xf0\x1f\x2d\x97\x54\x47\xc1\x81\x99\x06\x2c\xc0\x17\xec\xec\xfd\x85\x7c\xa8\x29\x5f\x6a\x55\x71\x8e\xce\x86\x86\x55\x2b\x78\x5a\x19\x9f\x3e\xb3\x16\xe8\x04\xee\x32\x9c\xe9\x09\xde\x0c\x87\xde\xfa\x00\x04\x99\x13\xff\xd5\x34\x7c\x35\xdd\x2d\x29\x72\x79\x98\x1c\x8e\xe3\x4d\x2f\x9f\x5f\x8d\xc9\x61\x32\x19\x06\x78\x01\x00\x87\x3a\xc7\xe5\xf0\xd0\x7a\xab\x35\x72\x1d\xb5\xd0\x3b\xbf\x82\x67\x55\x61\x04\x51\xfb\xf5\x58\x57\x44\xcf\x3b\xbb\xa2\x82\xb7\xa2\xdb\xb2\x2b\x32\x29\x2b\xba\x02\xa3\x9f\xa9\x0e\x64\x43\x6f\x0e\x9e\x70\xf4\x1f\xb3\x76\xfe\x31\xd5\xae\x7f\xa8\x86\x07\x2b\x66\xdf\xc6\x8e\x53\xe2\xeb\xdc\x4e\xe0\x75\x7e\x8b\x05\xd9\xeb\xd4\x25\xee\x12\x1f\xfe\x75\x0e\xde\x10\x08\x45\x8b\x60\x5e\x47\x5c\x26\xad\x2e\x06\x17\xfe\xc8\x1b\x56\x74\x4c\x48\xef\x85\x25\xff\xcd\xb6\x2c\x77\x2f\x48\x98\x7f\x7b\xbd\x65\xea\x1a\x3c\x9a\xc3\x72\xd4\x73\x48\xbb\xf9\x80\x57\xcc\x3f\x77\x3a\x54\xa3\xcf\x98\x0c\x7c\x29\xfd\x9f\x72\x2e\xbc\xb7\x10\x3f\x6f\x2a\xfa\x5b\x9b\xe4\xab\x55\x49\xcd\xc2\x3c\x87\xd3\x7f\xbb\x2a\x8a\x2d\x5c\x3e\x89\x3d\x8e\x50\x75\xf0\x21\x08\x09\x03\x34\xc1\x77\xd4\xe9\xcb\xb9\xd4\xe1\x24\xa8\xea\xd6\xf8\x54\x7b\x88\xd1\x97\xab\x20\x20\xad\x27\x53\xd6\x43\x9a\x14\x54\xc8\x86\x3f\x0c\x54\xff\x42\xd0\x40\xe6\x6d\x93\xd3\x19\xc1\xa4\x03\x28\xda\xb0\xea\x2d\x38\x45\x67\xe4\xb9\x2e\x52\x87\x81\x12\xa2\xac\x2c\x13\x5d\x94\xb5\x92\xe3\x8d\x76\xf0\x95\xa3\x25\x93\x17\x5c\xbf\xb4\x3c\x83\x78\x64\xa7\x24\xf4\x28\x2c\x39\xbf\x01\xbd\xc5\x4f\xd0\x41\x28\xa5\x35\x40\x29\x6f\x65\xa8\x38\x68\x2d\x3f\x3a\x3d\xfa\x1a\x0e\x31\x2c\x97\x1e\x0a\x5f\xa9\xf8\x3a\x35\x2f\x43\x8f\x26\x0d\xcd\x8a\x87\x6e\xf0\x81\xb3\x7b\x6c\xe6\x5f\xa0\xc6\xc0\x40\xbd\x26\xfd\x24\x35\x35\xc5\x56\xc0\x1c\x3a\x2f\xe5\xab\x83\x30\x8e\xce\x4b\x5f\x3b\x8e\xa0\xb3\xaf\x75\xec\x49\x9d\x7d\xa2\x24\x4e\x5d\xf0\x1a\x45\x5c\x6f\xeb\x0e\xd6\x3d\xc5\x71\xdc\x43\x67\xdf\x66\xd9\x4d\x5d\xf8\x9e\x4d\x9c\x3a\x93\x24\xd6\x45\x36\x84\x4e\x27\x0e\x1d\x13\x32\x34\x58\x9b\x99\x15\xe2\xec\xa1\xeb\xe5\x7b\x1d\xf7\xd1\xb9\x14\xa0\x2e\x81\x3d\x74\x9d\xcc\xa6\x63\x42\x42\x9f\x56\xdb\xa5\xa8\x17\x1a\x13\x46\x1c\x79\x6d\x5d\x28\xc5\xae\xf6\x41\x60\x90\x7f\xeb\x25\x6e\x77\xb5\xb6\xe1\xe4\x09\x00\x77\x10\xfc\x24\x7a\x1c\xdd\x82\x01\xa1\x3b\x28\x30\xc8\x7e\x4f\x0c\x00\x7c\x1c\x75\x0b\x76\x90\x24\xb2\x89\x9a\x20\x41\xa2\x6c\x33\xf4\x1a\x94\x35\xba\xd0\x54\x8c\xbf\x00\x15\x50\x11\x9a\xaa\x96\x8a\x49\xff\x03\x1e\x03\x64\x61\x4f\x75\xd6\xc0\x8e\xa2\x4f\x8f\xe8\xd3\xfe\xfb\xf7\x0a\x86\xc2\x6f\xdf\x5d\xc9\xa2\x9d\x84\xcf\xdd\xcf\x3b\x49\x7b\x86\x04\xa7\x0e\xd8\xd4\x3d\x88\x93\x0d\xeb\x55\x59\xa2\xed\x51\x2f\x7d\xc3\x59\x5b\xdd\x3c\x0a\x3f\x03\x3a\x36\x91\xdd\xfa\x5d\x0f\x93\x38\xb8\xf0\x71\x80\x31\x89\xa6\xff\xbb\x3b\xfc\xce\xc3\x3a\xa6\x14\x3d\x48\x43\x5f\xd7\x94\xd9\xc2\xf7\x23\xe9\xa7\xe5\x83\x43\xca\x39\x93\x22\xe7\x91\xad\x0b\x3c\x37\x3e\x1a\x77\x38\x45\x10\xf9\x27\xd1\x6e\x54\xfe\x49\x12\xa7\xca\x3b\x38\x76\xa3\x33\xe7\x5c\x04\x95\x3b\xd2\xe2\x68\x3a\x2f\xb6\x0e\xb3\x26\x7e\xb3\xd0\x95\x6b\x08\x33\xd6\x4f\x07\x46\x25\x3c\xe6\x75\xb7\xd6\x99\x7d\xba\x35\x94\xe4\x50\xa1\xd5\x6e\xa9\x56\xfb\x84\x7d\x40\xed\x7e\xd5\x18\x77\xbc\x29\x8c\x0b\x75\xd7\x96\xb8\x8e\x8e\xa2\x93\x9f\xed\xb4\xa0\x78\x86\x76\x10\xe4\xd4\x7b\x5c\x20\x1c\x83\xff\x14\x40\xf8\xdc\x40\x0f\x54\x3f\x05\x80\x0e\x83\x97\xe7\x67\x09\xf9\x56\x3f\xd3\x4b\x66\x58\x60\xd7\x7d\xbf\x59\xe4\x69\x87\x7d\x5f\x12\xd8\x22\x08\xc5\xa2\x74\xbc\x2b\xc8\x29\x29\xd8\xed\x50\xdc\xac\xa1\x6a\x9d\x89\x34\x51\xb6\x4e\xd6\xd0\x2c\x89\x07\x1b\x45\xbc\x80\x4e\x0e\xe6\xe1\x67\x4e\x2c\x5e\xf8\x7e\xca\xc8\x0b\x3d\x33\x8f\x49\x1b\x74\x3e\x2a\x43\x80\x0e\x88\xb7\x3f\x21\x26\xde\x6c\xb3\xf4\x5e\xa6\x88\xc4\x14\x79\x99\xc4\xbc\xd9\xc0\x67\xb9\x1a\x5e\xba\x26\x39\x3e\xb4\x58\x80\x20\x19\xca\xbc\xc2\x91\xef\x77\x00\xd4\xc6\x5d\xf3\xd7\xca\xe3\xc5\x9c\xfc\xc1\x31\xc3\xd0\x86\x78\xd6\x94\xad\xd6\x90\x48\xfd\xcd\xf3\xe7\xf5\x7d\xd2\xcd\x4a\xf0\x98\x61\x1a\x6a\x0a\x36\xd9\xfd\x51\xb4\x75\x30\xbd\xcb\x86\x8a\xf5\xb5\xfe\xb8\xe6\x0e\xb5\xa3\x97\xf0\xe9\x63\x0a\xae\x3d\x14\x1a\xbc\xfa\xd8\x7b\x9b\xd8\xf7\xd4\xb1\x53\x9a\x77\x57\xd8\x92\x55\x45\xfc\x22\xc8\x5b\x70\xf6\x4d\x9c\x68\xab\xf0\x61\x1b\xd7\x0a\xa3\xa0\xfd\x48\x26\xbf\x71\xf0\xcc\x8d\x3f\xe3\xba\x33\xfb\x5c\x4d\xf8\xd0\xcc\x95\xf3\xfa\xc1\x2b\x33\x10\xc4\x6c\x43\xad\xbb\x5e\xf4\xce\xc3\x1b\x35\xcd\xd9\xd2\x3c\xa1\x43\x82\xb4\x1d\xb3\x9e\x7c\x41\x89\xef\xfb\xbe\xfb\x1a\x82\xe0\x8f\xf4\x37\x9c\xb7\xcb\x81\xff\x45\xe4\xba\xe1\x9b\xba\x4f\x57\x46\x2a\x7a\x87\x61\x1f\xe6\x73\xd3\xe1\x4b\x48\x3e\x9b\x10\xd7\xd3\xa7\x1a\xe9\x69\xef\xe3\x4e\xfb\x68\x22\x24\xfc\x10\xb6\xf9\xf0\xf2\xe7\x1c\x7b\x5d\x21\x88\x27\x07\x50\x9f\x0d\x91\x1b\x14\x6f\x60\x97\xf8\x31\xf2\xb1\xf7\xe5\x71\x37\xe9\xb7\x59\x39\x52\x63\x0d\x53\x99\x7a\xca\x13\x3a\xa9\x7d\x57\xfe\x60\x68\x5f\xb7\x69\xcc\x7f\x1d\xbc\xd4\x05\xd7\x0a\x47\x2e\xef\x69\xeb\xac\x7f\x9e\xaa\xb7\x53\xe8\xa6\x53\xf2\xb2\x95\x7c\x93\x49\x96\x67\x65\xf9\x40\xea\x86\x2e\x59\x59\xea\x74\x3d\x10\x6b\xb6\x24\x4c\x1e\x0a\x88\x88\x15\x54\x92\x07\x2a\x0f\xa2\xb7\xbd\xbb\xd3\x38\x06\xee\x7d\x47\xdd\x65\xb6\xe7\xfd\xf0\xe0\xdd\x70\x70\x59\xd8\xd1\xaa\xfc\x64\x33\xff\x45\xb7\x9d\x8b\xcd\x44\x4b\xef\xf5\x45\x74\x7d\xe0\x71\x99\x95\x64\xe7\x47\xd4\x45\xe0\x19\xc7\x46\xa7\x24\xc8\x0b\xd8\x67\x11\x1a\x02\x9f\xf9\xcb\x71\xf8\xd1\xa6\x20\x96\xc3\x6b\xab\x26\x03\x68\x08\x83\x69\xb5\x69\xb6\xf3\xc5\x42\x77\x15\x7b\xd0\x45\x7e\xda\xdd\x52\xf6\x79\x46\x70\xeb\x04\xfa\x08\x7e\xcd\x09\xdc\x4d\xfa\x3e\x13\x71\xe4\x4d\xc4\xbe\xef\x0e\xee\x9c\xb4\x01\xde\x76\x3f\xc6\xb7\xef\x74\x75\x23\x8e\xfb\x3e\xe5\xed\x97\xa7\xbd\x97\x66\x4c\x4b\xd1\x7d\x6a\xa6\xa1\xa2\xf6\x14\xac\x2d\xce\x53\x12\xf7\x46\x03\x02\x4d\xb5\xdd\xc4\xec\xc7\xe1\xfa\xae\x6e\xdc\x8b\xd2\xf8\x99\x11\x08\x43\xcc\x4f\xee\x5c\xc4\x2b\x2a\x5f\x06\x76\xb9\xcf\xf7\xa7\x4f\x89\x37\x05\x71\x5c\xa8\x92\x38\x25\xa2\xf7\xe1\x67\xb4\xbb\x8b\x05\x7c\x5c\xba\x58\xcc\x2c\xca\x61\xfe\xd9\xaf\x57\xd4\xdd\x8c\x03\xa7\xa7\x78\xb5\xc7\xc4\xfb\xe7\x01\xfa\x4a\x3f\x71\x1f\x9a\x24\x18\x75\x11\x44\x80\x34\xf0\x9c\x49\xff\x1b\x44\x8f\x5d\x07\x60\xc1\xe4\xb0\x6b\x33\x7a\x2e\x68\x68\xf7\xe5\x46\x2d\x8c\x25\x17\xbe\x97\xb4\x1f\x04\x4f\x42\xe5\x2a\x96\x65\xd3\x75\x2e\x76\x70\xee\x43\xde\x1e\xdf\xbb\x71\x9f\xee\xfe\xb6\x67\x8d\x7a\x5f\x03\x8f\x64\x18\x6e\xa3\x9b\x0c\x7c\xfe\x69\x3b\x73\x82\x1b\xf5\x5d\xe3\x77\xcd\x76\xe9\x1d\xd1\x6c\x04\xd7\x57\x3c\x6d\x01\xd3\x0b\xf6\xd1\x0c\xfc\xf0\x97\xe1\xf0\xf2\x8e\x80\xc4\xd2\x0a\xcc\x16\x37\x74\x7c\xd8\x8c\x81\xed\x54\xf9\xb6\xef\x90\x6a\x89\xc6\xe4\xf6\x60\xf8\x0e\x4c\x2f\x1c\xfe\xef\x1e\xd1\x8e\x1c\x8d\x7d\x34\xca\x9d\x18\xbe\xd0\x89\xe9\x3b\x15\x2d\x9c\x97\x29\xd2\xf5\x25\x82\x83\xd8\xf7\xfd\x75\x93\xd0\xa0\xde\x86\x1a\xf9\xbb\x40\x0c\x5a\x27\x93\x58\x78\x7f\x61\xc4\xe0\xc5\x7a\x10\x34\xdc\x27\x35\xb5\x26\x59\x65\x5f\x7a\x77\xf5\xdf\xa5\x77\xd7\xf8\x3e\x97\x5e\xb1\xfe\x47\x20\xb5\x0b\x1a\xfa\x9d\x86\x36\x12\x3f\x63\x6a\x0f\x93\xd0\xc4\x27\x85\x9e\x02\x3f\x41\x27\xf8\xca\x7a\x12\x71\x20\xea\xc0\x27\xdb\xf1\xa5\x82\xbe\xea\x26\x35\xb2\xa5\x97\x50\x57\x95\x0f\x64\x9d\x09\x08\xa1\x14\x60\x26\xe9\x48\x69\x49\x37\x13\x55\xf8\xf4\x29\xb1\x7f\x3f\xe9\xd9\x3c\xf1\xbc\x2c\xd3\x20\xdc\xfd\xbd\x7d\x02\x17\xcd\x68\x02\x4b\x7c\x6b\x54\xdf\x74\x4a\xbe\x57\xd6\x1d\xab\x2c\xd1\xf0\xf8\x8f\xc1\x45\x4c\x96\xa7\xdd\xf3\xc3\xfc\x2a\x0c\xd8\xe2\xc2\x7f\xdd\x3d\xcc\xae\x02\x08\xf5\x3b\x84\xf0\x72\xab\x70\x38\xc2\xba\x44\xe3\x69\x54\x88\x47\x97\x85\x90\x36\x8f\x8a\x99\x20\x57\xa5\x6d\x79\x30\xd1\x3d\x14\x63\xc9\x44\xe9\x6f\xdf\xba\x6c\xad\xa6\xc5\x4f\xc8\x34\xe5\x71\x3f\xa9\xea\x42\x88\x75\x8f\x2f\x42\xac\x7d\xde\x38\xc0\x2e\x7b\x14\xa0\xc7\x22\x07\xd8\xe5\x92\x02\xf4\x38\xe5\x61\x8c\x30\x0b\xb0\x86\x0c\x73\x0d\x6e\xe8\x43\x17\xf6\x86\x3e\xc4\xc0\x06\x71\xdf\xd0\x87\x28\xfa\x80\xdb\xeb\xae\x18\x0e\xbc\xf8\xd2\xe1\x5f\x92\x6c\x63\x5a\xa4\xd6\xe3\x54\xac\x6d\x38\x84\x08\x84\xe3\x47\xbc\x72\x3b\x8a\xfd\x36\xe1\x2e\x73\xcc\x69\x11\xe1\x50\x7c\xcb\x5b\xf2\x66\xf3\x65\x6e\x5c\xbb\x87\xe1\x01\xde\x73\xb4\x6a\xcf\x13\xaf\x69\x75\x14\x3e\x0f\x4b\x6c\xda\x63\x93\x6d\x84\x77\x3d\xda\x36\xe5\x2c\x9e\x67\x6a\x8c\x00\xbb\xac\xb0\xad\xda\xb2\xe2\x99\x8d\x03\x29\xd1\x5f\x9a\x6e\xda\x4b\x11\x25\x9a\xfa\x4b\x28\xbd\x22\xc1\xbf\x39\x79\xd1\x07\x42\x59\xbc\xf2\x80\x22\x42\x6a\x27\xd6\x6f\x08\x62\x1a\x6b\xd8\xcf\x38\x0d\x1b\x82\x04\xc7\x1a\xf6\xd3\x50\x3b\x3d\x1a\xc9\xbc\xea\xf6\xd8\x49\x44\x8d\x35\x56\x72\x7f\x45\x22\xbd\xba\x05\x31\xd4\xce\xef\x78\xbe\x6d\xad\x8c\xfc\x89\xed\x88\x34\x18\xa3\x9d\x95\x62\x3e\x58\x10\x7d\x48\x52\xbf\x31\xa0\x1d\xdb\x77\x19\x93\x93\xc9\xc4\xaa\x96\x7d\x4b\x5a\xf7\x05\x5f\xb8\x50\xe4\x0f\xda\xcf\x43\xbd\xe2\xcb\x43\xba\x5b\x2d\xec\x49\xe8\x2b\x8b\x1b\xdd\xd6\x86\x25\xf3\x30\x75\x69\x90\x09\xf8\xde\x93\x43\xd6\xd1\x8c\x4c\x84\x60\xb0\x79\x86\xfd\x28\x16\xb9\x6e\x3a\x37\x43\x43\xdd\x47\xed\x44\x0b\x37\xf0\xed\x5f\x20\xb3\x5b\xd9\x69\xbb\xc1\xb0\x9f\xd8\x28\x02\x33\xd4\x7f\x90\x4a\x15\x47\xdf\x9d\xb1\x9f\x8d\x83\xbc\x07\xfc\x5e\x19\x3c\x0b\x21\xd7\x54\x6d\x46\xe6\x4b\x96\xf4\xe7\x8f\x90\xaf\x52\xd1\x3b\xf2\xf1\xc3\xdb\xb4\xe3\xb5\x80\x0e\xe0\x8e\xd7\xe4\xa2\xc0\x97\x0e\x55\xa3\x09\xfa\x67\x7e\xc4\xdd\x0a\x1e\x5b\xd1\x40\x89\xcb\x4c\x77\xcd\x9e\x3e\xf5\x70\x04\xba\xdb\xd0\xc7\xd3\x5c\x42\x4c\x32\x76\x6d\x35\x67\x34\x9d\x3a\x40\x0d\x9e\xca\x3e\x97\x99\xa4\xe9\x2f\x8f\x63\x62\x62\x03\xf1\x1b\x7e\xe3\x9e\x2b\xa6\xce\xe4\xda\xbd\xd3\x7f\xf0\x79\x6f\x7a\xfb\xeb\x60\x48\x9c\x07\x85\x79\x97\xb7\xa2\xb3\x89\x63\x38\xc0\x1e\x8f\x22\x0c\x49\x75\x5c\xa6\xbf\x5c\x52\x23\x72\xea\xad\xec\x27\xd0\xd8\xcc\x5b\xc9\xf3\x9b\xce\x83\x12\xbb\xc7\x16\xc9\x12\xdd\x92\x80\x74\x7c\xa0\x60\xfe\x77\x00\x00\x00\xff\xff\x5c\x35\xc2\x60\xcb\x98\x00\x00") func staticJsAppJsBytes() ([]byte, error) { return bindataRead( @@ -363,8 +371,8 @@ func staticJsAppJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/app.js", size: 38933, mode: os.FileMode(0644), modTime: time.Unix(1562365567, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xd9, 0x24, 0x92, 0x34, 0x81, 0x7c, 0x2a, 0x73, 0xc3, 0x1b, 0xf6, 0x5, 0x47, 0x73, 0x43, 0x33, 0x6, 0x22, 0x77, 0xa0, 0xc, 0x5f, 0x46, 0x3d, 0x26, 0xe2, 0xb1, 0x74, 0xaf, 0xcc, 0xd1, 0xf6}} + info := bindataFileInfo{name: "static/js/app.js", size: 39115, mode: os.FileMode(420), modTime: time.Unix(1572716502, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -383,8 +391,8 @@ func staticJsBase64Js() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/base64.js", size: 3517, mode: os.FileMode(0644), modTime: time.Unix(1562354498, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xea, 0xb5, 0x65, 0xce, 0x8e, 0x97, 0xd3, 0xea, 0xba, 0xce, 0xc9, 0x6c, 0xee, 0x24, 0x8c, 0x9d, 0xd1, 0x3b, 0x3e, 0x75, 0xa1, 0x61, 0xe6, 0xdb, 0xed, 0x9b, 0x7c, 0x2, 0x88, 0xc7, 0x74, 0xcd}} + info := bindataFileInfo{name: "static/js/base64.js", size: 3517, mode: os.FileMode(420), modTime: time.Unix(1569462078, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -403,8 +411,8 @@ func staticJsBootstrapContextmenuJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/bootstrap-contextmenu.js", size: 5300, mode: os.FileMode(0644), modTime: time.Unix(1502760883, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xf2, 0x60, 0x1f, 0x8b, 0x44, 0x4c, 0x50, 0x52, 0x7f, 0x6b, 0xe4, 0x1c, 0xce, 0x22, 0xb7, 0x3, 0x7, 0xb0, 0x28, 0xe4, 0x8f, 0x8f, 0x87, 0xe3, 0x1a, 0x37, 0x41, 0xa6, 0xcc, 0xd9, 0x1e, 0x5a}} + info := bindataFileInfo{name: "static/js/bootstrap-contextmenu.js", size: 5300, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -423,8 +431,8 @@ func staticJsBootstrap3TypeaheadMinJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/bootstrap3-typeahead.min.js", size: 9335, mode: os.FileMode(0644), modTime: time.Unix(1509762282, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x62, 0x1c, 0xd9, 0x22, 0xdd, 0xc5, 0x3d, 0x28, 0x77, 0x64, 0x36, 0x45, 0x94, 0xa9, 0xb9, 0xc0, 0x55, 0x13, 0xf1, 0x29, 0x8a, 0x93, 0x7f, 0xa, 0x44, 0x81, 0x68, 0xcf, 0xae, 0x67, 0x77, 0x17}} + info := bindataFileInfo{name: "static/js/bootstrap3-typeahead.min.js", size: 9335, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -443,8 +451,8 @@ func staticJsJqueryJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/jquery.js", size: 84245, mode: os.FileMode(0644), modTime: time.Unix(1467073917, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x87, 0x47, 0x6, 0xb2, 0xb1, 0x31, 0x1a, 0x7, 0x19, 0xb5, 0x26, 0x7f, 0x7d, 0x1c, 0xf8, 0x3, 0x5, 0x7e, 0x36, 0x7e, 0x94, 0xae, 0x1f, 0xf7, 0xbf, 0x78, 0xc5, 0x45, 0xd, 0x30, 0xf5, 0xd4}} + info := bindataFileInfo{name: "static/js/jquery.js", size: 84245, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -463,8 +471,8 @@ func staticJsThemeTomorrowJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/theme-tomorrow.js", size: 2556, mode: os.FileMode(0644), modTime: time.Unix(1512708940, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xaf, 0xb6, 0x90, 0x87, 0xbd, 0xf7, 0x90, 0x2f, 0xe8, 0x54, 0x22, 0x62, 0x13, 0x2, 0x71, 0x40, 0xd1, 0x54, 0x19, 0x8b, 0x8, 0x71, 0x82, 0xa6, 0x87, 0x5d, 0x14, 0x6b, 0xfb, 0x10, 0x41, 0x69}} + info := bindataFileInfo{name: "static/js/theme-tomorrow.js", size: 2556, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -483,8 +491,8 @@ func staticJsUtilsJs() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "static/js/utils.js", size: 831, mode: os.FileMode(0644), modTime: time.Unix(1509582243, 0)} - a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x89, 0x40, 0x16, 0x6a, 0x65, 0xd3, 0x18, 0x6d, 0x52, 0x8f, 0x73, 0x15, 0xcb, 0x5d, 0x41, 0xb9, 0x8a, 0x8a, 0x47, 0x27, 0xe7, 0x5e, 0x25, 0xc7, 0x78, 0xa9, 0x1c, 0x62, 0x58, 0xbd, 0xcf, 0x49}} + info := bindataFileInfo{name: "static/js/utils.js", size: 831, mode: os.FileMode(420), modTime: time.Unix(1568918072, 0)} + a := &asset{bytes: bytes, info: info} return a, nil } @@ -492,8 +500,8 @@ func staticJsUtilsJs() (*asset, error) { // It returns an error if the asset could not be found or // could not be loaded. func Asset(name string) ([]byte, error) { - canonicalName := strings.Replace(name, "\\", "/", -1) - if f, ok := _bindata[canonicalName]; ok { + cannonicalName := strings.Replace(name, "\\", "/", -1) + if f, ok := _bindata[cannonicalName]; ok { a, err := f() if err != nil { return nil, fmt.Errorf("Asset %s can't read by error: %v", name, err) @@ -503,12 +511,6 @@ func Asset(name string) ([]byte, error) { return nil, fmt.Errorf("Asset %s not found", name) } -// AssetString returns the asset contents as a string (instead of a []byte). -func AssetString(name string) (string, error) { - data, err := Asset(name) - return string(data), err -} - // MustAsset is like Asset but panics when Asset would return an error. // It simplifies safe initialization of global variables. func MustAsset(name string) []byte { @@ -520,18 +522,12 @@ func MustAsset(name string) []byte { return a } -// MustAssetString is like AssetString but panics when Asset would return an -// error. It simplifies safe initialization of global variables. -func MustAssetString(name string) string { - return string(MustAsset(name)) -} - // AssetInfo loads and returns the asset info for the given name. // It returns an error if the asset could not be found or // could not be loaded. func AssetInfo(name string) (os.FileInfo, error) { - canonicalName := strings.Replace(name, "\\", "/", -1) - if f, ok := _bindata[canonicalName]; ok { + cannonicalName := strings.Replace(name, "\\", "/", -1) + if f, ok := _bindata[cannonicalName]; ok { a, err := f() if err != nil { return nil, fmt.Errorf("AssetInfo %s can't read by error: %v", name, err) @@ -541,33 +537,6 @@ func AssetInfo(name string) (os.FileInfo, error) { return nil, fmt.Errorf("AssetInfo %s not found", name) } -// AssetDigest returns the digest of the file with the given name. It returns an -// error if the asset could not be found or the digest could not be loaded. -func AssetDigest(name string) ([sha256.Size]byte, error) { - canonicalName := strings.Replace(name, "\\", "/", -1) - if f, ok := _bindata[canonicalName]; ok { - a, err := f() - if err != nil { - return [sha256.Size]byte{}, fmt.Errorf("AssetDigest %s can't read by error: %v", name, err) - } - return a.digest, nil - } - return [sha256.Size]byte{}, fmt.Errorf("AssetDigest %s not found", name) -} - -// Digests returns a map of all known files and their checksums. -func Digests() (map[string][sha256.Size]byte, error) { - mp := make(map[string][sha256.Size]byte, len(_bindata)) - for name := range _bindata { - a, err := _bindata[name]() - if err != nil { - return nil, err - } - mp[name] = a.digest - } - return mp, nil -} - // AssetNames returns the names of the assets. func AssetNames() []string { names := make([]string, 0, len(_bindata)) @@ -579,45 +548,26 @@ func AssetNames() []string { // _bindata is a table, holding each asset generator, mapped to its name. var _bindata = map[string]func() (*asset, error){ - "static/css/app.css": staticCssAppCss, - - "static/css/bootstrap.css": staticCssBootstrapCss, - - "static/css/font-awesome.css": staticCssFontAwesomeCss, - - "static/fonts/FontAwesome.otf": staticFontsFontawesomeOtf, - - "static/fonts/fontawesome-webfont.eot": staticFontsFontawesomeWebfontEot, - - "static/fonts/fontawesome-webfont.svg": staticFontsFontawesomeWebfontSvg, - - "static/fonts/fontawesome-webfont.ttf": staticFontsFontawesomeWebfontTtf, - + "static/css/app.css": staticCssAppCss, + "static/css/bootstrap.css": staticCssBootstrapCss, + "static/css/font-awesome.css": staticCssFontAwesomeCss, + "static/fonts/FontAwesome.otf": staticFontsFontawesomeOtf, + "static/fonts/fontawesome-webfont.eot": staticFontsFontawesomeWebfontEot, + "static/fonts/fontawesome-webfont.svg": staticFontsFontawesomeWebfontSvg, + "static/fonts/fontawesome-webfont.ttf": staticFontsFontawesomeWebfontTtf, "static/fonts/fontawesome-webfont.woff": staticFontsFontawesomeWebfontWoff, - - "static/img/icon.ico": staticImgIconIco, - - "static/img/icon.png": staticImgIconPng, - - "static/index.html": staticIndexHtml, - - "static/js/ace-pgsql.js": staticJsAcePgsqlJs, - - "static/js/ace.js": staticJsAceJs, - - "static/js/app.js": staticJsAppJs, - - "static/js/base64.js": staticJsBase64Js, - - "static/js/bootstrap-contextmenu.js": staticJsBootstrapContextmenuJs, - + "static/img/icon.ico": staticImgIconIco, + "static/img/icon.png": staticImgIconPng, + "static/index.html": staticIndexHtml, + "static/js/ace-pgsql.js": staticJsAcePgsqlJs, + "static/js/ace.js": staticJsAceJs, + "static/js/app.js": staticJsAppJs, + "static/js/base64.js": staticJsBase64Js, + "static/js/bootstrap-contextmenu.js": staticJsBootstrapContextmenuJs, "static/js/bootstrap3-typeahead.min.js": staticJsBootstrap3TypeaheadMinJs, - - "static/js/jquery.js": staticJsJqueryJs, - - "static/js/theme-tomorrow.js": staticJsThemeTomorrowJs, - - "static/js/utils.js": staticJsUtilsJs, + "static/js/jquery.js": staticJsJqueryJs, + "static/js/theme-tomorrow.js": staticJsThemeTomorrowJs, + "static/js/utils.js": staticJsUtilsJs, } // AssetDir returns the file names below a certain @@ -629,15 +579,15 @@ var _bindata = map[string]func() (*asset, error){ // img/ // a.png // b.png -// then AssetDir("data") would return []string{"foo.txt", "img"}, -// AssetDir("data/img") would return []string{"a.png", "b.png"}, -// AssetDir("foo.txt") and AssetDir("notexist") would return an error, and +// then AssetDir("data") would return []string{"foo.txt", "img"} +// AssetDir("data/img") would return []string{"a.png", "b.png"} +// AssetDir("foo.txt") and AssetDir("notexist") would return an error // AssetDir("") will return []string{"data"}. func AssetDir(name string) ([]string, error) { node := _bintree if len(name) != 0 { - canonicalName := strings.Replace(name, "\\", "/", -1) - pathList := strings.Split(canonicalName, "/") + cannonicalName := strings.Replace(name, "\\", "/", -1) + pathList := strings.Split(cannonicalName, "/") for _, p := range pathList { node = node.Children[p] if node == nil { @@ -693,7 +643,7 @@ var _bintree = &bintree{nil, map[string]*bintree{ }}, }} -// RestoreAsset restores an asset under the given directory. +// RestoreAsset restores an asset under the given directory func RestoreAsset(dir, name string) error { data, err := Asset(name) if err != nil { @@ -711,10 +661,14 @@ func RestoreAsset(dir, name string) error { if err != nil { return err } - return os.Chtimes(_filePath(dir, name), info.ModTime(), info.ModTime()) + err = os.Chtimes(_filePath(dir, name), info.ModTime(), info.ModTime()) + if err != nil { + return err + } + return nil } -// RestoreAssets restores an asset under the given directory recursively. +// RestoreAssets restores an asset under the given directory recursively func RestoreAssets(dir, name string) error { children, err := AssetDir(name) // File @@ -732,6 +686,6 @@ func RestoreAssets(dir, name string) error { } func _filePath(dir, name string) string { - canonicalName := strings.Replace(name, "\\", "/", -1) - return filepath.Join(append([]string{dir}, strings.Split(canonicalName, "/")...)...) + cannonicalName := strings.Replace(name, "\\", "/", -1) + return filepath.Join(append([]string{dir}, strings.Split(cannonicalName, "/")...)...) } diff --git a/pkg/shared/ssh_info.go b/pkg/shared/ssh_info.go index c9a819a..c1d374a 100644 --- a/pkg/shared/ssh_info.go +++ b/pkg/shared/ssh_info.go @@ -4,12 +4,14 @@ import ( "fmt" ) +// SSHInfo contains ssh server configuration type SSHInfo struct { - Host string `json:"host,omitempty"` - Port string `json:"port,omitempty"` - User string `json:"user,omitempty"` - Password string `json:"password,omitempty"` - Key string `json:"key,omitempty"` + Host string `json:"host,omitempty"` + Port string `json:"port,omitempty"` + User string `json:"user,omitempty"` + Password string `json:"password,omitempty"` + Key string `json:"key,omitempty"` + KeyPassword string `json:"keypassword,omitempty"` } func (info SSHInfo) String() string { diff --git a/static/css/app.css b/static/css/app.css index 045ca98..d3259be 100644 --- a/static/css/app.css +++ b/static/css/app.css @@ -593,6 +593,7 @@ bottom: 0px; right: 0px; display: none; + overflow-y: auto; } .connection-actions { @@ -681,6 +682,12 @@ z-index: 1000; } +.connection-ssh-group h3 { + font-size: 18px; + margin: 0px 0px 20px 0px; + color: #aaa; +} + /* -------------------------------------------------------------------------- */ /* Sidebar Schema Objects */ /* -------------------------------------------------------------------------- */ diff --git a/static/index.html b/static/index.html index a5ce017..b360aed 100644 --- a/static/index.html +++ b/static/index.html @@ -213,8 +213,10 @@

+

SSH Connection

+
- +
@@ -224,25 +226,26 @@
- -
- + +
+ +
+
+
- -
- + +
+ +
+
+
-
- -
- -
-
+
diff --git a/static/js/app.js b/static/js/app.js index b63673c..5633291 100644 --- a/static/js/app.js +++ b/static/js/app.js @@ -1371,6 +1371,7 @@ $(document).ready(function() { $("#ssh_user").val(item.ssh.user); $("#ssh_password").val(item.ssh.password); $("#ssh_key").val(item.ssh.key); + $("#ssh_key_password").val(item.ssh.keypassword); $("#connection_ssh").click(); } else { @@ -1379,6 +1380,7 @@ $(document).ready(function() { $("#ssh_user").val(""); $("#ssh_password").val(""); $("#ssh_key").val(""); + $("#ssh_key_password").val(""); $(".connection-ssh-group").hide(); $("#connection_standard").click(); } @@ -1397,12 +1399,13 @@ $(document).ready(function() { } if ($(".connection-group-switch button.active").attr("data") == "ssh") { - params["ssh"] = 1 - params["ssh_host"] = $("#ssh_host").val(); - params["ssh_port"] = $("#ssh_port").val(); - params["ssh_user"] = $("#ssh_user").val(); - params["ssh_password"] = $("#ssh_password").val(); - params["ssh_key"] = $("#ssh_key").val(); + params["ssh"] = 1 + params["ssh_host"] = $("#ssh_host").val(); + params["ssh_port"] = $("#ssh_port").val(); + params["ssh_user"] = $("#ssh_user").val(); + params["ssh_password"] = $("#ssh_password").val(); + params["ssh_key"] = $("#ssh_key").val(); + params["ssh_key_password"] = $("#ssh_key_password").val() } $("#connection_error").hide(); diff --git a/vendor/github.com/BurntSushi/toml/COPYING b/vendor/github.com/BurntSushi/toml/COPYING index 5a8e332..01b5743 100644 --- a/vendor/github.com/BurntSushi/toml/COPYING +++ b/vendor/github.com/BurntSushi/toml/COPYING @@ -1,14 +1,21 @@ - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - Version 2, December 2004 +The MIT License (MIT) - Copyright (C) 2004 Sam Hocevar +Copyright (c) 2013 TOML authors - Everyone is permitted to copy and distribute verbatim or modified - copies of this license document, and changing it is allowed as long - as the name is changed. +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: - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. You just DO WHAT THE FUCK YOU WANT TO. +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. diff --git a/vendor/github.com/BurntSushi/toml/cmd/toml-test-decoder/COPYING b/vendor/github.com/BurntSushi/toml/cmd/toml-test-decoder/COPYING index 5a8e332..01b5743 100644 --- a/vendor/github.com/BurntSushi/toml/cmd/toml-test-decoder/COPYING +++ b/vendor/github.com/BurntSushi/toml/cmd/toml-test-decoder/COPYING @@ -1,14 +1,21 @@ - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - Version 2, December 2004 +The MIT License (MIT) - Copyright (C) 2004 Sam Hocevar +Copyright (c) 2013 TOML authors - Everyone is permitted to copy and distribute verbatim or modified - copies of this license document, and changing it is allowed as long - as the name is changed. +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: - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. You just DO WHAT THE FUCK YOU WANT TO. +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. diff --git a/vendor/github.com/BurntSushi/toml/cmd/toml-test-encoder/COPYING b/vendor/github.com/BurntSushi/toml/cmd/toml-test-encoder/COPYING index 5a8e332..01b5743 100644 --- a/vendor/github.com/BurntSushi/toml/cmd/toml-test-encoder/COPYING +++ b/vendor/github.com/BurntSushi/toml/cmd/toml-test-encoder/COPYING @@ -1,14 +1,21 @@ - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - Version 2, December 2004 +The MIT License (MIT) - Copyright (C) 2004 Sam Hocevar +Copyright (c) 2013 TOML authors - Everyone is permitted to copy and distribute verbatim or modified - copies of this license document, and changing it is allowed as long - as the name is changed. +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: - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. You just DO WHAT THE FUCK YOU WANT TO. +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. diff --git a/vendor/github.com/BurntSushi/toml/cmd/tomlv/COPYING b/vendor/github.com/BurntSushi/toml/cmd/tomlv/COPYING index 5a8e332..01b5743 100644 --- a/vendor/github.com/BurntSushi/toml/cmd/tomlv/COPYING +++ b/vendor/github.com/BurntSushi/toml/cmd/tomlv/COPYING @@ -1,14 +1,21 @@ - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - Version 2, December 2004 +The MIT License (MIT) - Copyright (C) 2004 Sam Hocevar +Copyright (c) 2013 TOML authors - Everyone is permitted to copy and distribute verbatim or modified - copies of this license document, and changing it is allowed as long - as the name is changed. +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: - DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. You just DO WHAT THE FUCK YOU WANT TO. +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. diff --git a/vendor/github.com/BurntSushi/toml/lex.go b/vendor/github.com/BurntSushi/toml/lex.go index 6dee7fc..e0a742a 100644 --- a/vendor/github.com/BurntSushi/toml/lex.go +++ b/vendor/github.com/BurntSushi/toml/lex.go @@ -775,7 +775,7 @@ func lexDatetime(lx *lexer) stateFn { return lexDatetime } switch r { - case '-', 'T', ':', '.', 'Z': + case '-', 'T', ':', '.', 'Z', '+': return lexDatetime } diff --git a/vendor/github.com/ScaleFT/sshkeys/.gitignore b/vendor/github.com/ScaleFT/sshkeys/.gitignore new file mode 100644 index 0000000..61ead86 --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/.gitignore @@ -0,0 +1 @@ +/vendor diff --git a/vendor/github.com/ScaleFT/sshkeys/.travis.yml b/vendor/github.com/ScaleFT/sshkeys/.travis.yml new file mode 100644 index 0000000..cedda7b --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/.travis.yml @@ -0,0 +1,17 @@ +language: go + +sudo: false + +go_import_path: github.com/ScaleFT/sshkeys + +go: + - 1.9.x + - 1.10.x + - 1.11.x + - tip + +before_install: + - go get -u github.com/stretchr/testify/require github.com/dchest/bcrypt_pbkdf golang.org/x/crypto/ed25519 golang.org/x/crypto/ssh + +script: + - go test -v ./... diff --git a/vendor/github.com/ScaleFT/sshkeys/CODE_OF_CONDUCT.md b/vendor/github.com/ScaleFT/sshkeys/CODE_OF_CONDUCT.md new file mode 100644 index 0000000..dae63bb --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/CODE_OF_CONDUCT.md @@ -0,0 +1,74 @@ +# Contributor Covenant Code of Conduct + +## Our Pledge + +In the interest of fostering an open and welcoming environment, we as +contributors and maintainers pledge to making participation in our project and +our community a harassment-free experience for everyone, regardless of age, body +size, disability, ethnicity, gender identity and expression, level of experience, +nationality, personal appearance, race, religion, or sexual identity and +orientation. + +## Our Standards + +Examples of behavior that contributes to creating a positive environment +include: + +* Using welcoming and inclusive language +* Being respectful of differing viewpoints and experiences +* Gracefully accepting constructive criticism +* Focusing on what is best for the community +* Showing empathy towards other community members + +Examples of unacceptable behavior by participants include: + +* The use of sexualized language or imagery and unwelcome sexual attention or +advances +* Trolling, insulting/derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or electronic + address, without explicit permission +* Other conduct which could reasonably be considered inappropriate in a + professional setting + +## Our Responsibilities + +Project maintainers are responsible for clarifying the standards of acceptable +behavior and are expected to take appropriate and fair corrective action in +response to any instances of unacceptable behavior. + +Project maintainers have the right and responsibility to remove, edit, or +reject comments, commits, code, wiki edits, issues, and other contributions +that are not aligned to this Code of Conduct, or to ban temporarily or +permanently any contributor for other behaviors that they deem inappropriate, +threatening, offensive, or harmful. + +## Scope + +This Code of Conduct applies both within project spaces and in public spaces +when an individual is representing the project or its community. Examples of +representing a project or community include using an official project e-mail +address, posting via an official social media account, or acting as an appointed +representative at an online or offline event. Representation of a project may be +further defined and clarified by project maintainers. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported by contacting the project team at [opensource@scaleft.com](mailto:opensource@scaleft.com). All +complaints will be reviewed and investigated and will result in a response that +is deemed necessary and appropriate to the circumstances. The project team is +obligated to maintain confidentiality with regard to the reporter of an incident. +Further details of specific enforcement policies may be posted separately. + +Project maintainers who do not follow or enforce the Code of Conduct in good +faith may face temporary or permanent repercussions as determined by other +members of the project's leadership. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, +available at [http://contributor-covenant.org/version/1/4][version] + +[homepage]: http://contributor-covenant.org +[version]: http://contributor-covenant.org/version/1/4/ diff --git a/vendor/github.com/ScaleFT/sshkeys/CONTRIBUTING.md b/vendor/github.com/ScaleFT/sshkeys/CONTRIBUTING.md new file mode 100644 index 0000000..35b7a64 --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/CONTRIBUTING.md @@ -0,0 +1,42 @@ +# How to Contribute + +ScaleFT's projects are [Apache 2.0 licensed](LICENSE) and accept contributions +via GitHub pull requests. This document outlines some of the conventions on +development workflow, contact points, community conduct and other resources +to make it easier to get your contribution accepted. + +# Code of Conduct + +This project adheres to the Contributor Covenant [code of conduct](CODE_OF_CONDUCT.md). +By participating, you are expected to uphold this code. + +Please report unacceptable behavior to [opensource@scaleft.com](mailto:opensource@scaleft.com). + +# Reporting Security Issues + +ScaleFT takes security seriously. If you discover a security issue, +please bring it to our attention right away! + +Please DO NOT file a public issue or pull request, +[instead send your report privately to the ScaleFT Security Team](https://www.scaleft.com/company/security/), +reachable at [security@scaleft.com](mailto:security@scaleft.com). + +Security reports are greatly appreciated and we will publicly thank you for them. + +# Getting Started + +- Fork the repository on GitHub +- Read the [README](README.md) for build and test instructions +- Play with the project, submit bugs, submit patches! + +# Contribution Flow + +This is a rough outline of what a contributor's workflow looks like: + +- Create a topic branch from where you want to base your work (usually master). +- Make commits of logical units, rebasing later is ok too! +- Push your changes to a topic branch in your fork of the repository. +- Make sure the tests pass, and add any new tests as appropriate. +- Submit a pull request to the original repository. + +Thanks for your contributions! diff --git a/vendor/github.com/ScaleFT/sshkeys/LICENSE b/vendor/github.com/ScaleFT/sshkeys/LICENSE new file mode 100644 index 0000000..d645695 --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/LICENSE @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/github.com/ScaleFT/sshkeys/NOTICE b/vendor/github.com/ScaleFT/sshkeys/NOTICE new file mode 100644 index 0000000..21302db --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/NOTICE @@ -0,0 +1,10 @@ +sshkeys +Copyright 2017 ScaleFT, Inc + +This product includes software developed at ScaleFT, Inc. +(https://www.scaleft.com/). + +Portions of this software are derived from +https://github.com/golang/crypto/blob/master/ssh/keys.go + +Copyright (c) 2009 The Go Authors. All rights reserved. diff --git a/vendor/github.com/ScaleFT/sshkeys/README.md b/vendor/github.com/ScaleFT/sshkeys/README.md new file mode 100644 index 0000000..89d8c21 --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/README.md @@ -0,0 +1,14 @@ +# sshkeys + +[![GoDoc](https://godoc.org/github.com/ScaleFT/sshkeys?status.svg)](https://godoc.org/github.com/ScaleFT/sshkeys) +[![Build Status](https://travis-ci.org/ScaleFT/sshkeys.svg?branch=master)](https://travis-ci.org/ScaleFT/sshkeys) + +`sshkeys` provides utilities for parsing and marshalling cryptographic keys used for SSH, in both cleartext and encrypted formats. + +[ssh.ParseRawPrivateKey](https://godoc.org/golang.org/x/crypto/ssh#ParseRawPrivateKey) only supports parsing a subset of the formats `sshkeys` supports, does not support parsing encrypted private keys, and does not support marshalling. + +## Supported Formats + +* OpenSSH's [PROTOCOL.key](https://github.com/openssh/openssh-portable/blob/master/PROTOCOL.key) for RSA and ED25519 keys. +* OpenSSH version >= 7.6 using aes256-ctr encryption +* "Classic" PEM containing RSA (PKCS#1), DSA (OpenSSL), and ECDSA private keys. diff --git a/vendor/github.com/ScaleFT/sshkeys/marshal.go b/vendor/github.com/ScaleFT/sshkeys/marshal.go new file mode 100644 index 0000000..5fdf209 --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/marshal.go @@ -0,0 +1,275 @@ +package sshkeys + +import ( + "crypto/aes" + "crypto/cipher" + "crypto/dsa" + "crypto/ecdsa" + "crypto/rand" + "crypto/rsa" + "crypto/x509" + "encoding/asn1" + "encoding/pem" + "fmt" + "math/big" + mrand "math/rand" + + "github.com/dchest/bcrypt_pbkdf" + "golang.org/x/crypto/ed25519" + "golang.org/x/crypto/ssh" +) + +// Format of private key to use when Marshaling. +type Format int + +const ( + // FormatOpenSSHv1 encodes a private key using OpenSSH's PROTOCOL.key format: https://github.com/openssh/openssh-portable/blob/master/PROTOCOL.key + FormatOpenSSHv1 Format = iota + // FormatClassicPEM encodes private keys in PEM, with a key-specific encoding, as used by OpenSSH. + FormatClassicPEM +) + +// MarshalOptions provides the Marshal function format and encryption options. +type MarshalOptions struct { + // Passphrase to encrypt private key with, if nil, the key will not be encrypted. + Passphrase []byte + // Format to encode the private key in. + Format Format +} + +// Marshal converts a private key into an optionally encrypted format. +func Marshal(pk interface{}, opts *MarshalOptions) ([]byte, error) { + switch opts.Format { + case FormatOpenSSHv1: + return marshalOpenssh(pk, opts) + case FormatClassicPEM: + return marshalPem(pk, opts) + default: + return nil, fmt.Errorf("sshkeys: invalid format %d", opts.Format) + } +} + +func marshalPem(pk interface{}, opts *MarshalOptions) ([]byte, error) { + var err error + var plain []byte + var pemType string + + switch key := pk.(type) { + case *rsa.PrivateKey: + pemType = "RSA PRIVATE KEY" + plain = x509.MarshalPKCS1PrivateKey(key) + case *ecdsa.PrivateKey: + pemType = "EC PRIVATE KEY" + plain, err = x509.MarshalECPrivateKey(key) + if err != nil { + return nil, err + } + case *dsa.PrivateKey: + pemType = "DSA PRIVATE KEY" + plain, err = marshalDSAPrivateKey(key) + if err != nil { + return nil, err + } + case *ed25519.PrivateKey: + return nil, fmt.Errorf("sshkeys: ed25519 keys must be marshaled with FormatOpenSSHv1") + default: + return nil, fmt.Errorf("sshkeys: unsupported key type %T", pk) + } + + if len(opts.Passphrase) > 0 { + block, err := x509.EncryptPEMBlock(rand.Reader, pemType, plain, opts.Passphrase, x509.PEMCipherAES128) + if err != nil { + return nil, err + } + return pem.EncodeToMemory(block), nil + } + + return pem.EncodeToMemory(&pem.Block{ + Type: pemType, + Bytes: plain, + }), nil +} + +type dsaOpenssl struct { + Version int + P *big.Int + Q *big.Int + G *big.Int + Pub *big.Int + Priv *big.Int +} + +// https://github.com/golang/crypto/blob/master/ssh/keys.go#L793-L804 +func marshalDSAPrivateKey(pk *dsa.PrivateKey) ([]byte, error) { + k := dsaOpenssl{ + Version: 0, + P: pk.P, + Q: pk.Q, + G: pk.G, + Pub: pk.Y, + Priv: pk.X, + } + + return asn1.Marshal(k) +} + +const opensshv1Magic = "openssh-key-v1" + +type opensshHeader struct { + CipherName string + KdfName string + KdfOpts string + NumKeys uint32 + PubKey string + PrivKeyBlock string +} + +type opensshKey struct { + Check1 uint32 + Check2 uint32 + Keytype string + Rest []byte `ssh:"rest"` +} + +type opensshRsa struct { + N *big.Int + E *big.Int + D *big.Int + Iqmp *big.Int + P *big.Int + Q *big.Int + Comment string + Pad []byte `ssh:"rest"` +} + +type opensshED25519 struct { + Pub []byte + Priv []byte + Comment string + Pad []byte `ssh:"rest"` +} + +func padBytes(data []byte, blocksize int) []byte { + if blocksize != 0 { + var i byte + for i = byte(1); len(data)%blocksize != 0; i++ { + data = append(data, i&0xFF) + } + } + return data +} + +func marshalOpenssh(pk interface{}, opts *MarshalOptions) ([]byte, error) { + var blocksize int + var keylen int + + out := opensshHeader{ + CipherName: "none", + KdfName: "none", + KdfOpts: "", + NumKeys: 1, + PubKey: "", + } + + if len(opts.Passphrase) > 0 { + out.CipherName = "aes256-cbc" + out.KdfName = "bcrypt" + keylen = keySizeAES256 + blocksize = aes.BlockSize + } + + check := mrand.Uint32() + pk1 := opensshKey{ + Check1: check, + Check2: check, + } + + switch key := pk.(type) { + case *rsa.PrivateKey: + k := &opensshRsa{ + N: key.N, + E: big.NewInt(int64(key.E)), + D: key.D, + Iqmp: key.Precomputed.Qinv, + P: key.Primes[0], + Q: key.Primes[1], + Comment: "", + } + + data := ssh.Marshal(k) + pk1.Keytype = ssh.KeyAlgoRSA + pk1.Rest = data + publicKey, err := ssh.NewPublicKey(&key.PublicKey) + if err != nil { + return nil, err + } + out.PubKey = string(publicKey.Marshal()) + + case ed25519.PrivateKey: + k := opensshED25519{ + Pub: key.Public().(ed25519.PublicKey), + Priv: key, + } + data := ssh.Marshal(k) + pk1.Keytype = ssh.KeyAlgoED25519 + pk1.Rest = data + + publicKey, err := ssh.NewPublicKey(key.Public()) + if err != nil { + return nil, err + } + out.PubKey = string(publicKey.Marshal()) + default: + return nil, fmt.Errorf("sshkeys: unsupported key type %T", pk) + } + + if len(opts.Passphrase) > 0 { + rounds := 16 + ivlen := blocksize + salt := make([]byte, blocksize) + _, err := rand.Read(salt) + if err != nil { + return nil, err + } + + kdfdata, err := bcrypt_pbkdf.Key(opts.Passphrase, salt, rounds, keylen+ivlen) + if err != nil { + return nil, err + } + iv := kdfdata[keylen : ivlen+keylen] + aeskey := kdfdata[0:keylen] + + block, err := aes.NewCipher(aeskey) + if err != nil { + return nil, err + } + + pkblock := padBytes(ssh.Marshal(pk1), blocksize) + + cbc := cipher.NewCBCEncrypter(block, iv) + cbc.CryptBlocks(pkblock, pkblock) + + out.PrivKeyBlock = string(pkblock) + + var opts struct { + Salt []byte + Rounds uint32 + } + + opts.Salt = salt + opts.Rounds = uint32(rounds) + + out.KdfOpts = string(ssh.Marshal(&opts)) + } else { + out.PrivKeyBlock = string(ssh.Marshal(pk1)) + } + + outBytes := []byte(opensshv1Magic) + outBytes = append(outBytes, 0) + outBytes = append(outBytes, ssh.Marshal(out)...) + block := &pem.Block{ + Type: "OPENSSH PRIVATE KEY", + Bytes: outBytes, + } + return pem.EncodeToMemory(block), nil +} diff --git a/vendor/github.com/ScaleFT/sshkeys/parse.go b/vendor/github.com/ScaleFT/sshkeys/parse.go new file mode 100644 index 0000000..7ccc2f8 --- /dev/null +++ b/vendor/github.com/ScaleFT/sshkeys/parse.go @@ -0,0 +1,244 @@ +// Portions of this file are based on https://github.com/golang/crypto/blob/master/ssh/keys.go +// +// Copyright 2012 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package sshkeys + +import ( + "bytes" + "crypto/aes" + "crypto/cipher" + "crypto/rsa" + "crypto/x509" + "encoding/pem" + "errors" + "fmt" + "math/big" + "strings" + + "github.com/dchest/bcrypt_pbkdf" + "golang.org/x/crypto/ed25519" + "golang.org/x/crypto/ssh" +) + +// ErrIncorrectPassword is returned when the supplied passphrase was not correct for an encrypted private key. +var ErrIncorrectPassword = errors.New("sshkeys: Invalid Passphrase") + +const keySizeAES256 = 32 + +// ParseEncryptedPrivateKey returns a Signer from an encrypted private key. It supports +// the same keys as ParseEncryptedRawPrivateKey. +func ParseEncryptedPrivateKey(data []byte, passphrase []byte) (ssh.Signer, error) { + key, err := ParseEncryptedRawPrivateKey(data, passphrase) + if err != nil { + return nil, err + } + + return ssh.NewSignerFromKey(key) +} + +// ParseEncryptedRawPrivateKey returns a private key from an encrypted private key. It +// supports RSA (PKCS#1 or OpenSSH), DSA (OpenSSL), and ECDSA private keys. +// +// ErrIncorrectPassword will be returned if the supplied passphrase is wrong, +// but some formats like RSA in PKCS#1 detecting a wrong passphrase is difficult, +// and other parse errors may be returned. +func ParseEncryptedRawPrivateKey(data []byte, passphrase []byte) (interface{}, error) { + var err error + + block, _ := pem.Decode(data) + if block == nil { + return nil, errors.New("no PEM block found") + } + + if x509.IsEncryptedPEMBlock(block) { + data, err = x509.DecryptPEMBlock(block, passphrase) + if err == x509.IncorrectPasswordError { + return nil, ErrIncorrectPassword + } + if err != nil { + return nil, err + } + } else { + data = block.Bytes + } + + switch block.Type { + case "RSA PRIVATE KEY": + pk, err := x509.ParsePKCS1PrivateKey(data) + if err != nil { + // The Algos for PEM Encryption do not include strong message authentication, + // so sometimes DecryptPEMBlock works, but ParsePKCS1PrivateKey fails with an asn1 error. + // We are just catching the most common prefix here... + if strings.HasPrefix(err.Error(), "asn1: structure error") { + return nil, ErrIncorrectPassword + } + return nil, err + } + return pk, nil + case "EC PRIVATE KEY": + return x509.ParseECPrivateKey(data) + case "DSA PRIVATE KEY": + return ssh.ParseDSAPrivateKey(data) + case "OPENSSH PRIVATE KEY": + return parseOpenSSHPrivateKey(data, passphrase) + default: + return nil, fmt.Errorf("sshkeys: unsupported key type %q", block.Type) + } +} + +func parseOpenSSHPrivateKey(data []byte, passphrase []byte) (interface{}, error) { + magic := append([]byte(opensshv1Magic), 0) + if !bytes.Equal(magic, data[0:len(magic)]) { + return nil, errors.New("sshkeys: invalid openssh private key format") + } + remaining := data[len(magic):] + + w := opensshHeader{} + + if err := ssh.Unmarshal(remaining, &w); err != nil { + return nil, err + } + + if w.NumKeys != 1 { + return nil, fmt.Errorf("sshkeys: NumKeys must be 1: %d", w.NumKeys) + } + + var privateKeyBytes []byte + var encrypted bool + + switch { + // OpenSSH supports bcrypt KDF w/ AES256-CBC or AES256-CTR mode + case w.KdfName == "bcrypt" && w.CipherName == "aes256-cbc": + iv, block, err := extractBcryptIvBlock(passphrase, w) + if err != nil { + return nil, err + } + + cbc := cipher.NewCBCDecrypter(block, iv) + privateKeyBytes = []byte(w.PrivKeyBlock) + cbc.CryptBlocks(privateKeyBytes, privateKeyBytes) + + encrypted = true + + case w.KdfName == "bcrypt" && w.CipherName == "aes256-ctr": + iv, block, err := extractBcryptIvBlock(passphrase, w) + if err != nil { + return nil, err + } + + stream := cipher.NewCTR(block, iv) + privateKeyBytes = []byte(w.PrivKeyBlock) + stream.XORKeyStream(privateKeyBytes, privateKeyBytes) + + encrypted = true + + case w.KdfName == "none" && w.CipherName == "none": + privateKeyBytes = []byte(w.PrivKeyBlock) + + default: + return nil, fmt.Errorf("sshkeys: unknown Cipher/KDF: %s:%s", w.CipherName, w.KdfName) + } + + pk1 := opensshKey{} + + if err := ssh.Unmarshal(privateKeyBytes, &pk1); err != nil { + if encrypted { + return nil, ErrIncorrectPassword + } + return nil, err + } + + if pk1.Check1 != pk1.Check2 { + return nil, ErrIncorrectPassword + } + + // we only handle ed25519 and rsa keys currently + switch pk1.Keytype { + case ssh.KeyAlgoRSA: + // https://github.com/openssh/openssh-portable/blob/V_7_4_P1/sshkey.c#L2760-L2773 + key := opensshRsa{} + + err := ssh.Unmarshal(pk1.Rest, &key) + if err != nil { + return nil, err + } + + for i, b := range key.Pad { + if int(b) != i+1 { + return nil, errors.New("sshkeys: padding not as expected") + } + } + + pk := &rsa.PrivateKey{ + PublicKey: rsa.PublicKey{ + N: key.N, + E: int(key.E.Int64()), + }, + D: key.D, + Primes: []*big.Int{key.P, key.Q}, + } + + err = pk.Validate() + if err != nil { + return nil, err + } + + pk.Precompute() + + return pk, nil + case ssh.KeyAlgoED25519: + key := opensshED25519{} + + err := ssh.Unmarshal(pk1.Rest, &key) + if err != nil { + return nil, err + } + + if len(key.Priv) != ed25519.PrivateKeySize { + return nil, errors.New("sshkeys: private key unexpected length") + } + + for i, b := range key.Pad { + if int(b) != i+1 { + return nil, errors.New("sshkeys: padding not as expected") + } + } + + pk := ed25519.PrivateKey(make([]byte, ed25519.PrivateKeySize)) + copy(pk, key.Priv) + return pk, nil + default: + return nil, errors.New("sshkeys: unhandled key type") + } +} + +func extractBcryptIvBlock(passphrase []byte, w opensshHeader) ([]byte, cipher.Block, error) { + cipherKeylen := keySizeAES256 + cipherIvLen := aes.BlockSize + + var opts struct { + Salt []byte + Rounds uint32 + } + + if err := ssh.Unmarshal([]byte(w.KdfOpts), &opts); err != nil { + return nil, nil, err + } + kdfdata, err := bcrypt_pbkdf.Key(passphrase, opts.Salt, int(opts.Rounds), cipherKeylen+cipherIvLen) + if err != nil { + return nil, nil, err + } + + iv := kdfdata[cipherKeylen : cipherIvLen+cipherKeylen] + aeskey := kdfdata[0:cipherKeylen] + block, err := aes.NewCipher(aeskey) + + if err != nil { + return nil, nil, err + } + + return iv, block, nil +} diff --git a/vendor/github.com/dchest/bcrypt_pbkdf/LICENSE b/vendor/github.com/dchest/bcrypt_pbkdf/LICENSE new file mode 100644 index 0000000..b99c5e3 --- /dev/null +++ b/vendor/github.com/dchest/bcrypt_pbkdf/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2014 Dmitry Chestnykh +Copyright (c) 2010 The Go Authors +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. diff --git a/vendor/github.com/dchest/bcrypt_pbkdf/README b/vendor/github.com/dchest/bcrypt_pbkdf/README new file mode 100644 index 0000000..8ce6874 --- /dev/null +++ b/vendor/github.com/dchest/bcrypt_pbkdf/README @@ -0,0 +1,21 @@ +Go implementation of bcrypt_pbkdf(3) from OpenBSD +(a variant of PBKDF2 with bcrypt-based PRF). + + +USAGE + + func Key(password, salt []byte, rounds, keyLen int) ([]byte, error) + + + Key derives a key from the password, salt and rounds count, returning a + []byte of length keyLen that can be used as cryptographic key. + + Remember to get a good random salt of at least 16 bytes. Using a higher + rounds count will increase the cost of an exhaustive search but will also + make derivation proportionally slower. + + +REFERENCES + +* http://www.tedunangst.com/flak/post/bcrypt-pbkdf +* http://cvsweb.openbsd.org/cgi-bin/cvsweb/src/lib/libutil/bcrypt_pbkdf.c diff --git a/vendor/github.com/dchest/bcrypt_pbkdf/bcrypt_pbkdf.go b/vendor/github.com/dchest/bcrypt_pbkdf/bcrypt_pbkdf.go new file mode 100644 index 0000000..60bba1c --- /dev/null +++ b/vendor/github.com/dchest/bcrypt_pbkdf/bcrypt_pbkdf.go @@ -0,0 +1,97 @@ +// Copyright 2014 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 bcrypt_pbkdf implements password-based key derivation function based +// on bcrypt compatible with bcrypt_pbkdf(3) from OpenBSD. +package bcrypt_pbkdf + +import ( + "crypto/sha512" + "errors" + + // NOTE! Requires blowfish package version from Aug 1, 2014 or later. + // Will produce incorrect results if the package is older. + // See commit message for details: http://goo.gl/wx6g8O + "golang.org/x/crypto/blowfish" +) + +// Key derives a key from the password, salt and rounds count, returning a +// []byte of length keyLen that can be used as cryptographic key. +// +// Remember to get a good random salt of at least 16 bytes. Using a higher +// rounds count will increase the cost of an exhaustive search but will also +// make derivation proportionally slower. +func Key(password, salt []byte, rounds, keyLen int) ([]byte, error) { + if rounds < 1 { + return nil, errors.New("bcrypt_pbkdf: number of rounds is too small") + } + if len(password) == 0 { + return nil, errors.New("bcrypt_pbkdf: empty password") + } + if len(salt) == 0 || len(salt) > 1<<20 { + return nil, errors.New("bcrypt_pbkdf: bad salt length") + } + if keyLen > 1024 { + return nil, errors.New("bcrypt_pbkdf: keyLen is too large") + } + var shapass, shasalt [sha512.Size]byte + var out, tmp [32]byte + var cnt [4]byte + + numBlocks := (keyLen + len(out) - 1) / len(out) + key := make([]byte, numBlocks*len(out)) + + h := sha512.New() + h.Write(password) + h.Sum(shapass[:0]) + + for block := 1; block <= numBlocks; block++ { + h.Reset() + h.Write(salt) + cnt[0] = byte(block >> 24) + cnt[1] = byte(block >> 16) + cnt[2] = byte(block >> 8) + cnt[3] = byte(block) + h.Write(cnt[:]) + bcryptHash(tmp[:], shapass[:], h.Sum(shasalt[:0])) + copy(out[:], tmp[:]) + + for i := 2; i <= rounds; i++ { + h.Reset() + h.Write(tmp[:]) + bcryptHash(tmp[:], shapass[:], h.Sum(shasalt[:0])) + for j := 0; j < len(out); j++ { + out[j] ^= tmp[j] + } + } + + for i, v := range out { + key[i*numBlocks+(block-1)] = v + } + } + return key[:keyLen], nil +} + +var magic = []byte("OxychromaticBlowfishSwatDynamite") + +func bcryptHash(out, shapass, shasalt []byte) { + c, err := blowfish.NewSaltedCipher(shapass, shasalt) + if err != nil { + panic(err) + } + for i := 0; i < 64; i++ { + blowfish.ExpandKey(shasalt, c) + blowfish.ExpandKey(shapass, c) + } + copy(out[:], magic) + for i := 0; i < 32; i += 8 { + for j := 0; j < 64; j++ { + c.Encrypt(out[i:i+8], out[i:i+8]) + } + } + // Swap bytes due to different endianness. + for i := 0; i < 32; i += 4 { + out[i+3], out[i+2], out[i+1], out[i] = out[i], out[i+1], out[i+2], out[i+3] + } +} diff --git a/vendor/golang.org/x/crypto/blowfish/block.go b/vendor/golang.org/x/crypto/blowfish/block.go new file mode 100644 index 0000000..9d80f19 --- /dev/null +++ b/vendor/golang.org/x/crypto/blowfish/block.go @@ -0,0 +1,159 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package blowfish + +// getNextWord returns the next big-endian uint32 value from the byte slice +// at the given position in a circular manner, updating the position. +func getNextWord(b []byte, pos *int) uint32 { + var w uint32 + j := *pos + for i := 0; i < 4; i++ { + w = w<<8 | uint32(b[j]) + j++ + if j >= len(b) { + j = 0 + } + } + *pos = j + return w +} + +// ExpandKey performs a key expansion on the given *Cipher. Specifically, it +// performs the Blowfish algorithm's key schedule which sets up the *Cipher's +// pi and substitution tables for calls to Encrypt. This is used, primarily, +// by the bcrypt package to reuse the Blowfish key schedule during its +// set up. It's unlikely that you need to use this directly. +func ExpandKey(key []byte, c *Cipher) { + j := 0 + for i := 0; i < 18; i++ { + // Using inlined getNextWord for performance. + var d uint32 + for k := 0; k < 4; k++ { + d = d<<8 | uint32(key[j]) + j++ + if j >= len(key) { + j = 0 + } + } + c.p[i] ^= d + } + + var l, r uint32 + for i := 0; i < 18; i += 2 { + l, r = encryptBlock(l, r, c) + c.p[i], c.p[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s0[i], c.s0[i+1] = l, r + } + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s1[i], c.s1[i+1] = l, r + } + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s2[i], c.s2[i+1] = l, r + } + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s3[i], c.s3[i+1] = l, r + } +} + +// This is similar to ExpandKey, but folds the salt during the key +// schedule. While ExpandKey is essentially expandKeyWithSalt with an all-zero +// salt passed in, reusing ExpandKey turns out to be a place of inefficiency +// and specializing it here is useful. +func expandKeyWithSalt(key []byte, salt []byte, c *Cipher) { + j := 0 + for i := 0; i < 18; i++ { + c.p[i] ^= getNextWord(key, &j) + } + + j = 0 + var l, r uint32 + for i := 0; i < 18; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.p[i], c.p[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s0[i], c.s0[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s1[i], c.s1[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s2[i], c.s2[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s3[i], c.s3[i+1] = l, r + } +} + +func encryptBlock(l, r uint32, c *Cipher) (uint32, uint32) { + xl, xr := l, r + xl ^= c.p[0] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[1] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[2] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[3] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[4] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[5] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[6] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[7] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[8] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[9] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[10] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[11] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[12] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[13] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[14] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[15] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[16] + xr ^= c.p[17] + return xr, xl +} + +func decryptBlock(l, r uint32, c *Cipher) (uint32, uint32) { + xl, xr := l, r + xl ^= c.p[17] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[16] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[15] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[14] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[13] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[12] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[11] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[10] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[9] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[8] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[7] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[6] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[5] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[4] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[3] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[2] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[1] + xr ^= c.p[0] + return xr, xl +} diff --git a/vendor/golang.org/x/crypto/blowfish/cipher.go b/vendor/golang.org/x/crypto/blowfish/cipher.go new file mode 100644 index 0000000..213bf20 --- /dev/null +++ b/vendor/golang.org/x/crypto/blowfish/cipher.go @@ -0,0 +1,99 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package blowfish implements Bruce Schneier's Blowfish encryption algorithm. +// +// Blowfish is a legacy cipher and its short block size makes it vulnerable to +// birthday bound attacks (see https://sweet32.info). It should only be used +// where compatibility with legacy systems, not security, is the goal. +// +// Deprecated: any new system should use AES (from crypto/aes, if necessary in +// an AEAD mode like crypto/cipher.NewGCM) or XChaCha20-Poly1305 (from +// golang.org/x/crypto/chacha20poly1305). +package blowfish // import "golang.org/x/crypto/blowfish" + +// The code is a port of Bruce Schneier's C implementation. +// See https://www.schneier.com/blowfish.html. + +import "strconv" + +// The Blowfish block size in bytes. +const BlockSize = 8 + +// A Cipher is an instance of Blowfish encryption using a particular key. +type Cipher struct { + p [18]uint32 + s0, s1, s2, s3 [256]uint32 +} + +type KeySizeError int + +func (k KeySizeError) Error() string { + return "crypto/blowfish: invalid key size " + strconv.Itoa(int(k)) +} + +// NewCipher creates and returns a Cipher. +// The key argument should be the Blowfish key, from 1 to 56 bytes. +func NewCipher(key []byte) (*Cipher, error) { + var result Cipher + if k := len(key); k < 1 || k > 56 { + return nil, KeySizeError(k) + } + initCipher(&result) + ExpandKey(key, &result) + return &result, nil +} + +// NewSaltedCipher creates a returns a Cipher that folds a salt into its key +// schedule. For most purposes, NewCipher, instead of NewSaltedCipher, is +// sufficient and desirable. For bcrypt compatibility, the key can be over 56 +// bytes. +func NewSaltedCipher(key, salt []byte) (*Cipher, error) { + if len(salt) == 0 { + return NewCipher(key) + } + var result Cipher + if k := len(key); k < 1 { + return nil, KeySizeError(k) + } + initCipher(&result) + expandKeyWithSalt(key, salt, &result) + return &result, nil +} + +// BlockSize returns the Blowfish block size, 8 bytes. +// It is necessary to satisfy the Block interface in the +// package "crypto/cipher". +func (c *Cipher) BlockSize() int { return BlockSize } + +// Encrypt encrypts the 8-byte buffer src using the key k +// and stores the result in dst. +// Note that for amounts of data larger than a block, +// it is not safe to just call Encrypt on successive blocks; +// instead, use an encryption mode like CBC (see crypto/cipher/cbc.go). +func (c *Cipher) Encrypt(dst, src []byte) { + l := uint32(src[0])<<24 | uint32(src[1])<<16 | uint32(src[2])<<8 | uint32(src[3]) + r := uint32(src[4])<<24 | uint32(src[5])<<16 | uint32(src[6])<<8 | uint32(src[7]) + l, r = encryptBlock(l, r, c) + dst[0], dst[1], dst[2], dst[3] = byte(l>>24), byte(l>>16), byte(l>>8), byte(l) + dst[4], dst[5], dst[6], dst[7] = byte(r>>24), byte(r>>16), byte(r>>8), byte(r) +} + +// Decrypt decrypts the 8-byte buffer src using the key k +// and stores the result in dst. +func (c *Cipher) Decrypt(dst, src []byte) { + l := uint32(src[0])<<24 | uint32(src[1])<<16 | uint32(src[2])<<8 | uint32(src[3]) + r := uint32(src[4])<<24 | uint32(src[5])<<16 | uint32(src[6])<<8 | uint32(src[7]) + l, r = decryptBlock(l, r, c) + dst[0], dst[1], dst[2], dst[3] = byte(l>>24), byte(l>>16), byte(l>>8), byte(l) + dst[4], dst[5], dst[6], dst[7] = byte(r>>24), byte(r>>16), byte(r>>8), byte(r) +} + +func initCipher(c *Cipher) { + copy(c.p[0:], p[0:]) + copy(c.s0[0:], s0[0:]) + copy(c.s1[0:], s1[0:]) + copy(c.s2[0:], s2[0:]) + copy(c.s3[0:], s3[0:]) +} diff --git a/vendor/golang.org/x/crypto/blowfish/const.go b/vendor/golang.org/x/crypto/blowfish/const.go new file mode 100644 index 0000000..d040775 --- /dev/null +++ b/vendor/golang.org/x/crypto/blowfish/const.go @@ -0,0 +1,199 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// The startup permutation array and substitution boxes. +// They are the hexadecimal digits of PI; see: +// https://www.schneier.com/code/constants.txt. + +package blowfish + +var s0 = [256]uint32{ + 0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96, + 0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16, + 0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658, + 0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013, + 0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e, + 0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60, + 0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6, + 0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a, + 0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c, + 0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193, + 0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1, + 0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239, + 0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a, + 0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3, + 0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176, + 0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe, + 0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706, + 0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b, + 0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b, + 0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463, + 0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c, + 0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3, + 0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a, + 0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8, + 0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760, + 0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db, + 0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8, + 0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b, + 0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33, + 0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4, + 0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0, + 0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c, + 0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777, + 0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299, + 0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705, + 0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf, + 0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e, + 0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa, + 0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9, + 0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915, + 0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f, + 0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664, + 0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a, +} + +var s1 = [256]uint32{ + 0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d, + 0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1, + 0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65, + 0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1, + 0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9, + 0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737, + 0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d, + 0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd, + 0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc, + 0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41, + 0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908, + 0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af, + 0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124, + 0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c, + 0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908, + 0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd, + 0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b, + 0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e, + 0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa, + 0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a, + 0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d, + 0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66, + 0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5, + 0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84, + 0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96, + 0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14, + 0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca, + 0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7, + 0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77, + 0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99, + 0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054, + 0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73, + 0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea, + 0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105, + 0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646, + 0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285, + 0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea, + 0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb, + 0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e, + 0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc, + 0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd, + 0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20, + 0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7, +} + +var s2 = [256]uint32{ + 0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7, + 0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af, + 0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af, + 0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504, + 0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4, + 0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee, + 0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec, + 0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b, + 0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332, + 0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527, + 0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58, + 0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c, + 0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22, + 0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17, + 0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60, + 0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115, + 0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99, + 0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0, + 0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74, + 0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d, + 0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3, + 0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3, + 0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979, + 0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c, + 0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa, + 0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a, + 0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086, + 0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc, + 0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24, + 0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2, + 0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84, + 0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c, + 0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09, + 0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10, + 0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe, + 0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027, + 0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0, + 0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634, + 0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188, + 0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc, + 0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8, + 0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837, + 0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0, +} + +var s3 = [256]uint32{ + 0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742, + 0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b, + 0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79, + 0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6, + 0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a, + 0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4, + 0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1, + 0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59, + 0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797, + 0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28, + 0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6, + 0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28, + 0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba, + 0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a, + 0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5, + 0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f, + 0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce, + 0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680, + 0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd, + 0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb, + 0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb, + 0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370, + 0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc, + 0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048, + 0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc, + 0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9, + 0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a, + 0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f, + 0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a, + 0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1, + 0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b, + 0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e, + 0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e, + 0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f, + 0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623, + 0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc, + 0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a, + 0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6, + 0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3, + 0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060, + 0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c, + 0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f, + 0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6, +} + +var p = [18]uint32{ + 0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0, + 0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c, + 0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b, +} diff --git a/vendor/golang.org/x/crypto/curve25519/curve25519.go b/vendor/golang.org/x/crypto/curve25519/curve25519.go index cb8fbc5..75f24ba 100644 --- a/vendor/golang.org/x/crypto/curve25519/curve25519.go +++ b/vendor/golang.org/x/crypto/curve25519/curve25519.go @@ -86,7 +86,7 @@ func feFromBytes(dst *fieldElement, src *[32]byte) { h6 := load3(src[20:]) << 7 h7 := load3(src[23:]) << 5 h8 := load3(src[26:]) << 4 - h9 := load3(src[29:]) << 2 + h9 := (load3(src[29:]) & 0x7fffff) << 2 var carry [10]int64 carry[9] = (h9 + 1<<24) >> 25 diff --git a/vendor/golang.org/x/crypto/curve25519/ladderstep_amd64.s b/vendor/golang.org/x/crypto/curve25519/ladderstep_amd64.s index 9e9040b..e0ac30c 100644 --- a/vendor/golang.org/x/crypto/curve25519/ladderstep_amd64.s +++ b/vendor/golang.org/x/crypto/curve25519/ladderstep_amd64.s @@ -121,18 +121,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -236,18 +236,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -441,18 +441,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -591,18 +591,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -731,18 +731,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -846,18 +846,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -996,18 +996,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -1146,18 +1146,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX @@ -1332,18 +1332,18 @@ TEXT ·ladderstep(SB),0,$296-8 ADDQ AX,R12 ADCQ DX,R13 MOVQ $REDMASK51,DX - SHLQ $13,CX:SI + SHLQ $13,SI,CX ANDQ DX,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ DX,R8 ADDQ CX,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ DX,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ DX,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ DX,R14 ADDQ R13,R14 IMUL3Q $19,R15,CX diff --git a/vendor/golang.org/x/crypto/curve25519/mul_amd64.s b/vendor/golang.org/x/crypto/curve25519/mul_amd64.s index 5ce80a2..1f76d1a 100644 --- a/vendor/golang.org/x/crypto/curve25519/mul_amd64.s +++ b/vendor/golang.org/x/crypto/curve25519/mul_amd64.s @@ -124,18 +124,18 @@ TEXT ·mul(SB),0,$16-24 ADDQ AX,R14 ADCQ DX,R15 MOVQ $REDMASK51,SI - SHLQ $13,R9:R8 + SHLQ $13,R8,R9 ANDQ SI,R8 - SHLQ $13,R11:R10 + SHLQ $13,R10,R11 ANDQ SI,R10 ADDQ R9,R10 - SHLQ $13,R13:R12 + SHLQ $13,R12,R13 ANDQ SI,R12 ADDQ R11,R12 - SHLQ $13,R15:R14 + SHLQ $13,R14,R15 ANDQ SI,R14 ADDQ R13,R14 - SHLQ $13,BP:BX + SHLQ $13,BX,BP ANDQ SI,BX ADDQ R15,BX IMUL3Q $19,BP,DX diff --git a/vendor/golang.org/x/crypto/curve25519/square_amd64.s b/vendor/golang.org/x/crypto/curve25519/square_amd64.s index 12f7373..07511a4 100644 --- a/vendor/golang.org/x/crypto/curve25519/square_amd64.s +++ b/vendor/golang.org/x/crypto/curve25519/square_amd64.s @@ -87,18 +87,18 @@ TEXT ·square(SB),7,$0-16 ADDQ AX,R13 ADCQ DX,R14 MOVQ $REDMASK51,SI - SHLQ $13,R8:CX + SHLQ $13,CX,R8 ANDQ SI,CX - SHLQ $13,R10:R9 + SHLQ $13,R9,R10 ANDQ SI,R9 ADDQ R8,R9 - SHLQ $13,R12:R11 + SHLQ $13,R11,R12 ANDQ SI,R11 ADDQ R10,R11 - SHLQ $13,R14:R13 + SHLQ $13,R13,R14 ANDQ SI,R13 ADDQ R12,R13 - SHLQ $13,BX:R15 + SHLQ $13,R15,BX ANDQ SI,R15 ADDQ R14,R15 IMUL3Q $19,BX,DX diff --git a/vendor/golang.org/x/crypto/ed25519/ed25519.go b/vendor/golang.org/x/crypto/ed25519/ed25519.go index d6f683b..c7f8c7e 100644 --- a/vendor/golang.org/x/crypto/ed25519/ed25519.go +++ b/vendor/golang.org/x/crypto/ed25519/ed25519.go @@ -2,6 +2,11 @@ // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. +// In Go 1.13, the ed25519 package was promoted to the standard library as +// crypto/ed25519, and this package became a wrapper for the standard library one. +// +// +build !go1.13 + // Package ed25519 implements the Ed25519 signature algorithm. See // https://ed25519.cr.yp.to/. // diff --git a/vendor/golang.org/x/crypto/ed25519/ed25519_go113.go b/vendor/golang.org/x/crypto/ed25519/ed25519_go113.go new file mode 100644 index 0000000..d1448d8 --- /dev/null +++ b/vendor/golang.org/x/crypto/ed25519/ed25519_go113.go @@ -0,0 +1,73 @@ +// Copyright 2019 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build go1.13 + +// Package ed25519 implements the Ed25519 signature algorithm. See +// https://ed25519.cr.yp.to/. +// +// These functions are also compatible with the “Ed25519” function defined in +// RFC 8032. However, unlike RFC 8032's formulation, this package's private key +// representation includes a public key suffix to make multiple signing +// operations with the same key more efficient. This package refers to the RFC +// 8032 private key as the “seed”. +// +// Beginning with Go 1.13, the functionality of this package was moved to the +// standard library as crypto/ed25519. This package only acts as a compatibility +// wrapper. +package ed25519 + +import ( + "crypto/ed25519" + "io" +) + +const ( + // PublicKeySize is the size, in bytes, of public keys as used in this package. + PublicKeySize = 32 + // PrivateKeySize is the size, in bytes, of private keys as used in this package. + PrivateKeySize = 64 + // SignatureSize is the size, in bytes, of signatures generated and verified by this package. + SignatureSize = 64 + // SeedSize is the size, in bytes, of private key seeds. These are the private key representations used by RFC 8032. + SeedSize = 32 +) + +// PublicKey is the type of Ed25519 public keys. +// +// This type is an alias for crypto/ed25519's PublicKey type. +// See the crypto/ed25519 package for the methods on this type. +type PublicKey = ed25519.PublicKey + +// PrivateKey is the type of Ed25519 private keys. It implements crypto.Signer. +// +// This type is an alias for crypto/ed25519's PrivateKey type. +// See the crypto/ed25519 package for the methods on this type. +type PrivateKey = ed25519.PrivateKey + +// GenerateKey generates a public/private key pair using entropy from rand. +// If rand is nil, crypto/rand.Reader will be used. +func GenerateKey(rand io.Reader) (PublicKey, PrivateKey, error) { + return ed25519.GenerateKey(rand) +} + +// NewKeyFromSeed calculates a private key from a seed. It will panic if +// len(seed) is not SeedSize. This function is provided for interoperability +// with RFC 8032. RFC 8032's private keys correspond to seeds in this +// package. +func NewKeyFromSeed(seed []byte) PrivateKey { + return ed25519.NewKeyFromSeed(seed) +} + +// Sign signs the message with privateKey and returns a signature. It will +// panic if len(privateKey) is not PrivateKeySize. +func Sign(privateKey PrivateKey, message []byte) []byte { + return ed25519.Sign(privateKey, message) +} + +// Verify reports whether sig is a valid signature of message by publicKey. It +// will panic if len(publicKey) is not PublicKeySize. +func Verify(publicKey PublicKey, message, sig []byte) bool { + return ed25519.Verify(publicKey, message, sig) +} diff --git a/vendor/golang.org/x/crypto/internal/chacha20/asm_arm64.s b/vendor/golang.org/x/crypto/internal/chacha20/asm_arm64.s new file mode 100644 index 0000000..b3a16ef --- /dev/null +++ b/vendor/golang.org/x/crypto/internal/chacha20/asm_arm64.s @@ -0,0 +1,308 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build go1.11 +// +build !gccgo,!appengine + +#include "textflag.h" + +#define NUM_ROUNDS 10 + +// func xorKeyStreamVX(dst, src []byte, key *[8]uint32, nonce *[3]uint32, counter *uint32) +TEXT ·xorKeyStreamVX(SB), NOSPLIT, $0 + MOVD dst+0(FP), R1 + MOVD src+24(FP), R2 + MOVD src_len+32(FP), R3 + MOVD key+48(FP), R4 + MOVD nonce+56(FP), R6 + MOVD counter+64(FP), R7 + + MOVD $·constants(SB), R10 + MOVD $·incRotMatrix(SB), R11 + + MOVW (R7), R20 + + AND $~255, R3, R13 + ADD R2, R13, R12 // R12 for block end + AND $255, R3, R13 +loop: + MOVD $NUM_ROUNDS, R21 + VLD1 (R11), [V30.S4, V31.S4] + + // load contants + // VLD4R (R10), [V0.S4, V1.S4, V2.S4, V3.S4] + WORD $0x4D60E940 + + // load keys + // VLD4R 16(R4), [V4.S4, V5.S4, V6.S4, V7.S4] + WORD $0x4DFFE884 + // VLD4R 16(R4), [V8.S4, V9.S4, V10.S4, V11.S4] + WORD $0x4DFFE888 + SUB $32, R4 + + // load counter + nonce + // VLD1R (R7), [V12.S4] + WORD $0x4D40C8EC + + // VLD3R (R6), [V13.S4, V14.S4, V15.S4] + WORD $0x4D40E8CD + + // update counter + VADD V30.S4, V12.S4, V12.S4 + +chacha: + // V0..V3 += V4..V7 + // V12..V15 <<<= ((V12..V15 XOR V0..V3), 16) + VADD V0.S4, V4.S4, V0.S4 + VADD V1.S4, V5.S4, V1.S4 + VADD V2.S4, V6.S4, V2.S4 + VADD V3.S4, V7.S4, V3.S4 + VEOR V12.B16, V0.B16, V12.B16 + VEOR V13.B16, V1.B16, V13.B16 + VEOR V14.B16, V2.B16, V14.B16 + VEOR V15.B16, V3.B16, V15.B16 + VREV32 V12.H8, V12.H8 + VREV32 V13.H8, V13.H8 + VREV32 V14.H8, V14.H8 + VREV32 V15.H8, V15.H8 + // V8..V11 += V12..V15 + // V4..V7 <<<= ((V4..V7 XOR V8..V11), 12) + VADD V8.S4, V12.S4, V8.S4 + VADD V9.S4, V13.S4, V9.S4 + VADD V10.S4, V14.S4, V10.S4 + VADD V11.S4, V15.S4, V11.S4 + VEOR V8.B16, V4.B16, V16.B16 + VEOR V9.B16, V5.B16, V17.B16 + VEOR V10.B16, V6.B16, V18.B16 + VEOR V11.B16, V7.B16, V19.B16 + VSHL $12, V16.S4, V4.S4 + VSHL $12, V17.S4, V5.S4 + VSHL $12, V18.S4, V6.S4 + VSHL $12, V19.S4, V7.S4 + VSRI $20, V16.S4, V4.S4 + VSRI $20, V17.S4, V5.S4 + VSRI $20, V18.S4, V6.S4 + VSRI $20, V19.S4, V7.S4 + + // V0..V3 += V4..V7 + // V12..V15 <<<= ((V12..V15 XOR V0..V3), 8) + VADD V0.S4, V4.S4, V0.S4 + VADD V1.S4, V5.S4, V1.S4 + VADD V2.S4, V6.S4, V2.S4 + VADD V3.S4, V7.S4, V3.S4 + VEOR V12.B16, V0.B16, V12.B16 + VEOR V13.B16, V1.B16, V13.B16 + VEOR V14.B16, V2.B16, V14.B16 + VEOR V15.B16, V3.B16, V15.B16 + VTBL V31.B16, [V12.B16], V12.B16 + VTBL V31.B16, [V13.B16], V13.B16 + VTBL V31.B16, [V14.B16], V14.B16 + VTBL V31.B16, [V15.B16], V15.B16 + + // V8..V11 += V12..V15 + // V4..V7 <<<= ((V4..V7 XOR V8..V11), 7) + VADD V12.S4, V8.S4, V8.S4 + VADD V13.S4, V9.S4, V9.S4 + VADD V14.S4, V10.S4, V10.S4 + VADD V15.S4, V11.S4, V11.S4 + VEOR V8.B16, V4.B16, V16.B16 + VEOR V9.B16, V5.B16, V17.B16 + VEOR V10.B16, V6.B16, V18.B16 + VEOR V11.B16, V7.B16, V19.B16 + VSHL $7, V16.S4, V4.S4 + VSHL $7, V17.S4, V5.S4 + VSHL $7, V18.S4, V6.S4 + VSHL $7, V19.S4, V7.S4 + VSRI $25, V16.S4, V4.S4 + VSRI $25, V17.S4, V5.S4 + VSRI $25, V18.S4, V6.S4 + VSRI $25, V19.S4, V7.S4 + + // V0..V3 += V5..V7, V4 + // V15,V12-V14 <<<= ((V15,V12-V14 XOR V0..V3), 16) + VADD V0.S4, V5.S4, V0.S4 + VADD V1.S4, V6.S4, V1.S4 + VADD V2.S4, V7.S4, V2.S4 + VADD V3.S4, V4.S4, V3.S4 + VEOR V15.B16, V0.B16, V15.B16 + VEOR V12.B16, V1.B16, V12.B16 + VEOR V13.B16, V2.B16, V13.B16 + VEOR V14.B16, V3.B16, V14.B16 + VREV32 V12.H8, V12.H8 + VREV32 V13.H8, V13.H8 + VREV32 V14.H8, V14.H8 + VREV32 V15.H8, V15.H8 + + // V10 += V15; V5 <<<= ((V10 XOR V5), 12) + // ... + VADD V15.S4, V10.S4, V10.S4 + VADD V12.S4, V11.S4, V11.S4 + VADD V13.S4, V8.S4, V8.S4 + VADD V14.S4, V9.S4, V9.S4 + VEOR V10.B16, V5.B16, V16.B16 + VEOR V11.B16, V6.B16, V17.B16 + VEOR V8.B16, V7.B16, V18.B16 + VEOR V9.B16, V4.B16, V19.B16 + VSHL $12, V16.S4, V5.S4 + VSHL $12, V17.S4, V6.S4 + VSHL $12, V18.S4, V7.S4 + VSHL $12, V19.S4, V4.S4 + VSRI $20, V16.S4, V5.S4 + VSRI $20, V17.S4, V6.S4 + VSRI $20, V18.S4, V7.S4 + VSRI $20, V19.S4, V4.S4 + + // V0 += V5; V15 <<<= ((V0 XOR V15), 8) + // ... + VADD V5.S4, V0.S4, V0.S4 + VADD V6.S4, V1.S4, V1.S4 + VADD V7.S4, V2.S4, V2.S4 + VADD V4.S4, V3.S4, V3.S4 + VEOR V0.B16, V15.B16, V15.B16 + VEOR V1.B16, V12.B16, V12.B16 + VEOR V2.B16, V13.B16, V13.B16 + VEOR V3.B16, V14.B16, V14.B16 + VTBL V31.B16, [V12.B16], V12.B16 + VTBL V31.B16, [V13.B16], V13.B16 + VTBL V31.B16, [V14.B16], V14.B16 + VTBL V31.B16, [V15.B16], V15.B16 + + // V10 += V15; V5 <<<= ((V10 XOR V5), 7) + // ... + VADD V15.S4, V10.S4, V10.S4 + VADD V12.S4, V11.S4, V11.S4 + VADD V13.S4, V8.S4, V8.S4 + VADD V14.S4, V9.S4, V9.S4 + VEOR V10.B16, V5.B16, V16.B16 + VEOR V11.B16, V6.B16, V17.B16 + VEOR V8.B16, V7.B16, V18.B16 + VEOR V9.B16, V4.B16, V19.B16 + VSHL $7, V16.S4, V5.S4 + VSHL $7, V17.S4, V6.S4 + VSHL $7, V18.S4, V7.S4 + VSHL $7, V19.S4, V4.S4 + VSRI $25, V16.S4, V5.S4 + VSRI $25, V17.S4, V6.S4 + VSRI $25, V18.S4, V7.S4 + VSRI $25, V19.S4, V4.S4 + + SUB $1, R21 + CBNZ R21, chacha + + // VLD4R (R10), [V16.S4, V17.S4, V18.S4, V19.S4] + WORD $0x4D60E950 + + // VLD4R 16(R4), [V20.S4, V21.S4, V22.S4, V23.S4] + WORD $0x4DFFE894 + VADD V30.S4, V12.S4, V12.S4 + VADD V16.S4, V0.S4, V0.S4 + VADD V17.S4, V1.S4, V1.S4 + VADD V18.S4, V2.S4, V2.S4 + VADD V19.S4, V3.S4, V3.S4 + // VLD4R 16(R4), [V24.S4, V25.S4, V26.S4, V27.S4] + WORD $0x4DFFE898 + // restore R4 + SUB $32, R4 + + // load counter + nonce + // VLD1R (R7), [V28.S4] + WORD $0x4D40C8FC + // VLD3R (R6), [V29.S4, V30.S4, V31.S4] + WORD $0x4D40E8DD + + VADD V20.S4, V4.S4, V4.S4 + VADD V21.S4, V5.S4, V5.S4 + VADD V22.S4, V6.S4, V6.S4 + VADD V23.S4, V7.S4, V7.S4 + VADD V24.S4, V8.S4, V8.S4 + VADD V25.S4, V9.S4, V9.S4 + VADD V26.S4, V10.S4, V10.S4 + VADD V27.S4, V11.S4, V11.S4 + VADD V28.S4, V12.S4, V12.S4 + VADD V29.S4, V13.S4, V13.S4 + VADD V30.S4, V14.S4, V14.S4 + VADD V31.S4, V15.S4, V15.S4 + + VZIP1 V1.S4, V0.S4, V16.S4 + VZIP2 V1.S4, V0.S4, V17.S4 + VZIP1 V3.S4, V2.S4, V18.S4 + VZIP2 V3.S4, V2.S4, V19.S4 + VZIP1 V5.S4, V4.S4, V20.S4 + VZIP2 V5.S4, V4.S4, V21.S4 + VZIP1 V7.S4, V6.S4, V22.S4 + VZIP2 V7.S4, V6.S4, V23.S4 + VZIP1 V9.S4, V8.S4, V24.S4 + VZIP2 V9.S4, V8.S4, V25.S4 + VZIP1 V11.S4, V10.S4, V26.S4 + VZIP2 V11.S4, V10.S4, V27.S4 + VZIP1 V13.S4, V12.S4, V28.S4 + VZIP2 V13.S4, V12.S4, V29.S4 + VZIP1 V15.S4, V14.S4, V30.S4 + VZIP2 V15.S4, V14.S4, V31.S4 + VZIP1 V18.D2, V16.D2, V0.D2 + VZIP2 V18.D2, V16.D2, V4.D2 + VZIP1 V19.D2, V17.D2, V8.D2 + VZIP2 V19.D2, V17.D2, V12.D2 + VLD1.P 64(R2), [V16.B16, V17.B16, V18.B16, V19.B16] + + VZIP1 V22.D2, V20.D2, V1.D2 + VZIP2 V22.D2, V20.D2, V5.D2 + VZIP1 V23.D2, V21.D2, V9.D2 + VZIP2 V23.D2, V21.D2, V13.D2 + VLD1.P 64(R2), [V20.B16, V21.B16, V22.B16, V23.B16] + VZIP1 V26.D2, V24.D2, V2.D2 + VZIP2 V26.D2, V24.D2, V6.D2 + VZIP1 V27.D2, V25.D2, V10.D2 + VZIP2 V27.D2, V25.D2, V14.D2 + VLD1.P 64(R2), [V24.B16, V25.B16, V26.B16, V27.B16] + VZIP1 V30.D2, V28.D2, V3.D2 + VZIP2 V30.D2, V28.D2, V7.D2 + VZIP1 V31.D2, V29.D2, V11.D2 + VZIP2 V31.D2, V29.D2, V15.D2 + VLD1.P 64(R2), [V28.B16, V29.B16, V30.B16, V31.B16] + VEOR V0.B16, V16.B16, V16.B16 + VEOR V1.B16, V17.B16, V17.B16 + VEOR V2.B16, V18.B16, V18.B16 + VEOR V3.B16, V19.B16, V19.B16 + VST1.P [V16.B16, V17.B16, V18.B16, V19.B16], 64(R1) + VEOR V4.B16, V20.B16, V20.B16 + VEOR V5.B16, V21.B16, V21.B16 + VEOR V6.B16, V22.B16, V22.B16 + VEOR V7.B16, V23.B16, V23.B16 + VST1.P [V20.B16, V21.B16, V22.B16, V23.B16], 64(R1) + VEOR V8.B16, V24.B16, V24.B16 + VEOR V9.B16, V25.B16, V25.B16 + VEOR V10.B16, V26.B16, V26.B16 + VEOR V11.B16, V27.B16, V27.B16 + VST1.P [V24.B16, V25.B16, V26.B16, V27.B16], 64(R1) + VEOR V12.B16, V28.B16, V28.B16 + VEOR V13.B16, V29.B16, V29.B16 + VEOR V14.B16, V30.B16, V30.B16 + VEOR V15.B16, V31.B16, V31.B16 + VST1.P [V28.B16, V29.B16, V30.B16, V31.B16], 64(R1) + + ADD $4, R20 + MOVW R20, (R7) // update counter + + CMP R2, R12 + BGT loop + + RET + + +DATA ·constants+0x00(SB)/4, $0x61707865 +DATA ·constants+0x04(SB)/4, $0x3320646e +DATA ·constants+0x08(SB)/4, $0x79622d32 +DATA ·constants+0x0c(SB)/4, $0x6b206574 +GLOBL ·constants(SB), NOPTR|RODATA, $32 + +DATA ·incRotMatrix+0x00(SB)/4, $0x00000000 +DATA ·incRotMatrix+0x04(SB)/4, $0x00000001 +DATA ·incRotMatrix+0x08(SB)/4, $0x00000002 +DATA ·incRotMatrix+0x0c(SB)/4, $0x00000003 +DATA ·incRotMatrix+0x10(SB)/4, $0x02010003 +DATA ·incRotMatrix+0x14(SB)/4, $0x06050407 +DATA ·incRotMatrix+0x18(SB)/4, $0x0A09080B +DATA ·incRotMatrix+0x1c(SB)/4, $0x0E0D0C0F +GLOBL ·incRotMatrix(SB), NOPTR|RODATA, $32 diff --git a/vendor/golang.org/x/crypto/internal/chacha20/asm_ppc64le.s b/vendor/golang.org/x/crypto/internal/chacha20/asm_ppc64le.s new file mode 100644 index 0000000..5441852 --- /dev/null +++ b/vendor/golang.org/x/crypto/internal/chacha20/asm_ppc64le.s @@ -0,0 +1,465 @@ +// Copyright 2019 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Based on CRYPTOGAMS code with the following comment: +// # ==================================================================== +// # Written by Andy Polyakov for the OpenSSL +// # project. The module is, however, dual licensed under OpenSSL and +// # CRYPTOGAMS licenses depending on where you obtain it. For further +// # details see http://www.openssl.org/~appro/cryptogams/. +// # ==================================================================== + +// Code for the perl script that generates the ppc64 assembler +// can be found in the cryptogams repository at the link below. It is based on +// the original from openssl. + +// https://github.com/dot-asm/cryptogams/commit/a60f5b50ed908e91 + +// The differences in this and the original implementation are +// due to the calling conventions and initialization of constants. + +// +build ppc64le,!gccgo,!appengine + +#include "textflag.h" + +#define OUT R3 +#define INP R4 +#define LEN R5 +#define KEY R6 +#define CNT R7 +#define TMP R15 + +#define CONSTBASE R16 + +#define X0 R11 +#define X1 R12 +#define X2 R14 +#define X3 R15 +#define X4 R16 +#define X5 R17 +#define X6 R18 +#define X7 R19 +#define X8 R20 +#define X9 R21 +#define X10 R22 +#define X11 R23 +#define X12 R24 +#define X13 R25 +#define X14 R26 +#define X15 R27 + + +DATA consts<>+0x00(SB)/8, $0x3320646e61707865 +DATA consts<>+0x08(SB)/8, $0x6b20657479622d32 +DATA consts<>+0x10(SB)/8, $0x0000000000000001 +DATA consts<>+0x18(SB)/8, $0x0000000000000000 +DATA consts<>+0x20(SB)/8, $0x0000000000000004 +DATA consts<>+0x28(SB)/8, $0x0000000000000000 +DATA consts<>+0x30(SB)/8, $0x0a0b08090e0f0c0d +DATA consts<>+0x38(SB)/8, $0x0203000106070405 +DATA consts<>+0x40(SB)/8, $0x090a0b080d0e0f0c +DATA consts<>+0x48(SB)/8, $0x0102030005060704 +DATA consts<>+0x50(SB)/8, $0x6170786561707865 +DATA consts<>+0x58(SB)/8, $0x6170786561707865 +DATA consts<>+0x60(SB)/8, $0x3320646e3320646e +DATA consts<>+0x68(SB)/8, $0x3320646e3320646e +DATA consts<>+0x70(SB)/8, $0x79622d3279622d32 +DATA consts<>+0x78(SB)/8, $0x79622d3279622d32 +DATA consts<>+0x80(SB)/8, $0x6b2065746b206574 +DATA consts<>+0x88(SB)/8, $0x6b2065746b206574 +DATA consts<>+0x90(SB)/8, $0x0000000100000000 +DATA consts<>+0x98(SB)/8, $0x0000000300000002 +GLOBL consts<>(SB), RODATA, $0xa0 + +//func chaCha20_ctr32_vsx(out, inp []byte, len int, key *[32]byte, counter *[16]byte) +TEXT ·chaCha20_ctr32_vsx(SB),NOSPLIT,$64-40 + MOVD out+0(FP), OUT + MOVD inp+8(FP), INP + MOVD len+16(FP), LEN + MOVD key+24(FP), KEY + MOVD cnt+32(FP), CNT + + // Addressing for constants + MOVD $consts<>+0x00(SB), CONSTBASE + MOVD $16, R8 + MOVD $32, R9 + MOVD $48, R10 + MOVD $64, R11 + // V16 + LXVW4X (CONSTBASE)(R0), VS48 + ADD $80,CONSTBASE + + // Load key into V17,V18 + LXVW4X (KEY)(R0), VS49 + LXVW4X (KEY)(R8), VS50 + + // Load CNT, NONCE into V19 + LXVW4X (CNT)(R0), VS51 + + // Clear V27 + VXOR V27, V27, V27 + + // V28 + LXVW4X (CONSTBASE)(R11), VS60 + + // splat slot from V19 -> V26 + VSPLTW $0, V19, V26 + + VSLDOI $4, V19, V27, V19 + VSLDOI $12, V27, V19, V19 + + VADDUWM V26, V28, V26 + + MOVD $10, R14 + MOVD R14, CTR + +loop_outer_vsx: + // V0, V1, V2, V3 + LXVW4X (R0)(CONSTBASE), VS32 + LXVW4X (R8)(CONSTBASE), VS33 + LXVW4X (R9)(CONSTBASE), VS34 + LXVW4X (R10)(CONSTBASE), VS35 + + // splat values from V17, V18 into V4-V11 + VSPLTW $0, V17, V4 + VSPLTW $1, V17, V5 + VSPLTW $2, V17, V6 + VSPLTW $3, V17, V7 + VSPLTW $0, V18, V8 + VSPLTW $1, V18, V9 + VSPLTW $2, V18, V10 + VSPLTW $3, V18, V11 + + // VOR + VOR V26, V26, V12 + + // splat values from V19 -> V13, V14, V15 + VSPLTW $1, V19, V13 + VSPLTW $2, V19, V14 + VSPLTW $3, V19, V15 + + // splat const values + VSPLTISW $-16, V27 + VSPLTISW $12, V28 + VSPLTISW $8, V29 + VSPLTISW $7, V30 + +loop_vsx: + VADDUWM V0, V4, V0 + VADDUWM V1, V5, V1 + VADDUWM V2, V6, V2 + VADDUWM V3, V7, V3 + + VXOR V12, V0, V12 + VXOR V13, V1, V13 + VXOR V14, V2, V14 + VXOR V15, V3, V15 + + VRLW V12, V27, V12 + VRLW V13, V27, V13 + VRLW V14, V27, V14 + VRLW V15, V27, V15 + + VADDUWM V8, V12, V8 + VADDUWM V9, V13, V9 + VADDUWM V10, V14, V10 + VADDUWM V11, V15, V11 + + VXOR V4, V8, V4 + VXOR V5, V9, V5 + VXOR V6, V10, V6 + VXOR V7, V11, V7 + + VRLW V4, V28, V4 + VRLW V5, V28, V5 + VRLW V6, V28, V6 + VRLW V7, V28, V7 + + VADDUWM V0, V4, V0 + VADDUWM V1, V5, V1 + VADDUWM V2, V6, V2 + VADDUWM V3, V7, V3 + + VXOR V12, V0, V12 + VXOR V13, V1, V13 + VXOR V14, V2, V14 + VXOR V15, V3, V15 + + VRLW V12, V29, V12 + VRLW V13, V29, V13 + VRLW V14, V29, V14 + VRLW V15, V29, V15 + + VADDUWM V8, V12, V8 + VADDUWM V9, V13, V9 + VADDUWM V10, V14, V10 + VADDUWM V11, V15, V11 + + VXOR V4, V8, V4 + VXOR V5, V9, V5 + VXOR V6, V10, V6 + VXOR V7, V11, V7 + + VRLW V4, V30, V4 + VRLW V5, V30, V5 + VRLW V6, V30, V6 + VRLW V7, V30, V7 + + VADDUWM V0, V5, V0 + VADDUWM V1, V6, V1 + VADDUWM V2, V7, V2 + VADDUWM V3, V4, V3 + + VXOR V15, V0, V15 + VXOR V12, V1, V12 + VXOR V13, V2, V13 + VXOR V14, V3, V14 + + VRLW V15, V27, V15 + VRLW V12, V27, V12 + VRLW V13, V27, V13 + VRLW V14, V27, V14 + + VADDUWM V10, V15, V10 + VADDUWM V11, V12, V11 + VADDUWM V8, V13, V8 + VADDUWM V9, V14, V9 + + VXOR V5, V10, V5 + VXOR V6, V11, V6 + VXOR V7, V8, V7 + VXOR V4, V9, V4 + + VRLW V5, V28, V5 + VRLW V6, V28, V6 + VRLW V7, V28, V7 + VRLW V4, V28, V4 + + VADDUWM V0, V5, V0 + VADDUWM V1, V6, V1 + VADDUWM V2, V7, V2 + VADDUWM V3, V4, V3 + + VXOR V15, V0, V15 + VXOR V12, V1, V12 + VXOR V13, V2, V13 + VXOR V14, V3, V14 + + VRLW V15, V29, V15 + VRLW V12, V29, V12 + VRLW V13, V29, V13 + VRLW V14, V29, V14 + + VADDUWM V10, V15, V10 + VADDUWM V11, V12, V11 + VADDUWM V8, V13, V8 + VADDUWM V9, V14, V9 + + VXOR V5, V10, V5 + VXOR V6, V11, V6 + VXOR V7, V8, V7 + VXOR V4, V9, V4 + + VRLW V5, V30, V5 + VRLW V6, V30, V6 + VRLW V7, V30, V7 + VRLW V4, V30, V4 + BC 16, LT, loop_vsx + + VADDUWM V12, V26, V12 + + WORD $0x13600F8C // VMRGEW V0, V1, V27 + WORD $0x13821F8C // VMRGEW V2, V3, V28 + + WORD $0x10000E8C // VMRGOW V0, V1, V0 + WORD $0x10421E8C // VMRGOW V2, V3, V2 + + WORD $0x13A42F8C // VMRGEW V4, V5, V29 + WORD $0x13C63F8C // VMRGEW V6, V7, V30 + + XXPERMDI VS32, VS34, $0, VS33 + XXPERMDI VS32, VS34, $3, VS35 + XXPERMDI VS59, VS60, $0, VS32 + XXPERMDI VS59, VS60, $3, VS34 + + WORD $0x10842E8C // VMRGOW V4, V5, V4 + WORD $0x10C63E8C // VMRGOW V6, V7, V6 + + WORD $0x13684F8C // VMRGEW V8, V9, V27 + WORD $0x138A5F8C // VMRGEW V10, V11, V28 + + XXPERMDI VS36, VS38, $0, VS37 + XXPERMDI VS36, VS38, $3, VS39 + XXPERMDI VS61, VS62, $0, VS36 + XXPERMDI VS61, VS62, $3, VS38 + + WORD $0x11084E8C // VMRGOW V8, V9, V8 + WORD $0x114A5E8C // VMRGOW V10, V11, V10 + + WORD $0x13AC6F8C // VMRGEW V12, V13, V29 + WORD $0x13CE7F8C // VMRGEW V14, V15, V30 + + XXPERMDI VS40, VS42, $0, VS41 + XXPERMDI VS40, VS42, $3, VS43 + XXPERMDI VS59, VS60, $0, VS40 + XXPERMDI VS59, VS60, $3, VS42 + + WORD $0x118C6E8C // VMRGOW V12, V13, V12 + WORD $0x11CE7E8C // VMRGOW V14, V15, V14 + + VSPLTISW $4, V27 + VADDUWM V26, V27, V26 + + XXPERMDI VS44, VS46, $0, VS45 + XXPERMDI VS44, VS46, $3, VS47 + XXPERMDI VS61, VS62, $0, VS44 + XXPERMDI VS61, VS62, $3, VS46 + + VADDUWM V0, V16, V0 + VADDUWM V4, V17, V4 + VADDUWM V8, V18, V8 + VADDUWM V12, V19, V12 + + CMPU LEN, $64 + BLT tail_vsx + + // Bottom of loop + LXVW4X (INP)(R0), VS59 + LXVW4X (INP)(R8), VS60 + LXVW4X (INP)(R9), VS61 + LXVW4X (INP)(R10), VS62 + + VXOR V27, V0, V27 + VXOR V28, V4, V28 + VXOR V29, V8, V29 + VXOR V30, V12, V30 + + STXVW4X VS59, (OUT)(R0) + STXVW4X VS60, (OUT)(R8) + ADD $64, INP + STXVW4X VS61, (OUT)(R9) + ADD $-64, LEN + STXVW4X VS62, (OUT)(R10) + ADD $64, OUT + BEQ done_vsx + + VADDUWM V1, V16, V0 + VADDUWM V5, V17, V4 + VADDUWM V9, V18, V8 + VADDUWM V13, V19, V12 + + CMPU LEN, $64 + BLT tail_vsx + + LXVW4X (INP)(R0), VS59 + LXVW4X (INP)(R8), VS60 + LXVW4X (INP)(R9), VS61 + LXVW4X (INP)(R10), VS62 + VXOR V27, V0, V27 + + VXOR V28, V4, V28 + VXOR V29, V8, V29 + VXOR V30, V12, V30 + + STXVW4X VS59, (OUT)(R0) + STXVW4X VS60, (OUT)(R8) + ADD $64, INP + STXVW4X VS61, (OUT)(R9) + ADD $-64, LEN + STXVW4X VS62, (OUT)(V10) + ADD $64, OUT + BEQ done_vsx + + VADDUWM V2, V16, V0 + VADDUWM V6, V17, V4 + VADDUWM V10, V18, V8 + VADDUWM V14, V19, V12 + + CMPU LEN, $64 + BLT tail_vsx + + LXVW4X (INP)(R0), VS59 + LXVW4X (INP)(R8), VS60 + LXVW4X (INP)(R9), VS61 + LXVW4X (INP)(R10), VS62 + + VXOR V27, V0, V27 + VXOR V28, V4, V28 + VXOR V29, V8, V29 + VXOR V30, V12, V30 + + STXVW4X VS59, (OUT)(R0) + STXVW4X VS60, (OUT)(R8) + ADD $64, INP + STXVW4X VS61, (OUT)(R9) + ADD $-64, LEN + STXVW4X VS62, (OUT)(R10) + ADD $64, OUT + BEQ done_vsx + + VADDUWM V3, V16, V0 + VADDUWM V7, V17, V4 + VADDUWM V11, V18, V8 + VADDUWM V15, V19, V12 + + CMPU LEN, $64 + BLT tail_vsx + + LXVW4X (INP)(R0), VS59 + LXVW4X (INP)(R8), VS60 + LXVW4X (INP)(R9), VS61 + LXVW4X (INP)(R10), VS62 + + VXOR V27, V0, V27 + VXOR V28, V4, V28 + VXOR V29, V8, V29 + VXOR V30, V12, V30 + + STXVW4X VS59, (OUT)(R0) + STXVW4X VS60, (OUT)(R8) + ADD $64, INP + STXVW4X VS61, (OUT)(R9) + ADD $-64, LEN + STXVW4X VS62, (OUT)(R10) + ADD $64, OUT + + MOVD $10, R14 + MOVD R14, CTR + BNE loop_outer_vsx + +done_vsx: + // Increment counter by 4 + MOVD (CNT), R14 + ADD $4, R14 + MOVD R14, (CNT) + RET + +tail_vsx: + ADD $32, R1, R11 + MOVD LEN, CTR + + // Save values on stack to copy from + STXVW4X VS32, (R11)(R0) + STXVW4X VS36, (R11)(R8) + STXVW4X VS40, (R11)(R9) + STXVW4X VS44, (R11)(R10) + ADD $-1, R11, R12 + ADD $-1, INP + ADD $-1, OUT + +looptail_vsx: + // Copying the result to OUT + // in bytes. + MOVBZU 1(R12), KEY + MOVBZU 1(INP), TMP + XOR KEY, TMP, KEY + MOVBU KEY, 1(OUT) + BC 16, LT, looptail_vsx + + // Clear the stack values + STXVW4X VS48, (R11)(R0) + STXVW4X VS48, (R11)(R8) + STXVW4X VS48, (R11)(R9) + STXVW4X VS48, (R11)(R10) + BR done_vsx diff --git a/vendor/golang.org/x/crypto/internal/chacha20/chacha_arm64.go b/vendor/golang.org/x/crypto/internal/chacha20/chacha_arm64.go new file mode 100644 index 0000000..ad74e23 --- /dev/null +++ b/vendor/golang.org/x/crypto/internal/chacha20/chacha_arm64.go @@ -0,0 +1,31 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build go1.11 +// +build !gccgo + +package chacha20 + +const ( + haveAsm = true + bufSize = 256 +) + +//go:noescape +func xorKeyStreamVX(dst, src []byte, key *[8]uint32, nonce *[3]uint32, counter *uint32) + +func (c *Cipher) xorKeyStreamAsm(dst, src []byte) { + + if len(src) >= bufSize { + xorKeyStreamVX(dst, src, &c.key, &c.nonce, &c.counter) + } + + if len(src)%bufSize != 0 { + i := len(src) - len(src)%bufSize + c.buf = [bufSize]byte{} + copy(c.buf[:], src[i:]) + xorKeyStreamVX(c.buf[:], c.buf[:], &c.key, &c.nonce, &c.counter) + c.len = bufSize - copy(dst[i:], c.buf[:len(src)%bufSize]) + } +} diff --git a/vendor/golang.org/x/crypto/internal/chacha20/chacha_noasm.go b/vendor/golang.org/x/crypto/internal/chacha20/chacha_noasm.go index 91520d1..fc26825 100644 --- a/vendor/golang.org/x/crypto/internal/chacha20/chacha_noasm.go +++ b/vendor/golang.org/x/crypto/internal/chacha20/chacha_noasm.go @@ -2,7 +2,7 @@ // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. -// +build !s390x gccgo appengine +// +build !arm64,!s390x,!ppc64le arm64,!go1.11 gccgo appengine package chacha20 diff --git a/vendor/golang.org/x/crypto/internal/chacha20/chacha_ppc64le.go b/vendor/golang.org/x/crypto/internal/chacha20/chacha_ppc64le.go new file mode 100644 index 0000000..d38a7d3 --- /dev/null +++ b/vendor/golang.org/x/crypto/internal/chacha20/chacha_ppc64le.go @@ -0,0 +1,53 @@ +// Copyright 2019 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build ppc64le,!gccgo,!appengine + +package chacha20 + +import ( + "encoding/binary" +) + +var haveAsm = true + +const bufSize = 256 + +//go:noescape +func chaCha20_ctr32_vsx(out, inp *byte, len int, key *[8]uint32, counter *uint32) + +func (c *Cipher) xorKeyStreamAsm(dst, src []byte) { + // This implementation can handle buffers that aren't multiples of + // 256. + if len(src) >= bufSize { + chaCha20_ctr32_vsx(&dst[0], &src[0], len(src), &c.key, &c.counter) + } else if len(src)%bufSize != 0 { + chaCha20_ctr32_vsx(&c.buf[0], &c.buf[0], bufSize, &c.key, &c.counter) + start := len(src) - len(src)%bufSize + ts, td, tb := src[start:], dst[start:], c.buf[:] + // Unroll loop to XOR 32 bytes per iteration. + for i := 0; i < len(ts)-32; i += 32 { + td, tb = td[:len(ts)], tb[:len(ts)] // bounds check elimination + s0 := binary.LittleEndian.Uint64(ts[0:8]) + s1 := binary.LittleEndian.Uint64(ts[8:16]) + s2 := binary.LittleEndian.Uint64(ts[16:24]) + s3 := binary.LittleEndian.Uint64(ts[24:32]) + b0 := binary.LittleEndian.Uint64(tb[0:8]) + b1 := binary.LittleEndian.Uint64(tb[8:16]) + b2 := binary.LittleEndian.Uint64(tb[16:24]) + b3 := binary.LittleEndian.Uint64(tb[24:32]) + binary.LittleEndian.PutUint64(td[0:8], s0^b0) + binary.LittleEndian.PutUint64(td[8:16], s1^b1) + binary.LittleEndian.PutUint64(td[16:24], s2^b2) + binary.LittleEndian.PutUint64(td[24:32], s3^b3) + ts, td, tb = ts[32:], td[32:], tb[32:] + } + td, tb = td[:len(ts)], tb[:len(ts)] // bounds check elimination + for i, v := range ts { + td[i] = tb[i] ^ v + } + c.len = bufSize - (len(src) % bufSize) + } + +} diff --git a/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.go b/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.go index 0c1c671..aad645b 100644 --- a/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.go +++ b/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.go @@ -6,15 +6,14 @@ package chacha20 -var haveAsm = hasVectorFacility() +import ( + "golang.org/x/sys/cpu" +) + +var haveAsm = cpu.S390X.HasVX const bufSize = 256 -// hasVectorFacility reports whether the machine supports the vector -// facility (vx). -// Implementation in asm_s390x.s. -func hasVectorFacility() bool - // xorKeyStreamVX is an assembly implementation of XORKeyStream. It must only // be called when the vector facility is available. // Implementation in asm_s390x.s. diff --git a/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.s b/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.s index 98427c5..57df404 100644 --- a/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.s +++ b/vendor/golang.org/x/crypto/internal/chacha20/chacha_s390x.s @@ -258,26 +258,3 @@ tail: MOVD R8, R3 MOVD $0, R4 JMP continue - -// func hasVectorFacility() bool -TEXT ·hasVectorFacility(SB), NOSPLIT, $24-1 - MOVD $x-24(SP), R1 - XC $24, 0(R1), 0(R1) // clear the storage - MOVD $2, R0 // R0 is the number of double words stored -1 - WORD $0xB2B01000 // STFLE 0(R1) - XOR R0, R0 // reset the value of R0 - MOVBZ z-8(SP), R1 - AND $0x40, R1 - BEQ novector - -vectorinstalled: - // check if the vector instruction has been enabled - VLEIB $0, $0xF, V16 - VLGVB $0, V16, R1 - CMPBNE R1, $0xF, novector - MOVB $1, ret+0(FP) // have vx - RET - -novector: - MOVB $0, ret+0(FP) // no vx - RET diff --git a/vendor/golang.org/x/crypto/poly1305/mac_noasm.go b/vendor/golang.org/x/crypto/poly1305/mac_noasm.go new file mode 100644 index 0000000..a8dd589 --- /dev/null +++ b/vendor/golang.org/x/crypto/poly1305/mac_noasm.go @@ -0,0 +1,11 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build !amd64,!ppc64le gccgo appengine + +package poly1305 + +type mac struct{ macGeneric } + +func newMAC(key *[32]byte) mac { return mac{newMACGeneric(key)} } diff --git a/vendor/golang.org/x/crypto/poly1305/poly1305.go b/vendor/golang.org/x/crypto/poly1305/poly1305.go index f562fa5..d076a56 100644 --- a/vendor/golang.org/x/crypto/poly1305/poly1305.go +++ b/vendor/golang.org/x/crypto/poly1305/poly1305.go @@ -2,21 +2,19 @@ // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. -/* -Package poly1305 implements Poly1305 one-time message authentication code as -specified in https://cr.yp.to/mac/poly1305-20050329.pdf. - -Poly1305 is a fast, one-time authentication function. It is infeasible for an -attacker to generate an authenticator for a message without the key. However, a -key must only be used for a single message. Authenticating two different -messages with the same key allows an attacker to forge authenticators for other -messages with the same key. - -Poly1305 was originally coupled with AES in order to make Poly1305-AES. AES was -used with a fixed key in order to generate one-time keys from an nonce. -However, in this package AES isn't used and the one-time key is specified -directly. -*/ +// Package poly1305 implements Poly1305 one-time message authentication code as +// specified in https://cr.yp.to/mac/poly1305-20050329.pdf. +// +// Poly1305 is a fast, one-time authentication function. It is infeasible for an +// attacker to generate an authenticator for a message without the key. However, a +// key must only be used for a single message. Authenticating two different +// messages with the same key allows an attacker to forge authenticators for other +// messages with the same key. +// +// Poly1305 was originally coupled with AES in order to make Poly1305-AES. AES was +// used with a fixed key in order to generate one-time keys from an nonce. +// However, in this package AES isn't used and the one-time key is specified +// directly. package poly1305 // import "golang.org/x/crypto/poly1305" import "crypto/subtle" @@ -31,3 +29,55 @@ func Verify(mac *[16]byte, m []byte, key *[32]byte) bool { Sum(&tmp, m, key) return subtle.ConstantTimeCompare(tmp[:], mac[:]) == 1 } + +// New returns a new MAC computing an authentication +// tag of all data written to it with the given key. +// This allows writing the message progressively instead +// of passing it as a single slice. Common users should use +// the Sum function instead. +// +// The key must be unique for each message, as authenticating +// two different messages with the same key allows an attacker +// to forge messages at will. +func New(key *[32]byte) *MAC { + return &MAC{ + mac: newMAC(key), + finalized: false, + } +} + +// MAC is an io.Writer computing an authentication tag +// of the data written to it. +// +// MAC cannot be used like common hash.Hash implementations, +// because using a poly1305 key twice breaks its security. +// Therefore writing data to a running MAC after calling +// Sum causes it to panic. +type MAC struct { + mac // platform-dependent implementation + + finalized bool +} + +// Size returns the number of bytes Sum will return. +func (h *MAC) Size() int { return TagSize } + +// Write adds more data to the running message authentication code. +// It never returns an error. +// +// It must not be called after the first call of Sum. +func (h *MAC) Write(p []byte) (n int, err error) { + if h.finalized { + panic("poly1305: write to MAC after Sum") + } + return h.mac.Write(p) +} + +// Sum computes the authenticator of all data written to the +// message authentication code. +func (h *MAC) Sum(b []byte) []byte { + var mac [TagSize]byte + h.mac.Sum(&mac) + h.finalized = true + return append(b, mac[:]...) +} diff --git a/vendor/golang.org/x/crypto/poly1305/sum_amd64.go b/vendor/golang.org/x/crypto/poly1305/sum_amd64.go index 4dd72fe..2dbf42a 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_amd64.go +++ b/vendor/golang.org/x/crypto/poly1305/sum_amd64.go @@ -6,17 +6,63 @@ package poly1305 -// This function is implemented in sum_amd64.s //go:noescape -func poly1305(out *[16]byte, m *byte, mlen uint64, key *[32]byte) +func initialize(state *[7]uint64, key *[32]byte) + +//go:noescape +func update(state *[7]uint64, msg []byte) + +//go:noescape +func finalize(tag *[TagSize]byte, state *[7]uint64) // Sum generates an authenticator for m using a one-time key and puts the // 16-byte result into out. Authenticating two different messages with the same // key allows an attacker to forge messages at will. func Sum(out *[16]byte, m []byte, key *[32]byte) { - var mPtr *byte - if len(m) > 0 { - mPtr = &m[0] - } - poly1305(out, mPtr, uint64(len(m)), key) + h := newMAC(key) + h.Write(m) + h.Sum(out) +} + +func newMAC(key *[32]byte) (h mac) { + initialize(&h.state, key) + return +} + +type mac struct { + state [7]uint64 // := uint64{ h0, h1, h2, r0, r1, pad0, pad1 } + + buffer [TagSize]byte + offset int +} + +func (h *mac) Write(p []byte) (n int, err error) { + n = len(p) + if h.offset > 0 { + remaining := TagSize - h.offset + if n < remaining { + h.offset += copy(h.buffer[h.offset:], p) + return n, nil + } + copy(h.buffer[h.offset:], p[:remaining]) + p = p[remaining:] + h.offset = 0 + update(&h.state, h.buffer[:]) + } + if nn := len(p) - (len(p) % TagSize); nn > 0 { + update(&h.state, p[:nn]) + p = p[nn:] + } + if len(p) > 0 { + h.offset += copy(h.buffer[h.offset:], p) + } + return n, nil +} + +func (h *mac) Sum(out *[16]byte) { + state := h.state + if h.offset > 0 { + update(&state, h.buffer[:h.offset]) + } + finalize(out, &state) } diff --git a/vendor/golang.org/x/crypto/poly1305/sum_amd64.s b/vendor/golang.org/x/crypto/poly1305/sum_amd64.s index 2edae63..7d600f1 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_amd64.s +++ b/vendor/golang.org/x/crypto/poly1305/sum_amd64.s @@ -58,20 +58,17 @@ DATA ·poly1305Mask<>+0x00(SB)/8, $0x0FFFFFFC0FFFFFFF DATA ·poly1305Mask<>+0x08(SB)/8, $0x0FFFFFFC0FFFFFFC GLOBL ·poly1305Mask<>(SB), RODATA, $16 -// func poly1305(out *[16]byte, m *byte, mlen uint64, key *[32]key) -TEXT ·poly1305(SB), $0-32 - MOVQ out+0(FP), DI - MOVQ m+8(FP), SI - MOVQ mlen+16(FP), R15 - MOVQ key+24(FP), AX +// func update(state *[7]uint64, msg []byte) +TEXT ·update(SB), $0-32 + MOVQ state+0(FP), DI + MOVQ msg_base+8(FP), SI + MOVQ msg_len+16(FP), R15 - MOVQ 0(AX), R11 - MOVQ 8(AX), R12 - ANDQ ·poly1305Mask<>(SB), R11 // r0 - ANDQ ·poly1305Mask<>+8(SB), R12 // r1 - XORQ R8, R8 // h0 - XORQ R9, R9 // h1 - XORQ R10, R10 // h2 + MOVQ 0(DI), R8 // h0 + MOVQ 8(DI), R9 // h1 + MOVQ 16(DI), R10 // h2 + MOVQ 24(DI), R11 // r0 + MOVQ 32(DI), R12 // r1 CMPQ R15, $16 JB bytes_between_0_and_15 @@ -109,16 +106,42 @@ flush_buffer: JMP multiply done: - MOVQ R8, AX - MOVQ R9, BX + MOVQ R8, 0(DI) + MOVQ R9, 8(DI) + MOVQ R10, 16(DI) + RET + +// func initialize(state *[7]uint64, key *[32]byte) +TEXT ·initialize(SB), $0-16 + MOVQ state+0(FP), DI + MOVQ key+8(FP), SI + + // state[0...7] is initialized with zero + MOVOU 0(SI), X0 + MOVOU 16(SI), X1 + MOVOU ·poly1305Mask<>(SB), X2 + PAND X2, X0 + MOVOU X0, 24(DI) + MOVOU X1, 40(DI) + RET + +// func finalize(tag *[TagSize]byte, state *[7]uint64) +TEXT ·finalize(SB), $0-16 + MOVQ tag+0(FP), DI + MOVQ state+8(FP), SI + + MOVQ 0(SI), AX + MOVQ 8(SI), BX + MOVQ 16(SI), CX + MOVQ AX, R8 + MOVQ BX, R9 SUBQ $0xFFFFFFFFFFFFFFFB, AX SBBQ $0xFFFFFFFFFFFFFFFF, BX - SBBQ $3, R10 + SBBQ $3, CX CMOVQCS R8, AX CMOVQCS R9, BX - MOVQ key+24(FP), R8 - ADDQ 16(R8), AX - ADCQ 24(R8), BX + ADDQ 40(SI), AX + ADCQ 48(SI), BX MOVQ AX, 0(DI) MOVQ BX, 8(DI) diff --git a/vendor/golang.org/x/crypto/poly1305/sum_ref.go b/vendor/golang.org/x/crypto/poly1305/sum_generic.go similarity index 54% rename from vendor/golang.org/x/crypto/poly1305/sum_ref.go rename to vendor/golang.org/x/crypto/poly1305/sum_generic.go index c4d59bd..bab76ef 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_ref.go +++ b/vendor/golang.org/x/crypto/poly1305/sum_generic.go @@ -1,4 +1,4 @@ -// Copyright 2012 The Go Authors. All rights reserved. +// Copyright 2018 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. @@ -6,21 +6,79 @@ package poly1305 import "encoding/binary" +const ( + msgBlock = uint32(1 << 24) + finalBlock = uint32(0) +) + // sumGeneric generates an authenticator for msg using a one-time key and // puts the 16-byte result into out. This is the generic implementation of // Sum and should be called if no assembly implementation is available. func sumGeneric(out *[TagSize]byte, msg []byte, key *[32]byte) { - var ( - h0, h1, h2, h3, h4 uint32 // the hash accumulators - r0, r1, r2, r3, r4 uint64 // the r part of the key - ) + h := newMACGeneric(key) + h.Write(msg) + h.Sum(out) +} - r0 = uint64(binary.LittleEndian.Uint32(key[0:]) & 0x3ffffff) - r1 = uint64((binary.LittleEndian.Uint32(key[3:]) >> 2) & 0x3ffff03) - r2 = uint64((binary.LittleEndian.Uint32(key[6:]) >> 4) & 0x3ffc0ff) - r3 = uint64((binary.LittleEndian.Uint32(key[9:]) >> 6) & 0x3f03fff) - r4 = uint64((binary.LittleEndian.Uint32(key[12:]) >> 8) & 0x00fffff) +func newMACGeneric(key *[32]byte) (h macGeneric) { + h.r[0] = binary.LittleEndian.Uint32(key[0:]) & 0x3ffffff + h.r[1] = (binary.LittleEndian.Uint32(key[3:]) >> 2) & 0x3ffff03 + h.r[2] = (binary.LittleEndian.Uint32(key[6:]) >> 4) & 0x3ffc0ff + h.r[3] = (binary.LittleEndian.Uint32(key[9:]) >> 6) & 0x3f03fff + h.r[4] = (binary.LittleEndian.Uint32(key[12:]) >> 8) & 0x00fffff + h.s[0] = binary.LittleEndian.Uint32(key[16:]) + h.s[1] = binary.LittleEndian.Uint32(key[20:]) + h.s[2] = binary.LittleEndian.Uint32(key[24:]) + h.s[3] = binary.LittleEndian.Uint32(key[28:]) + return +} + +type macGeneric struct { + h, r [5]uint32 + s [4]uint32 + + buffer [TagSize]byte + offset int +} + +func (h *macGeneric) Write(p []byte) (n int, err error) { + n = len(p) + if h.offset > 0 { + remaining := TagSize - h.offset + if n < remaining { + h.offset += copy(h.buffer[h.offset:], p) + return n, nil + } + copy(h.buffer[h.offset:], p[:remaining]) + p = p[remaining:] + h.offset = 0 + updateGeneric(h.buffer[:], msgBlock, &(h.h), &(h.r)) + } + if nn := len(p) - (len(p) % TagSize); nn > 0 { + updateGeneric(p, msgBlock, &(h.h), &(h.r)) + p = p[nn:] + } + if len(p) > 0 { + h.offset += copy(h.buffer[h.offset:], p) + } + return n, nil +} + +func (h *macGeneric) Sum(out *[16]byte) { + H, R := h.h, h.r + if h.offset > 0 { + var buffer [TagSize]byte + copy(buffer[:], h.buffer[:h.offset]) + buffer[h.offset] = 1 // invariant: h.offset < TagSize + updateGeneric(buffer[:], finalBlock, &H, &R) + } + finalizeGeneric(out, &H, &(h.s)) +} + +func updateGeneric(msg []byte, flag uint32, h, r *[5]uint32) { + h0, h1, h2, h3, h4 := h[0], h[1], h[2], h[3], h[4] + r0, r1, r2, r3, r4 := uint64(r[0]), uint64(r[1]), uint64(r[2]), uint64(r[3]), uint64(r[4]) R1, R2, R3, R4 := r1*5, r2*5, r3*5, r4*5 for len(msg) >= TagSize { @@ -29,7 +87,7 @@ func sumGeneric(out *[TagSize]byte, msg []byte, key *[32]byte) { h1 += (binary.LittleEndian.Uint32(msg[3:]) >> 2) & 0x3ffffff h2 += (binary.LittleEndian.Uint32(msg[6:]) >> 4) & 0x3ffffff h3 += (binary.LittleEndian.Uint32(msg[9:]) >> 6) & 0x3ffffff - h4 += (binary.LittleEndian.Uint32(msg[12:]) >> 8) | (1 << 24) + h4 += (binary.LittleEndian.Uint32(msg[12:]) >> 8) | flag // h *= r d0 := (uint64(h0) * r0) + (uint64(h1) * R4) + (uint64(h2) * R3) + (uint64(h3) * R2) + (uint64(h4) * R1) @@ -52,36 +110,11 @@ func sumGeneric(out *[TagSize]byte, msg []byte, key *[32]byte) { msg = msg[TagSize:] } - if len(msg) > 0 { - var block [TagSize]byte - off := copy(block[:], msg) - block[off] = 0x01 + h[0], h[1], h[2], h[3], h[4] = h0, h1, h2, h3, h4 +} - // h += msg - h0 += binary.LittleEndian.Uint32(block[0:]) & 0x3ffffff - h1 += (binary.LittleEndian.Uint32(block[3:]) >> 2) & 0x3ffffff - h2 += (binary.LittleEndian.Uint32(block[6:]) >> 4) & 0x3ffffff - h3 += (binary.LittleEndian.Uint32(block[9:]) >> 6) & 0x3ffffff - h4 += (binary.LittleEndian.Uint32(block[12:]) >> 8) - - // h *= r - d0 := (uint64(h0) * r0) + (uint64(h1) * R4) + (uint64(h2) * R3) + (uint64(h3) * R2) + (uint64(h4) * R1) - d1 := (d0 >> 26) + (uint64(h0) * r1) + (uint64(h1) * r0) + (uint64(h2) * R4) + (uint64(h3) * R3) + (uint64(h4) * R2) - d2 := (d1 >> 26) + (uint64(h0) * r2) + (uint64(h1) * r1) + (uint64(h2) * r0) + (uint64(h3) * R4) + (uint64(h4) * R3) - d3 := (d2 >> 26) + (uint64(h0) * r3) + (uint64(h1) * r2) + (uint64(h2) * r1) + (uint64(h3) * r0) + (uint64(h4) * R4) - d4 := (d3 >> 26) + (uint64(h0) * r4) + (uint64(h1) * r3) + (uint64(h2) * r2) + (uint64(h3) * r1) + (uint64(h4) * r0) - - // h %= p - h0 = uint32(d0) & 0x3ffffff - h1 = uint32(d1) & 0x3ffffff - h2 = uint32(d2) & 0x3ffffff - h3 = uint32(d3) & 0x3ffffff - h4 = uint32(d4) & 0x3ffffff - - h0 += uint32(d4>>26) * 5 - h1 += h0 >> 26 - h0 = h0 & 0x3ffffff - } +func finalizeGeneric(out *[TagSize]byte, h *[5]uint32, s *[4]uint32) { + h0, h1, h2, h3, h4 := h[0], h[1], h[2], h[3], h[4] // h %= p reduction h2 += h1 >> 26 @@ -123,13 +156,13 @@ func sumGeneric(out *[TagSize]byte, msg []byte, key *[32]byte) { // s: the s part of the key // tag = (h + s) % (2^128) - t := uint64(h0) + uint64(binary.LittleEndian.Uint32(key[16:])) + t := uint64(h0) + uint64(s[0]) h0 = uint32(t) - t = uint64(h1) + uint64(binary.LittleEndian.Uint32(key[20:])) + (t >> 32) + t = uint64(h1) + uint64(s[1]) + (t >> 32) h1 = uint32(t) - t = uint64(h2) + uint64(binary.LittleEndian.Uint32(key[24:])) + (t >> 32) + t = uint64(h2) + uint64(s[2]) + (t >> 32) h2 = uint32(t) - t = uint64(h3) + uint64(binary.LittleEndian.Uint32(key[28:])) + (t >> 32) + t = uint64(h3) + uint64(s[3]) + (t >> 32) h3 = uint32(t) binary.LittleEndian.PutUint32(out[0:], h0) diff --git a/vendor/golang.org/x/crypto/poly1305/sum_noasm.go b/vendor/golang.org/x/crypto/poly1305/sum_noasm.go index 751eec5..8a9c207 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_noasm.go +++ b/vendor/golang.org/x/crypto/poly1305/sum_noasm.go @@ -2,7 +2,7 @@ // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. -// +build s390x,!go1.11 !arm,!amd64,!s390x gccgo appengine nacl +// +build s390x,!go1.11 !arm,!amd64,!s390x,!ppc64le gccgo appengine nacl package poly1305 @@ -10,5 +10,7 @@ package poly1305 // 16-byte result into out. Authenticating two different messages with the same // key allows an attacker to forge messages at will. func Sum(out *[TagSize]byte, msg []byte, key *[32]byte) { - sumGeneric(out, msg, key) + h := newMAC(key) + h.Write(msg) + h.Sum(out) } diff --git a/vendor/golang.org/x/crypto/poly1305/sum_ppc64le.go b/vendor/golang.org/x/crypto/poly1305/sum_ppc64le.go new file mode 100644 index 0000000..2402b63 --- /dev/null +++ b/vendor/golang.org/x/crypto/poly1305/sum_ppc64le.go @@ -0,0 +1,68 @@ +// Copyright 2019 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build ppc64le,!gccgo,!appengine + +package poly1305 + +//go:noescape +func initialize(state *[7]uint64, key *[32]byte) + +//go:noescape +func update(state *[7]uint64, msg []byte) + +//go:noescape +func finalize(tag *[TagSize]byte, state *[7]uint64) + +// Sum generates an authenticator for m using a one-time key and puts the +// 16-byte result into out. Authenticating two different messages with the same +// key allows an attacker to forge messages at will. +func Sum(out *[16]byte, m []byte, key *[32]byte) { + h := newMAC(key) + h.Write(m) + h.Sum(out) +} + +func newMAC(key *[32]byte) (h mac) { + initialize(&h.state, key) + return +} + +type mac struct { + state [7]uint64 // := uint64{ h0, h1, h2, r0, r1, pad0, pad1 } + + buffer [TagSize]byte + offset int +} + +func (h *mac) Write(p []byte) (n int, err error) { + n = len(p) + if h.offset > 0 { + remaining := TagSize - h.offset + if n < remaining { + h.offset += copy(h.buffer[h.offset:], p) + return n, nil + } + copy(h.buffer[h.offset:], p[:remaining]) + p = p[remaining:] + h.offset = 0 + update(&h.state, h.buffer[:]) + } + if nn := len(p) - (len(p) % TagSize); nn > 0 { + update(&h.state, p[:nn]) + p = p[nn:] + } + if len(p) > 0 { + h.offset += copy(h.buffer[h.offset:], p) + } + return n, nil +} + +func (h *mac) Sum(out *[16]byte) { + state := h.state + if h.offset > 0 { + update(&state, h.buffer[:h.offset]) + } + finalize(out, &state) +} diff --git a/vendor/golang.org/x/crypto/poly1305/sum_ppc64le.s b/vendor/golang.org/x/crypto/poly1305/sum_ppc64le.s new file mode 100644 index 0000000..55c7167 --- /dev/null +++ b/vendor/golang.org/x/crypto/poly1305/sum_ppc64le.s @@ -0,0 +1,247 @@ +// Copyright 2019 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build ppc64le,!gccgo,!appengine + +#include "textflag.h" + +// This was ported from the amd64 implementation. + +#define POLY1305_ADD(msg, h0, h1, h2, t0, t1, t2) \ + MOVD (msg), t0; \ + MOVD 8(msg), t1; \ + MOVD $1, t2; \ + ADDC t0, h0, h0; \ + ADDE t1, h1, h1; \ + ADDE t2, h2; \ + ADD $16, msg + +#define POLY1305_MUL(h0, h1, h2, r0, r1, t0, t1, t2, t3, t4, t5) \ + MULLD r0, h0, t0; \ + MULLD r0, h1, t4; \ + MULHDU r0, h0, t1; \ + MULHDU r0, h1, t5; \ + ADDC t4, t1, t1; \ + MULLD r0, h2, t2; \ + ADDZE t5; \ + MULHDU r1, h0, t4; \ + MULLD r1, h0, h0; \ + ADD t5, t2, t2; \ + ADDC h0, t1, t1; \ + MULLD h2, r1, t3; \ + ADDZE t4, h0; \ + MULHDU r1, h1, t5; \ + MULLD r1, h1, t4; \ + ADDC t4, t2, t2; \ + ADDE t5, t3, t3; \ + ADDC h0, t2, t2; \ + MOVD $-4, t4; \ + MOVD t0, h0; \ + MOVD t1, h1; \ + ADDZE t3; \ + ANDCC $3, t2, h2; \ + AND t2, t4, t0; \ + ADDC t0, h0, h0; \ + ADDE t3, h1, h1; \ + SLD $62, t3, t4; \ + SRD $2, t2; \ + ADDZE h2; \ + OR t4, t2, t2; \ + SRD $2, t3; \ + ADDC t2, h0, h0; \ + ADDE t3, h1, h1; \ + ADDZE h2 + +DATA ·poly1305Mask<>+0x00(SB)/8, $0x0FFFFFFC0FFFFFFF +DATA ·poly1305Mask<>+0x08(SB)/8, $0x0FFFFFFC0FFFFFFC +GLOBL ·poly1305Mask<>(SB), RODATA, $16 + +// func update(state *[7]uint64, msg []byte) + +TEXT ·update(SB), $0-32 + MOVD state+0(FP), R3 + MOVD msg_base+8(FP), R4 + MOVD msg_len+16(FP), R5 + + MOVD 0(R3), R8 // h0 + MOVD 8(R3), R9 // h1 + MOVD 16(R3), R10 // h2 + MOVD 24(R3), R11 // r0 + MOVD 32(R3), R12 // r1 + + CMP R5, $16 + BLT bytes_between_0_and_15 + +loop: + POLY1305_ADD(R4, R8, R9, R10, R20, R21, R22) + +multiply: + POLY1305_MUL(R8, R9, R10, R11, R12, R16, R17, R18, R14, R20, R21) + ADD $-16, R5 + CMP R5, $16 + BGE loop + +bytes_between_0_and_15: + CMP $0, R5 + BEQ done + MOVD $0, R16 // h0 + MOVD $0, R17 // h1 + +flush_buffer: + CMP R5, $8 + BLE just1 + + MOVD $8, R21 + SUB R21, R5, R21 + + // Greater than 8 -- load the rightmost remaining bytes in msg + // and put into R17 (h1) + MOVD (R4)(R21), R17 + MOVD $16, R22 + + // Find the offset to those bytes + SUB R5, R22, R22 + SLD $3, R22 + + // Shift to get only the bytes in msg + SRD R22, R17, R17 + + // Put 1 at high end + MOVD $1, R23 + SLD $3, R21 + SLD R21, R23, R23 + OR R23, R17, R17 + + // Remainder is 8 + MOVD $8, R5 + +just1: + CMP R5, $8 + BLT less8 + + // Exactly 8 + MOVD (R4), R16 + + CMP $0, R17 + + // Check if we've already set R17; if not + // set 1 to indicate end of msg. + BNE carry + MOVD $1, R17 + BR carry + +less8: + MOVD $0, R16 // h0 + MOVD $0, R22 // shift count + CMP R5, $4 + BLT less4 + MOVWZ (R4), R16 + ADD $4, R4 + ADD $-4, R5 + MOVD $32, R22 + +less4: + CMP R5, $2 + BLT less2 + MOVHZ (R4), R21 + SLD R22, R21, R21 + OR R16, R21, R16 + ADD $16, R22 + ADD $-2, R5 + ADD $2, R4 + +less2: + CMP $0, R5 + BEQ insert1 + MOVBZ (R4), R21 + SLD R22, R21, R21 + OR R16, R21, R16 + ADD $8, R22 + +insert1: + // Insert 1 at end of msg + MOVD $1, R21 + SLD R22, R21, R21 + OR R16, R21, R16 + +carry: + // Add new values to h0, h1, h2 + ADDC R16, R8 + ADDE R17, R9 + ADDE $0, R10 + MOVD $16, R5 + ADD R5, R4 + BR multiply + +done: + // Save h0, h1, h2 in state + MOVD R8, 0(R3) + MOVD R9, 8(R3) + MOVD R10, 16(R3) + RET + +// func initialize(state *[7]uint64, key *[32]byte) +TEXT ·initialize(SB), $0-16 + MOVD state+0(FP), R3 + MOVD key+8(FP), R4 + + // state[0...7] is initialized with zero + // Load key + MOVD 0(R4), R5 + MOVD 8(R4), R6 + MOVD 16(R4), R7 + MOVD 24(R4), R8 + + // Address of key mask + MOVD $·poly1305Mask<>(SB), R9 + + // Save original key in state + MOVD R7, 40(R3) + MOVD R8, 48(R3) + + // Get mask + MOVD (R9), R7 + MOVD 8(R9), R8 + + // And with key + AND R5, R7, R5 + AND R6, R8, R6 + + // Save masked key in state + MOVD R5, 24(R3) + MOVD R6, 32(R3) + RET + +// func finalize(tag *[TagSize]byte, state *[7]uint64) +TEXT ·finalize(SB), $0-16 + MOVD tag+0(FP), R3 + MOVD state+8(FP), R4 + + // Get h0, h1, h2 from state + MOVD 0(R4), R5 + MOVD 8(R4), R6 + MOVD 16(R4), R7 + + // Save h0, h1 + MOVD R5, R8 + MOVD R6, R9 + MOVD $3, R20 + MOVD $-1, R21 + SUBC $-5, R5 + SUBE R21, R6 + SUBE R20, R7 + MOVD $0, R21 + SUBZE R21 + + // Check for carry + CMP $0, R21 + ISEL $2, R5, R8, R5 + ISEL $2, R6, R9, R6 + MOVD 40(R4), R8 + MOVD 48(R4), R9 + ADDC R8, R5 + ADDE R9, R6 + MOVD R5, 0(R3) + MOVD R6, 8(R3) + RET diff --git a/vendor/golang.org/x/crypto/poly1305/sum_s390x.go b/vendor/golang.org/x/crypto/poly1305/sum_s390x.go index 7a266ce..ec99e07 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_s390x.go +++ b/vendor/golang.org/x/crypto/poly1305/sum_s390x.go @@ -6,16 +6,9 @@ package poly1305 -// hasVectorFacility reports whether the machine supports -// the vector facility (vx). -func hasVectorFacility() bool - -// hasVMSLFacility reports whether the machine supports -// Vector Multiply Sum Logical (VMSL). -func hasVMSLFacility() bool - -var hasVX = hasVectorFacility() -var hasVMSL = hasVMSLFacility() +import ( + "golang.org/x/sys/cpu" +) // poly1305vx is an assembly implementation of Poly1305 that uses vector // instructions. It must only be called if the vector facility (vx) is @@ -33,12 +26,12 @@ func poly1305vmsl(out *[16]byte, m *byte, mlen uint64, key *[32]byte) // 16-byte result into out. Authenticating two different messages with the same // key allows an attacker to forge messages at will. func Sum(out *[16]byte, m []byte, key *[32]byte) { - if hasVX { + if cpu.S390X.HasVX { var mPtr *byte if len(m) > 0 { mPtr = &m[0] } - if hasVMSL && len(m) > 256 { + if cpu.S390X.HasVXE && len(m) > 256 { poly1305vmsl(out, mPtr, uint64(len(m)), key) } else { poly1305vx(out, mPtr, uint64(len(m)), key) diff --git a/vendor/golang.org/x/crypto/poly1305/sum_s390x.s b/vendor/golang.org/x/crypto/poly1305/sum_s390x.s index 356c07a..ca5a309 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_s390x.s +++ b/vendor/golang.org/x/crypto/poly1305/sum_s390x.s @@ -376,25 +376,3 @@ b1: MOVD $0, R3 BR multiply - -TEXT ·hasVectorFacility(SB), NOSPLIT, $24-1 - MOVD $x-24(SP), R1 - XC $24, 0(R1), 0(R1) // clear the storage - MOVD $2, R0 // R0 is the number of double words stored -1 - WORD $0xB2B01000 // STFLE 0(R1) - XOR R0, R0 // reset the value of R0 - MOVBZ z-8(SP), R1 - AND $0x40, R1 - BEQ novector - -vectorinstalled: - // check if the vector instruction has been enabled - VLEIB $0, $0xF, V16 - VLGVB $0, V16, R1 - CMPBNE R1, $0xF, novector - MOVB $1, ret+0(FP) // have vx - RET - -novector: - MOVB $0, ret+0(FP) // no vx - RET diff --git a/vendor/golang.org/x/crypto/poly1305/sum_vmsl_s390x.s b/vendor/golang.org/x/crypto/poly1305/sum_vmsl_s390x.s index e548020..e60bbc1 100644 --- a/vendor/golang.org/x/crypto/poly1305/sum_vmsl_s390x.s +++ b/vendor/golang.org/x/crypto/poly1305/sum_vmsl_s390x.s @@ -907,25 +907,3 @@ square: MULTIPLY(H0_0, H1_0, H2_0, H0_1, H1_1, H2_1, R_0, R_1, R_2, R5_1, R5_2, M0, M1, M2, M3, M4, M5, T_0, T_1, T_2, T_3, T_4, T_5, T_6, T_7, T_8, T_9) REDUCE2(H0_0, H1_0, H2_0, M0, M1, M2, M3, M4, T_9, T_10, H0_1, M5) BR next - -TEXT ·hasVMSLFacility(SB), NOSPLIT, $24-1 - MOVD $x-24(SP), R1 - XC $24, 0(R1), 0(R1) // clear the storage - MOVD $2, R0 // R0 is the number of double words stored -1 - WORD $0xB2B01000 // STFLE 0(R1) - XOR R0, R0 // reset the value of R0 - MOVBZ z-8(SP), R1 - AND $0x01, R1 - BEQ novmsl - -vectorinstalled: - // check if the vector instruction has been enabled - VLEIB $0, $0xF, V16 - VLGVB $0, V16, R1 - CMPBNE R1, $0xF, novmsl - MOVB $1, ret+0(FP) // have vx - RET - -novmsl: - MOVB $0, ret+0(FP) // no vx - RET diff --git a/vendor/golang.org/x/crypto/ssh/certs.go b/vendor/golang.org/x/crypto/ssh/certs.go index 42106f3..00ed992 100644 --- a/vendor/golang.org/x/crypto/ssh/certs.go +++ b/vendor/golang.org/x/crypto/ssh/certs.go @@ -222,6 +222,11 @@ type openSSHCertSigner struct { signer Signer } +type algorithmOpenSSHCertSigner struct { + *openSSHCertSigner + algorithmSigner AlgorithmSigner +} + // NewCertSigner returns a Signer that signs with the given Certificate, whose // private key is held by signer. It returns an error if the public key in cert // doesn't match the key used by signer. @@ -230,7 +235,12 @@ func NewCertSigner(cert *Certificate, signer Signer) (Signer, error) { return nil, errors.New("ssh: signer and cert have different public key") } - return &openSSHCertSigner{cert, signer}, nil + if algorithmSigner, ok := signer.(AlgorithmSigner); ok { + return &algorithmOpenSSHCertSigner{ + &openSSHCertSigner{cert, signer}, algorithmSigner}, nil + } else { + return &openSSHCertSigner{cert, signer}, nil + } } func (s *openSSHCertSigner) Sign(rand io.Reader, data []byte) (*Signature, error) { @@ -241,6 +251,10 @@ func (s *openSSHCertSigner) PublicKey() PublicKey { return s.pub } +func (s *algorithmOpenSSHCertSigner) SignWithAlgorithm(rand io.Reader, data []byte, algorithm string) (*Signature, error) { + return s.algorithmSigner.SignWithAlgorithm(rand, data, algorithm) +} + const sourceAddressCriticalOption = "source-address" // CertChecker does the work of verifying a certificate. Its methods diff --git a/vendor/golang.org/x/crypto/ssh/cipher.go b/vendor/golang.org/x/crypto/ssh/cipher.go index 67b0126..a65a923 100644 --- a/vendor/golang.org/x/crypto/ssh/cipher.go +++ b/vendor/golang.org/x/crypto/ssh/cipher.go @@ -149,8 +149,8 @@ type streamPacketCipher struct { macResult []byte } -// readPacket reads and decrypt a single packet from the reader argument. -func (s *streamPacketCipher) readPacket(seqNum uint32, r io.Reader) ([]byte, error) { +// readCipherPacket reads and decrypt a single packet from the reader argument. +func (s *streamPacketCipher) readCipherPacket(seqNum uint32, r io.Reader) ([]byte, error) { if _, err := io.ReadFull(r, s.prefix[:]); err != nil { return nil, err } @@ -221,8 +221,8 @@ func (s *streamPacketCipher) readPacket(seqNum uint32, r io.Reader) ([]byte, err return s.packetData[:length-paddingLength-1], nil } -// writePacket encrypts and sends a packet of data to the writer argument -func (s *streamPacketCipher) writePacket(seqNum uint32, w io.Writer, rand io.Reader, packet []byte) error { +// writeCipherPacket encrypts and sends a packet of data to the writer argument +func (s *streamPacketCipher) writeCipherPacket(seqNum uint32, w io.Writer, rand io.Reader, packet []byte) error { if len(packet) > maxPacket { return errors.New("ssh: packet too large") } @@ -327,7 +327,7 @@ func newGCMCipher(key, iv, unusedMacKey []byte, unusedAlgs directionAlgorithms) const gcmTagSize = 16 -func (c *gcmCipher) writePacket(seqNum uint32, w io.Writer, rand io.Reader, packet []byte) error { +func (c *gcmCipher) writeCipherPacket(seqNum uint32, w io.Writer, rand io.Reader, packet []byte) error { // Pad out to multiple of 16 bytes. This is different from the // stream cipher because that encrypts the length too. padding := byte(packetSizeMultiple - (1+len(packet))%packetSizeMultiple) @@ -370,7 +370,7 @@ func (c *gcmCipher) incIV() { } } -func (c *gcmCipher) readPacket(seqNum uint32, r io.Reader) ([]byte, error) { +func (c *gcmCipher) readCipherPacket(seqNum uint32, r io.Reader) ([]byte, error) { if _, err := io.ReadFull(r, c.prefix[:]); err != nil { return nil, err } @@ -486,8 +486,8 @@ type cbcError string func (e cbcError) Error() string { return string(e) } -func (c *cbcCipher) readPacket(seqNum uint32, r io.Reader) ([]byte, error) { - p, err := c.readPacketLeaky(seqNum, r) +func (c *cbcCipher) readCipherPacket(seqNum uint32, r io.Reader) ([]byte, error) { + p, err := c.readCipherPacketLeaky(seqNum, r) if err != nil { if _, ok := err.(cbcError); ok { // Verification error: read a fixed amount of @@ -500,7 +500,7 @@ func (c *cbcCipher) readPacket(seqNum uint32, r io.Reader) ([]byte, error) { return p, err } -func (c *cbcCipher) readPacketLeaky(seqNum uint32, r io.Reader) ([]byte, error) { +func (c *cbcCipher) readCipherPacketLeaky(seqNum uint32, r io.Reader) ([]byte, error) { blockSize := c.decrypter.BlockSize() // Read the header, which will include some of the subsequent data in the @@ -576,7 +576,7 @@ func (c *cbcCipher) readPacketLeaky(seqNum uint32, r io.Reader) ([]byte, error) return c.packetData[prefixLen:paddingStart], nil } -func (c *cbcCipher) writePacket(seqNum uint32, w io.Writer, rand io.Reader, packet []byte) error { +func (c *cbcCipher) writeCipherPacket(seqNum uint32, w io.Writer, rand io.Reader, packet []byte) error { effectiveBlockSize := maxUInt32(cbcMinPacketSizeMultiple, c.encrypter.BlockSize()) // Length of encrypted portion of the packet (header, payload, padding). @@ -665,7 +665,7 @@ func newChaCha20Cipher(key, unusedIV, unusedMACKey []byte, unusedAlgs directionA return c, nil } -func (c *chacha20Poly1305Cipher) readPacket(seqNum uint32, r io.Reader) ([]byte, error) { +func (c *chacha20Poly1305Cipher) readCipherPacket(seqNum uint32, r io.Reader) ([]byte, error) { nonce := [3]uint32{0, 0, bits.ReverseBytes32(seqNum)} s := chacha20.New(c.contentKey, nonce) var polyKey [32]byte @@ -723,7 +723,7 @@ func (c *chacha20Poly1305Cipher) readPacket(seqNum uint32, r io.Reader) ([]byte, return plain, nil } -func (c *chacha20Poly1305Cipher) writePacket(seqNum uint32, w io.Writer, rand io.Reader, payload []byte) error { +func (c *chacha20Poly1305Cipher) writeCipherPacket(seqNum uint32, w io.Writer, rand io.Reader, payload []byte) error { nonce := [3]uint32{0, 0, bits.ReverseBytes32(seqNum)} s := chacha20.New(c.contentKey, nonce) var polyKey [32]byte diff --git a/vendor/golang.org/x/crypto/ssh/client.go b/vendor/golang.org/x/crypto/ssh/client.go index ae6ca77..7b00bff 100644 --- a/vendor/golang.org/x/crypto/ssh/client.go +++ b/vendor/golang.org/x/crypto/ssh/client.go @@ -185,7 +185,7 @@ func Dial(network, addr string, config *ClientConfig) (*Client, error) { // keys. A HostKeyCallback must return nil if the host key is OK, or // an error to reject it. It receives the hostname as passed to Dial // or NewClientConn. The remote address is the RemoteAddr of the -// net.Conn underlying the the SSH connection. +// net.Conn underlying the SSH connection. type HostKeyCallback func(hostname string, remote net.Addr, key PublicKey) error // BannerCallback is the function type used for treat the banner sent by diff --git a/vendor/golang.org/x/crypto/ssh/client_auth.go b/vendor/golang.org/x/crypto/ssh/client_auth.go index 5f44b77..0590070 100644 --- a/vendor/golang.org/x/crypto/ssh/client_auth.go +++ b/vendor/golang.org/x/crypto/ssh/client_auth.go @@ -523,3 +523,117 @@ func (r *retryableAuthMethod) method() string { func RetryableAuthMethod(auth AuthMethod, maxTries int) AuthMethod { return &retryableAuthMethod{authMethod: auth, maxTries: maxTries} } + +// GSSAPIWithMICAuthMethod is an AuthMethod with "gssapi-with-mic" authentication. +// See RFC 4462 section 3 +// gssAPIClient is implementation of the GSSAPIClient interface, see the definition of the interface for details. +// target is the server host you want to log in to. +func GSSAPIWithMICAuthMethod(gssAPIClient GSSAPIClient, target string) AuthMethod { + if gssAPIClient == nil { + panic("gss-api client must be not nil with enable gssapi-with-mic") + } + return &gssAPIWithMICCallback{gssAPIClient: gssAPIClient, target: target} +} + +type gssAPIWithMICCallback struct { + gssAPIClient GSSAPIClient + target string +} + +func (g *gssAPIWithMICCallback) auth(session []byte, user string, c packetConn, rand io.Reader) (authResult, []string, error) { + m := &userAuthRequestMsg{ + User: user, + Service: serviceSSH, + Method: g.method(), + } + // The GSS-API authentication method is initiated when the client sends an SSH_MSG_USERAUTH_REQUEST. + // See RFC 4462 section 3.2. + m.Payload = appendU32(m.Payload, 1) + m.Payload = appendString(m.Payload, string(krb5OID)) + if err := c.writePacket(Marshal(m)); err != nil { + return authFailure, nil, err + } + // The server responds to the SSH_MSG_USERAUTH_REQUEST with either an + // SSH_MSG_USERAUTH_FAILURE if none of the mechanisms are supported or + // with an SSH_MSG_USERAUTH_GSSAPI_RESPONSE. + // See RFC 4462 section 3.3. + // OpenSSH supports Kerberos V5 mechanism only for GSS-API authentication,so I don't want to check + // selected mech if it is valid. + packet, err := c.readPacket() + if err != nil { + return authFailure, nil, err + } + userAuthGSSAPIResp := &userAuthGSSAPIResponse{} + if err := Unmarshal(packet, userAuthGSSAPIResp); err != nil { + return authFailure, nil, err + } + // Start the loop into the exchange token. + // See RFC 4462 section 3.4. + var token []byte + defer g.gssAPIClient.DeleteSecContext() + for { + // Initiates the establishment of a security context between the application and a remote peer. + nextToken, needContinue, err := g.gssAPIClient.InitSecContext("host@"+g.target, token, false) + if err != nil { + return authFailure, nil, err + } + if len(nextToken) > 0 { + if err := c.writePacket(Marshal(&userAuthGSSAPIToken{ + Token: nextToken, + })); err != nil { + return authFailure, nil, err + } + } + if !needContinue { + break + } + packet, err = c.readPacket() + if err != nil { + return authFailure, nil, err + } + switch packet[0] { + case msgUserAuthFailure: + var msg userAuthFailureMsg + if err := Unmarshal(packet, &msg); err != nil { + return authFailure, nil, err + } + if msg.PartialSuccess { + return authPartialSuccess, msg.Methods, nil + } + return authFailure, msg.Methods, nil + case msgUserAuthGSSAPIError: + userAuthGSSAPIErrorResp := &userAuthGSSAPIError{} + if err := Unmarshal(packet, userAuthGSSAPIErrorResp); err != nil { + return authFailure, nil, err + } + return authFailure, nil, fmt.Errorf("GSS-API Error:\n"+ + "Major Status: %d\n"+ + "Minor Status: %d\n"+ + "Error Message: %s\n", userAuthGSSAPIErrorResp.MajorStatus, userAuthGSSAPIErrorResp.MinorStatus, + userAuthGSSAPIErrorResp.Message) + case msgUserAuthGSSAPIToken: + userAuthGSSAPITokenReq := &userAuthGSSAPIToken{} + if err := Unmarshal(packet, userAuthGSSAPITokenReq); err != nil { + return authFailure, nil, err + } + token = userAuthGSSAPITokenReq.Token + } + } + // Binding Encryption Keys. + // See RFC 4462 section 3.5. + micField := buildMIC(string(session), user, "ssh-connection", "gssapi-with-mic") + micToken, err := g.gssAPIClient.GetMIC(micField) + if err != nil { + return authFailure, nil, err + } + if err := c.writePacket(Marshal(&userAuthGSSAPIMIC{ + MIC: micToken, + })); err != nil { + return authFailure, nil, err + } + return handleAuthResponse(c) +} + +func (g *gssAPIWithMICCallback) method() string { + return "gssapi-with-mic" +} diff --git a/vendor/golang.org/x/crypto/ssh/common.go b/vendor/golang.org/x/crypto/ssh/common.go index 04f3620..290382d 100644 --- a/vendor/golang.org/x/crypto/ssh/common.go +++ b/vendor/golang.org/x/crypto/ssh/common.go @@ -51,6 +51,21 @@ var supportedKexAlgos = []string{ kexAlgoDH14SHA1, kexAlgoDH1SHA1, } +// serverForbiddenKexAlgos contains key exchange algorithms, that are forbidden +// for the server half. +var serverForbiddenKexAlgos = map[string]struct{}{ + kexAlgoDHGEXSHA1: {}, // server half implementation is only minimal to satisfy the automated tests + kexAlgoDHGEXSHA256: {}, // server half implementation is only minimal to satisfy the automated tests +} + +// preferredKexAlgos specifies the default preference for key-exchange algorithms +// in preference order. +var preferredKexAlgos = []string{ + kexAlgoCurve25519SHA256, + kexAlgoECDH256, kexAlgoECDH384, kexAlgoECDH521, + kexAlgoDH14SHA1, +} + // supportedHostKeyAlgos specifies the supported host-key algorithms (i.e. methods // of authenticating servers) in preference order. var supportedHostKeyAlgos = []string{ @@ -109,6 +124,7 @@ func findCommon(what string, client []string, server []string) (common string, e return "", fmt.Errorf("ssh: no common algorithm for %s; client offered: %v, server offered: %v", what, client, server) } +// directionAlgorithms records algorithm choices in one direction (either read or write) type directionAlgorithms struct { Cipher string MAC string @@ -137,7 +153,7 @@ type algorithms struct { r directionAlgorithms } -func findAgreedAlgorithms(clientKexInit, serverKexInit *kexInitMsg) (algs *algorithms, err error) { +func findAgreedAlgorithms(isClient bool, clientKexInit, serverKexInit *kexInitMsg) (algs *algorithms, err error) { result := &algorithms{} result.kex, err = findCommon("key exchange", clientKexInit.KexAlgos, serverKexInit.KexAlgos) @@ -150,32 +166,37 @@ func findAgreedAlgorithms(clientKexInit, serverKexInit *kexInitMsg) (algs *algor return } - result.w.Cipher, err = findCommon("client to server cipher", clientKexInit.CiphersClientServer, serverKexInit.CiphersClientServer) + stoc, ctos := &result.w, &result.r + if isClient { + ctos, stoc = stoc, ctos + } + + ctos.Cipher, err = findCommon("client to server cipher", clientKexInit.CiphersClientServer, serverKexInit.CiphersClientServer) if err != nil { return } - result.r.Cipher, err = findCommon("server to client cipher", clientKexInit.CiphersServerClient, serverKexInit.CiphersServerClient) + stoc.Cipher, err = findCommon("server to client cipher", clientKexInit.CiphersServerClient, serverKexInit.CiphersServerClient) if err != nil { return } - result.w.MAC, err = findCommon("client to server MAC", clientKexInit.MACsClientServer, serverKexInit.MACsClientServer) + ctos.MAC, err = findCommon("client to server MAC", clientKexInit.MACsClientServer, serverKexInit.MACsClientServer) if err != nil { return } - result.r.MAC, err = findCommon("server to client MAC", clientKexInit.MACsServerClient, serverKexInit.MACsServerClient) + stoc.MAC, err = findCommon("server to client MAC", clientKexInit.MACsServerClient, serverKexInit.MACsServerClient) if err != nil { return } - result.w.Compression, err = findCommon("client to server compression", clientKexInit.CompressionClientServer, serverKexInit.CompressionClientServer) + ctos.Compression, err = findCommon("client to server compression", clientKexInit.CompressionClientServer, serverKexInit.CompressionClientServer) if err != nil { return } - result.r.Compression, err = findCommon("server to client compression", clientKexInit.CompressionServerClient, serverKexInit.CompressionServerClient) + stoc.Compression, err = findCommon("server to client compression", clientKexInit.CompressionServerClient, serverKexInit.CompressionServerClient) if err != nil { return } @@ -233,7 +254,7 @@ func (c *Config) SetDefaults() { c.Ciphers = ciphers if c.KeyExchanges == nil { - c.KeyExchanges = supportedKexAlgos + c.KeyExchanges = preferredKexAlgos } if c.MACs == nil { diff --git a/vendor/golang.org/x/crypto/ssh/handshake.go b/vendor/golang.org/x/crypto/ssh/handshake.go index 4f7912e..2b10b05 100644 --- a/vendor/golang.org/x/crypto/ssh/handshake.go +++ b/vendor/golang.org/x/crypto/ssh/handshake.go @@ -543,7 +543,8 @@ func (t *handshakeTransport) enterKeyExchange(otherInitPacket []byte) error { clientInit := otherInit serverInit := t.sentInitMsg - if len(t.hostKeys) == 0 { + isClient := len(t.hostKeys) == 0 + if isClient { clientInit, serverInit = serverInit, clientInit magics.clientKexInit = t.sentInitPacket @@ -551,7 +552,7 @@ func (t *handshakeTransport) enterKeyExchange(otherInitPacket []byte) error { } var err error - t.algorithms, err = findAgreedAlgorithms(clientInit, serverInit) + t.algorithms, err = findAgreedAlgorithms(isClient, clientInit, serverInit) if err != nil { return err } diff --git a/vendor/golang.org/x/crypto/ssh/kex.go b/vendor/golang.org/x/crypto/ssh/kex.go index f34bcc0..1607200 100644 --- a/vendor/golang.org/x/crypto/ssh/kex.go +++ b/vendor/golang.org/x/crypto/ssh/kex.go @@ -10,7 +10,9 @@ import ( "crypto/elliptic" "crypto/rand" "crypto/subtle" + "encoding/binary" "errors" + "fmt" "io" "math/big" @@ -24,6 +26,12 @@ const ( kexAlgoECDH384 = "ecdh-sha2-nistp384" kexAlgoECDH521 = "ecdh-sha2-nistp521" kexAlgoCurve25519SHA256 = "curve25519-sha256@libssh.org" + + // For the following kex only the client half contains a production + // ready implementation. The server half only consists of a minimal + // implementation to satisfy the automated tests. + kexAlgoDHGEXSHA1 = "diffie-hellman-group-exchange-sha1" + kexAlgoDHGEXSHA256 = "diffie-hellman-group-exchange-sha256" ) // kexResult captures the outcome of a key exchange. @@ -402,6 +410,8 @@ func init() { kexAlgoMap[kexAlgoECDH384] = &ecdh{elliptic.P384()} kexAlgoMap[kexAlgoECDH256] = &ecdh{elliptic.P256()} kexAlgoMap[kexAlgoCurve25519SHA256] = &curve25519sha256{} + kexAlgoMap[kexAlgoDHGEXSHA1] = &dhGEXSHA{hashFunc: crypto.SHA1} + kexAlgoMap[kexAlgoDHGEXSHA256] = &dhGEXSHA{hashFunc: crypto.SHA256} } // curve25519sha256 implements the curve25519-sha256@libssh.org key @@ -538,3 +548,242 @@ func (kex *curve25519sha256) Server(c packetConn, rand io.Reader, magics *handsh Hash: crypto.SHA256, }, nil } + +// dhGEXSHA implements the diffie-hellman-group-exchange-sha1 and +// diffie-hellman-group-exchange-sha256 key agreement protocols, +// as described in RFC 4419 +type dhGEXSHA struct { + g, p *big.Int + hashFunc crypto.Hash +} + +const numMRTests = 64 + +const ( + dhGroupExchangeMinimumBits = 2048 + dhGroupExchangePreferredBits = 2048 + dhGroupExchangeMaximumBits = 8192 +) + +func (gex *dhGEXSHA) diffieHellman(theirPublic, myPrivate *big.Int) (*big.Int, error) { + if theirPublic.Sign() <= 0 || theirPublic.Cmp(gex.p) >= 0 { + return nil, fmt.Errorf("ssh: DH parameter out of bounds") + } + return new(big.Int).Exp(theirPublic, myPrivate, gex.p), nil +} + +func (gex *dhGEXSHA) Client(c packetConn, randSource io.Reader, magics *handshakeMagics) (*kexResult, error) { + // Send GexRequest + kexDHGexRequest := kexDHGexRequestMsg{ + MinBits: dhGroupExchangeMinimumBits, + PreferedBits: dhGroupExchangePreferredBits, + MaxBits: dhGroupExchangeMaximumBits, + } + if err := c.writePacket(Marshal(&kexDHGexRequest)); err != nil { + return nil, err + } + + // Receive GexGroup + packet, err := c.readPacket() + if err != nil { + return nil, err + } + + var kexDHGexGroup kexDHGexGroupMsg + if err = Unmarshal(packet, &kexDHGexGroup); err != nil { + return nil, err + } + + // reject if p's bit length < dhGroupExchangeMinimumBits or > dhGroupExchangeMaximumBits + if kexDHGexGroup.P.BitLen() < dhGroupExchangeMinimumBits || kexDHGexGroup.P.BitLen() > dhGroupExchangeMaximumBits { + return nil, fmt.Errorf("ssh: server-generated gex p is out of range (%d bits)", kexDHGexGroup.P.BitLen()) + } + + gex.p = kexDHGexGroup.P + gex.g = kexDHGexGroup.G + + // Check if p is safe by verifing that p and (p-1)/2 are primes + one := big.NewInt(1) + var pHalf = &big.Int{} + pHalf.Rsh(gex.p, 1) + if !gex.p.ProbablyPrime(numMRTests) || !pHalf.ProbablyPrime(numMRTests) { + return nil, fmt.Errorf("ssh: server provided gex p is not safe") + } + + // Check if g is safe by verifing that g > 1 and g < p - 1 + var pMinusOne = &big.Int{} + pMinusOne.Sub(gex.p, one) + if gex.g.Cmp(one) != 1 && gex.g.Cmp(pMinusOne) != -1 { + return nil, fmt.Errorf("ssh: server provided gex g is not safe") + } + + // Send GexInit + x, err := rand.Int(randSource, pHalf) + if err != nil { + return nil, err + } + X := new(big.Int).Exp(gex.g, x, gex.p) + kexDHGexInit := kexDHGexInitMsg{ + X: X, + } + if err := c.writePacket(Marshal(&kexDHGexInit)); err != nil { + return nil, err + } + + // Receive GexReply + packet, err = c.readPacket() + if err != nil { + return nil, err + } + + var kexDHGexReply kexDHGexReplyMsg + if err = Unmarshal(packet, &kexDHGexReply); err != nil { + return nil, err + } + + kInt, err := gex.diffieHellman(kexDHGexReply.Y, x) + if err != nil { + return nil, err + } + + // Check if k is safe by verifing that k > 1 and k < p - 1 + if kInt.Cmp(one) != 1 && kInt.Cmp(pMinusOne) != -1 { + return nil, fmt.Errorf("ssh: derived k is not safe") + } + + h := gex.hashFunc.New() + magics.write(h) + writeString(h, kexDHGexReply.HostKey) + binary.Write(h, binary.BigEndian, uint32(dhGroupExchangeMinimumBits)) + binary.Write(h, binary.BigEndian, uint32(dhGroupExchangePreferredBits)) + binary.Write(h, binary.BigEndian, uint32(dhGroupExchangeMaximumBits)) + writeInt(h, gex.p) + writeInt(h, gex.g) + writeInt(h, X) + writeInt(h, kexDHGexReply.Y) + K := make([]byte, intLength(kInt)) + marshalInt(K, kInt) + h.Write(K) + + return &kexResult{ + H: h.Sum(nil), + K: K, + HostKey: kexDHGexReply.HostKey, + Signature: kexDHGexReply.Signature, + Hash: gex.hashFunc, + }, nil +} + +// Server half implementation of the Diffie Hellman Key Exchange with SHA1 and SHA256. +// +// This is a minimal implementation to satisfy the automated tests. +func (gex *dhGEXSHA) Server(c packetConn, randSource io.Reader, magics *handshakeMagics, priv Signer) (result *kexResult, err error) { + // Receive GexRequest + packet, err := c.readPacket() + if err != nil { + return + } + var kexDHGexRequest kexDHGexRequestMsg + if err = Unmarshal(packet, &kexDHGexRequest); err != nil { + return + } + + // smoosh the user's preferred size into our own limits + if kexDHGexRequest.PreferedBits > dhGroupExchangeMaximumBits { + kexDHGexRequest.PreferedBits = dhGroupExchangeMaximumBits + } + if kexDHGexRequest.PreferedBits < dhGroupExchangeMinimumBits { + kexDHGexRequest.PreferedBits = dhGroupExchangeMinimumBits + } + // fix min/max if they're inconsistent. technically, we could just pout + // and hang up, but there's no harm in giving them the benefit of the + // doubt and just picking a bitsize for them. + if kexDHGexRequest.MinBits > kexDHGexRequest.PreferedBits { + kexDHGexRequest.MinBits = kexDHGexRequest.PreferedBits + } + if kexDHGexRequest.MaxBits < kexDHGexRequest.PreferedBits { + kexDHGexRequest.MaxBits = kexDHGexRequest.PreferedBits + } + + // Send GexGroup + // This is the group called diffie-hellman-group14-sha1 in RFC + // 4253 and Oakley Group 14 in RFC 3526. + p, _ := new(big.Int).SetString("FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7EDEE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB9ED529077096966D670C354E4ABC9804F1746C08CA18217C32905E462E36CE3BE39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9DE2BCBF6955817183995497CEA956AE515D2261898FA051015728E5A8AACAA68FFFFFFFFFFFFFFFF", 16) + gex.p = p + gex.g = big.NewInt(2) + + kexDHGexGroup := kexDHGexGroupMsg{ + P: gex.p, + G: gex.g, + } + if err := c.writePacket(Marshal(&kexDHGexGroup)); err != nil { + return nil, err + } + + // Receive GexInit + packet, err = c.readPacket() + if err != nil { + return + } + var kexDHGexInit kexDHGexInitMsg + if err = Unmarshal(packet, &kexDHGexInit); err != nil { + return + } + + var pHalf = &big.Int{} + pHalf.Rsh(gex.p, 1) + + y, err := rand.Int(randSource, pHalf) + if err != nil { + return + } + + Y := new(big.Int).Exp(gex.g, y, gex.p) + kInt, err := gex.diffieHellman(kexDHGexInit.X, y) + if err != nil { + return nil, err + } + + hostKeyBytes := priv.PublicKey().Marshal() + + h := gex.hashFunc.New() + magics.write(h) + writeString(h, hostKeyBytes) + binary.Write(h, binary.BigEndian, uint32(dhGroupExchangeMinimumBits)) + binary.Write(h, binary.BigEndian, uint32(dhGroupExchangePreferredBits)) + binary.Write(h, binary.BigEndian, uint32(dhGroupExchangeMaximumBits)) + writeInt(h, gex.p) + writeInt(h, gex.g) + writeInt(h, kexDHGexInit.X) + writeInt(h, Y) + + K := make([]byte, intLength(kInt)) + marshalInt(K, kInt) + h.Write(K) + + H := h.Sum(nil) + + // H is already a hash, but the hostkey signing will apply its + // own key-specific hash algorithm. + sig, err := signAndMarshal(priv, randSource, H) + if err != nil { + return nil, err + } + + kexDHGexReply := kexDHGexReplyMsg{ + HostKey: hostKeyBytes, + Y: Y, + Signature: sig, + } + packet = Marshal(&kexDHGexReply) + + err = c.writePacket(packet) + + return &kexResult{ + H: H, + K: K, + HostKey: hostKeyBytes, + Signature: sig, + Hash: gex.hashFunc, + }, err +} diff --git a/vendor/golang.org/x/crypto/ssh/keys.go b/vendor/golang.org/x/crypto/ssh/keys.go index 34d9582..9698047 100644 --- a/vendor/golang.org/x/crypto/ssh/keys.go +++ b/vendor/golang.org/x/crypto/ssh/keys.go @@ -38,6 +38,16 @@ const ( KeyAlgoED25519 = "ssh-ed25519" ) +// These constants represent non-default signature algorithms that are supported +// as algorithm parameters to AlgorithmSigner.SignWithAlgorithm methods. See +// [PROTOCOL.agent] section 4.5.1 and +// https://tools.ietf.org/html/draft-ietf-curdle-rsa-sha2-10 +const ( + SigAlgoRSA = "ssh-rsa" + SigAlgoRSASHA2256 = "rsa-sha2-256" + SigAlgoRSASHA2512 = "rsa-sha2-512" +) + // parsePubKey parses a public key of the given algorithm. // Use ParsePublicKey for keys with prepended algorithm. func parsePubKey(in []byte, algo string) (pubKey PublicKey, rest []byte, err error) { @@ -301,6 +311,19 @@ type Signer interface { Sign(rand io.Reader, data []byte) (*Signature, error) } +// A AlgorithmSigner is a Signer that also supports specifying a specific +// algorithm to use for signing. +type AlgorithmSigner interface { + Signer + + // SignWithAlgorithm is like Signer.Sign, but allows specification of a + // non-default signing algorithm. See the SigAlgo* constants in this + // package for signature algorithms supported by this package. Callers may + // pass an empty string for the algorithm in which case the AlgorithmSigner + // will use its default algorithm. + SignWithAlgorithm(rand io.Reader, data []byte, algorithm string) (*Signature, error) +} + type rsaPublicKey rsa.PublicKey func (r *rsaPublicKey) Type() string { @@ -349,13 +372,21 @@ func (r *rsaPublicKey) Marshal() []byte { } func (r *rsaPublicKey) Verify(data []byte, sig *Signature) error { - if sig.Format != r.Type() { + var hash crypto.Hash + switch sig.Format { + case SigAlgoRSA: + hash = crypto.SHA1 + case SigAlgoRSASHA2256: + hash = crypto.SHA256 + case SigAlgoRSASHA2512: + hash = crypto.SHA512 + default: return fmt.Errorf("ssh: signature type %s for key type %s", sig.Format, r.Type()) } - h := crypto.SHA1.New() + h := hash.New() h.Write(data) digest := h.Sum(nil) - return rsa.VerifyPKCS1v15((*rsa.PublicKey)(r), crypto.SHA1, digest, sig.Blob) + return rsa.VerifyPKCS1v15((*rsa.PublicKey)(r), hash, digest, sig.Blob) } func (r *rsaPublicKey) CryptoPublicKey() crypto.PublicKey { @@ -459,6 +490,14 @@ func (k *dsaPrivateKey) PublicKey() PublicKey { } func (k *dsaPrivateKey) Sign(rand io.Reader, data []byte) (*Signature, error) { + return k.SignWithAlgorithm(rand, data, "") +} + +func (k *dsaPrivateKey) SignWithAlgorithm(rand io.Reader, data []byte, algorithm string) (*Signature, error) { + if algorithm != "" && algorithm != k.PublicKey().Type() { + return nil, fmt.Errorf("ssh: unsupported signature algorithm %s", algorithm) + } + h := crypto.SHA1.New() h.Write(data) digest := h.Sum(nil) @@ -691,16 +730,42 @@ func (s *wrappedSigner) PublicKey() PublicKey { } func (s *wrappedSigner) Sign(rand io.Reader, data []byte) (*Signature, error) { + return s.SignWithAlgorithm(rand, data, "") +} + +func (s *wrappedSigner) SignWithAlgorithm(rand io.Reader, data []byte, algorithm string) (*Signature, error) { var hashFunc crypto.Hash - switch key := s.pubKey.(type) { - case *rsaPublicKey, *dsaPublicKey: - hashFunc = crypto.SHA1 - case *ecdsaPublicKey: - hashFunc = ecHash(key.Curve) - case ed25519PublicKey: - default: - return nil, fmt.Errorf("ssh: unsupported key type %T", key) + if _, ok := s.pubKey.(*rsaPublicKey); ok { + // RSA keys support a few hash functions determined by the requested signature algorithm + switch algorithm { + case "", SigAlgoRSA: + algorithm = SigAlgoRSA + hashFunc = crypto.SHA1 + case SigAlgoRSASHA2256: + hashFunc = crypto.SHA256 + case SigAlgoRSASHA2512: + hashFunc = crypto.SHA512 + default: + return nil, fmt.Errorf("ssh: unsupported signature algorithm %s", algorithm) + } + } else { + // The only supported algorithm for all other key types is the same as the type of the key + if algorithm == "" { + algorithm = s.pubKey.Type() + } else if algorithm != s.pubKey.Type() { + return nil, fmt.Errorf("ssh: unsupported signature algorithm %s", algorithm) + } + + switch key := s.pubKey.(type) { + case *dsaPublicKey: + hashFunc = crypto.SHA1 + case *ecdsaPublicKey: + hashFunc = ecHash(key.Curve) + case ed25519PublicKey: + default: + return nil, fmt.Errorf("ssh: unsupported key type %T", key) + } } var digest []byte @@ -745,7 +810,7 @@ func (s *wrappedSigner) Sign(rand io.Reader, data []byte) (*Signature, error) { } return &Signature{ - Format: s.pubKey.Type(), + Format: algorithm, Blob: signature, }, nil } @@ -903,8 +968,8 @@ func ParseDSAPrivateKey(der []byte) (*dsa.PrivateKey, error) { // Implemented based on the documentation at // https://github.com/openssh/openssh-portable/blob/master/PROTOCOL.key func parseOpenSSHPrivateKey(key []byte) (crypto.PrivateKey, error) { - magic := append([]byte("openssh-key-v1"), 0) - if !bytes.Equal(magic, key[0:len(magic)]) { + const magic = "openssh-key-v1\x00" + if len(key) < len(magic) || string(key[:len(magic)]) != magic { return nil, errors.New("ssh: invalid openssh private key format") } remaining := key[len(magic):] diff --git a/vendor/golang.org/x/crypto/ssh/messages.go b/vendor/golang.org/x/crypto/ssh/messages.go index 08d2811..ac41a41 100644 --- a/vendor/golang.org/x/crypto/ssh/messages.go +++ b/vendor/golang.org/x/crypto/ssh/messages.go @@ -97,6 +97,36 @@ type kexDHReplyMsg struct { Signature []byte } +// See RFC 4419, section 5. +const msgKexDHGexGroup = 31 + +type kexDHGexGroupMsg struct { + P *big.Int `sshtype:"31"` + G *big.Int +} + +const msgKexDHGexInit = 32 + +type kexDHGexInitMsg struct { + X *big.Int `sshtype:"32"` +} + +const msgKexDHGexReply = 33 + +type kexDHGexReplyMsg struct { + HostKey []byte `sshtype:"33"` + Y *big.Int + Signature []byte +} + +const msgKexDHGexRequest = 34 + +type kexDHGexRequestMsg struct { + MinBits uint32 `sshtype:"34"` + PreferedBits uint32 + MaxBits uint32 +} + // See RFC 4253, section 10. const msgServiceRequest = 5 @@ -275,6 +305,42 @@ type userAuthPubKeyOkMsg struct { PubKey []byte } +// See RFC 4462, section 3 +const msgUserAuthGSSAPIResponse = 60 + +type userAuthGSSAPIResponse struct { + SupportMech []byte `sshtype:"60"` +} + +const msgUserAuthGSSAPIToken = 61 + +type userAuthGSSAPIToken struct { + Token []byte `sshtype:"61"` +} + +const msgUserAuthGSSAPIMIC = 66 + +type userAuthGSSAPIMIC struct { + MIC []byte `sshtype:"66"` +} + +// See RFC 4462, section 3.9 +const msgUserAuthGSSAPIErrTok = 64 + +type userAuthGSSAPIErrTok struct { + ErrorToken []byte `sshtype:"64"` +} + +// See RFC 4462, section 3.8 +const msgUserAuthGSSAPIError = 65 + +type userAuthGSSAPIError struct { + MajorStatus uint32 `sshtype:"65"` + MinorStatus uint32 + Message string + LanguageTag string +} + // typeTags returns the possible type bytes for the given reflect.Type, which // should be a struct. The possible values are separated by a '|' character. func typeTags(structType reflect.Type) (tags []byte) { @@ -756,6 +822,14 @@ func decode(packet []byte) (interface{}, error) { msg = new(channelRequestSuccessMsg) case msgChannelFailure: msg = new(channelRequestFailureMsg) + case msgUserAuthGSSAPIToken: + msg = new(userAuthGSSAPIToken) + case msgUserAuthGSSAPIMIC: + msg = new(userAuthGSSAPIMIC) + case msgUserAuthGSSAPIErrTok: + msg = new(userAuthGSSAPIErrTok) + case msgUserAuthGSSAPIError: + msg = new(userAuthGSSAPIError) default: return nil, unexpectedMessageError(0, packet[0]) } @@ -764,3 +838,29 @@ func decode(packet []byte) (interface{}, error) { } return msg, nil } + +var packetTypeNames = map[byte]string{ + msgDisconnect: "disconnectMsg", + msgServiceRequest: "serviceRequestMsg", + msgServiceAccept: "serviceAcceptMsg", + msgKexInit: "kexInitMsg", + msgKexDHInit: "kexDHInitMsg", + msgKexDHReply: "kexDHReplyMsg", + msgUserAuthRequest: "userAuthRequestMsg", + msgUserAuthSuccess: "userAuthSuccessMsg", + msgUserAuthFailure: "userAuthFailureMsg", + msgUserAuthPubKeyOk: "userAuthPubKeyOkMsg", + msgGlobalRequest: "globalRequestMsg", + msgRequestSuccess: "globalRequestSuccessMsg", + msgRequestFailure: "globalRequestFailureMsg", + msgChannelOpen: "channelOpenMsg", + msgChannelData: "channelDataMsg", + msgChannelOpenConfirm: "channelOpenConfirmMsg", + msgChannelOpenFailure: "channelOpenFailureMsg", + msgChannelWindowAdjust: "windowAdjustMsg", + msgChannelEOF: "channelEOFMsg", + msgChannelClose: "channelCloseMsg", + msgChannelRequest: "channelRequestMsg", + msgChannelSuccess: "channelRequestSuccessMsg", + msgChannelFailure: "channelRequestFailureMsg", +} diff --git a/vendor/golang.org/x/crypto/ssh/server.go b/vendor/golang.org/x/crypto/ssh/server.go index d0f4825..7a5a1d7 100644 --- a/vendor/golang.org/x/crypto/ssh/server.go +++ b/vendor/golang.org/x/crypto/ssh/server.go @@ -45,6 +45,20 @@ type Permissions struct { Extensions map[string]string } +type GSSAPIWithMICConfig struct { + // AllowLogin, must be set, is called when gssapi-with-mic + // authentication is selected (RFC 4462 section 3). The srcName is from the + // results of the GSS-API authentication. The format is username@DOMAIN. + // GSSAPI just guarantees to the server who the user is, but not if they can log in, and with what permissions. + // This callback is called after the user identity is established with GSSAPI to decide if the user can login with + // which permissions. If the user is allowed to login, it should return a nil error. + AllowLogin func(conn ConnMetadata, srcName string) (*Permissions, error) + + // Server must be set. It's the implementation + // of the GSSAPIServer interface. See GSSAPIServer interface for details. + Server GSSAPIServer +} + // ServerConfig holds server specific configuration data. type ServerConfig struct { // Config contains configuration shared between client and server. @@ -99,6 +113,10 @@ type ServerConfig struct { // BannerCallback, if present, is called and the return string is sent to // the client after key exchange completed but before authentication. BannerCallback func(conn ConnMetadata) string + + // GSSAPIWithMICConfig includes gssapi server and callback, which if both non-nil, is used + // when gssapi-with-mic authentication is selected (RFC 4462 section 3). + GSSAPIWithMICConfig *GSSAPIWithMICConfig } // AddHostKey adds a private key as a host key. If an existing host @@ -175,6 +193,12 @@ func NewServerConn(c net.Conn, config *ServerConfig) (*ServerConn, <-chan NewCha if fullConf.MaxAuthTries == 0 { fullConf.MaxAuthTries = 6 } + // Check if the config contains any unsupported key exchanges + for _, kex := range fullConf.KeyExchanges { + if _, ok := serverForbiddenKexAlgos[kex]; ok { + return nil, nil, nil, fmt.Errorf("ssh: unsupported key exchange %s for server", kex) + } + } s := &connection{ sshConn: sshConn{conn: c}, @@ -204,7 +228,9 @@ func (s *connection) serverHandshake(config *ServerConfig) (*Permissions, error) return nil, errors.New("ssh: server has no host keys") } - if !config.NoClientAuth && config.PasswordCallback == nil && config.PublicKeyCallback == nil && config.KeyboardInteractiveCallback == nil { + if !config.NoClientAuth && config.PasswordCallback == nil && config.PublicKeyCallback == nil && + config.KeyboardInteractiveCallback == nil && (config.GSSAPIWithMICConfig == nil || + config.GSSAPIWithMICConfig.AllowLogin == nil || config.GSSAPIWithMICConfig.Server == nil) { return nil, errors.New("ssh: no authentication methods configured but NoClientAuth is also false") } @@ -295,6 +321,55 @@ func checkSourceAddress(addr net.Addr, sourceAddrs string) error { return fmt.Errorf("ssh: remote address %v is not allowed because of source-address restriction", addr) } +func gssExchangeToken(gssapiConfig *GSSAPIWithMICConfig, firstToken []byte, s *connection, + sessionID []byte, userAuthReq userAuthRequestMsg) (authErr error, perms *Permissions, err error) { + gssAPIServer := gssapiConfig.Server + defer gssAPIServer.DeleteSecContext() + var srcName string + for { + var ( + outToken []byte + needContinue bool + ) + outToken, srcName, needContinue, err = gssAPIServer.AcceptSecContext(firstToken) + if err != nil { + return err, nil, nil + } + if len(outToken) != 0 { + if err := s.transport.writePacket(Marshal(&userAuthGSSAPIToken{ + Token: outToken, + })); err != nil { + return nil, nil, err + } + } + if !needContinue { + break + } + packet, err := s.transport.readPacket() + if err != nil { + return nil, nil, err + } + userAuthGSSAPITokenReq := &userAuthGSSAPIToken{} + if err := Unmarshal(packet, userAuthGSSAPITokenReq); err != nil { + return nil, nil, err + } + } + packet, err := s.transport.readPacket() + if err != nil { + return nil, nil, err + } + userAuthGSSAPIMICReq := &userAuthGSSAPIMIC{} + if err := Unmarshal(packet, userAuthGSSAPIMICReq); err != nil { + return nil, nil, err + } + mic := buildMIC(string(sessionID), userAuthReq.User, userAuthReq.Service, userAuthReq.Method) + if err := gssAPIServer.VerifyMIC(mic, userAuthGSSAPIMICReq.MIC); err != nil { + return err, nil, nil + } + perms, authErr = gssapiConfig.AllowLogin(s, srcName) + return authErr, perms, nil +} + // ServerAuthError represents server authentication errors and is // sometimes returned by NewServerConn. It appends any authentication // errors that may occur, and is returned if all of the authentication @@ -404,7 +479,7 @@ userAuthLoop: perms, authErr = config.PasswordCallback(s, password) case "keyboard-interactive": if config.KeyboardInteractiveCallback == nil { - authErr = errors.New("ssh: keyboard-interactive auth not configubred") + authErr = errors.New("ssh: keyboard-interactive auth not configured") break } @@ -484,6 +559,7 @@ userAuthLoop: // sig.Format. This is usually the same, but // for certs, the names differ. if !isAcceptableAlgo(sig.Format) { + authErr = fmt.Errorf("ssh: algorithm %q not accepted", sig.Format) break } signedData := buildDataSignedForAuth(sessionID, userAuthReq, algoBytes, pubKeyData) @@ -495,6 +571,49 @@ userAuthLoop: authErr = candidate.result perms = candidate.perms } + case "gssapi-with-mic": + gssapiConfig := config.GSSAPIWithMICConfig + userAuthRequestGSSAPI, err := parseGSSAPIPayload(userAuthReq.Payload) + if err != nil { + return nil, parseError(msgUserAuthRequest) + } + // OpenSSH supports Kerberos V5 mechanism only for GSS-API authentication. + if userAuthRequestGSSAPI.N == 0 { + authErr = fmt.Errorf("ssh: Mechanism negotiation is not supported") + break + } + var i uint32 + present := false + for i = 0; i < userAuthRequestGSSAPI.N; i++ { + if userAuthRequestGSSAPI.OIDS[i].Equal(krb5Mesh) { + present = true + break + } + } + if !present { + authErr = fmt.Errorf("ssh: GSSAPI authentication must use the Kerberos V5 mechanism") + break + } + // Initial server response, see RFC 4462 section 3.3. + if err := s.transport.writePacket(Marshal(&userAuthGSSAPIResponse{ + SupportMech: krb5OID, + })); err != nil { + return nil, err + } + // Exchange token, see RFC 4462 section 3.4. + packet, err := s.transport.readPacket() + if err != nil { + return nil, err + } + userAuthGSSAPITokenReq := &userAuthGSSAPIToken{} + if err := Unmarshal(packet, userAuthGSSAPITokenReq); err != nil { + return nil, err + } + authErr, perms, err = gssExchangeToken(gssapiConfig, userAuthGSSAPITokenReq.Token, s, sessionID, + userAuthReq) + if err != nil { + return nil, err + } default: authErr = fmt.Errorf("ssh: unknown method %q", userAuthReq.Method) } @@ -521,6 +640,10 @@ userAuthLoop: if config.KeyboardInteractiveCallback != nil { failureMsg.Methods = append(failureMsg.Methods, "keyboard-interactive") } + if config.GSSAPIWithMICConfig != nil && config.GSSAPIWithMICConfig.Server != nil && + config.GSSAPIWithMICConfig.AllowLogin != nil { + failureMsg.Methods = append(failureMsg.Methods, "gssapi-with-mic") + } if len(failureMsg.Methods) == 0 { return nil, errors.New("ssh: no authentication methods configured but NoClientAuth is also false") diff --git a/vendor/golang.org/x/crypto/ssh/ssh_gss.go b/vendor/golang.org/x/crypto/ssh/ssh_gss.go new file mode 100644 index 0000000..24bd7c8 --- /dev/null +++ b/vendor/golang.org/x/crypto/ssh/ssh_gss.go @@ -0,0 +1,139 @@ +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package ssh + +import ( + "encoding/asn1" + "errors" +) + +var krb5OID []byte + +func init() { + krb5OID, _ = asn1.Marshal(krb5Mesh) +} + +// GSSAPIClient provides the API to plug-in GSSAPI authentication for client logins. +type GSSAPIClient interface { + // InitSecContext initiates the establishment of a security context for GSS-API between the + // ssh client and ssh server. Initially the token parameter should be specified as nil. + // The routine may return a outputToken which should be transferred to + // the ssh server, where the ssh server will present it to + // AcceptSecContext. If no token need be sent, InitSecContext will indicate this by setting + // needContinue to false. To complete the context + // establishment, one or more reply tokens may be required from the ssh + // server;if so, InitSecContext will return a needContinue which is true. + // In this case, InitSecContext should be called again when the + // reply token is received from the ssh server, passing the reply + // token to InitSecContext via the token parameters. + // See RFC 2743 section 2.2.1 and RFC 4462 section 3.4. + InitSecContext(target string, token []byte, isGSSDelegCreds bool) (outputToken []byte, needContinue bool, err error) + // GetMIC generates a cryptographic MIC for the SSH2 message, and places + // the MIC in a token for transfer to the ssh server. + // The contents of the MIC field are obtained by calling GSS_GetMIC() + // over the following, using the GSS-API context that was just + // established: + // string session identifier + // byte SSH_MSG_USERAUTH_REQUEST + // string user name + // string service + // string "gssapi-with-mic" + // See RFC 2743 section 2.3.1 and RFC 4462 3.5. + GetMIC(micFiled []byte) ([]byte, error) + // Whenever possible, it should be possible for + // DeleteSecContext() calls to be successfully processed even + // if other calls cannot succeed, thereby enabling context-related + // resources to be released. + // In addition to deleting established security contexts, + // gss_delete_sec_context must also be able to delete "half-built" + // security contexts resulting from an incomplete sequence of + // InitSecContext()/AcceptSecContext() calls. + // See RFC 2743 section 2.2.3. + DeleteSecContext() error +} + +// GSSAPIServer provides the API to plug in GSSAPI authentication for server logins. +type GSSAPIServer interface { + // AcceptSecContext allows a remotely initiated security context between the application + // and a remote peer to be established by the ssh client. The routine may return a + // outputToken which should be transferred to the ssh client, + // where the ssh client will present it to InitSecContext. + // If no token need be sent, AcceptSecContext will indicate this + // by setting the needContinue to false. To + // complete the context establishment, one or more reply tokens may be + // required from the ssh client. if so, AcceptSecContext + // will return a needContinue which is true, in which case it + // should be called again when the reply token is received from the ssh + // client, passing the token to AcceptSecContext via the + // token parameters. + // The srcName return value is the authenticated username. + // See RFC 2743 section 2.2.2 and RFC 4462 section 3.4. + AcceptSecContext(token []byte) (outputToken []byte, srcName string, needContinue bool, err error) + // VerifyMIC verifies that a cryptographic MIC, contained in the token parameter, + // fits the supplied message is received from the ssh client. + // See RFC 2743 section 2.3.2. + VerifyMIC(micField []byte, micToken []byte) error + // Whenever possible, it should be possible for + // DeleteSecContext() calls to be successfully processed even + // if other calls cannot succeed, thereby enabling context-related + // resources to be released. + // In addition to deleting established security contexts, + // gss_delete_sec_context must also be able to delete "half-built" + // security contexts resulting from an incomplete sequence of + // InitSecContext()/AcceptSecContext() calls. + // See RFC 2743 section 2.2.3. + DeleteSecContext() error +} + +var ( + // OpenSSH supports Kerberos V5 mechanism only for GSS-API authentication, + // so we also support the krb5 mechanism only. + // See RFC 1964 section 1. + krb5Mesh = asn1.ObjectIdentifier{1, 2, 840, 113554, 1, 2, 2} +) + +// The GSS-API authentication method is initiated when the client sends an SSH_MSG_USERAUTH_REQUEST +// See RFC 4462 section 3.2. +type userAuthRequestGSSAPI struct { + N uint32 + OIDS []asn1.ObjectIdentifier +} + +func parseGSSAPIPayload(payload []byte) (*userAuthRequestGSSAPI, error) { + n, rest, ok := parseUint32(payload) + if !ok { + return nil, errors.New("parse uint32 failed") + } + s := &userAuthRequestGSSAPI{ + N: n, + OIDS: make([]asn1.ObjectIdentifier, n), + } + for i := 0; i < int(n); i++ { + var ( + desiredMech []byte + err error + ) + desiredMech, rest, ok = parseString(rest) + if !ok { + return nil, errors.New("parse string failed") + } + if rest, err = asn1.Unmarshal(desiredMech, &s.OIDS[i]); err != nil { + return nil, err + } + + } + return s, nil +} + +// See RFC 4462 section 3.6. +func buildMIC(sessionID string, username string, service string, authMethod string) []byte { + out := make([]byte, 0, 0) + out = appendString(out, sessionID) + out = append(out, msgUserAuthRequest) + out = appendString(out, username) + out = appendString(out, service) + out = appendString(out, authMethod) + return out +} diff --git a/vendor/golang.org/x/crypto/ssh/transport.go b/vendor/golang.org/x/crypto/ssh/transport.go index f6fae1d..49ddc2e 100644 --- a/vendor/golang.org/x/crypto/ssh/transport.go +++ b/vendor/golang.org/x/crypto/ssh/transport.go @@ -53,14 +53,14 @@ type transport struct { // packetCipher represents a combination of SSH encryption/MAC // protocol. A single instance should be used for one direction only. type packetCipher interface { - // writePacket encrypts the packet and writes it to w. The + // writeCipherPacket encrypts the packet and writes it to w. The // contents of the packet are generally scrambled. - writePacket(seqnum uint32, w io.Writer, rand io.Reader, packet []byte) error + writeCipherPacket(seqnum uint32, w io.Writer, rand io.Reader, packet []byte) error - // readPacket reads and decrypts a packet of data. The + // readCipherPacket reads and decrypts a packet of data. The // returned packet may be overwritten by future calls of // readPacket. - readPacket(seqnum uint32, r io.Reader) ([]byte, error) + readCipherPacket(seqnum uint32, r io.Reader) ([]byte, error) } // connectionState represents one side (read or write) of the @@ -127,7 +127,7 @@ func (t *transport) readPacket() (p []byte, err error) { } func (s *connectionState) readPacket(r *bufio.Reader) ([]byte, error) { - packet, err := s.packetCipher.readPacket(s.seqNum, r) + packet, err := s.packetCipher.readCipherPacket(s.seqNum, r) s.seqNum++ if err == nil && len(packet) == 0 { err = errors.New("ssh: zero length packet") @@ -175,7 +175,7 @@ func (t *transport) writePacket(packet []byte) error { func (s *connectionState) writePacket(w *bufio.Writer, rand io.Reader, packet []byte) error { changeKeys := len(packet) > 0 && packet[0] == msgNewKeys - err := s.packetCipher.writePacket(s.seqNum, w, rand, packet) + err := s.packetCipher.writeCipherPacket(s.seqNum, w, rand, packet) if err != nil { return err } diff --git a/vendor/golang.org/x/sys/cpu/cpu.go b/vendor/golang.org/x/sys/cpu/cpu.go new file mode 100644 index 0000000..3d88f86 --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu.go @@ -0,0 +1,38 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package cpu implements processor feature detection for +// various CPU architectures. +package cpu + +// CacheLinePad is used to pad structs to avoid false sharing. +type CacheLinePad struct{ _ [cacheLineSize]byte } + +// X86 contains the supported CPU features of the +// current X86/AMD64 platform. If the current platform +// is not X86/AMD64 then all feature flags are false. +// +// X86 is padded to avoid false sharing. Further the HasAVX +// and HasAVX2 are only set if the OS supports XMM and YMM +// registers in addition to the CPUID feature bit being set. +var X86 struct { + _ CacheLinePad + HasAES bool // AES hardware implementation (AES NI) + HasADX bool // Multi-precision add-carry instruction extensions + HasAVX bool // Advanced vector extension + HasAVX2 bool // Advanced vector extension 2 + HasBMI1 bool // Bit manipulation instruction set 1 + HasBMI2 bool // Bit manipulation instruction set 2 + HasERMS bool // Enhanced REP for MOVSB and STOSB + HasFMA bool // Fused-multiply-add instructions + HasOSXSAVE bool // OS supports XSAVE/XRESTOR for saving/restoring XMM registers. + HasPCLMULQDQ bool // PCLMULQDQ instruction - most often used for AES-GCM + HasPOPCNT bool // Hamming weight instruction POPCNT. + HasSSE2 bool // Streaming SIMD extension 2 (always available on amd64) + HasSSE3 bool // Streaming SIMD extension 3 + HasSSSE3 bool // Supplemental streaming SIMD extension 3 + HasSSE41 bool // Streaming SIMD extension 4 and 4.1 + HasSSE42 bool // Streaming SIMD extension 4 and 4.2 + _ CacheLinePad +} diff --git a/vendor/golang.org/x/sys/cpu/cpu_arm.go b/vendor/golang.org/x/sys/cpu/cpu_arm.go new file mode 100644 index 0000000..d93036f --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_arm.go @@ -0,0 +1,7 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package cpu + +const cacheLineSize = 32 diff --git a/vendor/golang.org/x/sys/cpu/cpu_arm64.go b/vendor/golang.org/x/sys/cpu/cpu_arm64.go new file mode 100644 index 0000000..1d2ab29 --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_arm64.go @@ -0,0 +1,7 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package cpu + +const cacheLineSize = 64 diff --git a/vendor/golang.org/x/sys/cpu/cpu_gc_x86.go b/vendor/golang.org/x/sys/cpu/cpu_gc_x86.go new file mode 100644 index 0000000..f7cb469 --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_gc_x86.go @@ -0,0 +1,16 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build 386 amd64 amd64p32 +// +build !gccgo + +package cpu + +// cpuid is implemented in cpu_x86.s for gc compiler +// and in cpu_gccgo.c for gccgo. +func cpuid(eaxArg, ecxArg uint32) (eax, ebx, ecx, edx uint32) + +// xgetbv with ecx = 0 is implemented in cpu_x86.s for gc compiler +// and in cpu_gccgo.c for gccgo. +func xgetbv() (eax, edx uint32) diff --git a/vendor/golang.org/x/sys/cpu/cpu_gccgo.c b/vendor/golang.org/x/sys/cpu/cpu_gccgo.c new file mode 100644 index 0000000..e363c7d --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_gccgo.c @@ -0,0 +1,43 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build 386 amd64 amd64p32 +// +build gccgo + +#include +#include + +// Need to wrap __get_cpuid_count because it's declared as static. +int +gccgoGetCpuidCount(uint32_t leaf, uint32_t subleaf, + uint32_t *eax, uint32_t *ebx, + uint32_t *ecx, uint32_t *edx) +{ + return __get_cpuid_count(leaf, subleaf, eax, ebx, ecx, edx); +} + +// xgetbv reads the contents of an XCR (Extended Control Register) +// specified in the ECX register into registers EDX:EAX. +// Currently, the only supported value for XCR is 0. +// +// TODO: Replace with a better alternative: +// +// #include +// +// #pragma GCC target("xsave") +// +// void gccgoXgetbv(uint32_t *eax, uint32_t *edx) { +// unsigned long long x = _xgetbv(0); +// *eax = x & 0xffffffff; +// *edx = (x >> 32) & 0xffffffff; +// } +// +// Note that _xgetbv is defined starting with GCC 8. +void +gccgoXgetbv(uint32_t *eax, uint32_t *edx) +{ + __asm(" xorl %%ecx, %%ecx\n" + " xgetbv" + : "=a"(*eax), "=d"(*edx)); +} diff --git a/vendor/golang.org/x/sys/cpu/cpu_gccgo.go b/vendor/golang.org/x/sys/cpu/cpu_gccgo.go new file mode 100644 index 0000000..ba49b91 --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_gccgo.go @@ -0,0 +1,26 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build 386 amd64 amd64p32 +// +build gccgo + +package cpu + +//extern gccgoGetCpuidCount +func gccgoGetCpuidCount(eaxArg, ecxArg uint32, eax, ebx, ecx, edx *uint32) + +func cpuid(eaxArg, ecxArg uint32) (eax, ebx, ecx, edx uint32) { + var a, b, c, d uint32 + gccgoGetCpuidCount(eaxArg, ecxArg, &a, &b, &c, &d) + return a, b, c, d +} + +//extern gccgoXgetbv +func gccgoXgetbv(eax, edx *uint32) + +func xgetbv() (eax, edx uint32) { + var a, d uint32 + gccgoXgetbv(&a, &d) + return a, d +} diff --git a/vendor/golang.org/x/sys/cpu/cpu_mips64x.go b/vendor/golang.org/x/sys/cpu/cpu_mips64x.go new file mode 100644 index 0000000..6165f12 --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_mips64x.go @@ -0,0 +1,9 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build mips64 mips64le + +package cpu + +const cacheLineSize = 32 diff --git a/vendor/golang.org/x/sys/cpu/cpu_mipsx.go b/vendor/golang.org/x/sys/cpu/cpu_mipsx.go new file mode 100644 index 0000000..1269eee --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_mipsx.go @@ -0,0 +1,9 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build mips mipsle + +package cpu + +const cacheLineSize = 32 diff --git a/vendor/golang.org/x/sys/cpu/cpu_ppc64x.go b/vendor/golang.org/x/sys/cpu/cpu_ppc64x.go new file mode 100644 index 0000000..d10759a --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_ppc64x.go @@ -0,0 +1,9 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build ppc64 ppc64le + +package cpu + +const cacheLineSize = 128 diff --git a/vendor/golang.org/x/sys/cpu/cpu_s390x.go b/vendor/golang.org/x/sys/cpu/cpu_s390x.go new file mode 100644 index 0000000..684c4f0 --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_s390x.go @@ -0,0 +1,7 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package cpu + +const cacheLineSize = 256 diff --git a/vendor/golang.org/x/sys/cpu/cpu_x86.go b/vendor/golang.org/x/sys/cpu/cpu_x86.go new file mode 100644 index 0000000..71e288b --- /dev/null +++ b/vendor/golang.org/x/sys/cpu/cpu_x86.go @@ -0,0 +1,55 @@ +// Copyright 2018 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build 386 amd64 amd64p32 + +package cpu + +const cacheLineSize = 64 + +func init() { + maxID, _, _, _ := cpuid(0, 0) + + if maxID < 1 { + return + } + + _, _, ecx1, edx1 := cpuid(1, 0) + X86.HasSSE2 = isSet(26, edx1) + + X86.HasSSE3 = isSet(0, ecx1) + X86.HasPCLMULQDQ = isSet(1, ecx1) + X86.HasSSSE3 = isSet(9, ecx1) + X86.HasFMA = isSet(12, ecx1) + X86.HasSSE41 = isSet(19, ecx1) + X86.HasSSE42 = isSet(20, ecx1) + X86.HasPOPCNT = isSet(23, ecx1) + X86.HasAES = isSet(25, ecx1) + X86.HasOSXSAVE = isSet(27, ecx1) + + osSupportsAVX := false + // For XGETBV, OSXSAVE bit is required and sufficient. + if X86.HasOSXSAVE { + eax, _ := xgetbv() + // Check if XMM and YMM registers have OS support. + osSupportsAVX = isSet(1, eax) && isSet(2, eax) + } + + X86.HasAVX = isSet(28, ecx1) && osSupportsAVX + + if maxID < 7 { + return + } + + _, ebx7, _, _ := cpuid(7, 0) + X86.HasBMI1 = isSet(3, ebx7) + X86.HasAVX2 = isSet(5, ebx7) && osSupportsAVX + X86.HasBMI2 = isSet(8, ebx7) + X86.HasERMS = isSet(9, ebx7) + X86.HasADX = isSet(19, ebx7) +} + +func isSet(bitpos uint, value uint32) bool { + return value&(1<