mirror of
https://github.com/minio/minio.git
synced 2025-01-11 15:03:22 -05:00
Update federation target to etcd/clientv3 (#6119)
With CoreDNS now supporting etcdv3 as the DNS backend, we can update our federation target to etcdv3. Users will now be able to use etcdv3 server as the federation backbone. Minio will update bucket data to etcdv3 and CoreDNS can pick that data up and serve it as bucket style DNS path.
This commit is contained in:
parent
adf7340394
commit
2aa18cafc6
@ -22,8 +22,8 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/coreos/etcd/client"
|
|
||||||
"github.com/minio/minio/pkg/auth"
|
"github.com/minio/minio/pkg/auth"
|
||||||
|
"github.com/minio/minio/pkg/dns"
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
"github.com/minio/minio/pkg/hash"
|
"github.com/minio/minio/pkg/hash"
|
||||||
)
|
)
|
||||||
@ -903,10 +903,8 @@ func toAPIErrorCode(err error) (apiErr APIErrorCode) {
|
|||||||
|
|
||||||
// etcd specific errors, a key is always a bucket for us return
|
// etcd specific errors, a key is always a bucket for us return
|
||||||
// ErrNoSuchBucket in such a case.
|
// ErrNoSuchBucket in such a case.
|
||||||
if e, ok := err.(*client.Error); ok {
|
if err == dns.ErrNoEntriesFound {
|
||||||
if e.Code == client.ErrorCodeKeyNotFound {
|
return ErrNoSuchBucket
|
||||||
return ErrNoSuchBucket
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
switch err.(type) {
|
switch err.(type) {
|
||||||
|
@ -30,8 +30,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
|
||||||
|
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
|
|
||||||
"github.com/minio/minio-go/pkg/set"
|
"github.com/minio/minio-go/pkg/set"
|
||||||
@ -64,7 +62,7 @@ func initFederatorBackend(objLayer ObjectLayer) {
|
|||||||
g.Go(func() error {
|
g.Go(func() error {
|
||||||
r, gerr := globalDNSConfig.Get(b[index].Name)
|
r, gerr := globalDNSConfig.Get(b[index].Name)
|
||||||
if gerr != nil {
|
if gerr != nil {
|
||||||
if etcd.IsKeyNotFound(gerr) || gerr == dns.ErrNoEntriesFound {
|
if gerr == dns.ErrNoEntriesFound {
|
||||||
return globalDNSConfig.Put(b[index].Name)
|
return globalDNSConfig.Put(b[index].Name)
|
||||||
}
|
}
|
||||||
return gerr
|
return gerr
|
||||||
@ -211,7 +209,7 @@ func (api objectAPIHandlers) ListBucketsHandler(w http.ResponseWriter, r *http.R
|
|||||||
var bucketsInfo []BucketInfo
|
var bucketsInfo []BucketInfo
|
||||||
if globalDNSConfig != nil {
|
if globalDNSConfig != nil {
|
||||||
dnsBuckets, err := globalDNSConfig.List()
|
dnsBuckets, err := globalDNSConfig.List()
|
||||||
if err != nil && !etcd.IsKeyNotFound(err) && err != dns.ErrNoEntriesFound {
|
if err != nil && err != dns.ErrNoEntriesFound {
|
||||||
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -432,7 +430,7 @@ func (api objectAPIHandlers) PutBucketHandler(w http.ResponseWriter, r *http.Req
|
|||||||
|
|
||||||
if globalDNSConfig != nil {
|
if globalDNSConfig != nil {
|
||||||
if _, err := globalDNSConfig.Get(bucket); err != nil {
|
if _, err := globalDNSConfig.Get(bucket); err != nil {
|
||||||
if etcd.IsKeyNotFound(err) || err == dns.ErrNoEntriesFound {
|
if err == dns.ErrNoEntriesFound {
|
||||||
// Proceed to creating a bucket.
|
// Proceed to creating a bucket.
|
||||||
if err = objectAPI.MakeBucketWithLocation(ctx, bucket, location); err != nil {
|
if err = objectAPI.MakeBucketWithLocation(ctx, bucket, location); err != nil {
|
||||||
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
||||||
|
@ -26,7 +26,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
etcd "github.com/coreos/etcd/clientv3"
|
||||||
|
|
||||||
"github.com/minio/cli"
|
"github.com/minio/cli"
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
@ -51,17 +51,19 @@ func checkUpdate(mode string) {
|
|||||||
// Initialize and load config from remote etcd or local config directory
|
// Initialize and load config from remote etcd or local config directory
|
||||||
func initConfig() {
|
func initConfig() {
|
||||||
if globalEtcdClient != nil {
|
if globalEtcdClient != nil {
|
||||||
kapi := etcd.NewKeysAPI(globalEtcdClient)
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
|
||||||
_, err := kapi.Get(ctx, getConfigFile(), nil)
|
resp, err := globalEtcdClient.Get(ctx, getConfigFile())
|
||||||
cancel()
|
cancel()
|
||||||
if err == nil {
|
// This means there are no entries in etcd with config file
|
||||||
logger.FatalIf(migrateConfig(), "Config migration failed.")
|
// So create a new config
|
||||||
logger.FatalIf(loadConfig(), "Unable to load config version: '%s'.", serverConfigVersion)
|
if err == nil && resp.Count == 0 {
|
||||||
|
logger.FatalIf(newConfig(), "Unable to initialize minio config for the first time.")
|
||||||
|
logger.Info("Created minio configuration file successfully at %v", globalEtcdClient.Endpoints())
|
||||||
} else {
|
} else {
|
||||||
if etcd.IsKeyNotFound(err) {
|
// This means there is an entry in etcd, update it if required and proceed
|
||||||
logger.FatalIf(newConfig(), "Unable to initialize minio config for the first time.")
|
if err == nil && resp.Count > 0 {
|
||||||
logger.Info("Created minio configuration file successfully at %v", globalEtcdClient.Endpoints())
|
logger.FatalIf(migrateConfig(), "Config migration failed.")
|
||||||
|
logger.FatalIf(loadConfig(), "Unable to load config version: '%s'.", serverConfigVersion)
|
||||||
} else {
|
} else {
|
||||||
logger.FatalIf(err, "Unable to load config version: '%s'.", serverConfigVersion)
|
logger.FatalIf(err, "Unable to load config version: '%s'.", serverConfigVersion)
|
||||||
}
|
}
|
||||||
@ -155,8 +157,9 @@ func handleCommonEnvVars() {
|
|||||||
etcdEndpoints := strings.Split(etcdEndpointsEnv, ",")
|
etcdEndpoints := strings.Split(etcdEndpointsEnv, ",")
|
||||||
var err error
|
var err error
|
||||||
globalEtcdClient, err = etcd.New(etcd.Config{
|
globalEtcdClient, err = etcd.New(etcd.Config{
|
||||||
Endpoints: etcdEndpoints,
|
Endpoints: etcdEndpoints,
|
||||||
Transport: NewCustomHTTPTransport(),
|
DialTimeout: defaultDialTimeout,
|
||||||
|
DialKeepAliveTime: defaultDialKeepAlive,
|
||||||
})
|
})
|
||||||
logger.FatalIf(err, "Unable to initialize etcd with %s", etcdEndpoints)
|
logger.FatalIf(err, "Unable to initialize etcd with %s", etcdEndpoints)
|
||||||
}
|
}
|
||||||
|
@ -21,9 +21,9 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
"github.com/minio/minio/pkg/auth"
|
"github.com/minio/minio/pkg/auth"
|
||||||
|
"github.com/minio/minio/pkg/dns"
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
"github.com/minio/minio/pkg/event/target"
|
"github.com/minio/minio/pkg/event/target"
|
||||||
xnet "github.com/minio/minio/pkg/net"
|
xnet "github.com/minio/minio/pkg/net"
|
||||||
@ -201,7 +201,7 @@ func purgeV1() error {
|
|||||||
|
|
||||||
cv1 := &configV1{}
|
cv1 := &configV1{}
|
||||||
_, err := Load(configFile, cv1)
|
_, err := Load(configFile, cv1)
|
||||||
if os.IsNotExist(err) || etcd.IsKeyNotFound(err) {
|
if os.IsNotExist(err) || err == dns.ErrNoEntriesFound {
|
||||||
return nil
|
return nil
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
return fmt.Errorf("Unable to load config version ‘1’. %v", err)
|
return fmt.Errorf("Unable to load config version ‘1’. %v", err)
|
||||||
|
@ -27,7 +27,6 @@ import (
|
|||||||
|
|
||||||
"github.com/minio/minio-go/pkg/set"
|
"github.com/minio/minio-go/pkg/set"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
|
||||||
humanize "github.com/dustin/go-humanize"
|
humanize "github.com/dustin/go-humanize"
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
"github.com/minio/minio/pkg/dns"
|
"github.com/minio/minio/pkg/dns"
|
||||||
@ -653,7 +652,7 @@ func (f bucketForwardingHandler) ServeHTTP(w http.ResponseWriter, r *http.Reques
|
|||||||
}
|
}
|
||||||
sr, err := globalDNSConfig.Get(bucket)
|
sr, err := globalDNSConfig.Get(bucket)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if etcd.IsKeyNotFound(err) || err == dns.ErrNoEntriesFound {
|
if err == dns.ErrNoEntriesFound {
|
||||||
writeErrorResponse(w, ErrNoSuchBucket, r.URL)
|
writeErrorResponse(w, ErrNoSuchBucket, r.URL)
|
||||||
} else {
|
} else {
|
||||||
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
||||||
|
@ -24,7 +24,7 @@ import (
|
|||||||
|
|
||||||
"github.com/minio/minio-go/pkg/set"
|
"github.com/minio/minio-go/pkg/set"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
etcd "github.com/coreos/etcd/clientv3"
|
||||||
humanize "github.com/dustin/go-humanize"
|
humanize "github.com/dustin/go-humanize"
|
||||||
"github.com/fatih/color"
|
"github.com/fatih/color"
|
||||||
xhttp "github.com/minio/minio/cmd/http"
|
xhttp "github.com/minio/minio/cmd/http"
|
||||||
@ -204,7 +204,7 @@ var (
|
|||||||
globalRPCAPIVersion = RPCVersion{3, 0, 0}
|
globalRPCAPIVersion = RPCVersion{3, 0, 0}
|
||||||
|
|
||||||
// Allocated etcd endpoint for config and bucket DNS.
|
// Allocated etcd endpoint for config and bucket DNS.
|
||||||
globalEtcdClient etcd.Client
|
globalEtcdClient *etcd.Client
|
||||||
|
|
||||||
// Allocated DNS config wrapper over etcd client.
|
// Allocated DNS config wrapper over etcd client.
|
||||||
globalDNSConfig dns.Config
|
globalDNSConfig dns.Config
|
||||||
|
@ -31,7 +31,6 @@ import (
|
|||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
miniogo "github.com/minio/minio-go"
|
miniogo "github.com/minio/minio-go"
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
@ -1547,7 +1546,7 @@ func (api objectAPIHandlers) DeleteObjectHandler(w http.ResponseWriter, r *http.
|
|||||||
if globalDNSConfig != nil {
|
if globalDNSConfig != nil {
|
||||||
_, err := globalDNSConfig.Get(bucket)
|
_, err := globalDNSConfig.Get(bucket)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if etcd.IsKeyNotFound(err) || err == dns.ErrNoEntriesFound {
|
if err == dns.ErrNoEntriesFound {
|
||||||
writeErrorResponse(w, ErrNoSuchBucket, r.URL)
|
writeErrorResponse(w, ErrNoSuchBucket, r.URL)
|
||||||
} else {
|
} else {
|
||||||
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
writeErrorResponse(w, toAPIErrorCode(err), r.URL)
|
||||||
|
@ -142,6 +142,10 @@ const (
|
|||||||
// Maximum Part ID for multipart upload is 10000
|
// Maximum Part ID for multipart upload is 10000
|
||||||
// (Acceptable values range from 1 to 10000 inclusive)
|
// (Acceptable values range from 1 to 10000 inclusive)
|
||||||
globalMaxPartID = 10000
|
globalMaxPartID = 10000
|
||||||
|
|
||||||
|
// Default values used while communicating with the cloud backends
|
||||||
|
defaultDialTimeout = 30 * time.Second
|
||||||
|
defaultDialKeepAlive = 30 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
// isMaxObjectSize - verify if max object size
|
// isMaxObjectSize - verify if max object size
|
||||||
@ -263,8 +267,8 @@ func NewCustomHTTPTransport() *http.Transport {
|
|||||||
return &http.Transport{
|
return &http.Transport{
|
||||||
Proxy: http.ProxyFromEnvironment,
|
Proxy: http.ProxyFromEnvironment,
|
||||||
DialContext: (&net.Dialer{
|
DialContext: (&net.Dialer{
|
||||||
Timeout: 30 * time.Second,
|
Timeout: defaultDialTimeout,
|
||||||
KeepAlive: 30 * time.Second,
|
KeepAlive: defaultDialKeepAlive,
|
||||||
}).DialContext,
|
}).DialContext,
|
||||||
MaxIdleConns: 1024,
|
MaxIdleConns: 1024,
|
||||||
MaxIdleConnsPerHost: 1024,
|
MaxIdleConnsPerHost: 1024,
|
||||||
|
@ -31,7 +31,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
|
||||||
humanize "github.com/dustin/go-humanize"
|
humanize "github.com/dustin/go-humanize"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"github.com/gorilla/rpc/v2/json2"
|
"github.com/gorilla/rpc/v2/json2"
|
||||||
@ -139,7 +138,7 @@ func (web *webAPIHandlers) MakeBucket(r *http.Request, args *MakeBucketArgs, rep
|
|||||||
|
|
||||||
if globalDNSConfig != nil {
|
if globalDNSConfig != nil {
|
||||||
if _, err := globalDNSConfig.Get(args.BucketName); err != nil {
|
if _, err := globalDNSConfig.Get(args.BucketName); err != nil {
|
||||||
if etcd.IsKeyNotFound(err) || err == dns.ErrNoEntriesFound {
|
if err == dns.ErrNoEntriesFound {
|
||||||
// Proceed to creating a bucket.
|
// Proceed to creating a bucket.
|
||||||
if err = objectAPI.MakeBucketWithLocation(context.Background(), args.BucketName, globalServerConfig.GetRegion()); err != nil {
|
if err = objectAPI.MakeBucketWithLocation(context.Background(), args.BucketName, globalServerConfig.GetRegion()); err != nil {
|
||||||
return toJSONError(err)
|
return toJSONError(err)
|
||||||
|
@ -10,7 +10,7 @@ Install Minio - [Minio Quickstart Guide](https://docs.minio.io/docs/minio-quicks
|
|||||||
Bucket lookup from DNS federation requires two dependencies
|
Bucket lookup from DNS federation requires two dependencies
|
||||||
|
|
||||||
- etcd (for config, bucket SRV records)
|
- etcd (for config, bucket SRV records)
|
||||||
- coredns (for DNS management based on populated bucket SRV records, optional)
|
- CoreDNS (for DNS management based on populated bucket SRV records, optional)
|
||||||
|
|
||||||
## Architecture
|
## Architecture
|
||||||
|
|
||||||
@ -73,10 +73,22 @@ NOTE: `mybucket` only exists on one cluster either `cluster1` or `cluster2` this
|
|||||||
is decided by how `domain.com` gets resolved, if there is a round-robin DNS on `domain.com` then
|
is decided by how `domain.com` gets resolved, if there is a round-robin DNS on `domain.com` then
|
||||||
it is randomized which cluster might provision the bucket.
|
it is randomized which cluster might provision the bucket.
|
||||||
|
|
||||||
### 3. Test your setup
|
### 3. Upgrading to `etcdv3` API
|
||||||
|
|
||||||
|
Users running Minio federation from release `RELEASE.2018-06-09T03-43-35Z` to `RELEASE.2018-07-10T01-42-11Z`, should migrate the existing bucket data on etcd server to `etcdv3` API, and update CoreDNS version to `1.2.0` before updating their Minio server to the latest version.
|
||||||
|
|
||||||
|
Here is some background on why this is needed - Minio server release `RELEASE.2018-06-09T03-43-35Z` to `RELEASE.2018-07-10T01-42-11Z` used etcdv2 API to store bucket data to etcd server. This was due to `etcdv3` support not available for CoreDNS server. So, even if Minio used `etcdv3` API to store bucket data, CoreDNS wouldn't be able to read and serve it as DNS records.
|
||||||
|
|
||||||
|
Now that CoreDNS [supports etcdv3](https://coredns.io/2018/07/11/coredns-1.2.0-release/), Minio server uses `etcdv3` API to store bucket data to etcd server. As `etcdv2` and `etcdv3` APIs are not compatible, data stored using `etcdv2` API is not visible to the `etcdv3` API. So, bucket data stored by previous Minio version will not be visible to current Minio version, until a migration is done.
|
||||||
|
|
||||||
|
CoreOS team has documented the steps required to migrate existing data from `etcdv2` to `etcdv3` in [this blog post](https://coreos.com/blog/migrating-applications-etcd-v3.html). Please refer the post and migrate etcd data to `etcdv3` API.
|
||||||
|
|
||||||
|
### 4. Test your setup
|
||||||
|
|
||||||
To test this setup, access the Minio server via browser or [`mc`](https://docs.minio.io/docs/minio-client-quickstart-guide). You’ll see the uploaded files are accessible from the all the Minio endpoints.
|
To test this setup, access the Minio server via browser or [`mc`](https://docs.minio.io/docs/minio-client-quickstart-guide). You’ll see the uploaded files are accessible from the all the Minio endpoints.
|
||||||
|
|
||||||
# Explore Further
|
# Explore Further
|
||||||
|
|
||||||
- [Use `mc` with Minio Server](https://docs.minio.io/docs/minio-client-quickstart-guide)
|
- [Use `mc` with Minio Server](https://docs.minio.io/docs/minio-client-quickstart-guide)
|
||||||
- [Use `aws-cli` with Minio Server](https://docs.minio.io/docs/aws-cli-with-minio)
|
- [Use `aws-cli` with Minio Server](https://docs.minio.io/docs/aws-cli-with-minio)
|
||||||
- [Use `s3cmd` with Minio Server](https://docs.minio.io/docs/s3cmd-with-minio)
|
- [Use `s3cmd` with Minio Server](https://docs.minio.io/docs/s3cmd-with-minio)
|
||||||
|
@ -29,7 +29,7 @@ import (
|
|||||||
"github.com/minio/minio-go/pkg/set"
|
"github.com/minio/minio-go/pkg/set"
|
||||||
|
|
||||||
"github.com/coredns/coredns/plugin/etcd/msg"
|
"github.com/coredns/coredns/plugin/etcd/msg"
|
||||||
etcd "github.com/coreos/etcd/client"
|
etcd "github.com/coreos/etcd/clientv3"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ErrNoEntriesFound - Indicates no entries were found for the given key (directory)
|
// ErrNoEntriesFound - Indicates no entries were found for the given key (directory)
|
||||||
@ -60,35 +60,33 @@ func (c *coreDNS) Get(bucket string) ([]SrvRecord, error) {
|
|||||||
// Retrieves list of entries under the key passed.
|
// Retrieves list of entries under the key passed.
|
||||||
// Note that this method fetches entries upto only two levels deep.
|
// Note that this method fetches entries upto only two levels deep.
|
||||||
func (c *coreDNS) list(key string) ([]SrvRecord, error) {
|
func (c *coreDNS) list(key string) ([]SrvRecord, error) {
|
||||||
kapi := etcd.NewKeysAPI(c.etcdClient)
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), defaultContextTimeout)
|
ctx, cancel := context.WithTimeout(context.Background(), defaultContextTimeout)
|
||||||
r, err := kapi.Get(ctx, key, &etcd.GetOptions{Recursive: true})
|
r, err := c.etcdClient.Get(ctx, key, etcd.WithPrefix())
|
||||||
cancel()
|
defer cancel()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
if r.Count == 0 {
|
||||||
|
key = strings.TrimSuffix(key, "/")
|
||||||
|
r, err = c.etcdClient.Get(ctx, key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if r.Count == 0 {
|
||||||
|
return nil, ErrNoEntriesFound
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var srvRecords []SrvRecord
|
var srvRecords []SrvRecord
|
||||||
for _, n := range r.Node.Nodes {
|
for _, n := range r.Kvs {
|
||||||
if !n.Dir {
|
var srvRecord SrvRecord
|
||||||
var srvRecord SrvRecord
|
if err = json.Unmarshal([]byte(n.Value), &srvRecord); err != nil {
|
||||||
if err = json.Unmarshal([]byte(n.Value), &srvRecord); err != nil {
|
return nil, err
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
srvRecord.Key = strings.TrimPrefix(n.Key, key)
|
|
||||||
srvRecords = append(srvRecords, srvRecord)
|
|
||||||
} else {
|
|
||||||
// As this is a directory, loop through all the nodes inside (assuming all nodes are non-directories)
|
|
||||||
for _, n1 := range n.Nodes {
|
|
||||||
var srvRecord SrvRecord
|
|
||||||
if err = json.Unmarshal([]byte(n1.Value), &srvRecord); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
srvRecord.Key = strings.TrimPrefix(n1.Key, key)
|
|
||||||
srvRecord.Key = strings.TrimSuffix(srvRecord.Key, srvRecord.Host)
|
|
||||||
srvRecords = append(srvRecords, srvRecord)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
srvRecord.Key = strings.TrimPrefix(string(n.Key), key)
|
||||||
|
srvRecord.Key = strings.TrimSuffix(srvRecord.Key, srvRecord.Host)
|
||||||
|
srvRecords = append(srvRecords, srvRecord)
|
||||||
|
|
||||||
}
|
}
|
||||||
if srvRecords != nil {
|
if srvRecords != nil {
|
||||||
sort.Slice(srvRecords, func(i int, j int) bool {
|
sort.Slice(srvRecords, func(i int, j int) bool {
|
||||||
@ -107,16 +105,15 @@ func (c *coreDNS) Put(bucket string) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
kapi := etcd.NewKeysAPI(c.etcdClient)
|
|
||||||
key := msg.Path(fmt.Sprintf("%s.%s", bucket, c.domainName), defaultPrefixPath)
|
key := msg.Path(fmt.Sprintf("%s.%s", bucket, c.domainName), defaultPrefixPath)
|
||||||
key = key + "/" + ip
|
key = key + "/" + ip
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), defaultContextTimeout)
|
ctx, cancel := context.WithTimeout(context.Background(), defaultContextTimeout)
|
||||||
_, err = kapi.Set(ctx, key, string(bucketMsg), nil)
|
_, err = c.etcdClient.Put(ctx, key, string(bucketMsg))
|
||||||
cancel()
|
defer cancel()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx, cancel = context.WithTimeout(context.Background(), defaultContextTimeout)
|
ctx, cancel = context.WithTimeout(context.Background(), defaultContextTimeout)
|
||||||
kapi.Delete(ctx, key, &etcd.DeleteOptions{Recursive: true})
|
c.etcdClient.Delete(ctx, key)
|
||||||
cancel()
|
defer cancel()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -125,11 +122,10 @@ func (c *coreDNS) Put(bucket string) error {
|
|||||||
|
|
||||||
// Removes DNS entries added in Put().
|
// Removes DNS entries added in Put().
|
||||||
func (c *coreDNS) Delete(bucket string) error {
|
func (c *coreDNS) Delete(bucket string) error {
|
||||||
kapi := etcd.NewKeysAPI(c.etcdClient)
|
|
||||||
key := msg.Path(fmt.Sprintf("%s.%s.", bucket, c.domainName), defaultPrefixPath)
|
key := msg.Path(fmt.Sprintf("%s.%s.", bucket, c.domainName), defaultPrefixPath)
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), defaultContextTimeout)
|
ctx, cancel := context.WithTimeout(context.Background(), defaultContextTimeout)
|
||||||
_, err := kapi.Delete(ctx, key, &etcd.DeleteOptions{Recursive: true})
|
_, err := c.etcdClient.Delete(ctx, key)
|
||||||
cancel()
|
defer cancel()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,12 +134,12 @@ type coreDNS struct {
|
|||||||
domainName string
|
domainName string
|
||||||
domainIPs set.StringSet
|
domainIPs set.StringSet
|
||||||
domainPort int
|
domainPort int
|
||||||
etcdClient etcd.Client
|
etcdClient *etcd.Client
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewCoreDNS - initialize a new coreDNS set/unset values.
|
// NewCoreDNS - initialize a new coreDNS set/unset values.
|
||||||
func NewCoreDNS(domainName string, domainIPs set.StringSet, domainPort string, etcdClient etcd.Client) (Config, error) {
|
func NewCoreDNS(domainName string, domainIPs set.StringSet, domainPort string, etcdClient *etcd.Client) (Config, error) {
|
||||||
if domainName == "" || domainIPs.IsEmpty() || etcdClient == nil {
|
if domainName == "" || domainIPs.IsEmpty() {
|
||||||
return nil, errors.New("invalid argument")
|
return nil, errors.New("invalid argument")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,7 +30,8 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
etcd "github.com/coreos/etcd/clientv3"
|
||||||
|
dns "github.com/minio/minio/pkg/dns"
|
||||||
yaml "gopkg.in/yaml.v2"
|
yaml "gopkg.in/yaml.v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -125,7 +126,7 @@ func saveFileConfig(filename string, v interface{}) error {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func saveFileConfigEtcd(filename string, clnt etcd.Client, v interface{}) error {
|
func saveFileConfigEtcd(filename string, clnt *etcd.Client, v interface{}) error {
|
||||||
// Fetch filename's extension
|
// Fetch filename's extension
|
||||||
ext := filepath.Ext(filename)
|
ext := filepath.Ext(filename)
|
||||||
// Marshal data
|
// Marshal data
|
||||||
@ -137,44 +138,34 @@ func saveFileConfigEtcd(filename string, clnt etcd.Client, v interface{}) error
|
|||||||
dataBytes = []byte(strings.Replace(string(dataBytes), "\n", "\r\n", -1))
|
dataBytes = []byte(strings.Replace(string(dataBytes), "\n", "\r\n", -1))
|
||||||
}
|
}
|
||||||
|
|
||||||
kapi := etcd.NewKeysAPI(clnt)
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute)
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute)
|
||||||
_, err = kapi.Update(ctx, filename, string(dataBytes))
|
_, err = clnt.Put(ctx, filename, string(dataBytes))
|
||||||
if etcd.IsKeyNotFound(err) {
|
defer cancel()
|
||||||
_, err = kapi.Create(ctx, filename, string(dataBytes))
|
|
||||||
}
|
|
||||||
cancel()
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func loadFileConfigEtcd(filename string, clnt etcd.Client, v interface{}) error {
|
func loadFileConfigEtcd(filename string, clnt *etcd.Client, v interface{}) error {
|
||||||
kapi := etcd.NewKeysAPI(clnt)
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute)
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute)
|
||||||
resp, err := kapi.Get(ctx, filename, nil)
|
resp, err := clnt.Get(ctx, filename)
|
||||||
cancel()
|
defer cancel()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
if resp.Count == 0 {
|
||||||
|
return dns.ErrNoEntriesFound
|
||||||
|
}
|
||||||
|
|
||||||
var ev *etcd.Node
|
for _, ev := range resp.Kvs {
|
||||||
switch {
|
if string(ev.Key) == filename {
|
||||||
case resp.Node.Dir:
|
fileData := ev.Value
|
||||||
for _, ev = range resp.Node.Nodes {
|
if runtime.GOOS == "windows" {
|
||||||
if string(ev.Key) == filename {
|
fileData = bytes.Replace(fileData, []byte("\r\n"), []byte("\n"), -1)
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
// Unmarshal file's content
|
||||||
|
return toUnmarshaller(filepath.Ext(filename))(fileData, v)
|
||||||
}
|
}
|
||||||
default:
|
|
||||||
ev = resp.Node
|
|
||||||
}
|
}
|
||||||
|
return dns.ErrNoEntriesFound
|
||||||
fileData := ev.Value
|
|
||||||
if runtime.GOOS == "windows" {
|
|
||||||
fileData = strings.Replace(ev.Value, "\r\n", "\n", -1)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unmarshal file's content
|
|
||||||
return toUnmarshaller(filepath.Ext(filename))([]byte(fileData), v)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// loadFileConfig unmarshals the file's content with the right
|
// loadFileConfig unmarshals the file's content with the right
|
||||||
|
@ -26,7 +26,7 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
etcd "github.com/coreos/etcd/client"
|
etcd "github.com/coreos/etcd/clientv3"
|
||||||
"github.com/fatih/structs"
|
"github.com/fatih/structs"
|
||||||
"github.com/minio/minio/pkg/safe"
|
"github.com/minio/minio/pkg/safe"
|
||||||
)
|
)
|
||||||
@ -45,7 +45,7 @@ type Config interface {
|
|||||||
// config - implements quick.Config interface
|
// config - implements quick.Config interface
|
||||||
type config struct {
|
type config struct {
|
||||||
data interface{}
|
data interface{}
|
||||||
clnt etcd.Client
|
clnt *etcd.Client
|
||||||
lock *sync.RWMutex
|
lock *sync.RWMutex
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -189,7 +189,7 @@ func writeFile(filename string, data []byte) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// GetVersion - extracts the version information.
|
// GetVersion - extracts the version information.
|
||||||
func GetVersion(filename string, clnt etcd.Client) (version string, err error) {
|
func GetVersion(filename string, clnt *etcd.Client) (version string, err error) {
|
||||||
var qc Config
|
var qc Config
|
||||||
qc, err = LoadConfig(filename, clnt, &struct {
|
qc, err = LoadConfig(filename, clnt, &struct {
|
||||||
Version string
|
Version string
|
||||||
@ -201,7 +201,7 @@ func GetVersion(filename string, clnt etcd.Client) (version string, err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// LoadConfig - loads json config from filename for the a given struct data
|
// LoadConfig - loads json config from filename for the a given struct data
|
||||||
func LoadConfig(filename string, clnt etcd.Client, data interface{}) (qc Config, err error) {
|
func LoadConfig(filename string, clnt *etcd.Client, data interface{}) (qc Config, err error) {
|
||||||
qc, err = NewConfig(data, clnt)
|
qc, err = NewConfig(data, clnt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -210,7 +210,7 @@ func LoadConfig(filename string, clnt etcd.Client, data interface{}) (qc Config,
|
|||||||
}
|
}
|
||||||
|
|
||||||
// SaveConfig - saves given configuration data into given file as JSON.
|
// SaveConfig - saves given configuration data into given file as JSON.
|
||||||
func SaveConfig(data interface{}, filename string, clnt etcd.Client) (err error) {
|
func SaveConfig(data interface{}, filename string, clnt *etcd.Client) (err error) {
|
||||||
if err = checkData(data); err != nil {
|
if err = checkData(data); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -224,7 +224,7 @@ func SaveConfig(data interface{}, filename string, clnt etcd.Client) (err error)
|
|||||||
|
|
||||||
// NewConfig loads config from etcd client if provided, otherwise loads from a local filename.
|
// NewConfig loads config from etcd client if provided, otherwise loads from a local filename.
|
||||||
// fails when all else fails.
|
// fails when all else fails.
|
||||||
func NewConfig(data interface{}, clnt etcd.Client) (cfg Config, err error) {
|
func NewConfig(data interface{}, clnt *etcd.Client) (cfg Config, err error) {
|
||||||
if err := checkData(data); err != nil {
|
if err := checkData(data); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
807
vendor/github.com/coreos/etcd/auth/authpb/auth.pb.go
generated
vendored
Normal file
807
vendor/github.com/coreos/etcd/auth/authpb/auth.pb.go
generated
vendored
Normal file
@ -0,0 +1,807 @@
|
|||||||
|
// Code generated by protoc-gen-gogo. DO NOT EDIT.
|
||||||
|
// source: auth.proto
|
||||||
|
|
||||||
|
/*
|
||||||
|
Package authpb is a generated protocol buffer package.
|
||||||
|
|
||||||
|
It is generated from these files:
|
||||||
|
auth.proto
|
||||||
|
|
||||||
|
It has these top-level messages:
|
||||||
|
User
|
||||||
|
Permission
|
||||||
|
Role
|
||||||
|
*/
|
||||||
|
package authpb
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
proto "github.com/golang/protobuf/proto"
|
||||||
|
|
||||||
|
math "math"
|
||||||
|
|
||||||
|
_ "github.com/gogo/protobuf/gogoproto"
|
||||||
|
|
||||||
|
io "io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Reference imports to suppress errors if they are not otherwise used.
|
||||||
|
var _ = proto.Marshal
|
||||||
|
var _ = fmt.Errorf
|
||||||
|
var _ = math.Inf
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the proto package it is being compiled against.
|
||||||
|
// A compilation error at this line likely means your copy of the
|
||||||
|
// proto package needs to be updated.
|
||||||
|
const _ = proto.ProtoPackageIsVersion2 // please upgrade the proto package
|
||||||
|
|
||||||
|
type Permission_Type int32
|
||||||
|
|
||||||
|
const (
|
||||||
|
READ Permission_Type = 0
|
||||||
|
WRITE Permission_Type = 1
|
||||||
|
READWRITE Permission_Type = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
var Permission_Type_name = map[int32]string{
|
||||||
|
0: "READ",
|
||||||
|
1: "WRITE",
|
||||||
|
2: "READWRITE",
|
||||||
|
}
|
||||||
|
var Permission_Type_value = map[string]int32{
|
||||||
|
"READ": 0,
|
||||||
|
"WRITE": 1,
|
||||||
|
"READWRITE": 2,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x Permission_Type) String() string {
|
||||||
|
return proto.EnumName(Permission_Type_name, int32(x))
|
||||||
|
}
|
||||||
|
func (Permission_Type) EnumDescriptor() ([]byte, []int) { return fileDescriptorAuth, []int{1, 0} }
|
||||||
|
|
||||||
|
// User is a single entry in the bucket authUsers
|
||||||
|
type User struct {
|
||||||
|
Name []byte `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
|
||||||
|
Password []byte `protobuf:"bytes,2,opt,name=password,proto3" json:"password,omitempty"`
|
||||||
|
Roles []string `protobuf:"bytes,3,rep,name=roles" json:"roles,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *User) Reset() { *m = User{} }
|
||||||
|
func (m *User) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*User) ProtoMessage() {}
|
||||||
|
func (*User) Descriptor() ([]byte, []int) { return fileDescriptorAuth, []int{0} }
|
||||||
|
|
||||||
|
// Permission is a single entity
|
||||||
|
type Permission struct {
|
||||||
|
PermType Permission_Type `protobuf:"varint,1,opt,name=permType,proto3,enum=authpb.Permission_Type" json:"permType,omitempty"`
|
||||||
|
Key []byte `protobuf:"bytes,2,opt,name=key,proto3" json:"key,omitempty"`
|
||||||
|
RangeEnd []byte `protobuf:"bytes,3,opt,name=range_end,json=rangeEnd,proto3" json:"range_end,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Permission) Reset() { *m = Permission{} }
|
||||||
|
func (m *Permission) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*Permission) ProtoMessage() {}
|
||||||
|
func (*Permission) Descriptor() ([]byte, []int) { return fileDescriptorAuth, []int{1} }
|
||||||
|
|
||||||
|
// Role is a single entry in the bucket authRoles
|
||||||
|
type Role struct {
|
||||||
|
Name []byte `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
|
||||||
|
KeyPermission []*Permission `protobuf:"bytes,2,rep,name=keyPermission" json:"keyPermission,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Role) Reset() { *m = Role{} }
|
||||||
|
func (m *Role) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*Role) ProtoMessage() {}
|
||||||
|
func (*Role) Descriptor() ([]byte, []int) { return fileDescriptorAuth, []int{2} }
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
proto.RegisterType((*User)(nil), "authpb.User")
|
||||||
|
proto.RegisterType((*Permission)(nil), "authpb.Permission")
|
||||||
|
proto.RegisterType((*Role)(nil), "authpb.Role")
|
||||||
|
proto.RegisterEnum("authpb.Permission_Type", Permission_Type_name, Permission_Type_value)
|
||||||
|
}
|
||||||
|
func (m *User) Marshal() (dAtA []byte, err error) {
|
||||||
|
size := m.Size()
|
||||||
|
dAtA = make([]byte, size)
|
||||||
|
n, err := m.MarshalTo(dAtA)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return dAtA[:n], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *User) MarshalTo(dAtA []byte) (int, error) {
|
||||||
|
var i int
|
||||||
|
_ = i
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if len(m.Name) > 0 {
|
||||||
|
dAtA[i] = 0xa
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(len(m.Name)))
|
||||||
|
i += copy(dAtA[i:], m.Name)
|
||||||
|
}
|
||||||
|
if len(m.Password) > 0 {
|
||||||
|
dAtA[i] = 0x12
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(len(m.Password)))
|
||||||
|
i += copy(dAtA[i:], m.Password)
|
||||||
|
}
|
||||||
|
if len(m.Roles) > 0 {
|
||||||
|
for _, s := range m.Roles {
|
||||||
|
dAtA[i] = 0x1a
|
||||||
|
i++
|
||||||
|
l = len(s)
|
||||||
|
for l >= 1<<7 {
|
||||||
|
dAtA[i] = uint8(uint64(l)&0x7f | 0x80)
|
||||||
|
l >>= 7
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
dAtA[i] = uint8(l)
|
||||||
|
i++
|
||||||
|
i += copy(dAtA[i:], s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Permission) Marshal() (dAtA []byte, err error) {
|
||||||
|
size := m.Size()
|
||||||
|
dAtA = make([]byte, size)
|
||||||
|
n, err := m.MarshalTo(dAtA)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return dAtA[:n], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Permission) MarshalTo(dAtA []byte) (int, error) {
|
||||||
|
var i int
|
||||||
|
_ = i
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if m.PermType != 0 {
|
||||||
|
dAtA[i] = 0x8
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(m.PermType))
|
||||||
|
}
|
||||||
|
if len(m.Key) > 0 {
|
||||||
|
dAtA[i] = 0x12
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(len(m.Key)))
|
||||||
|
i += copy(dAtA[i:], m.Key)
|
||||||
|
}
|
||||||
|
if len(m.RangeEnd) > 0 {
|
||||||
|
dAtA[i] = 0x1a
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(len(m.RangeEnd)))
|
||||||
|
i += copy(dAtA[i:], m.RangeEnd)
|
||||||
|
}
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Role) Marshal() (dAtA []byte, err error) {
|
||||||
|
size := m.Size()
|
||||||
|
dAtA = make([]byte, size)
|
||||||
|
n, err := m.MarshalTo(dAtA)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return dAtA[:n], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Role) MarshalTo(dAtA []byte) (int, error) {
|
||||||
|
var i int
|
||||||
|
_ = i
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if len(m.Name) > 0 {
|
||||||
|
dAtA[i] = 0xa
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(len(m.Name)))
|
||||||
|
i += copy(dAtA[i:], m.Name)
|
||||||
|
}
|
||||||
|
if len(m.KeyPermission) > 0 {
|
||||||
|
for _, msg := range m.KeyPermission {
|
||||||
|
dAtA[i] = 0x12
|
||||||
|
i++
|
||||||
|
i = encodeVarintAuth(dAtA, i, uint64(msg.Size()))
|
||||||
|
n, err := msg.MarshalTo(dAtA[i:])
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
i += n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func encodeVarintAuth(dAtA []byte, offset int, v uint64) int {
|
||||||
|
for v >= 1<<7 {
|
||||||
|
dAtA[offset] = uint8(v&0x7f | 0x80)
|
||||||
|
v >>= 7
|
||||||
|
offset++
|
||||||
|
}
|
||||||
|
dAtA[offset] = uint8(v)
|
||||||
|
return offset + 1
|
||||||
|
}
|
||||||
|
func (m *User) Size() (n int) {
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
l = len(m.Name)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
l = len(m.Password)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
if len(m.Roles) > 0 {
|
||||||
|
for _, s := range m.Roles {
|
||||||
|
l = len(s)
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Permission) Size() (n int) {
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if m.PermType != 0 {
|
||||||
|
n += 1 + sovAuth(uint64(m.PermType))
|
||||||
|
}
|
||||||
|
l = len(m.Key)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
l = len(m.RangeEnd)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Role) Size() (n int) {
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
l = len(m.Name)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
if len(m.KeyPermission) > 0 {
|
||||||
|
for _, e := range m.KeyPermission {
|
||||||
|
l = e.Size()
|
||||||
|
n += 1 + l + sovAuth(uint64(l))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func sovAuth(x uint64) (n int) {
|
||||||
|
for {
|
||||||
|
n++
|
||||||
|
x >>= 7
|
||||||
|
if x == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
func sozAuth(x uint64) (n int) {
|
||||||
|
return sovAuth(uint64((x << 1) ^ uint64((int64(x) >> 63))))
|
||||||
|
}
|
||||||
|
func (m *User) Unmarshal(dAtA []byte) error {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
preIndex := iNdEx
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fieldNum := int32(wire >> 3)
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
if wireType == 4 {
|
||||||
|
return fmt.Errorf("proto: User: wiretype end group for non-group")
|
||||||
|
}
|
||||||
|
if fieldNum <= 0 {
|
||||||
|
return fmt.Errorf("proto: User: illegal tag %d (wire type %d)", fieldNum, wire)
|
||||||
|
}
|
||||||
|
switch fieldNum {
|
||||||
|
case 1:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Name = append(m.Name[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.Name == nil {
|
||||||
|
m.Name = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 2:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Password", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Password = append(m.Password[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.Password == nil {
|
||||||
|
m.Password = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 3:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Roles", wireType)
|
||||||
|
}
|
||||||
|
var stringLen uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
stringLen |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
intStringLen := int(stringLen)
|
||||||
|
if intStringLen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + intStringLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Roles = append(m.Roles, string(dAtA[iNdEx:postIndex]))
|
||||||
|
iNdEx = postIndex
|
||||||
|
default:
|
||||||
|
iNdEx = preIndex
|
||||||
|
skippy, err := skipAuth(dAtA[iNdEx:])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if skippy < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
if (iNdEx + skippy) > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx += skippy
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if iNdEx > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func (m *Permission) Unmarshal(dAtA []byte) error {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
preIndex := iNdEx
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fieldNum := int32(wire >> 3)
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
if wireType == 4 {
|
||||||
|
return fmt.Errorf("proto: Permission: wiretype end group for non-group")
|
||||||
|
}
|
||||||
|
if fieldNum <= 0 {
|
||||||
|
return fmt.Errorf("proto: Permission: illegal tag %d (wire type %d)", fieldNum, wire)
|
||||||
|
}
|
||||||
|
switch fieldNum {
|
||||||
|
case 1:
|
||||||
|
if wireType != 0 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field PermType", wireType)
|
||||||
|
}
|
||||||
|
m.PermType = 0
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
m.PermType |= (Permission_Type(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case 2:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Key = append(m.Key[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.Key == nil {
|
||||||
|
m.Key = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 3:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field RangeEnd", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.RangeEnd = append(m.RangeEnd[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.RangeEnd == nil {
|
||||||
|
m.RangeEnd = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
default:
|
||||||
|
iNdEx = preIndex
|
||||||
|
skippy, err := skipAuth(dAtA[iNdEx:])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if skippy < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
if (iNdEx + skippy) > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx += skippy
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if iNdEx > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func (m *Role) Unmarshal(dAtA []byte) error {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
preIndex := iNdEx
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fieldNum := int32(wire >> 3)
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
if wireType == 4 {
|
||||||
|
return fmt.Errorf("proto: Role: wiretype end group for non-group")
|
||||||
|
}
|
||||||
|
if fieldNum <= 0 {
|
||||||
|
return fmt.Errorf("proto: Role: illegal tag %d (wire type %d)", fieldNum, wire)
|
||||||
|
}
|
||||||
|
switch fieldNum {
|
||||||
|
case 1:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Name = append(m.Name[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.Name == nil {
|
||||||
|
m.Name = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 2:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field KeyPermission", wireType)
|
||||||
|
}
|
||||||
|
var msglen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
msglen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if msglen < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + msglen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.KeyPermission = append(m.KeyPermission, &Permission{})
|
||||||
|
if err := m.KeyPermission[len(m.KeyPermission)-1].Unmarshal(dAtA[iNdEx:postIndex]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
default:
|
||||||
|
iNdEx = preIndex
|
||||||
|
skippy, err := skipAuth(dAtA[iNdEx:])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if skippy < 0 {
|
||||||
|
return ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
if (iNdEx + skippy) > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx += skippy
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if iNdEx > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func skipAuth(dAtA []byte) (n int, err error) {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
switch wireType {
|
||||||
|
case 0:
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx++
|
||||||
|
if dAtA[iNdEx-1] < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return iNdEx, nil
|
||||||
|
case 1:
|
||||||
|
iNdEx += 8
|
||||||
|
return iNdEx, nil
|
||||||
|
case 2:
|
||||||
|
var length int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
length |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
iNdEx += length
|
||||||
|
if length < 0 {
|
||||||
|
return 0, ErrInvalidLengthAuth
|
||||||
|
}
|
||||||
|
return iNdEx, nil
|
||||||
|
case 3:
|
||||||
|
for {
|
||||||
|
var innerWire uint64
|
||||||
|
var start int = iNdEx
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowAuth
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
innerWire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
innerWireType := int(innerWire & 0x7)
|
||||||
|
if innerWireType == 4 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
next, err := skipAuth(dAtA[start:])
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
iNdEx = start + next
|
||||||
|
}
|
||||||
|
return iNdEx, nil
|
||||||
|
case 4:
|
||||||
|
return iNdEx, nil
|
||||||
|
case 5:
|
||||||
|
iNdEx += 4
|
||||||
|
return iNdEx, nil
|
||||||
|
default:
|
||||||
|
return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panic("unreachable")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrInvalidLengthAuth = fmt.Errorf("proto: negative length found during unmarshaling")
|
||||||
|
ErrIntOverflowAuth = fmt.Errorf("proto: integer overflow")
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() { proto.RegisterFile("auth.proto", fileDescriptorAuth) }
|
||||||
|
|
||||||
|
var fileDescriptorAuth = []byte{
|
||||||
|
// 288 bytes of a gzipped FileDescriptorProto
|
||||||
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x6c, 0x90, 0xc1, 0x4a, 0xc3, 0x30,
|
||||||
|
0x1c, 0xc6, 0x9b, 0xb6, 0x1b, 0xed, 0x5f, 0x27, 0x25, 0x0c, 0x0c, 0x13, 0x42, 0xe9, 0xa9, 0x78,
|
||||||
|
0xa8, 0xb0, 0x5d, 0xbc, 0x2a, 0xf6, 0x20, 0x78, 0x90, 0x50, 0xf1, 0x28, 0x1d, 0x0d, 0x75, 0x6c,
|
||||||
|
0x6d, 0x4a, 0x32, 0x91, 0xbe, 0x89, 0x07, 0x1f, 0x68, 0xc7, 0x3d, 0x82, 0xab, 0x2f, 0x22, 0x4d,
|
||||||
|
0x64, 0x43, 0xdc, 0xed, 0xfb, 0xbe, 0xff, 0x97, 0xe4, 0x97, 0x3f, 0x40, 0xfe, 0xb6, 0x7e, 0x4d,
|
||||||
|
0x1a, 0x29, 0xd6, 0x02, 0x0f, 0x7b, 0xdd, 0xcc, 0x27, 0xe3, 0x52, 0x94, 0x42, 0x47, 0x57, 0xbd,
|
||||||
|
0x32, 0xd3, 0xe8, 0x01, 0xdc, 0x27, 0xc5, 0x25, 0xc6, 0xe0, 0xd6, 0x79, 0xc5, 0x09, 0x0a, 0x51,
|
||||||
|
0x7c, 0xca, 0xb4, 0xc6, 0x13, 0xf0, 0x9a, 0x5c, 0xa9, 0x77, 0x21, 0x0b, 0x62, 0xeb, 0x7c, 0xef,
|
||||||
|
0xf1, 0x18, 0x06, 0x52, 0xac, 0xb8, 0x22, 0x4e, 0xe8, 0xc4, 0x3e, 0x33, 0x26, 0xfa, 0x44, 0x00,
|
||||||
|
0x8f, 0x5c, 0x56, 0x0b, 0xa5, 0x16, 0xa2, 0xc6, 0x33, 0xf0, 0x1a, 0x2e, 0xab, 0xac, 0x6d, 0xcc,
|
||||||
|
0xc5, 0x67, 0xd3, 0xf3, 0xc4, 0xd0, 0x24, 0x87, 0x56, 0xd2, 0x8f, 0xd9, 0xbe, 0x88, 0x03, 0x70,
|
||||||
|
0x96, 0xbc, 0xfd, 0x7d, 0xb0, 0x97, 0xf8, 0x02, 0x7c, 0x99, 0xd7, 0x25, 0x7f, 0xe1, 0x75, 0x41,
|
||||||
|
0x1c, 0x03, 0xa2, 0x83, 0xb4, 0x2e, 0xa2, 0x4b, 0x70, 0xf5, 0x31, 0x0f, 0x5c, 0x96, 0xde, 0xdc,
|
||||||
|
0x05, 0x16, 0xf6, 0x61, 0xf0, 0xcc, 0xee, 0xb3, 0x34, 0x40, 0x78, 0x04, 0x7e, 0x1f, 0x1a, 0x6b,
|
||||||
|
0x47, 0x19, 0xb8, 0x4c, 0xac, 0xf8, 0xd1, 0xcf, 0x5e, 0xc3, 0x68, 0xc9, 0xdb, 0x03, 0x16, 0xb1,
|
||||||
|
0x43, 0x27, 0x3e, 0x99, 0xe2, 0xff, 0xc0, 0xec, 0x6f, 0xf1, 0x96, 0x6c, 0x76, 0xd4, 0xda, 0xee,
|
||||||
|
0xa8, 0xb5, 0xe9, 0x28, 0xda, 0x76, 0x14, 0x7d, 0x75, 0x14, 0x7d, 0x7c, 0x53, 0x6b, 0x3e, 0xd4,
|
||||||
|
0x3b, 0x9e, 0xfd, 0x04, 0x00, 0x00, 0xff, 0xff, 0xcc, 0x76, 0x8d, 0x4f, 0x8f, 0x01, 0x00, 0x00,
|
||||||
|
}
|
37
vendor/github.com/coreos/etcd/auth/authpb/auth.proto
generated
vendored
Normal file
37
vendor/github.com/coreos/etcd/auth/authpb/auth.proto
generated
vendored
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
package authpb;
|
||||||
|
|
||||||
|
import "gogoproto/gogo.proto";
|
||||||
|
|
||||||
|
option (gogoproto.marshaler_all) = true;
|
||||||
|
option (gogoproto.sizer_all) = true;
|
||||||
|
option (gogoproto.unmarshaler_all) = true;
|
||||||
|
option (gogoproto.goproto_getters_all) = false;
|
||||||
|
option (gogoproto.goproto_enum_prefix_all) = false;
|
||||||
|
|
||||||
|
// User is a single entry in the bucket authUsers
|
||||||
|
message User {
|
||||||
|
bytes name = 1;
|
||||||
|
bytes password = 2;
|
||||||
|
repeated string roles = 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Permission is a single entity
|
||||||
|
message Permission {
|
||||||
|
enum Type {
|
||||||
|
READ = 0;
|
||||||
|
WRITE = 1;
|
||||||
|
READWRITE = 2;
|
||||||
|
}
|
||||||
|
Type permType = 1;
|
||||||
|
|
||||||
|
bytes key = 2;
|
||||||
|
bytes range_end = 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Role is a single entry in the bucket authRoles
|
||||||
|
message Role {
|
||||||
|
bytes name = 1;
|
||||||
|
|
||||||
|
repeated Permission keyPermission = 2;
|
||||||
|
}
|
117
vendor/github.com/coreos/etcd/client/README.md
generated
vendored
117
vendor/github.com/coreos/etcd/client/README.md
generated
vendored
@ -1,117 +0,0 @@
|
|||||||
# etcd/client
|
|
||||||
|
|
||||||
etcd/client is the Go client library for etcd.
|
|
||||||
|
|
||||||
[![GoDoc](https://godoc.org/github.com/coreos/etcd/client?status.png)](https://godoc.org/github.com/coreos/etcd/client)
|
|
||||||
|
|
||||||
etcd uses `cmd/vendor` directory to store external dependencies, which are
|
|
||||||
to be compiled into etcd release binaries. `client` can be imported without
|
|
||||||
vendoring. For full compatibility, it is recommended to vendor builds using
|
|
||||||
etcd's vendored packages, using tools like godep, as in
|
|
||||||
[vendor directories](https://golang.org/cmd/go/#hdr-Vendor_Directories).
|
|
||||||
For more detail, please read [Go vendor design](https://golang.org/s/go15vendor).
|
|
||||||
|
|
||||||
## Install
|
|
||||||
|
|
||||||
```bash
|
|
||||||
go get github.com/coreos/etcd/client
|
|
||||||
```
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
```go
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
"github.com/coreos/etcd/client"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
cfg := client.Config{
|
|
||||||
Endpoints: []string{"http://127.0.0.1:2379"},
|
|
||||||
Transport: client.DefaultTransport,
|
|
||||||
// set timeout per request to fail fast when the target endpoint is unavailable
|
|
||||||
HeaderTimeoutPerRequest: time.Second,
|
|
||||||
}
|
|
||||||
c, err := client.New(cfg)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
kapi := client.NewKeysAPI(c)
|
|
||||||
// set "/foo" key with "bar" value
|
|
||||||
log.Print("Setting '/foo' key with 'bar' value")
|
|
||||||
resp, err := kapi.Set(context.Background(), "/foo", "bar", nil)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
} else {
|
|
||||||
// print common key info
|
|
||||||
log.Printf("Set is done. Metadata is %q\n", resp)
|
|
||||||
}
|
|
||||||
// get "/foo" key's value
|
|
||||||
log.Print("Getting '/foo' key value")
|
|
||||||
resp, err = kapi.Get(context.Background(), "/foo", nil)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
} else {
|
|
||||||
// print common key info
|
|
||||||
log.Printf("Get is done. Metadata is %q\n", resp)
|
|
||||||
// print value
|
|
||||||
log.Printf("%q key has %q value\n", resp.Node.Key, resp.Node.Value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Error Handling
|
|
||||||
|
|
||||||
etcd client might return three types of errors.
|
|
||||||
|
|
||||||
- context error
|
|
||||||
|
|
||||||
Each API call has its first parameter as `context`. A context can be canceled or have an attached deadline. If the context is canceled or reaches its deadline, the responding context error will be returned no matter what internal errors the API call has already encountered.
|
|
||||||
|
|
||||||
- cluster error
|
|
||||||
|
|
||||||
Each API call tries to send request to the cluster endpoints one by one until it successfully gets a response. If a requests to an endpoint fails, due to exceeding per request timeout or connection issues, the error will be added into a list of errors. If all possible endpoints fail, a cluster error that includes all encountered errors will be returned.
|
|
||||||
|
|
||||||
- response error
|
|
||||||
|
|
||||||
If the response gets from the cluster is invalid, a plain string error will be returned. For example, it might be a invalid JSON error.
|
|
||||||
|
|
||||||
Here is the example code to handle client errors:
|
|
||||||
|
|
||||||
```go
|
|
||||||
cfg := client.Config{Endpoints: []string{"http://etcd1:2379","http://etcd2:2379","http://etcd3:2379"}}
|
|
||||||
c, err := client.New(cfg)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
kapi := client.NewKeysAPI(c)
|
|
||||||
resp, err := kapi.Set(ctx, "test", "bar", nil)
|
|
||||||
if err != nil {
|
|
||||||
if err == context.Canceled {
|
|
||||||
// ctx is canceled by another routine
|
|
||||||
} else if err == context.DeadlineExceeded {
|
|
||||||
// ctx is attached with a deadline and it exceeded
|
|
||||||
} else if cerr, ok := err.(*client.ClusterError); ok {
|
|
||||||
// process (cerr.Errors)
|
|
||||||
} else {
|
|
||||||
// bad cluster endpoints, which are not etcd servers
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## Caveat
|
|
||||||
|
|
||||||
1. etcd/client prefers to use the same endpoint as long as the endpoint continues to work well. This saves socket resources, and improves efficiency for both client and server side. This preference doesn't remove consistency from the data consumed by the client because data replicated to each etcd member has already passed through the consensus process.
|
|
||||||
|
|
||||||
2. etcd/client does round-robin rotation on other available endpoints if the preferred endpoint isn't functioning properly. For example, if the member that etcd/client connects to is hard killed, etcd/client will fail on the first attempt with the killed member, and succeed on the second attempt with another member. If it fails to talk to all available endpoints, it will return all errors happened.
|
|
||||||
|
|
||||||
3. Default etcd/client cannot handle the case that the remote server is SIGSTOPed now. TCP keepalive mechanism doesn't help in this scenario because operating system may still send TCP keep-alive packets. Over time we'd like to improve this functionality, but solving this issue isn't high priority because a real-life case in which a server is stopped, but the connection is kept alive, hasn't been brought to our attention.
|
|
||||||
|
|
||||||
4. etcd/client cannot detect whether a member is healthy with watches and non-quorum read requests. If the member is isolated from the cluster, etcd/client may retrieve outdated data. Instead, users can either issue quorum read requests or monitor the /health endpoint for member health information.
|
|
237
vendor/github.com/coreos/etcd/client/auth_role.go
generated
vendored
237
vendor/github.com/coreos/etcd/client/auth_role.go
generated
vendored
@ -1,237 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"encoding/json"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Role struct {
|
|
||||||
Role string `json:"role"`
|
|
||||||
Permissions Permissions `json:"permissions"`
|
|
||||||
Grant *Permissions `json:"grant,omitempty"`
|
|
||||||
Revoke *Permissions `json:"revoke,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Permissions struct {
|
|
||||||
KV rwPermission `json:"kv"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type rwPermission struct {
|
|
||||||
Read []string `json:"read"`
|
|
||||||
Write []string `json:"write"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type PermissionType int
|
|
||||||
|
|
||||||
const (
|
|
||||||
ReadPermission PermissionType = iota
|
|
||||||
WritePermission
|
|
||||||
ReadWritePermission
|
|
||||||
)
|
|
||||||
|
|
||||||
// NewAuthRoleAPI constructs a new AuthRoleAPI that uses HTTP to
|
|
||||||
// interact with etcd's role creation and modification features.
|
|
||||||
func NewAuthRoleAPI(c Client) AuthRoleAPI {
|
|
||||||
return &httpAuthRoleAPI{
|
|
||||||
client: c,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type AuthRoleAPI interface {
|
|
||||||
// AddRole adds a role.
|
|
||||||
AddRole(ctx context.Context, role string) error
|
|
||||||
|
|
||||||
// RemoveRole removes a role.
|
|
||||||
RemoveRole(ctx context.Context, role string) error
|
|
||||||
|
|
||||||
// GetRole retrieves role details.
|
|
||||||
GetRole(ctx context.Context, role string) (*Role, error)
|
|
||||||
|
|
||||||
// GrantRoleKV grants a role some permission prefixes for the KV store.
|
|
||||||
GrantRoleKV(ctx context.Context, role string, prefixes []string, permType PermissionType) (*Role, error)
|
|
||||||
|
|
||||||
// RevokeRoleKV revokes some permission prefixes for a role on the KV store.
|
|
||||||
RevokeRoleKV(ctx context.Context, role string, prefixes []string, permType PermissionType) (*Role, error)
|
|
||||||
|
|
||||||
// ListRoles lists roles.
|
|
||||||
ListRoles(ctx context.Context) ([]string, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpAuthRoleAPI struct {
|
|
||||||
client httpClient
|
|
||||||
}
|
|
||||||
|
|
||||||
type authRoleAPIAction struct {
|
|
||||||
verb string
|
|
||||||
name string
|
|
||||||
role *Role
|
|
||||||
}
|
|
||||||
|
|
||||||
type authRoleAPIList struct{}
|
|
||||||
|
|
||||||
func (list *authRoleAPIList) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2AuthURL(ep, "roles", "")
|
|
||||||
req, _ := http.NewRequest("GET", u.String(), nil)
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *authRoleAPIAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2AuthURL(ep, "roles", l.name)
|
|
||||||
if l.role == nil {
|
|
||||||
req, _ := http.NewRequest(l.verb, u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
b, err := json.Marshal(l.role)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
body := bytes.NewReader(b)
|
|
||||||
req, _ := http.NewRequest(l.verb, u.String(), body)
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) ListRoles(ctx context.Context) ([]string, error) {
|
|
||||||
resp, body, err := r.client.Do(ctx, &authRoleAPIList{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err = assertStatusCode(resp.StatusCode, http.StatusOK); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var roleList struct {
|
|
||||||
Roles []Role `json:"roles"`
|
|
||||||
}
|
|
||||||
if err = json.Unmarshal(body, &roleList); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
ret := make([]string, 0, len(roleList.Roles))
|
|
||||||
for _, r := range roleList.Roles {
|
|
||||||
ret = append(ret, r.Role)
|
|
||||||
}
|
|
||||||
return ret, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) AddRole(ctx context.Context, rolename string) error {
|
|
||||||
role := &Role{
|
|
||||||
Role: rolename,
|
|
||||||
}
|
|
||||||
return r.addRemoveRole(ctx, &authRoleAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
name: rolename,
|
|
||||||
role: role,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) RemoveRole(ctx context.Context, rolename string) error {
|
|
||||||
return r.addRemoveRole(ctx, &authRoleAPIAction{
|
|
||||||
verb: "DELETE",
|
|
||||||
name: rolename,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) addRemoveRole(ctx context.Context, req *authRoleAPIAction) error {
|
|
||||||
resp, body, err := r.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := assertStatusCode(resp.StatusCode, http.StatusOK, http.StatusCreated); err != nil {
|
|
||||||
var sec authError
|
|
||||||
err := json.Unmarshal(body, &sec)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return sec
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) GetRole(ctx context.Context, rolename string) (*Role, error) {
|
|
||||||
return r.modRole(ctx, &authRoleAPIAction{
|
|
||||||
verb: "GET",
|
|
||||||
name: rolename,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func buildRWPermission(prefixes []string, permType PermissionType) rwPermission {
|
|
||||||
var out rwPermission
|
|
||||||
switch permType {
|
|
||||||
case ReadPermission:
|
|
||||||
out.Read = prefixes
|
|
||||||
case WritePermission:
|
|
||||||
out.Write = prefixes
|
|
||||||
case ReadWritePermission:
|
|
||||||
out.Read = prefixes
|
|
||||||
out.Write = prefixes
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) GrantRoleKV(ctx context.Context, rolename string, prefixes []string, permType PermissionType) (*Role, error) {
|
|
||||||
rwp := buildRWPermission(prefixes, permType)
|
|
||||||
role := &Role{
|
|
||||||
Role: rolename,
|
|
||||||
Grant: &Permissions{
|
|
||||||
KV: rwp,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
return r.modRole(ctx, &authRoleAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
name: rolename,
|
|
||||||
role: role,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) RevokeRoleKV(ctx context.Context, rolename string, prefixes []string, permType PermissionType) (*Role, error) {
|
|
||||||
rwp := buildRWPermission(prefixes, permType)
|
|
||||||
role := &Role{
|
|
||||||
Role: rolename,
|
|
||||||
Revoke: &Permissions{
|
|
||||||
KV: rwp,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
return r.modRole(ctx, &authRoleAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
name: rolename,
|
|
||||||
role: role,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *httpAuthRoleAPI) modRole(ctx context.Context, req *authRoleAPIAction) (*Role, error) {
|
|
||||||
resp, body, err := r.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err = assertStatusCode(resp.StatusCode, http.StatusOK); err != nil {
|
|
||||||
var sec authError
|
|
||||||
err = json.Unmarshal(body, &sec)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return nil, sec
|
|
||||||
}
|
|
||||||
var role Role
|
|
||||||
if err = json.Unmarshal(body, &role); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return &role, nil
|
|
||||||
}
|
|
320
vendor/github.com/coreos/etcd/client/auth_user.go
generated
vendored
320
vendor/github.com/coreos/etcd/client/auth_user.go
generated
vendored
@ -1,320 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"encoding/json"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"path"
|
|
||||||
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
defaultV2AuthPrefix = "/v2/auth"
|
|
||||||
)
|
|
||||||
|
|
||||||
type User struct {
|
|
||||||
User string `json:"user"`
|
|
||||||
Password string `json:"password,omitempty"`
|
|
||||||
Roles []string `json:"roles"`
|
|
||||||
Grant []string `json:"grant,omitempty"`
|
|
||||||
Revoke []string `json:"revoke,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// userListEntry is the user representation given by the server for ListUsers
|
|
||||||
type userListEntry struct {
|
|
||||||
User string `json:"user"`
|
|
||||||
Roles []Role `json:"roles"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type UserRoles struct {
|
|
||||||
User string `json:"user"`
|
|
||||||
Roles []Role `json:"roles"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func v2AuthURL(ep url.URL, action string, name string) *url.URL {
|
|
||||||
if name != "" {
|
|
||||||
ep.Path = path.Join(ep.Path, defaultV2AuthPrefix, action, name)
|
|
||||||
return &ep
|
|
||||||
}
|
|
||||||
ep.Path = path.Join(ep.Path, defaultV2AuthPrefix, action)
|
|
||||||
return &ep
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewAuthAPI constructs a new AuthAPI that uses HTTP to
|
|
||||||
// interact with etcd's general auth features.
|
|
||||||
func NewAuthAPI(c Client) AuthAPI {
|
|
||||||
return &httpAuthAPI{
|
|
||||||
client: c,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type AuthAPI interface {
|
|
||||||
// Enable auth.
|
|
||||||
Enable(ctx context.Context) error
|
|
||||||
|
|
||||||
// Disable auth.
|
|
||||||
Disable(ctx context.Context) error
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpAuthAPI struct {
|
|
||||||
client httpClient
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *httpAuthAPI) Enable(ctx context.Context) error {
|
|
||||||
return s.enableDisable(ctx, &authAPIAction{"PUT"})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *httpAuthAPI) Disable(ctx context.Context) error {
|
|
||||||
return s.enableDisable(ctx, &authAPIAction{"DELETE"})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *httpAuthAPI) enableDisable(ctx context.Context, req httpAction) error {
|
|
||||||
resp, body, err := s.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err = assertStatusCode(resp.StatusCode, http.StatusOK, http.StatusCreated); err != nil {
|
|
||||||
var sec authError
|
|
||||||
err = json.Unmarshal(body, &sec)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return sec
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type authAPIAction struct {
|
|
||||||
verb string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *authAPIAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2AuthURL(ep, "enable", "")
|
|
||||||
req, _ := http.NewRequest(l.verb, u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type authError struct {
|
|
||||||
Message string `json:"message"`
|
|
||||||
Code int `json:"-"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e authError) Error() string {
|
|
||||||
return e.Message
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewAuthUserAPI constructs a new AuthUserAPI that uses HTTP to
|
|
||||||
// interact with etcd's user creation and modification features.
|
|
||||||
func NewAuthUserAPI(c Client) AuthUserAPI {
|
|
||||||
return &httpAuthUserAPI{
|
|
||||||
client: c,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type AuthUserAPI interface {
|
|
||||||
// AddUser adds a user.
|
|
||||||
AddUser(ctx context.Context, username string, password string) error
|
|
||||||
|
|
||||||
// RemoveUser removes a user.
|
|
||||||
RemoveUser(ctx context.Context, username string) error
|
|
||||||
|
|
||||||
// GetUser retrieves user details.
|
|
||||||
GetUser(ctx context.Context, username string) (*User, error)
|
|
||||||
|
|
||||||
// GrantUser grants a user some permission roles.
|
|
||||||
GrantUser(ctx context.Context, username string, roles []string) (*User, error)
|
|
||||||
|
|
||||||
// RevokeUser revokes some permission roles from a user.
|
|
||||||
RevokeUser(ctx context.Context, username string, roles []string) (*User, error)
|
|
||||||
|
|
||||||
// ChangePassword changes the user's password.
|
|
||||||
ChangePassword(ctx context.Context, username string, password string) (*User, error)
|
|
||||||
|
|
||||||
// ListUsers lists the users.
|
|
||||||
ListUsers(ctx context.Context) ([]string, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpAuthUserAPI struct {
|
|
||||||
client httpClient
|
|
||||||
}
|
|
||||||
|
|
||||||
type authUserAPIAction struct {
|
|
||||||
verb string
|
|
||||||
username string
|
|
||||||
user *User
|
|
||||||
}
|
|
||||||
|
|
||||||
type authUserAPIList struct{}
|
|
||||||
|
|
||||||
func (list *authUserAPIList) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2AuthURL(ep, "users", "")
|
|
||||||
req, _ := http.NewRequest("GET", u.String(), nil)
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *authUserAPIAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2AuthURL(ep, "users", l.username)
|
|
||||||
if l.user == nil {
|
|
||||||
req, _ := http.NewRequest(l.verb, u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
b, err := json.Marshal(l.user)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
body := bytes.NewReader(b)
|
|
||||||
req, _ := http.NewRequest(l.verb, u.String(), body)
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) ListUsers(ctx context.Context) ([]string, error) {
|
|
||||||
resp, body, err := u.client.Do(ctx, &authUserAPIList{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err = assertStatusCode(resp.StatusCode, http.StatusOK); err != nil {
|
|
||||||
var sec authError
|
|
||||||
err = json.Unmarshal(body, &sec)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return nil, sec
|
|
||||||
}
|
|
||||||
|
|
||||||
var userList struct {
|
|
||||||
Users []userListEntry `json:"users"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if err = json.Unmarshal(body, &userList); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
ret := make([]string, 0, len(userList.Users))
|
|
||||||
for _, u := range userList.Users {
|
|
||||||
ret = append(ret, u.User)
|
|
||||||
}
|
|
||||||
return ret, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) AddUser(ctx context.Context, username string, password string) error {
|
|
||||||
user := &User{
|
|
||||||
User: username,
|
|
||||||
Password: password,
|
|
||||||
}
|
|
||||||
return u.addRemoveUser(ctx, &authUserAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
username: username,
|
|
||||||
user: user,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) RemoveUser(ctx context.Context, username string) error {
|
|
||||||
return u.addRemoveUser(ctx, &authUserAPIAction{
|
|
||||||
verb: "DELETE",
|
|
||||||
username: username,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) addRemoveUser(ctx context.Context, req *authUserAPIAction) error {
|
|
||||||
resp, body, err := u.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err = assertStatusCode(resp.StatusCode, http.StatusOK, http.StatusCreated); err != nil {
|
|
||||||
var sec authError
|
|
||||||
err = json.Unmarshal(body, &sec)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return sec
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) GetUser(ctx context.Context, username string) (*User, error) {
|
|
||||||
return u.modUser(ctx, &authUserAPIAction{
|
|
||||||
verb: "GET",
|
|
||||||
username: username,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) GrantUser(ctx context.Context, username string, roles []string) (*User, error) {
|
|
||||||
user := &User{
|
|
||||||
User: username,
|
|
||||||
Grant: roles,
|
|
||||||
}
|
|
||||||
return u.modUser(ctx, &authUserAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
username: username,
|
|
||||||
user: user,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) RevokeUser(ctx context.Context, username string, roles []string) (*User, error) {
|
|
||||||
user := &User{
|
|
||||||
User: username,
|
|
||||||
Revoke: roles,
|
|
||||||
}
|
|
||||||
return u.modUser(ctx, &authUserAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
username: username,
|
|
||||||
user: user,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) ChangePassword(ctx context.Context, username string, password string) (*User, error) {
|
|
||||||
user := &User{
|
|
||||||
User: username,
|
|
||||||
Password: password,
|
|
||||||
}
|
|
||||||
return u.modUser(ctx, &authUserAPIAction{
|
|
||||||
verb: "PUT",
|
|
||||||
username: username,
|
|
||||||
user: user,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *httpAuthUserAPI) modUser(ctx context.Context, req *authUserAPIAction) (*User, error) {
|
|
||||||
resp, body, err := u.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err = assertStatusCode(resp.StatusCode, http.StatusOK); err != nil {
|
|
||||||
var sec authError
|
|
||||||
err = json.Unmarshal(body, &sec)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return nil, sec
|
|
||||||
}
|
|
||||||
var user User
|
|
||||||
if err = json.Unmarshal(body, &user); err != nil {
|
|
||||||
var userR UserRoles
|
|
||||||
if urerr := json.Unmarshal(body, &userR); urerr != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
user.User = userR.User
|
|
||||||
for _, r := range userR.Roles {
|
|
||||||
user.Roles = append(user.Roles, r.Role)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return &user, nil
|
|
||||||
}
|
|
18
vendor/github.com/coreos/etcd/client/cancelreq.go
generated
vendored
18
vendor/github.com/coreos/etcd/client/cancelreq.go
generated
vendored
@ -1,18 +0,0 @@
|
|||||||
// Copyright 2015 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.
|
|
||||||
|
|
||||||
// borrowed from golang/net/context/ctxhttp/cancelreq.go
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import "net/http"
|
|
||||||
|
|
||||||
func requestCanceler(tr CancelableTransport, req *http.Request) func() {
|
|
||||||
ch := make(chan struct{})
|
|
||||||
req.Cancel = ch
|
|
||||||
|
|
||||||
return func() {
|
|
||||||
close(ch)
|
|
||||||
}
|
|
||||||
}
|
|
704
vendor/github.com/coreos/etcd/client/client.go
generated
vendored
704
vendor/github.com/coreos/etcd/client/client.go
generated
vendored
@ -1,704 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
|
||||||
"math/rand"
|
|
||||||
"net"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"sort"
|
|
||||||
"strconv"
|
|
||||||
"sync"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/coreos/etcd/version"
|
|
||||||
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrNoEndpoints = errors.New("client: no endpoints available")
|
|
||||||
ErrTooManyRedirects = errors.New("client: too many redirects")
|
|
||||||
ErrClusterUnavailable = errors.New("client: etcd cluster is unavailable or misconfigured")
|
|
||||||
ErrNoLeaderEndpoint = errors.New("client: no leader endpoint available")
|
|
||||||
errTooManyRedirectChecks = errors.New("client: too many redirect checks")
|
|
||||||
|
|
||||||
// oneShotCtxValue is set on a context using WithValue(&oneShotValue) so
|
|
||||||
// that Do() will not retry a request
|
|
||||||
oneShotCtxValue interface{}
|
|
||||||
)
|
|
||||||
|
|
||||||
var DefaultRequestTimeout = 5 * time.Second
|
|
||||||
|
|
||||||
var DefaultTransport CancelableTransport = &http.Transport{
|
|
||||||
Proxy: http.ProxyFromEnvironment,
|
|
||||||
Dial: (&net.Dialer{
|
|
||||||
Timeout: 30 * time.Second,
|
|
||||||
KeepAlive: 30 * time.Second,
|
|
||||||
}).Dial,
|
|
||||||
TLSHandshakeTimeout: 10 * time.Second,
|
|
||||||
}
|
|
||||||
|
|
||||||
type EndpointSelectionMode int
|
|
||||||
|
|
||||||
const (
|
|
||||||
// EndpointSelectionRandom is the default value of the 'SelectionMode'.
|
|
||||||
// As the name implies, the client object will pick a node from the members
|
|
||||||
// of the cluster in a random fashion. If the cluster has three members, A, B,
|
|
||||||
// and C, the client picks any node from its three members as its request
|
|
||||||
// destination.
|
|
||||||
EndpointSelectionRandom EndpointSelectionMode = iota
|
|
||||||
|
|
||||||
// If 'SelectionMode' is set to 'EndpointSelectionPrioritizeLeader',
|
|
||||||
// requests are sent directly to the cluster leader. This reduces
|
|
||||||
// forwarding roundtrips compared to making requests to etcd followers
|
|
||||||
// who then forward them to the cluster leader. In the event of a leader
|
|
||||||
// failure, however, clients configured this way cannot prioritize among
|
|
||||||
// the remaining etcd followers. Therefore, when a client sets 'SelectionMode'
|
|
||||||
// to 'EndpointSelectionPrioritizeLeader', it must use 'client.AutoSync()' to
|
|
||||||
// maintain its knowledge of current cluster state.
|
|
||||||
//
|
|
||||||
// This mode should be used with Client.AutoSync().
|
|
||||||
EndpointSelectionPrioritizeLeader
|
|
||||||
)
|
|
||||||
|
|
||||||
type Config struct {
|
|
||||||
// Endpoints defines a set of URLs (schemes, hosts and ports only)
|
|
||||||
// that can be used to communicate with a logical etcd cluster. For
|
|
||||||
// example, a three-node cluster could be provided like so:
|
|
||||||
//
|
|
||||||
// Endpoints: []string{
|
|
||||||
// "http://node1.example.com:2379",
|
|
||||||
// "http://node2.example.com:2379",
|
|
||||||
// "http://node3.example.com:2379",
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// If multiple endpoints are provided, the Client will attempt to
|
|
||||||
// use them all in the event that one or more of them are unusable.
|
|
||||||
//
|
|
||||||
// If Client.Sync is ever called, the Client may cache an alternate
|
|
||||||
// set of endpoints to continue operation.
|
|
||||||
Endpoints []string
|
|
||||||
|
|
||||||
// Transport is used by the Client to drive HTTP requests. If not
|
|
||||||
// provided, DefaultTransport will be used.
|
|
||||||
Transport CancelableTransport
|
|
||||||
|
|
||||||
// CheckRedirect specifies the policy for handling HTTP redirects.
|
|
||||||
// If CheckRedirect is not nil, the Client calls it before
|
|
||||||
// following an HTTP redirect. The sole argument is the number of
|
|
||||||
// requests that have already been made. If CheckRedirect returns
|
|
||||||
// an error, Client.Do will not make any further requests and return
|
|
||||||
// the error back it to the caller.
|
|
||||||
//
|
|
||||||
// If CheckRedirect is nil, the Client uses its default policy,
|
|
||||||
// which is to stop after 10 consecutive requests.
|
|
||||||
CheckRedirect CheckRedirectFunc
|
|
||||||
|
|
||||||
// Username specifies the user credential to add as an authorization header
|
|
||||||
Username string
|
|
||||||
|
|
||||||
// Password is the password for the specified user to add as an authorization header
|
|
||||||
// to the request.
|
|
||||||
Password string
|
|
||||||
|
|
||||||
// HeaderTimeoutPerRequest specifies the time limit to wait for response
|
|
||||||
// header in a single request made by the Client. The timeout includes
|
|
||||||
// connection time, any redirects, and header wait time.
|
|
||||||
//
|
|
||||||
// For non-watch GET request, server returns the response body immediately.
|
|
||||||
// For PUT/POST/DELETE request, server will attempt to commit request
|
|
||||||
// before responding, which is expected to take `100ms + 2 * RTT`.
|
|
||||||
// For watch request, server returns the header immediately to notify Client
|
|
||||||
// watch start. But if server is behind some kind of proxy, the response
|
|
||||||
// header may be cached at proxy, and Client cannot rely on this behavior.
|
|
||||||
//
|
|
||||||
// Especially, wait request will ignore this timeout.
|
|
||||||
//
|
|
||||||
// One API call may send multiple requests to different etcd servers until it
|
|
||||||
// succeeds. Use context of the API to specify the overall timeout.
|
|
||||||
//
|
|
||||||
// A HeaderTimeoutPerRequest of zero means no timeout.
|
|
||||||
HeaderTimeoutPerRequest time.Duration
|
|
||||||
|
|
||||||
// SelectionMode is an EndpointSelectionMode enum that specifies the
|
|
||||||
// policy for choosing the etcd cluster node to which requests are sent.
|
|
||||||
SelectionMode EndpointSelectionMode
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cfg *Config) transport() CancelableTransport {
|
|
||||||
if cfg.Transport == nil {
|
|
||||||
return DefaultTransport
|
|
||||||
}
|
|
||||||
return cfg.Transport
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cfg *Config) checkRedirect() CheckRedirectFunc {
|
|
||||||
if cfg.CheckRedirect == nil {
|
|
||||||
return DefaultCheckRedirect
|
|
||||||
}
|
|
||||||
return cfg.CheckRedirect
|
|
||||||
}
|
|
||||||
|
|
||||||
// CancelableTransport mimics net/http.Transport, but requires that
|
|
||||||
// the object also support request cancellation.
|
|
||||||
type CancelableTransport interface {
|
|
||||||
http.RoundTripper
|
|
||||||
CancelRequest(req *http.Request)
|
|
||||||
}
|
|
||||||
|
|
||||||
type CheckRedirectFunc func(via int) error
|
|
||||||
|
|
||||||
// DefaultCheckRedirect follows up to 10 redirects, but no more.
|
|
||||||
var DefaultCheckRedirect CheckRedirectFunc = func(via int) error {
|
|
||||||
if via > 10 {
|
|
||||||
return ErrTooManyRedirects
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type Client interface {
|
|
||||||
// Sync updates the internal cache of the etcd cluster's membership.
|
|
||||||
Sync(context.Context) error
|
|
||||||
|
|
||||||
// AutoSync periodically calls Sync() every given interval.
|
|
||||||
// The recommended sync interval is 10 seconds to 1 minute, which does
|
|
||||||
// not bring too much overhead to server and makes client catch up the
|
|
||||||
// cluster change in time.
|
|
||||||
//
|
|
||||||
// The example to use it:
|
|
||||||
//
|
|
||||||
// for {
|
|
||||||
// err := client.AutoSync(ctx, 10*time.Second)
|
|
||||||
// if err == context.DeadlineExceeded || err == context.Canceled {
|
|
||||||
// break
|
|
||||||
// }
|
|
||||||
// log.Print(err)
|
|
||||||
// }
|
|
||||||
AutoSync(context.Context, time.Duration) error
|
|
||||||
|
|
||||||
// Endpoints returns a copy of the current set of API endpoints used
|
|
||||||
// by Client to resolve HTTP requests. If Sync has ever been called,
|
|
||||||
// this may differ from the initial Endpoints provided in the Config.
|
|
||||||
Endpoints() []string
|
|
||||||
|
|
||||||
// SetEndpoints sets the set of API endpoints used by Client to resolve
|
|
||||||
// HTTP requests. If the given endpoints are not valid, an error will be
|
|
||||||
// returned
|
|
||||||
SetEndpoints(eps []string) error
|
|
||||||
|
|
||||||
// GetVersion retrieves the current etcd server and cluster version
|
|
||||||
GetVersion(ctx context.Context) (*version.Versions, error)
|
|
||||||
|
|
||||||
httpClient
|
|
||||||
}
|
|
||||||
|
|
||||||
func New(cfg Config) (Client, error) {
|
|
||||||
c := &httpClusterClient{
|
|
||||||
clientFactory: newHTTPClientFactory(cfg.transport(), cfg.checkRedirect(), cfg.HeaderTimeoutPerRequest),
|
|
||||||
rand: rand.New(rand.NewSource(int64(time.Now().Nanosecond()))),
|
|
||||||
selectionMode: cfg.SelectionMode,
|
|
||||||
}
|
|
||||||
if cfg.Username != "" {
|
|
||||||
c.credentials = &credentials{
|
|
||||||
username: cfg.Username,
|
|
||||||
password: cfg.Password,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if err := c.SetEndpoints(cfg.Endpoints); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return c, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpClient interface {
|
|
||||||
Do(context.Context, httpAction) (*http.Response, []byte, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
func newHTTPClientFactory(tr CancelableTransport, cr CheckRedirectFunc, headerTimeout time.Duration) httpClientFactory {
|
|
||||||
return func(ep url.URL) httpClient {
|
|
||||||
return &redirectFollowingHTTPClient{
|
|
||||||
checkRedirect: cr,
|
|
||||||
client: &simpleHTTPClient{
|
|
||||||
transport: tr,
|
|
||||||
endpoint: ep,
|
|
||||||
headerTimeout: headerTimeout,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type credentials struct {
|
|
||||||
username string
|
|
||||||
password string
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpClientFactory func(url.URL) httpClient
|
|
||||||
|
|
||||||
type httpAction interface {
|
|
||||||
HTTPRequest(url.URL) *http.Request
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpClusterClient struct {
|
|
||||||
clientFactory httpClientFactory
|
|
||||||
endpoints []url.URL
|
|
||||||
pinned int
|
|
||||||
credentials *credentials
|
|
||||||
sync.RWMutex
|
|
||||||
rand *rand.Rand
|
|
||||||
selectionMode EndpointSelectionMode
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) getLeaderEndpoint(ctx context.Context, eps []url.URL) (string, error) {
|
|
||||||
ceps := make([]url.URL, len(eps))
|
|
||||||
copy(ceps, eps)
|
|
||||||
|
|
||||||
// To perform a lookup on the new endpoint list without using the current
|
|
||||||
// client, we'll copy it
|
|
||||||
clientCopy := &httpClusterClient{
|
|
||||||
clientFactory: c.clientFactory,
|
|
||||||
credentials: c.credentials,
|
|
||||||
rand: c.rand,
|
|
||||||
|
|
||||||
pinned: 0,
|
|
||||||
endpoints: ceps,
|
|
||||||
}
|
|
||||||
|
|
||||||
mAPI := NewMembersAPI(clientCopy)
|
|
||||||
leader, err := mAPI.Leader(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
if len(leader.ClientURLs) == 0 {
|
|
||||||
return "", ErrNoLeaderEndpoint
|
|
||||||
}
|
|
||||||
|
|
||||||
return leader.ClientURLs[0], nil // TODO: how to handle multiple client URLs?
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) parseEndpoints(eps []string) ([]url.URL, error) {
|
|
||||||
if len(eps) == 0 {
|
|
||||||
return []url.URL{}, ErrNoEndpoints
|
|
||||||
}
|
|
||||||
|
|
||||||
neps := make([]url.URL, len(eps))
|
|
||||||
for i, ep := range eps {
|
|
||||||
u, err := url.Parse(ep)
|
|
||||||
if err != nil {
|
|
||||||
return []url.URL{}, err
|
|
||||||
}
|
|
||||||
neps[i] = *u
|
|
||||||
}
|
|
||||||
return neps, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) SetEndpoints(eps []string) error {
|
|
||||||
neps, err := c.parseEndpoints(eps)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
c.Lock()
|
|
||||||
defer c.Unlock()
|
|
||||||
|
|
||||||
c.endpoints = shuffleEndpoints(c.rand, neps)
|
|
||||||
// We're not doing anything for PrioritizeLeader here. This is
|
|
||||||
// due to not having a context meaning we can't call getLeaderEndpoint
|
|
||||||
// However, if you're using PrioritizeLeader, you've already been told
|
|
||||||
// to regularly call sync, where we do have a ctx, and can figure the
|
|
||||||
// leader. PrioritizeLeader is also quite a loose guarantee, so deal
|
|
||||||
// with it
|
|
||||||
c.pinned = 0
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) Do(ctx context.Context, act httpAction) (*http.Response, []byte, error) {
|
|
||||||
action := act
|
|
||||||
c.RLock()
|
|
||||||
leps := len(c.endpoints)
|
|
||||||
eps := make([]url.URL, leps)
|
|
||||||
n := copy(eps, c.endpoints)
|
|
||||||
pinned := c.pinned
|
|
||||||
|
|
||||||
if c.credentials != nil {
|
|
||||||
action = &authedAction{
|
|
||||||
act: act,
|
|
||||||
credentials: *c.credentials,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
c.RUnlock()
|
|
||||||
|
|
||||||
if leps == 0 {
|
|
||||||
return nil, nil, ErrNoEndpoints
|
|
||||||
}
|
|
||||||
|
|
||||||
if leps != n {
|
|
||||||
return nil, nil, errors.New("unable to pick endpoint: copy failed")
|
|
||||||
}
|
|
||||||
|
|
||||||
var resp *http.Response
|
|
||||||
var body []byte
|
|
||||||
var err error
|
|
||||||
cerr := &ClusterError{}
|
|
||||||
isOneShot := ctx.Value(&oneShotCtxValue) != nil
|
|
||||||
|
|
||||||
for i := pinned; i < leps+pinned; i++ {
|
|
||||||
k := i % leps
|
|
||||||
hc := c.clientFactory(eps[k])
|
|
||||||
resp, body, err = hc.Do(ctx, action)
|
|
||||||
if err != nil {
|
|
||||||
cerr.Errors = append(cerr.Errors, err)
|
|
||||||
if err == ctx.Err() {
|
|
||||||
return nil, nil, ctx.Err()
|
|
||||||
}
|
|
||||||
if err == context.Canceled || err == context.DeadlineExceeded {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
} else if resp.StatusCode/100 == 5 {
|
|
||||||
switch resp.StatusCode {
|
|
||||||
case http.StatusInternalServerError, http.StatusServiceUnavailable:
|
|
||||||
// TODO: make sure this is a no leader response
|
|
||||||
cerr.Errors = append(cerr.Errors, fmt.Errorf("client: etcd member %s has no leader", eps[k].String()))
|
|
||||||
default:
|
|
||||||
cerr.Errors = append(cerr.Errors, fmt.Errorf("client: etcd member %s returns server error [%s]", eps[k].String(), http.StatusText(resp.StatusCode)))
|
|
||||||
}
|
|
||||||
err = cerr.Errors[0]
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
if !isOneShot {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
c.Lock()
|
|
||||||
c.pinned = (k + 1) % leps
|
|
||||||
c.Unlock()
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
if k != pinned {
|
|
||||||
c.Lock()
|
|
||||||
c.pinned = k
|
|
||||||
c.Unlock()
|
|
||||||
}
|
|
||||||
return resp, body, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, nil, cerr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) Endpoints() []string {
|
|
||||||
c.RLock()
|
|
||||||
defer c.RUnlock()
|
|
||||||
|
|
||||||
eps := make([]string, len(c.endpoints))
|
|
||||||
for i, ep := range c.endpoints {
|
|
||||||
eps[i] = ep.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
return eps
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) Sync(ctx context.Context) error {
|
|
||||||
mAPI := NewMembersAPI(c)
|
|
||||||
ms, err := mAPI.List(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
var eps []string
|
|
||||||
for _, m := range ms {
|
|
||||||
eps = append(eps, m.ClientURLs...)
|
|
||||||
}
|
|
||||||
|
|
||||||
neps, err := c.parseEndpoints(eps)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
npin := 0
|
|
||||||
|
|
||||||
switch c.selectionMode {
|
|
||||||
case EndpointSelectionRandom:
|
|
||||||
c.RLock()
|
|
||||||
eq := endpointsEqual(c.endpoints, neps)
|
|
||||||
c.RUnlock()
|
|
||||||
|
|
||||||
if eq {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
// When items in the endpoint list changes, we choose a new pin
|
|
||||||
neps = shuffleEndpoints(c.rand, neps)
|
|
||||||
case EndpointSelectionPrioritizeLeader:
|
|
||||||
nle, err := c.getLeaderEndpoint(ctx, neps)
|
|
||||||
if err != nil {
|
|
||||||
return ErrNoLeaderEndpoint
|
|
||||||
}
|
|
||||||
|
|
||||||
for i, n := range neps {
|
|
||||||
if n.String() == nle {
|
|
||||||
npin = i
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("invalid endpoint selection mode: %d", c.selectionMode)
|
|
||||||
}
|
|
||||||
|
|
||||||
c.Lock()
|
|
||||||
defer c.Unlock()
|
|
||||||
c.endpoints = neps
|
|
||||||
c.pinned = npin
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) AutoSync(ctx context.Context, interval time.Duration) error {
|
|
||||||
ticker := time.NewTicker(interval)
|
|
||||||
defer ticker.Stop()
|
|
||||||
for {
|
|
||||||
err := c.Sync(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
select {
|
|
||||||
case <-ctx.Done():
|
|
||||||
return ctx.Err()
|
|
||||||
case <-ticker.C:
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *httpClusterClient) GetVersion(ctx context.Context) (*version.Versions, error) {
|
|
||||||
act := &getAction{Prefix: "/version"}
|
|
||||||
|
|
||||||
resp, body, err := c.Do(ctx, act)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
switch resp.StatusCode {
|
|
||||||
case http.StatusOK:
|
|
||||||
if len(body) == 0 {
|
|
||||||
return nil, ErrEmptyBody
|
|
||||||
}
|
|
||||||
var vresp version.Versions
|
|
||||||
if err := json.Unmarshal(body, &vresp); err != nil {
|
|
||||||
return nil, ErrInvalidJSON
|
|
||||||
}
|
|
||||||
return &vresp, nil
|
|
||||||
default:
|
|
||||||
var etcdErr Error
|
|
||||||
if err := json.Unmarshal(body, &etcdErr); err != nil {
|
|
||||||
return nil, ErrInvalidJSON
|
|
||||||
}
|
|
||||||
return nil, etcdErr
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type roundTripResponse struct {
|
|
||||||
resp *http.Response
|
|
||||||
err error
|
|
||||||
}
|
|
||||||
|
|
||||||
type simpleHTTPClient struct {
|
|
||||||
transport CancelableTransport
|
|
||||||
endpoint url.URL
|
|
||||||
headerTimeout time.Duration
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *simpleHTTPClient) Do(ctx context.Context, act httpAction) (*http.Response, []byte, error) {
|
|
||||||
req := act.HTTPRequest(c.endpoint)
|
|
||||||
|
|
||||||
if err := printcURL(req); err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
isWait := false
|
|
||||||
if req != nil && req.URL != nil {
|
|
||||||
ws := req.URL.Query().Get("wait")
|
|
||||||
if len(ws) != 0 {
|
|
||||||
var err error
|
|
||||||
isWait, err = strconv.ParseBool(ws)
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, fmt.Errorf("wrong wait value %s (%v for %+v)", ws, err, req)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var hctx context.Context
|
|
||||||
var hcancel context.CancelFunc
|
|
||||||
if !isWait && c.headerTimeout > 0 {
|
|
||||||
hctx, hcancel = context.WithTimeout(ctx, c.headerTimeout)
|
|
||||||
} else {
|
|
||||||
hctx, hcancel = context.WithCancel(ctx)
|
|
||||||
}
|
|
||||||
defer hcancel()
|
|
||||||
|
|
||||||
reqcancel := requestCanceler(c.transport, req)
|
|
||||||
|
|
||||||
rtchan := make(chan roundTripResponse, 1)
|
|
||||||
go func() {
|
|
||||||
resp, err := c.transport.RoundTrip(req)
|
|
||||||
rtchan <- roundTripResponse{resp: resp, err: err}
|
|
||||||
close(rtchan)
|
|
||||||
}()
|
|
||||||
|
|
||||||
var resp *http.Response
|
|
||||||
var err error
|
|
||||||
|
|
||||||
select {
|
|
||||||
case rtresp := <-rtchan:
|
|
||||||
resp, err = rtresp.resp, rtresp.err
|
|
||||||
case <-hctx.Done():
|
|
||||||
// cancel and wait for request to actually exit before continuing
|
|
||||||
reqcancel()
|
|
||||||
rtresp := <-rtchan
|
|
||||||
resp = rtresp.resp
|
|
||||||
switch {
|
|
||||||
case ctx.Err() != nil:
|
|
||||||
err = ctx.Err()
|
|
||||||
case hctx.Err() != nil:
|
|
||||||
err = fmt.Errorf("client: endpoint %s exceeded header timeout", c.endpoint.String())
|
|
||||||
default:
|
|
||||||
panic("failed to get error from context")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// always check for resp nil-ness to deal with possible
|
|
||||||
// race conditions between channels above
|
|
||||||
defer func() {
|
|
||||||
if resp != nil {
|
|
||||||
resp.Body.Close()
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var body []byte
|
|
||||||
done := make(chan struct{})
|
|
||||||
go func() {
|
|
||||||
body, err = ioutil.ReadAll(resp.Body)
|
|
||||||
done <- struct{}{}
|
|
||||||
}()
|
|
||||||
|
|
||||||
select {
|
|
||||||
case <-ctx.Done():
|
|
||||||
resp.Body.Close()
|
|
||||||
<-done
|
|
||||||
return nil, nil, ctx.Err()
|
|
||||||
case <-done:
|
|
||||||
}
|
|
||||||
|
|
||||||
return resp, body, err
|
|
||||||
}
|
|
||||||
|
|
||||||
type authedAction struct {
|
|
||||||
act httpAction
|
|
||||||
credentials credentials
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *authedAction) HTTPRequest(url url.URL) *http.Request {
|
|
||||||
r := a.act.HTTPRequest(url)
|
|
||||||
r.SetBasicAuth(a.credentials.username, a.credentials.password)
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
type redirectFollowingHTTPClient struct {
|
|
||||||
client httpClient
|
|
||||||
checkRedirect CheckRedirectFunc
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *redirectFollowingHTTPClient) Do(ctx context.Context, act httpAction) (*http.Response, []byte, error) {
|
|
||||||
next := act
|
|
||||||
for i := 0; i < 100; i++ {
|
|
||||||
if i > 0 {
|
|
||||||
if err := r.checkRedirect(i); err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
resp, body, err := r.client.Do(ctx, next)
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
if resp.StatusCode/100 == 3 {
|
|
||||||
hdr := resp.Header.Get("Location")
|
|
||||||
if hdr == "" {
|
|
||||||
return nil, nil, fmt.Errorf("Location header not set")
|
|
||||||
}
|
|
||||||
loc, err := url.Parse(hdr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, fmt.Errorf("Location header not valid URL: %s", hdr)
|
|
||||||
}
|
|
||||||
next = &redirectedHTTPAction{
|
|
||||||
action: act,
|
|
||||||
location: *loc,
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
return resp, body, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, nil, errTooManyRedirectChecks
|
|
||||||
}
|
|
||||||
|
|
||||||
type redirectedHTTPAction struct {
|
|
||||||
action httpAction
|
|
||||||
location url.URL
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *redirectedHTTPAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
orig := r.action.HTTPRequest(ep)
|
|
||||||
orig.URL = &r.location
|
|
||||||
return orig
|
|
||||||
}
|
|
||||||
|
|
||||||
func shuffleEndpoints(r *rand.Rand, eps []url.URL) []url.URL {
|
|
||||||
p := r.Perm(len(eps))
|
|
||||||
neps := make([]url.URL, len(eps))
|
|
||||||
for i, k := range p {
|
|
||||||
neps[i] = eps[k]
|
|
||||||
}
|
|
||||||
return neps
|
|
||||||
}
|
|
||||||
|
|
||||||
func endpointsEqual(left, right []url.URL) bool {
|
|
||||||
if len(left) != len(right) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
sLeft := make([]string, len(left))
|
|
||||||
sRight := make([]string, len(right))
|
|
||||||
for i, l := range left {
|
|
||||||
sLeft[i] = l.String()
|
|
||||||
}
|
|
||||||
for i, r := range right {
|
|
||||||
sRight[i] = r.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
sort.Strings(sLeft)
|
|
||||||
sort.Strings(sRight)
|
|
||||||
for i := range sLeft {
|
|
||||||
if sLeft[i] != sRight[i] {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
70
vendor/github.com/coreos/etcd/client/curl.go
generated
vendored
70
vendor/github.com/coreos/etcd/client/curl.go
generated
vendored
@ -1,70 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
|
||||||
"net/http"
|
|
||||||
"os"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
cURLDebug = false
|
|
||||||
)
|
|
||||||
|
|
||||||
func EnablecURLDebug() {
|
|
||||||
cURLDebug = true
|
|
||||||
}
|
|
||||||
|
|
||||||
func DisablecURLDebug() {
|
|
||||||
cURLDebug = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// printcURL prints the cURL equivalent request to stderr.
|
|
||||||
// It returns an error if the body of the request cannot
|
|
||||||
// be read.
|
|
||||||
// The caller MUST cancel the request if there is an error.
|
|
||||||
func printcURL(req *http.Request) error {
|
|
||||||
if !cURLDebug {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
var (
|
|
||||||
command string
|
|
||||||
b []byte
|
|
||||||
err error
|
|
||||||
)
|
|
||||||
|
|
||||||
if req.URL != nil {
|
|
||||||
command = fmt.Sprintf("curl -X %s %s", req.Method, req.URL.String())
|
|
||||||
}
|
|
||||||
|
|
||||||
if req.Body != nil {
|
|
||||||
b, err = ioutil.ReadAll(req.Body)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
command += fmt.Sprintf(" -d %q", string(b))
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Fprintf(os.Stderr, "cURL Command: %s\n", command)
|
|
||||||
|
|
||||||
// reset body
|
|
||||||
body := bytes.NewBuffer(b)
|
|
||||||
req.Body = ioutil.NopCloser(body)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
40
vendor/github.com/coreos/etcd/client/discover.go
generated
vendored
40
vendor/github.com/coreos/etcd/client/discover.go
generated
vendored
@ -1,40 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/coreos/etcd/pkg/srv"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Discoverer is an interface that wraps the Discover method.
|
|
||||||
type Discoverer interface {
|
|
||||||
// Discover looks up the etcd servers for the domain.
|
|
||||||
Discover(domain string) ([]string, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type srvDiscover struct{}
|
|
||||||
|
|
||||||
// NewSRVDiscover constructs a new Discoverer that uses the stdlib to lookup SRV records.
|
|
||||||
func NewSRVDiscover() Discoverer {
|
|
||||||
return &srvDiscover{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *srvDiscover) Discover(domain string) ([]string, error) {
|
|
||||||
srvs, err := srv.GetClient("etcd-client", domain)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return srvs.Endpoints, nil
|
|
||||||
}
|
|
73
vendor/github.com/coreos/etcd/client/doc.go
generated
vendored
73
vendor/github.com/coreos/etcd/client/doc.go
generated
vendored
@ -1,73 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
/*
|
|
||||||
Package client provides bindings for the etcd APIs.
|
|
||||||
|
|
||||||
Create a Config and exchange it for a Client:
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
|
|
||||||
"github.com/coreos/etcd/client"
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
)
|
|
||||||
|
|
||||||
cfg := client.Config{
|
|
||||||
Endpoints: []string{"http://127.0.0.1:2379"},
|
|
||||||
Transport: DefaultTransport,
|
|
||||||
}
|
|
||||||
|
|
||||||
c, err := client.New(cfg)
|
|
||||||
if err != nil {
|
|
||||||
// handle error
|
|
||||||
}
|
|
||||||
|
|
||||||
Clients are safe for concurrent use by multiple goroutines.
|
|
||||||
|
|
||||||
Create a KeysAPI using the Client, then use it to interact with etcd:
|
|
||||||
|
|
||||||
kAPI := client.NewKeysAPI(c)
|
|
||||||
|
|
||||||
// create a new key /foo with the value "bar"
|
|
||||||
_, err = kAPI.Create(context.Background(), "/foo", "bar")
|
|
||||||
if err != nil {
|
|
||||||
// handle error
|
|
||||||
}
|
|
||||||
|
|
||||||
// delete the newly created key only if the value is still "bar"
|
|
||||||
_, err = kAPI.Delete(context.Background(), "/foo", &DeleteOptions{PrevValue: "bar"})
|
|
||||||
if err != nil {
|
|
||||||
// handle error
|
|
||||||
}
|
|
||||||
|
|
||||||
Use a custom context to set timeouts on your operations:
|
|
||||||
|
|
||||||
import "time"
|
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
// set a new key, ignoring it's previous state
|
|
||||||
_, err := kAPI.Set(ctx, "/ping", "pong", nil)
|
|
||||||
if err != nil {
|
|
||||||
if err == context.DeadlineExceeded {
|
|
||||||
// request took longer than 5s
|
|
||||||
} else {
|
|
||||||
// handle error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
*/
|
|
||||||
package client
|
|
1087
vendor/github.com/coreos/etcd/client/keys.generated.go
generated
vendored
1087
vendor/github.com/coreos/etcd/client/keys.generated.go
generated
vendored
File diff suppressed because it is too large
Load Diff
682
vendor/github.com/coreos/etcd/client/keys.go
generated
vendored
682
vendor/github.com/coreos/etcd/client/keys.go
generated
vendored
@ -1,682 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
//go:generate codecgen -d 1819 -r "Node|Response|Nodes" -o keys.generated.go keys.go
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/coreos/etcd/pkg/pathutil"
|
|
||||||
"github.com/ugorji/go/codec"
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
ErrorCodeKeyNotFound = 100
|
|
||||||
ErrorCodeTestFailed = 101
|
|
||||||
ErrorCodeNotFile = 102
|
|
||||||
ErrorCodeNotDir = 104
|
|
||||||
ErrorCodeNodeExist = 105
|
|
||||||
ErrorCodeRootROnly = 107
|
|
||||||
ErrorCodeDirNotEmpty = 108
|
|
||||||
ErrorCodeUnauthorized = 110
|
|
||||||
|
|
||||||
ErrorCodePrevValueRequired = 201
|
|
||||||
ErrorCodeTTLNaN = 202
|
|
||||||
ErrorCodeIndexNaN = 203
|
|
||||||
ErrorCodeInvalidField = 209
|
|
||||||
ErrorCodeInvalidForm = 210
|
|
||||||
|
|
||||||
ErrorCodeRaftInternal = 300
|
|
||||||
ErrorCodeLeaderElect = 301
|
|
||||||
|
|
||||||
ErrorCodeWatcherCleared = 400
|
|
||||||
ErrorCodeEventIndexCleared = 401
|
|
||||||
)
|
|
||||||
|
|
||||||
type Error struct {
|
|
||||||
Code int `json:"errorCode"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
Cause string `json:"cause"`
|
|
||||||
Index uint64 `json:"index"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e Error) Error() string {
|
|
||||||
return fmt.Sprintf("%v: %v (%v) [%v]", e.Code, e.Message, e.Cause, e.Index)
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrInvalidJSON = errors.New("client: response is invalid json. The endpoint is probably not valid etcd cluster endpoint.")
|
|
||||||
ErrEmptyBody = errors.New("client: response body is empty")
|
|
||||||
)
|
|
||||||
|
|
||||||
// PrevExistType is used to define an existence condition when setting
|
|
||||||
// or deleting Nodes.
|
|
||||||
type PrevExistType string
|
|
||||||
|
|
||||||
const (
|
|
||||||
PrevIgnore = PrevExistType("")
|
|
||||||
PrevExist = PrevExistType("true")
|
|
||||||
PrevNoExist = PrevExistType("false")
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
defaultV2KeysPrefix = "/v2/keys"
|
|
||||||
)
|
|
||||||
|
|
||||||
// NewKeysAPI builds a KeysAPI that interacts with etcd's key-value
|
|
||||||
// API over HTTP.
|
|
||||||
func NewKeysAPI(c Client) KeysAPI {
|
|
||||||
return NewKeysAPIWithPrefix(c, defaultV2KeysPrefix)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewKeysAPIWithPrefix acts like NewKeysAPI, but allows the caller
|
|
||||||
// to provide a custom base URL path. This should only be used in
|
|
||||||
// very rare cases.
|
|
||||||
func NewKeysAPIWithPrefix(c Client, p string) KeysAPI {
|
|
||||||
return &httpKeysAPI{
|
|
||||||
client: c,
|
|
||||||
prefix: p,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type KeysAPI interface {
|
|
||||||
// Get retrieves a set of Nodes from etcd
|
|
||||||
Get(ctx context.Context, key string, opts *GetOptions) (*Response, error)
|
|
||||||
|
|
||||||
// Set assigns a new value to a Node identified by a given key. The caller
|
|
||||||
// may define a set of conditions in the SetOptions. If SetOptions.Dir=true
|
|
||||||
// then value is ignored.
|
|
||||||
Set(ctx context.Context, key, value string, opts *SetOptions) (*Response, error)
|
|
||||||
|
|
||||||
// Delete removes a Node identified by the given key, optionally destroying
|
|
||||||
// all of its children as well. The caller may define a set of required
|
|
||||||
// conditions in an DeleteOptions object.
|
|
||||||
Delete(ctx context.Context, key string, opts *DeleteOptions) (*Response, error)
|
|
||||||
|
|
||||||
// Create is an alias for Set w/ PrevExist=false
|
|
||||||
Create(ctx context.Context, key, value string) (*Response, error)
|
|
||||||
|
|
||||||
// CreateInOrder is used to atomically create in-order keys within the given directory.
|
|
||||||
CreateInOrder(ctx context.Context, dir, value string, opts *CreateInOrderOptions) (*Response, error)
|
|
||||||
|
|
||||||
// Update is an alias for Set w/ PrevExist=true
|
|
||||||
Update(ctx context.Context, key, value string) (*Response, error)
|
|
||||||
|
|
||||||
// Watcher builds a new Watcher targeted at a specific Node identified
|
|
||||||
// by the given key. The Watcher may be configured at creation time
|
|
||||||
// through a WatcherOptions object. The returned Watcher is designed
|
|
||||||
// to emit events that happen to a Node, and optionally to its children.
|
|
||||||
Watcher(key string, opts *WatcherOptions) Watcher
|
|
||||||
}
|
|
||||||
|
|
||||||
type WatcherOptions struct {
|
|
||||||
// AfterIndex defines the index after-which the Watcher should
|
|
||||||
// start emitting events. For example, if a value of 5 is
|
|
||||||
// provided, the first event will have an index >= 6.
|
|
||||||
//
|
|
||||||
// Setting AfterIndex to 0 (default) means that the Watcher
|
|
||||||
// should start watching for events starting at the current
|
|
||||||
// index, whatever that may be.
|
|
||||||
AfterIndex uint64
|
|
||||||
|
|
||||||
// Recursive specifies whether or not the Watcher should emit
|
|
||||||
// events that occur in children of the given keyspace. If set
|
|
||||||
// to false (default), events will be limited to those that
|
|
||||||
// occur for the exact key.
|
|
||||||
Recursive bool
|
|
||||||
}
|
|
||||||
|
|
||||||
type CreateInOrderOptions struct {
|
|
||||||
// TTL defines a period of time after-which the Node should
|
|
||||||
// expire and no longer exist. Values <= 0 are ignored. Given
|
|
||||||
// that the zero-value is ignored, TTL cannot be used to set
|
|
||||||
// a TTL of 0.
|
|
||||||
TTL time.Duration
|
|
||||||
}
|
|
||||||
|
|
||||||
type SetOptions struct {
|
|
||||||
// PrevValue specifies what the current value of the Node must
|
|
||||||
// be in order for the Set operation to succeed.
|
|
||||||
//
|
|
||||||
// Leaving this field empty means that the caller wishes to
|
|
||||||
// ignore the current value of the Node. This cannot be used
|
|
||||||
// to compare the Node's current value to an empty string.
|
|
||||||
//
|
|
||||||
// PrevValue is ignored if Dir=true
|
|
||||||
PrevValue string
|
|
||||||
|
|
||||||
// PrevIndex indicates what the current ModifiedIndex of the
|
|
||||||
// Node must be in order for the Set operation to succeed.
|
|
||||||
//
|
|
||||||
// If PrevIndex is set to 0 (default), no comparison is made.
|
|
||||||
PrevIndex uint64
|
|
||||||
|
|
||||||
// PrevExist specifies whether the Node must currently exist
|
|
||||||
// (PrevExist) or not (PrevNoExist). If the caller does not
|
|
||||||
// care about existence, set PrevExist to PrevIgnore, or simply
|
|
||||||
// leave it unset.
|
|
||||||
PrevExist PrevExistType
|
|
||||||
|
|
||||||
// TTL defines a period of time after-which the Node should
|
|
||||||
// expire and no longer exist. Values <= 0 are ignored. Given
|
|
||||||
// that the zero-value is ignored, TTL cannot be used to set
|
|
||||||
// a TTL of 0.
|
|
||||||
TTL time.Duration
|
|
||||||
|
|
||||||
// Refresh set to true means a TTL value can be updated
|
|
||||||
// without firing a watch or changing the node value. A
|
|
||||||
// value must not be provided when refreshing a key.
|
|
||||||
Refresh bool
|
|
||||||
|
|
||||||
// Dir specifies whether or not this Node should be created as a directory.
|
|
||||||
Dir bool
|
|
||||||
|
|
||||||
// NoValueOnSuccess specifies whether the response contains the current value of the Node.
|
|
||||||
// If set, the response will only contain the current value when the request fails.
|
|
||||||
NoValueOnSuccess bool
|
|
||||||
}
|
|
||||||
|
|
||||||
type GetOptions struct {
|
|
||||||
// Recursive defines whether or not all children of the Node
|
|
||||||
// should be returned.
|
|
||||||
Recursive bool
|
|
||||||
|
|
||||||
// Sort instructs the server whether or not to sort the Nodes.
|
|
||||||
// If true, the Nodes are sorted alphabetically by key in
|
|
||||||
// ascending order (A to z). If false (default), the Nodes will
|
|
||||||
// not be sorted and the ordering used should not be considered
|
|
||||||
// predictable.
|
|
||||||
Sort bool
|
|
||||||
|
|
||||||
// Quorum specifies whether it gets the latest committed value that
|
|
||||||
// has been applied in quorum of members, which ensures external
|
|
||||||
// consistency (or linearizability).
|
|
||||||
Quorum bool
|
|
||||||
}
|
|
||||||
|
|
||||||
type DeleteOptions struct {
|
|
||||||
// PrevValue specifies what the current value of the Node must
|
|
||||||
// be in order for the Delete operation to succeed.
|
|
||||||
//
|
|
||||||
// Leaving this field empty means that the caller wishes to
|
|
||||||
// ignore the current value of the Node. This cannot be used
|
|
||||||
// to compare the Node's current value to an empty string.
|
|
||||||
PrevValue string
|
|
||||||
|
|
||||||
// PrevIndex indicates what the current ModifiedIndex of the
|
|
||||||
// Node must be in order for the Delete operation to succeed.
|
|
||||||
//
|
|
||||||
// If PrevIndex is set to 0 (default), no comparison is made.
|
|
||||||
PrevIndex uint64
|
|
||||||
|
|
||||||
// Recursive defines whether or not all children of the Node
|
|
||||||
// should be deleted. If set to true, all children of the Node
|
|
||||||
// identified by the given key will be deleted. If left unset
|
|
||||||
// or explicitly set to false, only a single Node will be
|
|
||||||
// deleted.
|
|
||||||
Recursive bool
|
|
||||||
|
|
||||||
// Dir specifies whether or not this Node should be removed as a directory.
|
|
||||||
Dir bool
|
|
||||||
}
|
|
||||||
|
|
||||||
type Watcher interface {
|
|
||||||
// Next blocks until an etcd event occurs, then returns a Response
|
|
||||||
// representing that event. The behavior of Next depends on the
|
|
||||||
// WatcherOptions used to construct the Watcher. Next is designed to
|
|
||||||
// be called repeatedly, each time blocking until a subsequent event
|
|
||||||
// is available.
|
|
||||||
//
|
|
||||||
// If the provided context is cancelled, Next will return a non-nil
|
|
||||||
// error. Any other failures encountered while waiting for the next
|
|
||||||
// event (connection issues, deserialization failures, etc) will
|
|
||||||
// also result in a non-nil error.
|
|
||||||
Next(context.Context) (*Response, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type Response struct {
|
|
||||||
// Action is the name of the operation that occurred. Possible values
|
|
||||||
// include get, set, delete, update, create, compareAndSwap,
|
|
||||||
// compareAndDelete and expire.
|
|
||||||
Action string `json:"action"`
|
|
||||||
|
|
||||||
// Node represents the state of the relevant etcd Node.
|
|
||||||
Node *Node `json:"node"`
|
|
||||||
|
|
||||||
// PrevNode represents the previous state of the Node. PrevNode is non-nil
|
|
||||||
// only if the Node existed before the action occurred and the action
|
|
||||||
// caused a change to the Node.
|
|
||||||
PrevNode *Node `json:"prevNode"`
|
|
||||||
|
|
||||||
// Index holds the cluster-level index at the time the Response was generated.
|
|
||||||
// This index is not tied to the Node(s) contained in this Response.
|
|
||||||
Index uint64 `json:"-"`
|
|
||||||
|
|
||||||
// ClusterID holds the cluster-level ID reported by the server. This
|
|
||||||
// should be different for different etcd clusters.
|
|
||||||
ClusterID string `json:"-"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Node struct {
|
|
||||||
// Key represents the unique location of this Node (e.g. "/foo/bar").
|
|
||||||
Key string `json:"key"`
|
|
||||||
|
|
||||||
// Dir reports whether node describes a directory.
|
|
||||||
Dir bool `json:"dir,omitempty"`
|
|
||||||
|
|
||||||
// Value is the current data stored on this Node. If this Node
|
|
||||||
// is a directory, Value will be empty.
|
|
||||||
Value string `json:"value"`
|
|
||||||
|
|
||||||
// Nodes holds the children of this Node, only if this Node is a directory.
|
|
||||||
// This slice of will be arbitrarily deep (children, grandchildren, great-
|
|
||||||
// grandchildren, etc.) if a recursive Get or Watch request were made.
|
|
||||||
Nodes Nodes `json:"nodes"`
|
|
||||||
|
|
||||||
// CreatedIndex is the etcd index at-which this Node was created.
|
|
||||||
CreatedIndex uint64 `json:"createdIndex"`
|
|
||||||
|
|
||||||
// ModifiedIndex is the etcd index at-which this Node was last modified.
|
|
||||||
ModifiedIndex uint64 `json:"modifiedIndex"`
|
|
||||||
|
|
||||||
// Expiration is the server side expiration time of the key.
|
|
||||||
Expiration *time.Time `json:"expiration,omitempty"`
|
|
||||||
|
|
||||||
// TTL is the time to live of the key in second.
|
|
||||||
TTL int64 `json:"ttl,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *Node) String() string {
|
|
||||||
return fmt.Sprintf("{Key: %s, CreatedIndex: %d, ModifiedIndex: %d, TTL: %d}", n.Key, n.CreatedIndex, n.ModifiedIndex, n.TTL)
|
|
||||||
}
|
|
||||||
|
|
||||||
// TTLDuration returns the Node's TTL as a time.Duration object
|
|
||||||
func (n *Node) TTLDuration() time.Duration {
|
|
||||||
return time.Duration(n.TTL) * time.Second
|
|
||||||
}
|
|
||||||
|
|
||||||
type Nodes []*Node
|
|
||||||
|
|
||||||
// interfaces for sorting
|
|
||||||
|
|
||||||
func (ns Nodes) Len() int { return len(ns) }
|
|
||||||
func (ns Nodes) Less(i, j int) bool { return ns[i].Key < ns[j].Key }
|
|
||||||
func (ns Nodes) Swap(i, j int) { ns[i], ns[j] = ns[j], ns[i] }
|
|
||||||
|
|
||||||
type httpKeysAPI struct {
|
|
||||||
client httpClient
|
|
||||||
prefix string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) Set(ctx context.Context, key, val string, opts *SetOptions) (*Response, error) {
|
|
||||||
act := &setAction{
|
|
||||||
Prefix: k.prefix,
|
|
||||||
Key: key,
|
|
||||||
Value: val,
|
|
||||||
}
|
|
||||||
|
|
||||||
if opts != nil {
|
|
||||||
act.PrevValue = opts.PrevValue
|
|
||||||
act.PrevIndex = opts.PrevIndex
|
|
||||||
act.PrevExist = opts.PrevExist
|
|
||||||
act.TTL = opts.TTL
|
|
||||||
act.Refresh = opts.Refresh
|
|
||||||
act.Dir = opts.Dir
|
|
||||||
act.NoValueOnSuccess = opts.NoValueOnSuccess
|
|
||||||
}
|
|
||||||
|
|
||||||
doCtx := ctx
|
|
||||||
if act.PrevExist == PrevNoExist {
|
|
||||||
doCtx = context.WithValue(doCtx, &oneShotCtxValue, &oneShotCtxValue)
|
|
||||||
}
|
|
||||||
resp, body, err := k.client.Do(doCtx, act)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return unmarshalHTTPResponse(resp.StatusCode, resp.Header, body)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) Create(ctx context.Context, key, val string) (*Response, error) {
|
|
||||||
return k.Set(ctx, key, val, &SetOptions{PrevExist: PrevNoExist})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) CreateInOrder(ctx context.Context, dir, val string, opts *CreateInOrderOptions) (*Response, error) {
|
|
||||||
act := &createInOrderAction{
|
|
||||||
Prefix: k.prefix,
|
|
||||||
Dir: dir,
|
|
||||||
Value: val,
|
|
||||||
}
|
|
||||||
|
|
||||||
if opts != nil {
|
|
||||||
act.TTL = opts.TTL
|
|
||||||
}
|
|
||||||
|
|
||||||
resp, body, err := k.client.Do(ctx, act)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return unmarshalHTTPResponse(resp.StatusCode, resp.Header, body)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) Update(ctx context.Context, key, val string) (*Response, error) {
|
|
||||||
return k.Set(ctx, key, val, &SetOptions{PrevExist: PrevExist})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) Delete(ctx context.Context, key string, opts *DeleteOptions) (*Response, error) {
|
|
||||||
act := &deleteAction{
|
|
||||||
Prefix: k.prefix,
|
|
||||||
Key: key,
|
|
||||||
}
|
|
||||||
|
|
||||||
if opts != nil {
|
|
||||||
act.PrevValue = opts.PrevValue
|
|
||||||
act.PrevIndex = opts.PrevIndex
|
|
||||||
act.Dir = opts.Dir
|
|
||||||
act.Recursive = opts.Recursive
|
|
||||||
}
|
|
||||||
|
|
||||||
doCtx := context.WithValue(ctx, &oneShotCtxValue, &oneShotCtxValue)
|
|
||||||
resp, body, err := k.client.Do(doCtx, act)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return unmarshalHTTPResponse(resp.StatusCode, resp.Header, body)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) Get(ctx context.Context, key string, opts *GetOptions) (*Response, error) {
|
|
||||||
act := &getAction{
|
|
||||||
Prefix: k.prefix,
|
|
||||||
Key: key,
|
|
||||||
}
|
|
||||||
|
|
||||||
if opts != nil {
|
|
||||||
act.Recursive = opts.Recursive
|
|
||||||
act.Sorted = opts.Sort
|
|
||||||
act.Quorum = opts.Quorum
|
|
||||||
}
|
|
||||||
|
|
||||||
resp, body, err := k.client.Do(ctx, act)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return unmarshalHTTPResponse(resp.StatusCode, resp.Header, body)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (k *httpKeysAPI) Watcher(key string, opts *WatcherOptions) Watcher {
|
|
||||||
act := waitAction{
|
|
||||||
Prefix: k.prefix,
|
|
||||||
Key: key,
|
|
||||||
}
|
|
||||||
|
|
||||||
if opts != nil {
|
|
||||||
act.Recursive = opts.Recursive
|
|
||||||
if opts.AfterIndex > 0 {
|
|
||||||
act.WaitIndex = opts.AfterIndex + 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return &httpWatcher{
|
|
||||||
client: k.client,
|
|
||||||
nextWait: act,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpWatcher struct {
|
|
||||||
client httpClient
|
|
||||||
nextWait waitAction
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hw *httpWatcher) Next(ctx context.Context) (*Response, error) {
|
|
||||||
for {
|
|
||||||
httpresp, body, err := hw.client.Do(ctx, &hw.nextWait)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resp, err := unmarshalHTTPResponse(httpresp.StatusCode, httpresp.Header, body)
|
|
||||||
if err != nil {
|
|
||||||
if err == ErrEmptyBody {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
hw.nextWait.WaitIndex = resp.Node.ModifiedIndex + 1
|
|
||||||
return resp, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// v2KeysURL forms a URL representing the location of a key.
|
|
||||||
// The endpoint argument represents the base URL of an etcd
|
|
||||||
// server. The prefix is the path needed to route from the
|
|
||||||
// provided endpoint's path to the root of the keys API
|
|
||||||
// (typically "/v2/keys").
|
|
||||||
func v2KeysURL(ep url.URL, prefix, key string) *url.URL {
|
|
||||||
// We concatenate all parts together manually. We cannot use
|
|
||||||
// path.Join because it does not reserve trailing slash.
|
|
||||||
// We call CanonicalURLPath to further cleanup the path.
|
|
||||||
if prefix != "" && prefix[0] != '/' {
|
|
||||||
prefix = "/" + prefix
|
|
||||||
}
|
|
||||||
if key != "" && key[0] != '/' {
|
|
||||||
key = "/" + key
|
|
||||||
}
|
|
||||||
ep.Path = pathutil.CanonicalURLPath(ep.Path + prefix + key)
|
|
||||||
return &ep
|
|
||||||
}
|
|
||||||
|
|
||||||
type getAction struct {
|
|
||||||
Prefix string
|
|
||||||
Key string
|
|
||||||
Recursive bool
|
|
||||||
Sorted bool
|
|
||||||
Quorum bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *getAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2KeysURL(ep, g.Prefix, g.Key)
|
|
||||||
|
|
||||||
params := u.Query()
|
|
||||||
params.Set("recursive", strconv.FormatBool(g.Recursive))
|
|
||||||
params.Set("sorted", strconv.FormatBool(g.Sorted))
|
|
||||||
params.Set("quorum", strconv.FormatBool(g.Quorum))
|
|
||||||
u.RawQuery = params.Encode()
|
|
||||||
|
|
||||||
req, _ := http.NewRequest("GET", u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type waitAction struct {
|
|
||||||
Prefix string
|
|
||||||
Key string
|
|
||||||
WaitIndex uint64
|
|
||||||
Recursive bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *waitAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2KeysURL(ep, w.Prefix, w.Key)
|
|
||||||
|
|
||||||
params := u.Query()
|
|
||||||
params.Set("wait", "true")
|
|
||||||
params.Set("waitIndex", strconv.FormatUint(w.WaitIndex, 10))
|
|
||||||
params.Set("recursive", strconv.FormatBool(w.Recursive))
|
|
||||||
u.RawQuery = params.Encode()
|
|
||||||
|
|
||||||
req, _ := http.NewRequest("GET", u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type setAction struct {
|
|
||||||
Prefix string
|
|
||||||
Key string
|
|
||||||
Value string
|
|
||||||
PrevValue string
|
|
||||||
PrevIndex uint64
|
|
||||||
PrevExist PrevExistType
|
|
||||||
TTL time.Duration
|
|
||||||
Refresh bool
|
|
||||||
Dir bool
|
|
||||||
NoValueOnSuccess bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *setAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2KeysURL(ep, a.Prefix, a.Key)
|
|
||||||
|
|
||||||
params := u.Query()
|
|
||||||
form := url.Values{}
|
|
||||||
|
|
||||||
// we're either creating a directory or setting a key
|
|
||||||
if a.Dir {
|
|
||||||
params.Set("dir", strconv.FormatBool(a.Dir))
|
|
||||||
} else {
|
|
||||||
// These options are only valid for setting a key
|
|
||||||
if a.PrevValue != "" {
|
|
||||||
params.Set("prevValue", a.PrevValue)
|
|
||||||
}
|
|
||||||
form.Add("value", a.Value)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Options which apply to both setting a key and creating a dir
|
|
||||||
if a.PrevIndex != 0 {
|
|
||||||
params.Set("prevIndex", strconv.FormatUint(a.PrevIndex, 10))
|
|
||||||
}
|
|
||||||
if a.PrevExist != PrevIgnore {
|
|
||||||
params.Set("prevExist", string(a.PrevExist))
|
|
||||||
}
|
|
||||||
if a.TTL > 0 {
|
|
||||||
form.Add("ttl", strconv.FormatUint(uint64(a.TTL.Seconds()), 10))
|
|
||||||
}
|
|
||||||
|
|
||||||
if a.Refresh {
|
|
||||||
form.Add("refresh", "true")
|
|
||||||
}
|
|
||||||
if a.NoValueOnSuccess {
|
|
||||||
params.Set("noValueOnSuccess", strconv.FormatBool(a.NoValueOnSuccess))
|
|
||||||
}
|
|
||||||
|
|
||||||
u.RawQuery = params.Encode()
|
|
||||||
body := strings.NewReader(form.Encode())
|
|
||||||
|
|
||||||
req, _ := http.NewRequest("PUT", u.String(), body)
|
|
||||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
||||||
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type deleteAction struct {
|
|
||||||
Prefix string
|
|
||||||
Key string
|
|
||||||
PrevValue string
|
|
||||||
PrevIndex uint64
|
|
||||||
Dir bool
|
|
||||||
Recursive bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *deleteAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2KeysURL(ep, a.Prefix, a.Key)
|
|
||||||
|
|
||||||
params := u.Query()
|
|
||||||
if a.PrevValue != "" {
|
|
||||||
params.Set("prevValue", a.PrevValue)
|
|
||||||
}
|
|
||||||
if a.PrevIndex != 0 {
|
|
||||||
params.Set("prevIndex", strconv.FormatUint(a.PrevIndex, 10))
|
|
||||||
}
|
|
||||||
if a.Dir {
|
|
||||||
params.Set("dir", "true")
|
|
||||||
}
|
|
||||||
if a.Recursive {
|
|
||||||
params.Set("recursive", "true")
|
|
||||||
}
|
|
||||||
u.RawQuery = params.Encode()
|
|
||||||
|
|
||||||
req, _ := http.NewRequest("DELETE", u.String(), nil)
|
|
||||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
||||||
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type createInOrderAction struct {
|
|
||||||
Prefix string
|
|
||||||
Dir string
|
|
||||||
Value string
|
|
||||||
TTL time.Duration
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *createInOrderAction) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2KeysURL(ep, a.Prefix, a.Dir)
|
|
||||||
|
|
||||||
form := url.Values{}
|
|
||||||
form.Add("value", a.Value)
|
|
||||||
if a.TTL > 0 {
|
|
||||||
form.Add("ttl", strconv.FormatUint(uint64(a.TTL.Seconds()), 10))
|
|
||||||
}
|
|
||||||
body := strings.NewReader(form.Encode())
|
|
||||||
|
|
||||||
req, _ := http.NewRequest("POST", u.String(), body)
|
|
||||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
func unmarshalHTTPResponse(code int, header http.Header, body []byte) (res *Response, err error) {
|
|
||||||
switch code {
|
|
||||||
case http.StatusOK, http.StatusCreated:
|
|
||||||
if len(body) == 0 {
|
|
||||||
return nil, ErrEmptyBody
|
|
||||||
}
|
|
||||||
res, err = unmarshalSuccessfulKeysResponse(header, body)
|
|
||||||
default:
|
|
||||||
err = unmarshalFailedKeysResponse(body)
|
|
||||||
}
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func unmarshalSuccessfulKeysResponse(header http.Header, body []byte) (*Response, error) {
|
|
||||||
var res Response
|
|
||||||
err := codec.NewDecoderBytes(body, new(codec.JsonHandle)).Decode(&res)
|
|
||||||
if err != nil {
|
|
||||||
return nil, ErrInvalidJSON
|
|
||||||
}
|
|
||||||
if header.Get("X-Etcd-Index") != "" {
|
|
||||||
res.Index, err = strconv.ParseUint(header.Get("X-Etcd-Index"), 10, 64)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
res.ClusterID = header.Get("X-Etcd-Cluster-ID")
|
|
||||||
return &res, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func unmarshalFailedKeysResponse(body []byte) error {
|
|
||||||
var etcdErr Error
|
|
||||||
if err := json.Unmarshal(body, &etcdErr); err != nil {
|
|
||||||
return ErrInvalidJSON
|
|
||||||
}
|
|
||||||
return etcdErr
|
|
||||||
}
|
|
304
vendor/github.com/coreos/etcd/client/members.go
generated
vendored
304
vendor/github.com/coreos/etcd/client/members.go
generated
vendored
@ -1,304 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"path"
|
|
||||||
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
|
|
||||||
"github.com/coreos/etcd/pkg/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
defaultV2MembersPrefix = "/v2/members"
|
|
||||||
defaultLeaderSuffix = "/leader"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Member struct {
|
|
||||||
// ID is the unique identifier of this Member.
|
|
||||||
ID string `json:"id"`
|
|
||||||
|
|
||||||
// Name is a human-readable, non-unique identifier of this Member.
|
|
||||||
Name string `json:"name"`
|
|
||||||
|
|
||||||
// PeerURLs represents the HTTP(S) endpoints this Member uses to
|
|
||||||
// participate in etcd's consensus protocol.
|
|
||||||
PeerURLs []string `json:"peerURLs"`
|
|
||||||
|
|
||||||
// ClientURLs represents the HTTP(S) endpoints on which this Member
|
|
||||||
// serves it's client-facing APIs.
|
|
||||||
ClientURLs []string `json:"clientURLs"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type memberCollection []Member
|
|
||||||
|
|
||||||
func (c *memberCollection) UnmarshalJSON(data []byte) error {
|
|
||||||
d := struct {
|
|
||||||
Members []Member
|
|
||||||
}{}
|
|
||||||
|
|
||||||
if err := json.Unmarshal(data, &d); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if d.Members == nil {
|
|
||||||
*c = make([]Member, 0)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
*c = d.Members
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type memberCreateOrUpdateRequest struct {
|
|
||||||
PeerURLs types.URLs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *memberCreateOrUpdateRequest) MarshalJSON() ([]byte, error) {
|
|
||||||
s := struct {
|
|
||||||
PeerURLs []string `json:"peerURLs"`
|
|
||||||
}{
|
|
||||||
PeerURLs: make([]string, len(m.PeerURLs)),
|
|
||||||
}
|
|
||||||
|
|
||||||
for i, u := range m.PeerURLs {
|
|
||||||
s.PeerURLs[i] = u.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
return json.Marshal(&s)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewMembersAPI constructs a new MembersAPI that uses HTTP to
|
|
||||||
// interact with etcd's membership API.
|
|
||||||
func NewMembersAPI(c Client) MembersAPI {
|
|
||||||
return &httpMembersAPI{
|
|
||||||
client: c,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type MembersAPI interface {
|
|
||||||
// List enumerates the current cluster membership.
|
|
||||||
List(ctx context.Context) ([]Member, error)
|
|
||||||
|
|
||||||
// Add instructs etcd to accept a new Member into the cluster.
|
|
||||||
Add(ctx context.Context, peerURL string) (*Member, error)
|
|
||||||
|
|
||||||
// Remove demotes an existing Member out of the cluster.
|
|
||||||
Remove(ctx context.Context, mID string) error
|
|
||||||
|
|
||||||
// Update instructs etcd to update an existing Member in the cluster.
|
|
||||||
Update(ctx context.Context, mID string, peerURLs []string) error
|
|
||||||
|
|
||||||
// Leader gets current leader of the cluster
|
|
||||||
Leader(ctx context.Context) (*Member, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type httpMembersAPI struct {
|
|
||||||
client httpClient
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *httpMembersAPI) List(ctx context.Context) ([]Member, error) {
|
|
||||||
req := &membersAPIActionList{}
|
|
||||||
resp, body, err := m.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := assertStatusCode(resp.StatusCode, http.StatusOK); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var mCollection memberCollection
|
|
||||||
if err := json.Unmarshal(body, &mCollection); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return []Member(mCollection), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *httpMembersAPI) Add(ctx context.Context, peerURL string) (*Member, error) {
|
|
||||||
urls, err := types.NewURLs([]string{peerURL})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
req := &membersAPIActionAdd{peerURLs: urls}
|
|
||||||
resp, body, err := m.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := assertStatusCode(resp.StatusCode, http.StatusCreated, http.StatusConflict); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if resp.StatusCode != http.StatusCreated {
|
|
||||||
var merr membersError
|
|
||||||
if err := json.Unmarshal(body, &merr); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return nil, merr
|
|
||||||
}
|
|
||||||
|
|
||||||
var memb Member
|
|
||||||
if err := json.Unmarshal(body, &memb); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return &memb, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *httpMembersAPI) Update(ctx context.Context, memberID string, peerURLs []string) error {
|
|
||||||
urls, err := types.NewURLs(peerURLs)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
req := &membersAPIActionUpdate{peerURLs: urls, memberID: memberID}
|
|
||||||
resp, body, err := m.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := assertStatusCode(resp.StatusCode, http.StatusNoContent, http.StatusNotFound, http.StatusConflict); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if resp.StatusCode != http.StatusNoContent {
|
|
||||||
var merr membersError
|
|
||||||
if err := json.Unmarshal(body, &merr); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return merr
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *httpMembersAPI) Remove(ctx context.Context, memberID string) error {
|
|
||||||
req := &membersAPIActionRemove{memberID: memberID}
|
|
||||||
resp, _, err := m.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return assertStatusCode(resp.StatusCode, http.StatusNoContent, http.StatusGone)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *httpMembersAPI) Leader(ctx context.Context) (*Member, error) {
|
|
||||||
req := &membersAPIActionLeader{}
|
|
||||||
resp, body, err := m.client.Do(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := assertStatusCode(resp.StatusCode, http.StatusOK); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var leader Member
|
|
||||||
if err := json.Unmarshal(body, &leader); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return &leader, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type membersAPIActionList struct{}
|
|
||||||
|
|
||||||
func (l *membersAPIActionList) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2MembersURL(ep)
|
|
||||||
req, _ := http.NewRequest("GET", u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type membersAPIActionRemove struct {
|
|
||||||
memberID string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *membersAPIActionRemove) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2MembersURL(ep)
|
|
||||||
u.Path = path.Join(u.Path, d.memberID)
|
|
||||||
req, _ := http.NewRequest("DELETE", u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type membersAPIActionAdd struct {
|
|
||||||
peerURLs types.URLs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *membersAPIActionAdd) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2MembersURL(ep)
|
|
||||||
m := memberCreateOrUpdateRequest{PeerURLs: a.peerURLs}
|
|
||||||
b, _ := json.Marshal(&m)
|
|
||||||
req, _ := http.NewRequest("POST", u.String(), bytes.NewReader(b))
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
type membersAPIActionUpdate struct {
|
|
||||||
memberID string
|
|
||||||
peerURLs types.URLs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *membersAPIActionUpdate) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2MembersURL(ep)
|
|
||||||
m := memberCreateOrUpdateRequest{PeerURLs: a.peerURLs}
|
|
||||||
u.Path = path.Join(u.Path, a.memberID)
|
|
||||||
b, _ := json.Marshal(&m)
|
|
||||||
req, _ := http.NewRequest("PUT", u.String(), bytes.NewReader(b))
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
func assertStatusCode(got int, want ...int) (err error) {
|
|
||||||
for _, w := range want {
|
|
||||||
if w == got {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return fmt.Errorf("unexpected status code %d", got)
|
|
||||||
}
|
|
||||||
|
|
||||||
type membersAPIActionLeader struct{}
|
|
||||||
|
|
||||||
func (l *membersAPIActionLeader) HTTPRequest(ep url.URL) *http.Request {
|
|
||||||
u := v2MembersURL(ep)
|
|
||||||
u.Path = path.Join(u.Path, defaultLeaderSuffix)
|
|
||||||
req, _ := http.NewRequest("GET", u.String(), nil)
|
|
||||||
return req
|
|
||||||
}
|
|
||||||
|
|
||||||
// v2MembersURL add the necessary path to the provided endpoint
|
|
||||||
// to route requests to the default v2 members API.
|
|
||||||
func v2MembersURL(ep url.URL) *url.URL {
|
|
||||||
ep.Path = path.Join(ep.Path, defaultV2MembersPrefix)
|
|
||||||
return &ep
|
|
||||||
}
|
|
||||||
|
|
||||||
type membersError struct {
|
|
||||||
Message string `json:"message"`
|
|
||||||
Code int `json:"-"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e membersError) Error() string {
|
|
||||||
return e.Message
|
|
||||||
}
|
|
53
vendor/github.com/coreos/etcd/client/util.go
generated
vendored
53
vendor/github.com/coreos/etcd/client/util.go
generated
vendored
@ -1,53 +0,0 @@
|
|||||||
// Copyright 2016 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
package client
|
|
||||||
|
|
||||||
import (
|
|
||||||
"regexp"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
roleNotFoundRegExp *regexp.Regexp
|
|
||||||
userNotFoundRegExp *regexp.Regexp
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
roleNotFoundRegExp = regexp.MustCompile("auth: Role .* does not exist.")
|
|
||||||
userNotFoundRegExp = regexp.MustCompile("auth: User .* does not exist.")
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsKeyNotFound returns true if the error code is ErrorCodeKeyNotFound.
|
|
||||||
func IsKeyNotFound(err error) bool {
|
|
||||||
if cErr, ok := err.(Error); ok {
|
|
||||||
return cErr.Code == ErrorCodeKeyNotFound
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsRoleNotFound returns true if the error means role not found of v2 API.
|
|
||||||
func IsRoleNotFound(err error) bool {
|
|
||||||
if ae, ok := err.(authError); ok {
|
|
||||||
return roleNotFoundRegExp.MatchString(ae.Message)
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsUserNotFound returns true if the error means user not found of v2 API.
|
|
||||||
func IsUserNotFound(err error) bool {
|
|
||||||
if ae, ok := err.(authError); ok {
|
|
||||||
return userNotFoundRegExp.MatchString(ae.Message)
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
87
vendor/github.com/coreos/etcd/clientv3/README.md
generated
vendored
Normal file
87
vendor/github.com/coreos/etcd/clientv3/README.md
generated
vendored
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
# etcd/clientv3
|
||||||
|
|
||||||
|
[![Docs](https://readthedocs.org/projects/etcd/badge/?version=latest&style=flat-square)](https://etcd.readthedocs.io/en/latest/?badge=latest)
|
||||||
|
[![Godoc](https://img.shields.io/badge/go-documentation-blue.svg?style=flat-square)](https://godoc.org/github.com/coreos/etcd/clientv3)
|
||||||
|
|
||||||
|
`etcd/clientv3` is the official Go etcd client for v3.
|
||||||
|
|
||||||
|
See https://etcd.readthedocs.io/en/latest for latest client architecture.
|
||||||
|
|
||||||
|
## Install
|
||||||
|
|
||||||
|
```bash
|
||||||
|
go get github.com/coreos/etcd/clientv3
|
||||||
|
```
|
||||||
|
|
||||||
|
## Get started
|
||||||
|
|
||||||
|
Create client using `clientv3.New`:
|
||||||
|
|
||||||
|
```go
|
||||||
|
cli, err := clientv3.New(clientv3.Config{
|
||||||
|
Endpoints: []string{"localhost:2379", "localhost:22379", "localhost:32379"},
|
||||||
|
DialTimeout: 5 * time.Second,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
// handle error!
|
||||||
|
}
|
||||||
|
defer cli.Close()
|
||||||
|
```
|
||||||
|
|
||||||
|
etcd v3 uses [`gRPC`](http://www.grpc.io) for remote procedure calls. And `clientv3` uses
|
||||||
|
[`grpc-go`](https://github.com/grpc/grpc-go) to connect to etcd. Make sure to close the client after using it.
|
||||||
|
If the client is not closed, the connection will have leaky goroutines. To specify client request timeout,
|
||||||
|
pass `context.WithTimeout` to APIs:
|
||||||
|
|
||||||
|
```go
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), timeout)
|
||||||
|
resp, err := cli.Put(ctx, "sample_key", "sample_value")
|
||||||
|
cancel()
|
||||||
|
if err != nil {
|
||||||
|
// handle error!
|
||||||
|
}
|
||||||
|
// use the response
|
||||||
|
```
|
||||||
|
|
||||||
|
For full compatibility, it is recommended to vendor builds using etcd's vendored packages, using tools like `golang/dep`, as in [vendor directories](https://golang.org/cmd/go/#hdr-Vendor_Directories).
|
||||||
|
|
||||||
|
## Error Handling
|
||||||
|
|
||||||
|
etcd client returns 2 types of errors:
|
||||||
|
|
||||||
|
1. context error: canceled or deadline exceeded.
|
||||||
|
2. gRPC error: see [api/v3rpc/rpctypes](https://godoc.org/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes).
|
||||||
|
|
||||||
|
Here is the example code to handle client errors:
|
||||||
|
|
||||||
|
```go
|
||||||
|
resp, err := cli.Put(ctx, "", "")
|
||||||
|
if err != nil {
|
||||||
|
switch err {
|
||||||
|
case context.Canceled:
|
||||||
|
log.Fatalf("ctx is canceled by another routine: %v", err)
|
||||||
|
case context.DeadlineExceeded:
|
||||||
|
log.Fatalf("ctx is attached with a deadline is exceeded: %v", err)
|
||||||
|
case rpctypes.ErrEmptyKey:
|
||||||
|
log.Fatalf("client-side error: %v", err)
|
||||||
|
default:
|
||||||
|
log.Fatalf("bad cluster endpoints, which are not etcd servers: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Metrics
|
||||||
|
|
||||||
|
The etcd client optionally exposes RPC metrics through [go-grpc-prometheus](https://github.com/grpc-ecosystem/go-grpc-prometheus). See the [examples](https://github.com/coreos/etcd/blob/master/clientv3/example_metrics_test.go).
|
||||||
|
|
||||||
|
## Namespacing
|
||||||
|
|
||||||
|
The [namespace](https://godoc.org/github.com/coreos/etcd/clientv3/namespace) package provides `clientv3` interface wrappers to transparently isolate client requests to a user-defined prefix.
|
||||||
|
|
||||||
|
## Request size limit
|
||||||
|
|
||||||
|
Client request size limit is configurable via `clientv3.Config.MaxCallSendMsgSize` and `MaxCallRecvMsgSize` in bytes. If none given, client request send limit defaults to 2 MiB including gRPC overhead bytes. And receive limit defaults to `math.MaxInt32`.
|
||||||
|
|
||||||
|
## Examples
|
||||||
|
|
||||||
|
More code examples can be found at [GoDoc](https://godoc.org/github.com/coreos/etcd/clientv3).
|
232
vendor/github.com/coreos/etcd/clientv3/auth.go
generated
vendored
Normal file
232
vendor/github.com/coreos/etcd/clientv3/auth.go
generated
vendored
Normal file
@ -0,0 +1,232 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/auth/authpb"
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
AuthEnableResponse pb.AuthEnableResponse
|
||||||
|
AuthDisableResponse pb.AuthDisableResponse
|
||||||
|
AuthenticateResponse pb.AuthenticateResponse
|
||||||
|
AuthUserAddResponse pb.AuthUserAddResponse
|
||||||
|
AuthUserDeleteResponse pb.AuthUserDeleteResponse
|
||||||
|
AuthUserChangePasswordResponse pb.AuthUserChangePasswordResponse
|
||||||
|
AuthUserGrantRoleResponse pb.AuthUserGrantRoleResponse
|
||||||
|
AuthUserGetResponse pb.AuthUserGetResponse
|
||||||
|
AuthUserRevokeRoleResponse pb.AuthUserRevokeRoleResponse
|
||||||
|
AuthRoleAddResponse pb.AuthRoleAddResponse
|
||||||
|
AuthRoleGrantPermissionResponse pb.AuthRoleGrantPermissionResponse
|
||||||
|
AuthRoleGetResponse pb.AuthRoleGetResponse
|
||||||
|
AuthRoleRevokePermissionResponse pb.AuthRoleRevokePermissionResponse
|
||||||
|
AuthRoleDeleteResponse pb.AuthRoleDeleteResponse
|
||||||
|
AuthUserListResponse pb.AuthUserListResponse
|
||||||
|
AuthRoleListResponse pb.AuthRoleListResponse
|
||||||
|
|
||||||
|
PermissionType authpb.Permission_Type
|
||||||
|
Permission authpb.Permission
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
PermRead = authpb.READ
|
||||||
|
PermWrite = authpb.WRITE
|
||||||
|
PermReadWrite = authpb.READWRITE
|
||||||
|
)
|
||||||
|
|
||||||
|
type Auth interface {
|
||||||
|
// AuthEnable enables auth of an etcd cluster.
|
||||||
|
AuthEnable(ctx context.Context) (*AuthEnableResponse, error)
|
||||||
|
|
||||||
|
// AuthDisable disables auth of an etcd cluster.
|
||||||
|
AuthDisable(ctx context.Context) (*AuthDisableResponse, error)
|
||||||
|
|
||||||
|
// UserAdd adds a new user to an etcd cluster.
|
||||||
|
UserAdd(ctx context.Context, name string, password string) (*AuthUserAddResponse, error)
|
||||||
|
|
||||||
|
// UserDelete deletes a user from an etcd cluster.
|
||||||
|
UserDelete(ctx context.Context, name string) (*AuthUserDeleteResponse, error)
|
||||||
|
|
||||||
|
// UserChangePassword changes a password of a user.
|
||||||
|
UserChangePassword(ctx context.Context, name string, password string) (*AuthUserChangePasswordResponse, error)
|
||||||
|
|
||||||
|
// UserGrantRole grants a role to a user.
|
||||||
|
UserGrantRole(ctx context.Context, user string, role string) (*AuthUserGrantRoleResponse, error)
|
||||||
|
|
||||||
|
// UserGet gets a detailed information of a user.
|
||||||
|
UserGet(ctx context.Context, name string) (*AuthUserGetResponse, error)
|
||||||
|
|
||||||
|
// UserList gets a list of all users.
|
||||||
|
UserList(ctx context.Context) (*AuthUserListResponse, error)
|
||||||
|
|
||||||
|
// UserRevokeRole revokes a role of a user.
|
||||||
|
UserRevokeRole(ctx context.Context, name string, role string) (*AuthUserRevokeRoleResponse, error)
|
||||||
|
|
||||||
|
// RoleAdd adds a new role to an etcd cluster.
|
||||||
|
RoleAdd(ctx context.Context, name string) (*AuthRoleAddResponse, error)
|
||||||
|
|
||||||
|
// RoleGrantPermission grants a permission to a role.
|
||||||
|
RoleGrantPermission(ctx context.Context, name string, key, rangeEnd string, permType PermissionType) (*AuthRoleGrantPermissionResponse, error)
|
||||||
|
|
||||||
|
// RoleGet gets a detailed information of a role.
|
||||||
|
RoleGet(ctx context.Context, role string) (*AuthRoleGetResponse, error)
|
||||||
|
|
||||||
|
// RoleList gets a list of all roles.
|
||||||
|
RoleList(ctx context.Context) (*AuthRoleListResponse, error)
|
||||||
|
|
||||||
|
// RoleRevokePermission revokes a permission from a role.
|
||||||
|
RoleRevokePermission(ctx context.Context, role string, key, rangeEnd string) (*AuthRoleRevokePermissionResponse, error)
|
||||||
|
|
||||||
|
// RoleDelete deletes a role.
|
||||||
|
RoleDelete(ctx context.Context, role string) (*AuthRoleDeleteResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type authClient struct {
|
||||||
|
remote pb.AuthClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewAuth(c *Client) Auth {
|
||||||
|
api := &authClient{remote: RetryAuthClient(c)}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) AuthEnable(ctx context.Context) (*AuthEnableResponse, error) {
|
||||||
|
resp, err := auth.remote.AuthEnable(ctx, &pb.AuthEnableRequest{}, auth.callOpts...)
|
||||||
|
return (*AuthEnableResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) AuthDisable(ctx context.Context) (*AuthDisableResponse, error) {
|
||||||
|
resp, err := auth.remote.AuthDisable(ctx, &pb.AuthDisableRequest{}, auth.callOpts...)
|
||||||
|
return (*AuthDisableResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserAdd(ctx context.Context, name string, password string) (*AuthUserAddResponse, error) {
|
||||||
|
resp, err := auth.remote.UserAdd(ctx, &pb.AuthUserAddRequest{Name: name, Password: password}, auth.callOpts...)
|
||||||
|
return (*AuthUserAddResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserDelete(ctx context.Context, name string) (*AuthUserDeleteResponse, error) {
|
||||||
|
resp, err := auth.remote.UserDelete(ctx, &pb.AuthUserDeleteRequest{Name: name}, auth.callOpts...)
|
||||||
|
return (*AuthUserDeleteResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserChangePassword(ctx context.Context, name string, password string) (*AuthUserChangePasswordResponse, error) {
|
||||||
|
resp, err := auth.remote.UserChangePassword(ctx, &pb.AuthUserChangePasswordRequest{Name: name, Password: password}, auth.callOpts...)
|
||||||
|
return (*AuthUserChangePasswordResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserGrantRole(ctx context.Context, user string, role string) (*AuthUserGrantRoleResponse, error) {
|
||||||
|
resp, err := auth.remote.UserGrantRole(ctx, &pb.AuthUserGrantRoleRequest{User: user, Role: role}, auth.callOpts...)
|
||||||
|
return (*AuthUserGrantRoleResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserGet(ctx context.Context, name string) (*AuthUserGetResponse, error) {
|
||||||
|
resp, err := auth.remote.UserGet(ctx, &pb.AuthUserGetRequest{Name: name}, auth.callOpts...)
|
||||||
|
return (*AuthUserGetResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserList(ctx context.Context) (*AuthUserListResponse, error) {
|
||||||
|
resp, err := auth.remote.UserList(ctx, &pb.AuthUserListRequest{}, auth.callOpts...)
|
||||||
|
return (*AuthUserListResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) UserRevokeRole(ctx context.Context, name string, role string) (*AuthUserRevokeRoleResponse, error) {
|
||||||
|
resp, err := auth.remote.UserRevokeRole(ctx, &pb.AuthUserRevokeRoleRequest{Name: name, Role: role}, auth.callOpts...)
|
||||||
|
return (*AuthUserRevokeRoleResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) RoleAdd(ctx context.Context, name string) (*AuthRoleAddResponse, error) {
|
||||||
|
resp, err := auth.remote.RoleAdd(ctx, &pb.AuthRoleAddRequest{Name: name}, auth.callOpts...)
|
||||||
|
return (*AuthRoleAddResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) RoleGrantPermission(ctx context.Context, name string, key, rangeEnd string, permType PermissionType) (*AuthRoleGrantPermissionResponse, error) {
|
||||||
|
perm := &authpb.Permission{
|
||||||
|
Key: []byte(key),
|
||||||
|
RangeEnd: []byte(rangeEnd),
|
||||||
|
PermType: authpb.Permission_Type(permType),
|
||||||
|
}
|
||||||
|
resp, err := auth.remote.RoleGrantPermission(ctx, &pb.AuthRoleGrantPermissionRequest{Name: name, Perm: perm}, auth.callOpts...)
|
||||||
|
return (*AuthRoleGrantPermissionResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) RoleGet(ctx context.Context, role string) (*AuthRoleGetResponse, error) {
|
||||||
|
resp, err := auth.remote.RoleGet(ctx, &pb.AuthRoleGetRequest{Role: role}, auth.callOpts...)
|
||||||
|
return (*AuthRoleGetResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) RoleList(ctx context.Context) (*AuthRoleListResponse, error) {
|
||||||
|
resp, err := auth.remote.RoleList(ctx, &pb.AuthRoleListRequest{}, auth.callOpts...)
|
||||||
|
return (*AuthRoleListResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) RoleRevokePermission(ctx context.Context, role string, key, rangeEnd string) (*AuthRoleRevokePermissionResponse, error) {
|
||||||
|
resp, err := auth.remote.RoleRevokePermission(ctx, &pb.AuthRoleRevokePermissionRequest{Role: role, Key: []byte(key), RangeEnd: []byte(rangeEnd)}, auth.callOpts...)
|
||||||
|
return (*AuthRoleRevokePermissionResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authClient) RoleDelete(ctx context.Context, role string) (*AuthRoleDeleteResponse, error) {
|
||||||
|
resp, err := auth.remote.RoleDelete(ctx, &pb.AuthRoleDeleteRequest{Role: role}, auth.callOpts...)
|
||||||
|
return (*AuthRoleDeleteResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func StrToPermissionType(s string) (PermissionType, error) {
|
||||||
|
val, ok := authpb.Permission_Type_value[strings.ToUpper(s)]
|
||||||
|
if ok {
|
||||||
|
return PermissionType(val), nil
|
||||||
|
}
|
||||||
|
return PermissionType(-1), fmt.Errorf("invalid permission type: %s", s)
|
||||||
|
}
|
||||||
|
|
||||||
|
type authenticator struct {
|
||||||
|
conn *grpc.ClientConn // conn in-use
|
||||||
|
remote pb.AuthClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authenticator) authenticate(ctx context.Context, name string, password string) (*AuthenticateResponse, error) {
|
||||||
|
resp, err := auth.remote.Authenticate(ctx, &pb.AuthenticateRequest{Name: name, Password: password}, auth.callOpts...)
|
||||||
|
return (*AuthenticateResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (auth *authenticator) close() {
|
||||||
|
auth.conn.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func newAuthenticator(ctx context.Context, target string, opts []grpc.DialOption, c *Client) (*authenticator, error) {
|
||||||
|
conn, err := grpc.DialContext(ctx, target, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
api := &authenticator{
|
||||||
|
conn: conn,
|
||||||
|
remote: pb.NewAuthClient(conn),
|
||||||
|
}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api, nil
|
||||||
|
}
|
275
vendor/github.com/coreos/etcd/clientv3/balancer/balancer.go
generated
vendored
Normal file
275
vendor/github.com/coreos/etcd/clientv3/balancer/balancer.go
generated
vendored
Normal file
@ -0,0 +1,275 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package balancer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/clientv3/balancer/picker"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"google.golang.org/grpc/balancer"
|
||||||
|
"google.golang.org/grpc/connectivity"
|
||||||
|
"google.golang.org/grpc/resolver"
|
||||||
|
_ "google.golang.org/grpc/resolver/dns" // register DNS resolver
|
||||||
|
_ "google.golang.org/grpc/resolver/passthrough" // register passthrough resolver
|
||||||
|
)
|
||||||
|
|
||||||
|
// RegisterBuilder creates and registers a builder. Since this function calls balancer.Register, it
|
||||||
|
// must be invoked at initialization time.
|
||||||
|
func RegisterBuilder(cfg Config) {
|
||||||
|
bb := &builder{cfg}
|
||||||
|
balancer.Register(bb)
|
||||||
|
|
||||||
|
bb.cfg.Logger.Info(
|
||||||
|
"registered balancer",
|
||||||
|
zap.String("policy", bb.cfg.Policy.String()),
|
||||||
|
zap.String("name", bb.cfg.Name),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type builder struct {
|
||||||
|
cfg Config
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build is called initially when creating "ccBalancerWrapper".
|
||||||
|
// "grpc.Dial" is called to this client connection.
|
||||||
|
// Then, resolved addresses will be handled via "HandleResolvedAddrs".
|
||||||
|
func (b *builder) Build(cc balancer.ClientConn, opt balancer.BuildOptions) balancer.Balancer {
|
||||||
|
bb := &baseBalancer{
|
||||||
|
id: strconv.FormatInt(time.Now().UnixNano(), 36),
|
||||||
|
policy: b.cfg.Policy,
|
||||||
|
name: b.cfg.Policy.String(),
|
||||||
|
lg: b.cfg.Logger,
|
||||||
|
|
||||||
|
addrToSc: make(map[resolver.Address]balancer.SubConn),
|
||||||
|
scToAddr: make(map[balancer.SubConn]resolver.Address),
|
||||||
|
scToSt: make(map[balancer.SubConn]connectivity.State),
|
||||||
|
|
||||||
|
currentConn: nil,
|
||||||
|
csEvltr: &connectivityStateEvaluator{},
|
||||||
|
|
||||||
|
// initialize picker always returns "ErrNoSubConnAvailable"
|
||||||
|
Picker: picker.NewErr(balancer.ErrNoSubConnAvailable),
|
||||||
|
}
|
||||||
|
if b.cfg.Name != "" {
|
||||||
|
bb.name = b.cfg.Name
|
||||||
|
}
|
||||||
|
if bb.lg == nil {
|
||||||
|
bb.lg = zap.NewNop()
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: support multiple connections
|
||||||
|
bb.mu.Lock()
|
||||||
|
bb.currentConn = cc
|
||||||
|
bb.mu.Unlock()
|
||||||
|
|
||||||
|
bb.lg.Info(
|
||||||
|
"built balancer",
|
||||||
|
zap.String("balancer-id", bb.id),
|
||||||
|
zap.String("policy", bb.policy.String()),
|
||||||
|
zap.String("resolver-target", cc.Target()),
|
||||||
|
)
|
||||||
|
return bb
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name implements "grpc/balancer.Builder" interface.
|
||||||
|
func (b *builder) Name() string { return b.cfg.Name }
|
||||||
|
|
||||||
|
// Balancer defines client balancer interface.
|
||||||
|
type Balancer interface {
|
||||||
|
// Balancer is called on specified client connection. Client initiates gRPC
|
||||||
|
// connection with "grpc.Dial(addr, grpc.WithBalancerName)", and then those resolved
|
||||||
|
// addresses are passed to "grpc/balancer.Balancer.HandleResolvedAddrs".
|
||||||
|
// For each resolved address, balancer calls "balancer.ClientConn.NewSubConn".
|
||||||
|
// "grpc/balancer.Balancer.HandleSubConnStateChange" is called when connectivity state
|
||||||
|
// changes, thus requires failover logic in this method.
|
||||||
|
balancer.Balancer
|
||||||
|
|
||||||
|
// Picker calls "Pick" for every client request.
|
||||||
|
picker.Picker
|
||||||
|
}
|
||||||
|
|
||||||
|
type baseBalancer struct {
|
||||||
|
id string
|
||||||
|
policy picker.Policy
|
||||||
|
name string
|
||||||
|
lg *zap.Logger
|
||||||
|
|
||||||
|
mu sync.RWMutex
|
||||||
|
|
||||||
|
addrToSc map[resolver.Address]balancer.SubConn
|
||||||
|
scToAddr map[balancer.SubConn]resolver.Address
|
||||||
|
scToSt map[balancer.SubConn]connectivity.State
|
||||||
|
|
||||||
|
currentConn balancer.ClientConn
|
||||||
|
currentState connectivity.State
|
||||||
|
csEvltr *connectivityStateEvaluator
|
||||||
|
|
||||||
|
picker.Picker
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleResolvedAddrs implements "grpc/balancer.Balancer" interface.
|
||||||
|
// gRPC sends initial or updated resolved addresses from "Build".
|
||||||
|
func (bb *baseBalancer) HandleResolvedAddrs(addrs []resolver.Address, err error) {
|
||||||
|
if err != nil {
|
||||||
|
bb.lg.Warn("HandleResolvedAddrs called with error", zap.String("balancer-id", bb.id), zap.Error(err))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bb.lg.Info("resolved", zap.String("balancer-id", bb.id), zap.Strings("addresses", addrsToStrings(addrs)))
|
||||||
|
|
||||||
|
bb.mu.Lock()
|
||||||
|
defer bb.mu.Unlock()
|
||||||
|
|
||||||
|
resolved := make(map[resolver.Address]struct{})
|
||||||
|
for _, addr := range addrs {
|
||||||
|
resolved[addr] = struct{}{}
|
||||||
|
if _, ok := bb.addrToSc[addr]; !ok {
|
||||||
|
sc, err := bb.currentConn.NewSubConn([]resolver.Address{addr}, balancer.NewSubConnOptions{})
|
||||||
|
if err != nil {
|
||||||
|
bb.lg.Warn("NewSubConn failed", zap.String("balancer-id", bb.id), zap.Error(err), zap.String("address", addr.Addr))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
bb.addrToSc[addr] = sc
|
||||||
|
bb.scToAddr[sc] = addr
|
||||||
|
bb.scToSt[sc] = connectivity.Idle
|
||||||
|
sc.Connect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for addr, sc := range bb.addrToSc {
|
||||||
|
if _, ok := resolved[addr]; !ok {
|
||||||
|
// was removed by resolver or failed to create subconn
|
||||||
|
bb.currentConn.RemoveSubConn(sc)
|
||||||
|
delete(bb.addrToSc, addr)
|
||||||
|
|
||||||
|
bb.lg.Info(
|
||||||
|
"removed subconn",
|
||||||
|
zap.String("balancer-id", bb.id),
|
||||||
|
zap.String("address", addr.Addr),
|
||||||
|
zap.String("subconn", scToString(sc)),
|
||||||
|
)
|
||||||
|
|
||||||
|
// Keep the state of this sc in bb.scToSt until sc's state becomes Shutdown.
|
||||||
|
// The entry will be deleted in HandleSubConnStateChange.
|
||||||
|
// (DO NOT) delete(bb.scToAddr, sc)
|
||||||
|
// (DO NOT) delete(bb.scToSt, sc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleSubConnStateChange implements "grpc/balancer.Balancer" interface.
|
||||||
|
func (bb *baseBalancer) HandleSubConnStateChange(sc balancer.SubConn, s connectivity.State) {
|
||||||
|
bb.mu.Lock()
|
||||||
|
defer bb.mu.Unlock()
|
||||||
|
|
||||||
|
old, ok := bb.scToSt[sc]
|
||||||
|
if !ok {
|
||||||
|
bb.lg.Warn(
|
||||||
|
"state change for an unknown subconn",
|
||||||
|
zap.String("balancer-id", bb.id),
|
||||||
|
zap.String("subconn", scToString(sc)),
|
||||||
|
zap.String("state", s.String()),
|
||||||
|
)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
bb.lg.Info(
|
||||||
|
"state changed",
|
||||||
|
zap.String("balancer-id", bb.id),
|
||||||
|
zap.Bool("connected", s == connectivity.Ready),
|
||||||
|
zap.String("subconn", scToString(sc)),
|
||||||
|
zap.String("address", bb.scToAddr[sc].Addr),
|
||||||
|
zap.String("old-state", old.String()),
|
||||||
|
zap.String("new-state", s.String()),
|
||||||
|
)
|
||||||
|
|
||||||
|
bb.scToSt[sc] = s
|
||||||
|
switch s {
|
||||||
|
case connectivity.Idle:
|
||||||
|
sc.Connect()
|
||||||
|
case connectivity.Shutdown:
|
||||||
|
// When an address was removed by resolver, b called RemoveSubConn but
|
||||||
|
// kept the sc's state in scToSt. Remove state for this sc here.
|
||||||
|
delete(bb.scToAddr, sc)
|
||||||
|
delete(bb.scToSt, sc)
|
||||||
|
}
|
||||||
|
|
||||||
|
oldAggrState := bb.currentState
|
||||||
|
bb.currentState = bb.csEvltr.recordTransition(old, s)
|
||||||
|
|
||||||
|
// Regenerate picker when one of the following happens:
|
||||||
|
// - this sc became ready from not-ready
|
||||||
|
// - this sc became not-ready from ready
|
||||||
|
// - the aggregated state of balancer became TransientFailure from non-TransientFailure
|
||||||
|
// - the aggregated state of balancer became non-TransientFailure from TransientFailure
|
||||||
|
if (s == connectivity.Ready) != (old == connectivity.Ready) ||
|
||||||
|
(bb.currentState == connectivity.TransientFailure) != (oldAggrState == connectivity.TransientFailure) {
|
||||||
|
bb.regeneratePicker()
|
||||||
|
}
|
||||||
|
|
||||||
|
bb.currentConn.UpdateBalancerState(bb.currentState, bb.Picker)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bb *baseBalancer) regeneratePicker() {
|
||||||
|
if bb.currentState == connectivity.TransientFailure {
|
||||||
|
bb.lg.Info(
|
||||||
|
"generated transient error picker",
|
||||||
|
zap.String("balancer-id", bb.id),
|
||||||
|
zap.String("policy", bb.policy.String()),
|
||||||
|
)
|
||||||
|
bb.Picker = picker.NewErr(balancer.ErrTransientFailure)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// only pass ready subconns to picker
|
||||||
|
scs := make([]balancer.SubConn, 0)
|
||||||
|
addrToSc := make(map[resolver.Address]balancer.SubConn)
|
||||||
|
scToAddr := make(map[balancer.SubConn]resolver.Address)
|
||||||
|
for addr, sc := range bb.addrToSc {
|
||||||
|
if st, ok := bb.scToSt[sc]; ok && st == connectivity.Ready {
|
||||||
|
scs = append(scs, sc)
|
||||||
|
addrToSc[addr] = sc
|
||||||
|
scToAddr[sc] = addr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch bb.policy {
|
||||||
|
case picker.RoundrobinBalanced:
|
||||||
|
bb.Picker = picker.NewRoundrobinBalanced(bb.lg, scs, addrToSc, scToAddr)
|
||||||
|
|
||||||
|
default:
|
||||||
|
panic(fmt.Errorf("invalid balancer picker policy (%d)", bb.policy))
|
||||||
|
}
|
||||||
|
|
||||||
|
bb.lg.Info(
|
||||||
|
"generated picker",
|
||||||
|
zap.String("balancer-id", bb.id),
|
||||||
|
zap.String("policy", bb.policy.String()),
|
||||||
|
zap.Strings("subconn-ready", scsToStrings(addrToSc)),
|
||||||
|
zap.Int("subconn-size", len(addrToSc)),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close implements "grpc/balancer.Balancer" interface.
|
||||||
|
// Close is a nop because base balancer doesn't have internal state to clean up,
|
||||||
|
// and it doesn't need to call RemoveSubConn for the SubConns.
|
||||||
|
func (bb *baseBalancer) Close() {
|
||||||
|
// TODO
|
||||||
|
}
|
36
vendor/github.com/coreos/etcd/clientv3/balancer/config.go
generated
vendored
Normal file
36
vendor/github.com/coreos/etcd/clientv3/balancer/config.go
generated
vendored
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package balancer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/coreos/etcd/clientv3/balancer/picker"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Config defines balancer configurations.
|
||||||
|
type Config struct {
|
||||||
|
// Policy configures balancer policy.
|
||||||
|
Policy picker.Policy
|
||||||
|
|
||||||
|
// Name defines an additional name for balancer.
|
||||||
|
// Useful for balancer testing to avoid register conflicts.
|
||||||
|
// If empty, defaults to policy name.
|
||||||
|
Name string
|
||||||
|
|
||||||
|
// Logger configures balancer logging.
|
||||||
|
// If nil, logs are discarded.
|
||||||
|
Logger *zap.Logger
|
||||||
|
}
|
58
vendor/github.com/coreos/etcd/clientv3/balancer/connectivity.go
generated
vendored
Normal file
58
vendor/github.com/coreos/etcd/clientv3/balancer/connectivity.go
generated
vendored
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package balancer
|
||||||
|
|
||||||
|
import "google.golang.org/grpc/connectivity"
|
||||||
|
|
||||||
|
// connectivityStateEvaluator gets updated by addrConns when their
|
||||||
|
// states transition, based on which it evaluates the state of
|
||||||
|
// ClientConn.
|
||||||
|
type connectivityStateEvaluator struct {
|
||||||
|
numReady uint64 // Number of addrConns in ready state.
|
||||||
|
numConnecting uint64 // Number of addrConns in connecting state.
|
||||||
|
numTransientFailure uint64 // Number of addrConns in transientFailure.
|
||||||
|
}
|
||||||
|
|
||||||
|
// recordTransition records state change happening in every subConn and based on
|
||||||
|
// that it evaluates what aggregated state should be.
|
||||||
|
// It can only transition between Ready, Connecting and TransientFailure. Other states,
|
||||||
|
// Idle and Shutdown are transitioned into by ClientConn; in the beginning of the connection
|
||||||
|
// before any subConn is created ClientConn is in idle state. In the end when ClientConn
|
||||||
|
// closes it is in Shutdown state.
|
||||||
|
//
|
||||||
|
// recordTransition should only be called synchronously from the same goroutine.
|
||||||
|
func (cse *connectivityStateEvaluator) recordTransition(oldState, newState connectivity.State) connectivity.State {
|
||||||
|
// Update counters.
|
||||||
|
for idx, state := range []connectivity.State{oldState, newState} {
|
||||||
|
updateVal := 2*uint64(idx) - 1 // -1 for oldState and +1 for new.
|
||||||
|
switch state {
|
||||||
|
case connectivity.Ready:
|
||||||
|
cse.numReady += updateVal
|
||||||
|
case connectivity.Connecting:
|
||||||
|
cse.numConnecting += updateVal
|
||||||
|
case connectivity.TransientFailure:
|
||||||
|
cse.numTransientFailure += updateVal
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Evaluate.
|
||||||
|
if cse.numReady > 0 {
|
||||||
|
return connectivity.Ready
|
||||||
|
}
|
||||||
|
if cse.numConnecting > 0 {
|
||||||
|
return connectivity.Connecting
|
||||||
|
}
|
||||||
|
return connectivity.TransientFailure
|
||||||
|
}
|
16
vendor/github.com/coreos/etcd/clientv3/balancer/doc.go
generated
vendored
Normal file
16
vendor/github.com/coreos/etcd/clientv3/balancer/doc.go
generated
vendored
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package balancer implements client balancer.
|
||||||
|
package balancer
|
657
vendor/github.com/coreos/etcd/clientv3/balancer/grpc1.7-health.go
generated
vendored
Normal file
657
vendor/github.com/coreos/etcd/clientv3/balancer/grpc1.7-health.go
generated
vendored
Normal file
@ -0,0 +1,657 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package balancer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/grpclog"
|
||||||
|
healthpb "google.golang.org/grpc/health/grpc_health_v1"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TODO: replace with something better
|
||||||
|
var lg = grpclog.NewLoggerV2(ioutil.Discard, ioutil.Discard, ioutil.Discard)
|
||||||
|
|
||||||
|
const (
|
||||||
|
minHealthRetryDuration = 3 * time.Second
|
||||||
|
unknownService = "unknown service grpc.health.v1.Health"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ErrNoAddrAvilable is returned by Get() when the balancer does not have
|
||||||
|
// any active connection to endpoints at the time.
|
||||||
|
// This error is returned only when opts.BlockingWait is true.
|
||||||
|
var ErrNoAddrAvilable = status.Error(codes.Unavailable, "there is no address available")
|
||||||
|
|
||||||
|
type NotifyMsg int
|
||||||
|
|
||||||
|
const (
|
||||||
|
NotifyReset NotifyMsg = iota
|
||||||
|
NotifyNext
|
||||||
|
)
|
||||||
|
|
||||||
|
// GRPC17Health does the bare minimum to expose multiple eps
|
||||||
|
// to the grpc reconnection code path
|
||||||
|
type GRPC17Health struct {
|
||||||
|
// addrs are the client's endpoint addresses for grpc
|
||||||
|
addrs []grpc.Address
|
||||||
|
|
||||||
|
// eps holds the raw endpoints from the client
|
||||||
|
eps []string
|
||||||
|
|
||||||
|
// notifyCh notifies grpc of the set of addresses for connecting
|
||||||
|
notifyCh chan []grpc.Address
|
||||||
|
|
||||||
|
// readyc closes once the first connection is up
|
||||||
|
readyc chan struct{}
|
||||||
|
readyOnce sync.Once
|
||||||
|
|
||||||
|
// healthCheck checks an endpoint's health.
|
||||||
|
healthCheck func(ep string) (bool, error)
|
||||||
|
healthCheckTimeout time.Duration
|
||||||
|
|
||||||
|
unhealthyMu sync.RWMutex
|
||||||
|
unhealthyHostPorts map[string]time.Time
|
||||||
|
|
||||||
|
// mu protects all fields below.
|
||||||
|
mu sync.RWMutex
|
||||||
|
|
||||||
|
// upc closes when pinAddr transitions from empty to non-empty or the balancer closes.
|
||||||
|
upc chan struct{}
|
||||||
|
|
||||||
|
// downc closes when grpc calls down() on pinAddr
|
||||||
|
downc chan struct{}
|
||||||
|
|
||||||
|
// stopc is closed to signal updateNotifyLoop should stop.
|
||||||
|
stopc chan struct{}
|
||||||
|
stopOnce sync.Once
|
||||||
|
wg sync.WaitGroup
|
||||||
|
|
||||||
|
// donec closes when all goroutines are exited
|
||||||
|
donec chan struct{}
|
||||||
|
|
||||||
|
// updateAddrsC notifies updateNotifyLoop to update addrs.
|
||||||
|
updateAddrsC chan NotifyMsg
|
||||||
|
|
||||||
|
// grpc issues TLS cert checks using the string passed into dial so
|
||||||
|
// that string must be the host. To recover the full scheme://host URL,
|
||||||
|
// have a map from hosts to the original endpoint.
|
||||||
|
hostPort2ep map[string]string
|
||||||
|
|
||||||
|
// pinAddr is the currently pinned address; set to the empty string on
|
||||||
|
// initialization and shutdown.
|
||||||
|
pinAddr string
|
||||||
|
|
||||||
|
closed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// DialFunc defines gRPC dial function.
|
||||||
|
type DialFunc func(ep string, dopts ...grpc.DialOption) (*grpc.ClientConn, error)
|
||||||
|
|
||||||
|
// NewGRPC17Health returns a new health balancer with gRPC v1.7.
|
||||||
|
func NewGRPC17Health(
|
||||||
|
eps []string,
|
||||||
|
timeout time.Duration,
|
||||||
|
dialFunc DialFunc,
|
||||||
|
) *GRPC17Health {
|
||||||
|
notifyCh := make(chan []grpc.Address)
|
||||||
|
addrs := eps2addrs(eps)
|
||||||
|
hb := &GRPC17Health{
|
||||||
|
addrs: addrs,
|
||||||
|
eps: eps,
|
||||||
|
notifyCh: notifyCh,
|
||||||
|
readyc: make(chan struct{}),
|
||||||
|
healthCheck: func(ep string) (bool, error) { return grpcHealthCheck(ep, dialFunc) },
|
||||||
|
unhealthyHostPorts: make(map[string]time.Time),
|
||||||
|
upc: make(chan struct{}),
|
||||||
|
stopc: make(chan struct{}),
|
||||||
|
downc: make(chan struct{}),
|
||||||
|
donec: make(chan struct{}),
|
||||||
|
updateAddrsC: make(chan NotifyMsg),
|
||||||
|
hostPort2ep: getHostPort2ep(eps),
|
||||||
|
}
|
||||||
|
if timeout < minHealthRetryDuration {
|
||||||
|
timeout = minHealthRetryDuration
|
||||||
|
}
|
||||||
|
hb.healthCheckTimeout = timeout
|
||||||
|
|
||||||
|
close(hb.downc)
|
||||||
|
go hb.updateNotifyLoop()
|
||||||
|
hb.wg.Add(1)
|
||||||
|
go func() {
|
||||||
|
defer hb.wg.Done()
|
||||||
|
hb.updateUnhealthy()
|
||||||
|
}()
|
||||||
|
return hb
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Start(target string, config grpc.BalancerConfig) error { return nil }
|
||||||
|
|
||||||
|
func (b *GRPC17Health) ConnectNotify() <-chan struct{} {
|
||||||
|
b.mu.Lock()
|
||||||
|
defer b.mu.Unlock()
|
||||||
|
return b.upc
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) UpdateAddrsC() chan NotifyMsg { return b.updateAddrsC }
|
||||||
|
func (b *GRPC17Health) StopC() chan struct{} { return b.stopc }
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Ready() <-chan struct{} { return b.readyc }
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Endpoint(hostPort string) string {
|
||||||
|
b.mu.RLock()
|
||||||
|
defer b.mu.RUnlock()
|
||||||
|
return b.hostPort2ep[hostPort]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Pinned() string {
|
||||||
|
b.mu.RLock()
|
||||||
|
defer b.mu.RUnlock()
|
||||||
|
return b.pinAddr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) HostPortError(hostPort string, err error) {
|
||||||
|
if b.Endpoint(hostPort) == "" {
|
||||||
|
lg.Infof("clientv3/balancer: %q is stale (skip marking as unhealthy on %q)", hostPort, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
b.unhealthyMu.Lock()
|
||||||
|
b.unhealthyHostPorts[hostPort] = time.Now()
|
||||||
|
b.unhealthyMu.Unlock()
|
||||||
|
lg.Infof("clientv3/balancer: %q is marked unhealthy (%q)", hostPort, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) removeUnhealthy(hostPort, msg string) {
|
||||||
|
if b.Endpoint(hostPort) == "" {
|
||||||
|
lg.Infof("clientv3/balancer: %q was not in unhealthy (%q)", hostPort, msg)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
b.unhealthyMu.Lock()
|
||||||
|
delete(b.unhealthyHostPorts, hostPort)
|
||||||
|
b.unhealthyMu.Unlock()
|
||||||
|
lg.Infof("clientv3/balancer: %q is removed from unhealthy (%q)", hostPort, msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) countUnhealthy() (count int) {
|
||||||
|
b.unhealthyMu.RLock()
|
||||||
|
count = len(b.unhealthyHostPorts)
|
||||||
|
b.unhealthyMu.RUnlock()
|
||||||
|
return count
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) isUnhealthy(hostPort string) (unhealthy bool) {
|
||||||
|
b.unhealthyMu.RLock()
|
||||||
|
_, unhealthy = b.unhealthyHostPorts[hostPort]
|
||||||
|
b.unhealthyMu.RUnlock()
|
||||||
|
return unhealthy
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) cleanupUnhealthy() {
|
||||||
|
b.unhealthyMu.Lock()
|
||||||
|
for k, v := range b.unhealthyHostPorts {
|
||||||
|
if time.Since(v) > b.healthCheckTimeout {
|
||||||
|
delete(b.unhealthyHostPorts, k)
|
||||||
|
lg.Infof("clientv3/balancer: removed %q from unhealthy after %v", k, b.healthCheckTimeout)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.unhealthyMu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) liveAddrs() ([]grpc.Address, map[string]struct{}) {
|
||||||
|
unhealthyCnt := b.countUnhealthy()
|
||||||
|
|
||||||
|
b.mu.RLock()
|
||||||
|
defer b.mu.RUnlock()
|
||||||
|
|
||||||
|
hbAddrs := b.addrs
|
||||||
|
if len(b.addrs) == 1 || unhealthyCnt == 0 || unhealthyCnt == len(b.addrs) {
|
||||||
|
liveHostPorts := make(map[string]struct{}, len(b.hostPort2ep))
|
||||||
|
for k := range b.hostPort2ep {
|
||||||
|
liveHostPorts[k] = struct{}{}
|
||||||
|
}
|
||||||
|
return hbAddrs, liveHostPorts
|
||||||
|
}
|
||||||
|
|
||||||
|
addrs := make([]grpc.Address, 0, len(b.addrs)-unhealthyCnt)
|
||||||
|
liveHostPorts := make(map[string]struct{}, len(addrs))
|
||||||
|
for _, addr := range b.addrs {
|
||||||
|
if !b.isUnhealthy(addr.Addr) {
|
||||||
|
addrs = append(addrs, addr)
|
||||||
|
liveHostPorts[addr.Addr] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return addrs, liveHostPorts
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) updateUnhealthy() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-time.After(b.healthCheckTimeout):
|
||||||
|
b.cleanupUnhealthy()
|
||||||
|
pinned := b.Pinned()
|
||||||
|
if pinned == "" || b.isUnhealthy(pinned) {
|
||||||
|
select {
|
||||||
|
case b.updateAddrsC <- NotifyNext:
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NeedUpdate returns true if all connections are down or
|
||||||
|
// addresses do not include current pinned address.
|
||||||
|
func (b *GRPC17Health) NeedUpdate() bool {
|
||||||
|
// updating notifyCh can trigger new connections,
|
||||||
|
// need update addrs if all connections are down
|
||||||
|
// or addrs does not include pinAddr.
|
||||||
|
b.mu.RLock()
|
||||||
|
update := !hasAddr(b.addrs, b.pinAddr)
|
||||||
|
b.mu.RUnlock()
|
||||||
|
return update
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) UpdateAddrs(eps ...string) {
|
||||||
|
np := getHostPort2ep(eps)
|
||||||
|
|
||||||
|
b.mu.Lock()
|
||||||
|
defer b.mu.Unlock()
|
||||||
|
|
||||||
|
match := len(np) == len(b.hostPort2ep)
|
||||||
|
if match {
|
||||||
|
for k, v := range np {
|
||||||
|
if b.hostPort2ep[k] != v {
|
||||||
|
match = false
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if match {
|
||||||
|
// same endpoints, so no need to update address
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
b.hostPort2ep = np
|
||||||
|
b.addrs, b.eps = eps2addrs(eps), eps
|
||||||
|
|
||||||
|
b.unhealthyMu.Lock()
|
||||||
|
b.unhealthyHostPorts = make(map[string]time.Time)
|
||||||
|
b.unhealthyMu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Next() {
|
||||||
|
b.mu.RLock()
|
||||||
|
downc := b.downc
|
||||||
|
b.mu.RUnlock()
|
||||||
|
select {
|
||||||
|
case b.updateAddrsC <- NotifyNext:
|
||||||
|
case <-b.stopc:
|
||||||
|
}
|
||||||
|
// wait until disconnect so new RPCs are not issued on old connection
|
||||||
|
select {
|
||||||
|
case <-downc:
|
||||||
|
case <-b.stopc:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) updateNotifyLoop() {
|
||||||
|
defer close(b.donec)
|
||||||
|
|
||||||
|
for {
|
||||||
|
b.mu.RLock()
|
||||||
|
upc, downc, addr := b.upc, b.downc, b.pinAddr
|
||||||
|
b.mu.RUnlock()
|
||||||
|
// downc or upc should be closed
|
||||||
|
select {
|
||||||
|
case <-downc:
|
||||||
|
downc = nil
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case <-upc:
|
||||||
|
upc = nil
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
switch {
|
||||||
|
case downc == nil && upc == nil:
|
||||||
|
// stale
|
||||||
|
select {
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
case downc == nil:
|
||||||
|
b.notifyAddrs(NotifyReset)
|
||||||
|
select {
|
||||||
|
case <-upc:
|
||||||
|
case msg := <-b.updateAddrsC:
|
||||||
|
b.notifyAddrs(msg)
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
case upc == nil:
|
||||||
|
select {
|
||||||
|
// close connections that are not the pinned address
|
||||||
|
case b.notifyCh <- []grpc.Address{{Addr: addr}}:
|
||||||
|
case <-downc:
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case <-downc:
|
||||||
|
b.notifyAddrs(NotifyReset)
|
||||||
|
case msg := <-b.updateAddrsC:
|
||||||
|
b.notifyAddrs(msg)
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) notifyAddrs(msg NotifyMsg) {
|
||||||
|
if msg == NotifyNext {
|
||||||
|
select {
|
||||||
|
case b.notifyCh <- []grpc.Address{}:
|
||||||
|
case <-b.stopc:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.mu.RLock()
|
||||||
|
pinAddr := b.pinAddr
|
||||||
|
downc := b.downc
|
||||||
|
b.mu.RUnlock()
|
||||||
|
addrs, hostPorts := b.liveAddrs()
|
||||||
|
|
||||||
|
var waitDown bool
|
||||||
|
if pinAddr != "" {
|
||||||
|
_, ok := hostPorts[pinAddr]
|
||||||
|
waitDown = !ok
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case b.notifyCh <- addrs:
|
||||||
|
if waitDown {
|
||||||
|
select {
|
||||||
|
case <-downc:
|
||||||
|
case <-b.stopc:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case <-b.stopc:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Up(addr grpc.Address) func(error) {
|
||||||
|
if !b.mayPin(addr) {
|
||||||
|
return func(err error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
b.mu.Lock()
|
||||||
|
defer b.mu.Unlock()
|
||||||
|
|
||||||
|
// gRPC might call Up after it called Close. We add this check
|
||||||
|
// to "fix" it up at application layer. Otherwise, will panic
|
||||||
|
// if b.upc is already closed.
|
||||||
|
if b.closed {
|
||||||
|
return func(err error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
// gRPC might call Up on a stale address.
|
||||||
|
// Prevent updating pinAddr with a stale address.
|
||||||
|
if !hasAddr(b.addrs, addr.Addr) {
|
||||||
|
return func(err error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
if b.pinAddr != "" {
|
||||||
|
lg.Infof("clientv3/balancer: %q is up but not pinned (already pinned %q)", addr.Addr, b.pinAddr)
|
||||||
|
return func(err error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
// notify waiting Get()s and pin first connected address
|
||||||
|
close(b.upc)
|
||||||
|
b.downc = make(chan struct{})
|
||||||
|
b.pinAddr = addr.Addr
|
||||||
|
lg.Infof("clientv3/balancer: pin %q", addr.Addr)
|
||||||
|
|
||||||
|
// notify client that a connection is up
|
||||||
|
b.readyOnce.Do(func() { close(b.readyc) })
|
||||||
|
|
||||||
|
return func(err error) {
|
||||||
|
// If connected to a black hole endpoint or a killed server, the gRPC ping
|
||||||
|
// timeout will induce a network I/O error, and retrying until success;
|
||||||
|
// finding healthy endpoint on retry could take several timeouts and redials.
|
||||||
|
// To avoid wasting retries, gray-list unhealthy endpoints.
|
||||||
|
b.HostPortError(addr.Addr, err)
|
||||||
|
|
||||||
|
b.mu.Lock()
|
||||||
|
b.upc = make(chan struct{})
|
||||||
|
close(b.downc)
|
||||||
|
b.pinAddr = ""
|
||||||
|
b.mu.Unlock()
|
||||||
|
lg.Infof("clientv3/balancer: unpin %q (%q)", addr.Addr, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) mayPin(addr grpc.Address) bool {
|
||||||
|
if b.Endpoint(addr.Addr) == "" { // stale host:port
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
b.unhealthyMu.RLock()
|
||||||
|
unhealthyCnt := len(b.unhealthyHostPorts)
|
||||||
|
failedTime, bad := b.unhealthyHostPorts[addr.Addr]
|
||||||
|
b.unhealthyMu.RUnlock()
|
||||||
|
|
||||||
|
b.mu.RLock()
|
||||||
|
skip := len(b.addrs) == 1 || unhealthyCnt == 0 || len(b.addrs) == unhealthyCnt
|
||||||
|
b.mu.RUnlock()
|
||||||
|
if skip || !bad {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// prevent isolated member's endpoint from being infinitely retried, as follows:
|
||||||
|
// 1. keepalive pings detects GoAway with http2.ErrCodeEnhanceYourCalm
|
||||||
|
// 2. balancer 'Up' unpins with grpc: failed with network I/O error
|
||||||
|
// 3. grpc-healthcheck still SERVING, thus retry to pin
|
||||||
|
// instead, return before grpc-healthcheck if failed within healthcheck timeout
|
||||||
|
if elapsed := time.Since(failedTime); elapsed < b.healthCheckTimeout {
|
||||||
|
lg.Infof("clientv3/balancer: %q is up but not pinned (failed %v ago, require minimum %v after failure)", addr.Addr, elapsed, b.healthCheckTimeout)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if ok, _ := b.healthCheck(addr.Addr); ok {
|
||||||
|
b.removeUnhealthy(addr.Addr, "health check success")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
b.HostPortError(addr.Addr, errors.New("health check failed"))
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Get(ctx context.Context, opts grpc.BalancerGetOptions) (grpc.Address, func(), error) {
|
||||||
|
var (
|
||||||
|
addr string
|
||||||
|
closed bool
|
||||||
|
)
|
||||||
|
|
||||||
|
// If opts.BlockingWait is false (for fail-fast RPCs), it should return
|
||||||
|
// an address it has notified via Notify immediately instead of blocking.
|
||||||
|
if !opts.BlockingWait {
|
||||||
|
b.mu.RLock()
|
||||||
|
closed = b.closed
|
||||||
|
addr = b.pinAddr
|
||||||
|
b.mu.RUnlock()
|
||||||
|
if closed {
|
||||||
|
return grpc.Address{Addr: ""}, nil, grpc.ErrClientConnClosing
|
||||||
|
}
|
||||||
|
if addr == "" {
|
||||||
|
return grpc.Address{Addr: ""}, nil, ErrNoAddrAvilable
|
||||||
|
}
|
||||||
|
return grpc.Address{Addr: addr}, func() {}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
b.mu.RLock()
|
||||||
|
ch := b.upc
|
||||||
|
b.mu.RUnlock()
|
||||||
|
select {
|
||||||
|
case <-ch:
|
||||||
|
case <-b.donec:
|
||||||
|
return grpc.Address{Addr: ""}, nil, grpc.ErrClientConnClosing
|
||||||
|
case <-ctx.Done():
|
||||||
|
return grpc.Address{Addr: ""}, nil, ctx.Err()
|
||||||
|
}
|
||||||
|
b.mu.RLock()
|
||||||
|
closed = b.closed
|
||||||
|
addr = b.pinAddr
|
||||||
|
b.mu.RUnlock()
|
||||||
|
// Close() which sets b.closed = true can be called before Get(), Get() must exit if balancer is closed.
|
||||||
|
if closed {
|
||||||
|
return grpc.Address{Addr: ""}, nil, grpc.ErrClientConnClosing
|
||||||
|
}
|
||||||
|
if addr != "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return grpc.Address{Addr: addr}, func() {}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Notify() <-chan []grpc.Address { return b.notifyCh }
|
||||||
|
|
||||||
|
func (b *GRPC17Health) Close() error {
|
||||||
|
b.mu.Lock()
|
||||||
|
// In case gRPC calls close twice. TODO: remove the checking
|
||||||
|
// when we are sure that gRPC wont call close twice.
|
||||||
|
if b.closed {
|
||||||
|
b.mu.Unlock()
|
||||||
|
<-b.donec
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
b.closed = true
|
||||||
|
b.stopOnce.Do(func() { close(b.stopc) })
|
||||||
|
b.pinAddr = ""
|
||||||
|
|
||||||
|
// In the case of following scenario:
|
||||||
|
// 1. upc is not closed; no pinned address
|
||||||
|
// 2. client issues an RPC, calling invoke(), which calls Get(), enters for loop, blocks
|
||||||
|
// 3. client.conn.Close() calls balancer.Close(); closed = true
|
||||||
|
// 4. for loop in Get() never exits since ctx is the context passed in by the client and may not be canceled
|
||||||
|
// we must close upc so Get() exits from blocking on upc
|
||||||
|
select {
|
||||||
|
case <-b.upc:
|
||||||
|
default:
|
||||||
|
// terminate all waiting Get()s
|
||||||
|
close(b.upc)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.mu.Unlock()
|
||||||
|
b.wg.Wait()
|
||||||
|
|
||||||
|
// wait for updateNotifyLoop to finish
|
||||||
|
<-b.donec
|
||||||
|
close(b.notifyCh)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func grpcHealthCheck(ep string, dialFunc func(ep string, dopts ...grpc.DialOption) (*grpc.ClientConn, error)) (bool, error) {
|
||||||
|
conn, err := dialFunc(ep)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
defer conn.Close()
|
||||||
|
cli := healthpb.NewHealthClient(conn)
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
resp, err := cli.Check(ctx, &healthpb.HealthCheckRequest{})
|
||||||
|
cancel()
|
||||||
|
if err != nil {
|
||||||
|
if s, ok := status.FromError(err); ok && s.Code() == codes.Unavailable {
|
||||||
|
if s.Message() == unknownService { // etcd < v3.3.0
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
return resp.Status == healthpb.HealthCheckResponse_SERVING, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func hasAddr(addrs []grpc.Address, targetAddr string) bool {
|
||||||
|
for _, addr := range addrs {
|
||||||
|
if targetAddr == addr.Addr {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func getHost(ep string) string {
|
||||||
|
url, uerr := url.Parse(ep)
|
||||||
|
if uerr != nil || !strings.Contains(ep, "://") {
|
||||||
|
return ep
|
||||||
|
}
|
||||||
|
return url.Host
|
||||||
|
}
|
||||||
|
|
||||||
|
func eps2addrs(eps []string) []grpc.Address {
|
||||||
|
addrs := make([]grpc.Address, len(eps))
|
||||||
|
for i := range eps {
|
||||||
|
addrs[i].Addr = getHost(eps[i])
|
||||||
|
}
|
||||||
|
return addrs
|
||||||
|
}
|
||||||
|
|
||||||
|
func getHostPort2ep(eps []string) map[string]string {
|
||||||
|
hm := make(map[string]string, len(eps))
|
||||||
|
for i := range eps {
|
||||||
|
_, host, _ := parseEndpoint(eps[i])
|
||||||
|
hm[host] = eps[i]
|
||||||
|
}
|
||||||
|
return hm
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseEndpoint(endpoint string) (proto string, host string, scheme string) {
|
||||||
|
proto = "tcp"
|
||||||
|
host = endpoint
|
||||||
|
url, uerr := url.Parse(endpoint)
|
||||||
|
if uerr != nil || !strings.Contains(endpoint, "://") {
|
||||||
|
return proto, host, scheme
|
||||||
|
}
|
||||||
|
scheme = url.Scheme
|
||||||
|
|
||||||
|
// strip scheme:// prefix since grpc dials by host
|
||||||
|
host = url.Host
|
||||||
|
switch url.Scheme {
|
||||||
|
case "http", "https":
|
||||||
|
case "unix", "unixs":
|
||||||
|
proto = "unix"
|
||||||
|
host = url.Host + url.Path
|
||||||
|
default:
|
||||||
|
proto, host = "", ""
|
||||||
|
}
|
||||||
|
return proto, host, scheme
|
||||||
|
}
|
16
vendor/github.com/coreos/etcd/clientv3/balancer/picker/doc.go
generated
vendored
Normal file
16
vendor/github.com/coreos/etcd/clientv3/balancer/picker/doc.go
generated
vendored
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package picker defines/implements client balancer picker policy.
|
||||||
|
package picker
|
34
vendor/github.com/coreos/etcd/clientv3/balancer/picker/err.go
generated
vendored
Normal file
34
vendor/github.com/coreos/etcd/clientv3/balancer/picker/err.go
generated
vendored
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package picker
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"google.golang.org/grpc/balancer"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewErr returns a picker that always returns err on "Pick".
|
||||||
|
func NewErr(err error) Picker {
|
||||||
|
return &errPicker{err: err}
|
||||||
|
}
|
||||||
|
|
||||||
|
type errPicker struct {
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *errPicker) Pick(context.Context, balancer.PickOptions) (balancer.SubConn, func(balancer.DoneInfo), error) {
|
||||||
|
return nil, nil, p.err
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
// Copyright 2013-2015 CoreOS, Inc.
|
// Copyright 2018 The etcd Authors
|
||||||
//
|
//
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
// you may not use this file except in compliance with the License.
|
// you may not use this file except in compliance with the License.
|
||||||
@ -12,27 +12,13 @@
|
|||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
package semver
|
package picker
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sort"
|
"google.golang.org/grpc/balancer"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Versions []*Version
|
// Picker defines balancer Picker methods.
|
||||||
|
type Picker interface {
|
||||||
func (s Versions) Len() int {
|
balancer.Picker
|
||||||
return len(s)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s Versions) Swap(i, j int) {
|
|
||||||
s[i], s[j] = s[j], s[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s Versions) Less(i, j int) bool {
|
|
||||||
return s[i].LessThan(*s[j])
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sort sorts the given slice of Version
|
|
||||||
func Sort(versions []*Version) {
|
|
||||||
sort.Sort(Versions(versions))
|
|
||||||
}
|
}
|
49
vendor/github.com/coreos/etcd/clientv3/balancer/picker/picker_policy.go
generated
vendored
Normal file
49
vendor/github.com/coreos/etcd/clientv3/balancer/picker/picker_policy.go
generated
vendored
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package picker
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
// Policy defines balancer picker policy.
|
||||||
|
type Policy uint8
|
||||||
|
|
||||||
|
const (
|
||||||
|
// TODO: custom picker is not supported yet.
|
||||||
|
// custom defines custom balancer picker.
|
||||||
|
custom Policy = iota
|
||||||
|
|
||||||
|
// RoundrobinBalanced balance loads over multiple endpoints
|
||||||
|
// and implements failover in roundrobin fashion.
|
||||||
|
RoundrobinBalanced Policy = iota
|
||||||
|
|
||||||
|
// TODO: only send loads to pinned address "RoundrobinFailover"
|
||||||
|
// just like how 3.3 client works
|
||||||
|
//
|
||||||
|
// TODO: priotize leader
|
||||||
|
// TODO: health-check
|
||||||
|
// TODO: weighted roundrobin
|
||||||
|
// TODO: power of two random choice
|
||||||
|
)
|
||||||
|
|
||||||
|
func (p Policy) String() string {
|
||||||
|
switch p {
|
||||||
|
case custom:
|
||||||
|
panic("'custom' picker policy is not supported yet")
|
||||||
|
case RoundrobinBalanced:
|
||||||
|
return "etcd-client-roundrobin-balanced"
|
||||||
|
default:
|
||||||
|
panic(fmt.Errorf("invalid balancer picker policy (%d)", p))
|
||||||
|
}
|
||||||
|
}
|
92
vendor/github.com/coreos/etcd/clientv3/balancer/picker/roundrobin_balanced.go
generated
vendored
Normal file
92
vendor/github.com/coreos/etcd/clientv3/balancer/picker/roundrobin_balanced.go
generated
vendored
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package picker
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"go.uber.org/zap/zapcore"
|
||||||
|
"google.golang.org/grpc/balancer"
|
||||||
|
"google.golang.org/grpc/resolver"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewRoundrobinBalanced returns a new roundrobin balanced picker.
|
||||||
|
func NewRoundrobinBalanced(
|
||||||
|
lg *zap.Logger,
|
||||||
|
scs []balancer.SubConn,
|
||||||
|
addrToSc map[resolver.Address]balancer.SubConn,
|
||||||
|
scToAddr map[balancer.SubConn]resolver.Address,
|
||||||
|
) Picker {
|
||||||
|
return &rrBalanced{
|
||||||
|
lg: lg,
|
||||||
|
scs: scs,
|
||||||
|
addrToSc: addrToSc,
|
||||||
|
scToAddr: scToAddr,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type rrBalanced struct {
|
||||||
|
lg *zap.Logger
|
||||||
|
|
||||||
|
mu sync.RWMutex
|
||||||
|
next int
|
||||||
|
scs []balancer.SubConn
|
||||||
|
|
||||||
|
addrToSc map[resolver.Address]balancer.SubConn
|
||||||
|
scToAddr map[balancer.SubConn]resolver.Address
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pick is called for every client request.
|
||||||
|
func (rb *rrBalanced) Pick(ctx context.Context, opts balancer.PickOptions) (balancer.SubConn, func(balancer.DoneInfo), error) {
|
||||||
|
rb.mu.RLock()
|
||||||
|
n := len(rb.scs)
|
||||||
|
rb.mu.RUnlock()
|
||||||
|
if n == 0 {
|
||||||
|
return nil, nil, balancer.ErrNoSubConnAvailable
|
||||||
|
}
|
||||||
|
|
||||||
|
rb.mu.Lock()
|
||||||
|
cur := rb.next
|
||||||
|
sc := rb.scs[cur]
|
||||||
|
picked := rb.scToAddr[sc].Addr
|
||||||
|
rb.next = (rb.next + 1) % len(rb.scs)
|
||||||
|
rb.mu.Unlock()
|
||||||
|
|
||||||
|
rb.lg.Debug(
|
||||||
|
"picked",
|
||||||
|
zap.String("address", picked),
|
||||||
|
zap.Int("subconn-index", cur),
|
||||||
|
zap.Int("subconn-size", n),
|
||||||
|
)
|
||||||
|
|
||||||
|
doneFunc := func(info balancer.DoneInfo) {
|
||||||
|
// TODO: error handling?
|
||||||
|
fss := []zapcore.Field{
|
||||||
|
zap.Error(info.Err),
|
||||||
|
zap.String("address", picked),
|
||||||
|
zap.Bool("success", info.Err == nil),
|
||||||
|
zap.Bool("bytes-sent", info.BytesSent),
|
||||||
|
zap.Bool("bytes-received", info.BytesReceived),
|
||||||
|
}
|
||||||
|
if info.Err == nil {
|
||||||
|
rb.lg.Debug("balancer done", fss...)
|
||||||
|
} else {
|
||||||
|
rb.lg.Warn("balancer failed", fss...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sc, doneFunc, nil
|
||||||
|
}
|
229
vendor/github.com/coreos/etcd/clientv3/balancer/resolver/endpoint/endpoint.go
generated
vendored
Normal file
229
vendor/github.com/coreos/etcd/clientv3/balancer/resolver/endpoint/endpoint.go
generated
vendored
Normal file
@ -0,0 +1,229 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package endpoint resolves etcd entpoints using grpc targets of the form 'endpoint://<id>/<endpoint>'.
|
||||||
|
package endpoint
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"google.golang.org/grpc/resolver"
|
||||||
|
)
|
||||||
|
|
||||||
|
const scheme = "endpoint"
|
||||||
|
|
||||||
|
var (
|
||||||
|
targetPrefix = fmt.Sprintf("%s://", scheme)
|
||||||
|
|
||||||
|
bldr *builder
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
bldr = &builder{
|
||||||
|
resolverGroups: make(map[string]*ResolverGroup),
|
||||||
|
}
|
||||||
|
resolver.Register(bldr)
|
||||||
|
}
|
||||||
|
|
||||||
|
type builder struct {
|
||||||
|
mu sync.RWMutex
|
||||||
|
resolverGroups map[string]*ResolverGroup
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewResolverGroup creates a new ResolverGroup with the given id.
|
||||||
|
func NewResolverGroup(id string) (*ResolverGroup, error) {
|
||||||
|
return bldr.newResolverGroup(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResolverGroup keeps all endpoints of resolvers using a common endpoint://<id>/ target
|
||||||
|
// up-to-date.
|
||||||
|
type ResolverGroup struct {
|
||||||
|
mu sync.RWMutex
|
||||||
|
id string
|
||||||
|
endpoints []string
|
||||||
|
resolvers []*Resolver
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ResolverGroup) addResolver(r *Resolver) {
|
||||||
|
e.mu.Lock()
|
||||||
|
addrs := epsToAddrs(e.endpoints...)
|
||||||
|
e.resolvers = append(e.resolvers, r)
|
||||||
|
e.mu.Unlock()
|
||||||
|
r.cc.NewAddress(addrs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ResolverGroup) removeResolver(r *Resolver) {
|
||||||
|
e.mu.Lock()
|
||||||
|
for i, er := range e.resolvers {
|
||||||
|
if er == r {
|
||||||
|
e.resolvers = append(e.resolvers[:i], e.resolvers[i+1:]...)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
e.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetEndpoints updates the endpoints for ResolverGroup. All registered resolver are updated
|
||||||
|
// immediately with the new endpoints.
|
||||||
|
func (e *ResolverGroup) SetEndpoints(endpoints []string) {
|
||||||
|
addrs := epsToAddrs(endpoints...)
|
||||||
|
e.mu.Lock()
|
||||||
|
e.endpoints = endpoints
|
||||||
|
for _, r := range e.resolvers {
|
||||||
|
r.cc.NewAddress(addrs)
|
||||||
|
}
|
||||||
|
e.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Target constructs a endpoint target using the endpoint id of the ResolverGroup.
|
||||||
|
func (e *ResolverGroup) Target(endpoint string) string {
|
||||||
|
return Target(e.id, endpoint)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Target constructs a endpoint resolver target.
|
||||||
|
func Target(id, endpoint string) string {
|
||||||
|
return fmt.Sprintf("%s://%s/%s", scheme, id, endpoint)
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsTarget checks if a given target string in an endpoint resolver target.
|
||||||
|
func IsTarget(target string) bool {
|
||||||
|
return strings.HasPrefix(target, "endpoint://")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ResolverGroup) Close() {
|
||||||
|
bldr.close(e.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build creates or reuses an etcd resolver for the etcd cluster name identified by the authority part of the target.
|
||||||
|
func (b *builder) Build(target resolver.Target, cc resolver.ClientConn, opts resolver.BuildOption) (resolver.Resolver, error) {
|
||||||
|
if len(target.Authority) < 1 {
|
||||||
|
return nil, fmt.Errorf("'etcd' target scheme requires non-empty authority identifying etcd cluster being routed to")
|
||||||
|
}
|
||||||
|
id := target.Authority
|
||||||
|
es, err := b.getResolverGroup(id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to build resolver: %v", err)
|
||||||
|
}
|
||||||
|
r := &Resolver{
|
||||||
|
endpointID: id,
|
||||||
|
cc: cc,
|
||||||
|
}
|
||||||
|
es.addResolver(r)
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *builder) newResolverGroup(id string) (*ResolverGroup, error) {
|
||||||
|
b.mu.RLock()
|
||||||
|
_, ok := b.resolverGroups[id]
|
||||||
|
b.mu.RUnlock()
|
||||||
|
if ok {
|
||||||
|
return nil, fmt.Errorf("Endpoint already exists for id: %s", id)
|
||||||
|
}
|
||||||
|
|
||||||
|
es := &ResolverGroup{id: id}
|
||||||
|
b.mu.Lock()
|
||||||
|
b.resolverGroups[id] = es
|
||||||
|
b.mu.Unlock()
|
||||||
|
return es, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *builder) getResolverGroup(id string) (*ResolverGroup, error) {
|
||||||
|
b.mu.RLock()
|
||||||
|
es, ok := b.resolverGroups[id]
|
||||||
|
b.mu.RUnlock()
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("ResolverGroup not found for id: %s", id)
|
||||||
|
}
|
||||||
|
return es, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *builder) close(id string) {
|
||||||
|
b.mu.Lock()
|
||||||
|
delete(b.resolverGroups, id)
|
||||||
|
b.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *builder) Scheme() string {
|
||||||
|
return scheme
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resolver provides a resolver for a single etcd cluster, identified by name.
|
||||||
|
type Resolver struct {
|
||||||
|
endpointID string
|
||||||
|
cc resolver.ClientConn
|
||||||
|
sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: use balancer.epsToAddrs
|
||||||
|
func epsToAddrs(eps ...string) (addrs []resolver.Address) {
|
||||||
|
addrs = make([]resolver.Address, 0, len(eps))
|
||||||
|
for _, ep := range eps {
|
||||||
|
addrs = append(addrs, resolver.Address{Addr: ep})
|
||||||
|
}
|
||||||
|
return addrs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*Resolver) ResolveNow(o resolver.ResolveNowOption) {}
|
||||||
|
|
||||||
|
func (r *Resolver) Close() {
|
||||||
|
es, err := bldr.getResolverGroup(r.endpointID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
es.removeResolver(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ParseEndpoint endpoint parses an endpoint of the form
|
||||||
|
// (http|https)://<host>*|(unix|unixs)://<path>)
|
||||||
|
// and returns a protocol ('tcp' or 'unix'),
|
||||||
|
// host (or filepath if a unix socket),
|
||||||
|
// scheme (http, https, unix, unixs).
|
||||||
|
func ParseEndpoint(endpoint string) (proto string, host string, scheme string) {
|
||||||
|
proto = "tcp"
|
||||||
|
host = endpoint
|
||||||
|
url, uerr := url.Parse(endpoint)
|
||||||
|
if uerr != nil || !strings.Contains(endpoint, "://") {
|
||||||
|
return proto, host, scheme
|
||||||
|
}
|
||||||
|
scheme = url.Scheme
|
||||||
|
|
||||||
|
// strip scheme:// prefix since grpc dials by host
|
||||||
|
host = url.Host
|
||||||
|
switch url.Scheme {
|
||||||
|
case "http", "https":
|
||||||
|
case "unix", "unixs":
|
||||||
|
proto = "unix"
|
||||||
|
host = url.Host + url.Path
|
||||||
|
default:
|
||||||
|
proto, host = "", ""
|
||||||
|
}
|
||||||
|
return proto, host, scheme
|
||||||
|
}
|
||||||
|
|
||||||
|
// ParseTarget parses a endpoint://<id>/<endpoint> string and returns the parsed id and endpoint.
|
||||||
|
// If the target is malformed, an error is returned.
|
||||||
|
func ParseTarget(target string) (string, string, error) {
|
||||||
|
noPrefix := strings.TrimPrefix(target, targetPrefix)
|
||||||
|
if noPrefix == target {
|
||||||
|
return "", "", fmt.Errorf("malformed target, %s prefix is required: %s", targetPrefix, target)
|
||||||
|
}
|
||||||
|
parts := strings.SplitN(noPrefix, "/", 2)
|
||||||
|
if len(parts) != 2 {
|
||||||
|
return "", "", fmt.Errorf("malformed target, expected %s://<id>/<endpoint>, but got %s", scheme, target)
|
||||||
|
}
|
||||||
|
return parts[0], parts[1], nil
|
||||||
|
}
|
68
vendor/github.com/coreos/etcd/clientv3/balancer/utils.go
generated
vendored
Normal file
68
vendor/github.com/coreos/etcd/clientv3/balancer/utils.go
generated
vendored
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package balancer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
"sort"
|
||||||
|
"sync/atomic"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"google.golang.org/grpc/balancer"
|
||||||
|
"google.golang.org/grpc/resolver"
|
||||||
|
)
|
||||||
|
|
||||||
|
func scToString(sc balancer.SubConn) string {
|
||||||
|
return fmt.Sprintf("%p", sc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func scsToStrings(scs map[resolver.Address]balancer.SubConn) (ss []string) {
|
||||||
|
ss = make([]string, 0, len(scs))
|
||||||
|
for a, sc := range scs {
|
||||||
|
ss = append(ss, fmt.Sprintf("%s (%s)", a.Addr, scToString(sc)))
|
||||||
|
}
|
||||||
|
sort.Strings(ss)
|
||||||
|
return ss
|
||||||
|
}
|
||||||
|
|
||||||
|
func addrsToStrings(addrs []resolver.Address) (ss []string) {
|
||||||
|
ss = make([]string, len(addrs))
|
||||||
|
for i := range addrs {
|
||||||
|
ss[i] = addrs[i].Addr
|
||||||
|
}
|
||||||
|
sort.Strings(ss)
|
||||||
|
return ss
|
||||||
|
}
|
||||||
|
|
||||||
|
func epsToAddrs(eps ...string) (addrs []resolver.Address) {
|
||||||
|
addrs = make([]resolver.Address, 0, len(eps))
|
||||||
|
for _, ep := range eps {
|
||||||
|
u, err := url.Parse(ep)
|
||||||
|
if err != nil {
|
||||||
|
addrs = append(addrs, resolver.Address{Addr: ep, Type: resolver.Backend})
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
addrs = append(addrs, resolver.Address{Addr: u.Host, Type: resolver.Backend})
|
||||||
|
}
|
||||||
|
return addrs
|
||||||
|
}
|
||||||
|
|
||||||
|
var genN = new(uint32)
|
||||||
|
|
||||||
|
func genName() string {
|
||||||
|
now := time.Now().UnixNano()
|
||||||
|
return fmt.Sprintf("%X%X", now, atomic.AddUint32(genN, 1))
|
||||||
|
}
|
649
vendor/github.com/coreos/etcd/clientv3/client.go
generated
vendored
Normal file
649
vendor/github.com/coreos/etcd/clientv3/client.go
generated
vendored
Normal file
@ -0,0 +1,649 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/tls"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"net/url"
|
||||||
|
"os"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/clientv3/balancer"
|
||||||
|
"github.com/coreos/etcd/clientv3/balancer/picker"
|
||||||
|
"github.com/coreos/etcd/clientv3/balancer/resolver/endpoint"
|
||||||
|
"github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
|
||||||
|
"github.com/grpc-ecosystem/go-grpc-middleware/util/backoffutils"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/credentials"
|
||||||
|
"google.golang.org/grpc/keepalive"
|
||||||
|
"google.golang.org/grpc/metadata"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrNoAvailableEndpoints = errors.New("etcdclient: no available endpoints")
|
||||||
|
ErrOldCluster = errors.New("etcdclient: old cluster version")
|
||||||
|
|
||||||
|
roundRobinBalancerName = fmt.Sprintf("etcd-%s", picker.RoundrobinBalanced.String())
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
lg := zap.NewNop()
|
||||||
|
if os.Getenv("ETCD_CLIENT_DEBUG") != "" {
|
||||||
|
var err error
|
||||||
|
lg, err = zap.NewProductionConfig().Build() // info level logging
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
balancer.RegisterBuilder(balancer.Config{
|
||||||
|
Policy: picker.RoundrobinBalanced,
|
||||||
|
Name: roundRobinBalancerName,
|
||||||
|
Logger: lg,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Client provides and manages an etcd v3 client session.
|
||||||
|
type Client struct {
|
||||||
|
Cluster
|
||||||
|
KV
|
||||||
|
Lease
|
||||||
|
Watcher
|
||||||
|
Auth
|
||||||
|
Maintenance
|
||||||
|
|
||||||
|
conn *grpc.ClientConn
|
||||||
|
|
||||||
|
cfg Config
|
||||||
|
creds *credentials.TransportCredentials
|
||||||
|
balancer balancer.Balancer
|
||||||
|
resolverGroup *endpoint.ResolverGroup
|
||||||
|
mu *sync.Mutex
|
||||||
|
|
||||||
|
ctx context.Context
|
||||||
|
cancel context.CancelFunc
|
||||||
|
|
||||||
|
// Username is a user name for authentication.
|
||||||
|
Username string
|
||||||
|
// Password is a password for authentication.
|
||||||
|
Password string
|
||||||
|
// tokenCred is an instance of WithPerRPCCredentials()'s argument
|
||||||
|
tokenCred *authTokenCredential
|
||||||
|
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
|
||||||
|
lg *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a new etcdv3 client from a given configuration.
|
||||||
|
func New(cfg Config) (*Client, error) {
|
||||||
|
if len(cfg.Endpoints) == 0 {
|
||||||
|
return nil, ErrNoAvailableEndpoints
|
||||||
|
}
|
||||||
|
|
||||||
|
return newClient(&cfg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewCtxClient creates a client with a context but no underlying grpc
|
||||||
|
// connection. This is useful for embedded cases that override the
|
||||||
|
// service interface implementations and do not need connection management.
|
||||||
|
func NewCtxClient(ctx context.Context) *Client {
|
||||||
|
cctx, cancel := context.WithCancel(ctx)
|
||||||
|
return &Client{ctx: cctx, cancel: cancel}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewFromURL creates a new etcdv3 client from a URL.
|
||||||
|
func NewFromURL(url string) (*Client, error) {
|
||||||
|
return New(Config{Endpoints: []string{url}})
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewFromURLs creates a new etcdv3 client from URLs.
|
||||||
|
func NewFromURLs(urls []string) (*Client, error) {
|
||||||
|
return New(Config{Endpoints: urls})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close shuts down the client's etcd connections.
|
||||||
|
func (c *Client) Close() error {
|
||||||
|
c.cancel()
|
||||||
|
c.Watcher.Close()
|
||||||
|
c.Lease.Close()
|
||||||
|
if c.resolverGroup != nil {
|
||||||
|
c.resolverGroup.Close()
|
||||||
|
}
|
||||||
|
if c.conn != nil {
|
||||||
|
return toErr(c.ctx, c.conn.Close())
|
||||||
|
}
|
||||||
|
return c.ctx.Err()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ctx is a context for "out of band" messages (e.g., for sending
|
||||||
|
// "clean up" message when another context is canceled). It is
|
||||||
|
// canceled on client Close().
|
||||||
|
func (c *Client) Ctx() context.Context { return c.ctx }
|
||||||
|
|
||||||
|
// Endpoints lists the registered endpoints for the client.
|
||||||
|
func (c *Client) Endpoints() (eps []string) {
|
||||||
|
// copy the slice; protect original endpoints from being changed
|
||||||
|
eps = make([]string, len(c.cfg.Endpoints))
|
||||||
|
copy(eps, c.cfg.Endpoints)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetEndpoints updates client's endpoints.
|
||||||
|
func (c *Client) SetEndpoints(eps ...string) {
|
||||||
|
c.mu.Lock()
|
||||||
|
defer c.mu.Unlock()
|
||||||
|
c.cfg.Endpoints = eps
|
||||||
|
c.resolverGroup.SetEndpoints(eps)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sync synchronizes client's endpoints with the known endpoints from the etcd membership.
|
||||||
|
func (c *Client) Sync(ctx context.Context) error {
|
||||||
|
mresp, err := c.MemberList(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
var eps []string
|
||||||
|
for _, m := range mresp.Members {
|
||||||
|
eps = append(eps, m.ClientURLs...)
|
||||||
|
}
|
||||||
|
c.SetEndpoints(eps...)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) autoSync() {
|
||||||
|
if c.cfg.AutoSyncInterval == time.Duration(0) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-c.ctx.Done():
|
||||||
|
return
|
||||||
|
case <-time.After(c.cfg.AutoSyncInterval):
|
||||||
|
ctx, cancel := context.WithTimeout(c.ctx, 5*time.Second)
|
||||||
|
err := c.Sync(ctx)
|
||||||
|
cancel()
|
||||||
|
if err != nil && err != c.ctx.Err() {
|
||||||
|
lg.Lvl(4).Infof("Auto sync endpoints failed: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type authTokenCredential struct {
|
||||||
|
token string
|
||||||
|
tokenMu *sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cred authTokenCredential) RequireTransportSecurity() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cred authTokenCredential) GetRequestMetadata(ctx context.Context, s ...string) (map[string]string, error) {
|
||||||
|
cred.tokenMu.RLock()
|
||||||
|
defer cred.tokenMu.RUnlock()
|
||||||
|
return map[string]string{
|
||||||
|
rpctypes.TokenFieldNameGRPC: cred.token,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) processCreds(scheme string) (creds *credentials.TransportCredentials) {
|
||||||
|
creds = c.creds
|
||||||
|
switch scheme {
|
||||||
|
case "unix":
|
||||||
|
case "http":
|
||||||
|
creds = nil
|
||||||
|
case "https", "unixs":
|
||||||
|
if creds != nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
tlsconfig := &tls.Config{}
|
||||||
|
emptyCreds := credentials.NewTLS(tlsconfig)
|
||||||
|
creds = &emptyCreds
|
||||||
|
default:
|
||||||
|
creds = nil
|
||||||
|
}
|
||||||
|
return creds
|
||||||
|
}
|
||||||
|
|
||||||
|
// dialSetupOpts gives the dial opts prior to any authentication
|
||||||
|
func (c *Client) dialSetupOpts(target string, dopts ...grpc.DialOption) (opts []grpc.DialOption, err error) {
|
||||||
|
_, ep, err := endpoint.ParseTarget(target)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("unable to parse target: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if c.cfg.DialKeepAliveTime > 0 {
|
||||||
|
params := keepalive.ClientParameters{
|
||||||
|
Time: c.cfg.DialKeepAliveTime,
|
||||||
|
Timeout: c.cfg.DialKeepAliveTimeout,
|
||||||
|
}
|
||||||
|
opts = append(opts, grpc.WithKeepaliveParams(params))
|
||||||
|
}
|
||||||
|
opts = append(opts, dopts...)
|
||||||
|
|
||||||
|
f := func(dialEp string, t time.Duration) (net.Conn, error) {
|
||||||
|
proto, host, _ := endpoint.ParseEndpoint(dialEp)
|
||||||
|
if host == "" && ep != "" {
|
||||||
|
// dialing an endpoint not in the balancer; use
|
||||||
|
// endpoint passed into dial
|
||||||
|
proto, host, _ = endpoint.ParseEndpoint(ep)
|
||||||
|
}
|
||||||
|
if proto == "" {
|
||||||
|
return nil, fmt.Errorf("unknown scheme for %q", host)
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case <-c.ctx.Done():
|
||||||
|
return nil, c.ctx.Err()
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
dialer := &net.Dialer{Timeout: t}
|
||||||
|
return dialer.DialContext(c.ctx, proto, host)
|
||||||
|
}
|
||||||
|
opts = append(opts, grpc.WithDialer(f))
|
||||||
|
|
||||||
|
creds := c.creds
|
||||||
|
if _, _, scheme := endpoint.ParseEndpoint(ep); len(scheme) != 0 {
|
||||||
|
creds = c.processCreds(scheme)
|
||||||
|
}
|
||||||
|
if creds != nil {
|
||||||
|
opts = append(opts, grpc.WithTransportCredentials(*creds))
|
||||||
|
} else {
|
||||||
|
opts = append(opts, grpc.WithInsecure())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interceptor retry and backoff.
|
||||||
|
// TODO: Replace all of clientv3/retry.go with interceptor based retry, or with
|
||||||
|
// https://github.com/grpc/proposal/blob/master/A6-client-retries.md#retry-policy
|
||||||
|
// once it is available.
|
||||||
|
rrBackoff := withBackoff(c.roundRobinQuorumBackoff(defaultBackoffWaitBetween, defaultBackoffJitterFraction))
|
||||||
|
opts = append(opts,
|
||||||
|
// Disable stream retry by default since go-grpc-middleware/retry does not support client streams.
|
||||||
|
// Streams that are safe to retry are enabled individually.
|
||||||
|
grpc.WithStreamInterceptor(c.streamClientInterceptor(c.lg, withMax(0), rrBackoff)),
|
||||||
|
grpc.WithUnaryInterceptor(c.unaryClientInterceptor(c.lg, withMax(defaultUnaryMaxRetries), rrBackoff)),
|
||||||
|
)
|
||||||
|
|
||||||
|
return opts, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dial connects to a single endpoint using the client's config.
|
||||||
|
func (c *Client) Dial(endpoint string) (*grpc.ClientConn, error) {
|
||||||
|
return c.dial(endpoint)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) getToken(ctx context.Context) error {
|
||||||
|
var err error // return last error in a case of fail
|
||||||
|
var auth *authenticator
|
||||||
|
|
||||||
|
for i := 0; i < len(c.cfg.Endpoints); i++ {
|
||||||
|
ep := c.cfg.Endpoints[i]
|
||||||
|
// use dial options without dopts to avoid reusing the client balancer
|
||||||
|
var dOpts []grpc.DialOption
|
||||||
|
_, host, _ := endpoint.ParseEndpoint(ep)
|
||||||
|
target := c.resolverGroup.Target(host)
|
||||||
|
dOpts, err = c.dialSetupOpts(target, c.cfg.DialOptions...)
|
||||||
|
if err != nil {
|
||||||
|
err = fmt.Errorf("failed to configure auth dialer: %v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
dOpts = append(dOpts, grpc.WithBalancerName(roundRobinBalancerName))
|
||||||
|
auth, err = newAuthenticator(ctx, target, dOpts, c)
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
defer auth.close()
|
||||||
|
|
||||||
|
var resp *AuthenticateResponse
|
||||||
|
resp, err = auth.authenticate(ctx, c.Username, c.Password)
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
c.tokenCred.tokenMu.Lock()
|
||||||
|
c.tokenCred.token = resp.Token
|
||||||
|
c.tokenCred.tokenMu.Unlock()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) dial(ep string, dopts ...grpc.DialOption) (*grpc.ClientConn, error) {
|
||||||
|
// We pass a target to DialContext of the form: endpoint://<clusterName>/<host-part> that
|
||||||
|
// does not include scheme (http/https/unix/unixs) or path parts.
|
||||||
|
_, host, _ := endpoint.ParseEndpoint(ep)
|
||||||
|
target := c.resolverGroup.Target(host)
|
||||||
|
|
||||||
|
opts, err := c.dialSetupOpts(target, dopts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to configure dialer: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if c.Username != "" && c.Password != "" {
|
||||||
|
c.tokenCred = &authTokenCredential{
|
||||||
|
tokenMu: &sync.RWMutex{},
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := c.ctx, func() {}
|
||||||
|
if c.cfg.DialTimeout > 0 {
|
||||||
|
ctx, cancel = context.WithTimeout(ctx, c.cfg.DialTimeout)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = c.getToken(ctx)
|
||||||
|
if err != nil {
|
||||||
|
if toErr(ctx, err) != rpctypes.ErrAuthNotEnabled {
|
||||||
|
if err == ctx.Err() && ctx.Err() != c.ctx.Err() {
|
||||||
|
err = context.DeadlineExceeded
|
||||||
|
}
|
||||||
|
cancel()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
opts = append(opts, grpc.WithPerRPCCredentials(c.tokenCred))
|
||||||
|
}
|
||||||
|
cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
opts = append(opts, c.cfg.DialOptions...)
|
||||||
|
|
||||||
|
dctx := c.ctx
|
||||||
|
if c.cfg.DialTimeout > 0 {
|
||||||
|
var cancel context.CancelFunc
|
||||||
|
dctx, cancel = context.WithTimeout(c.ctx, c.cfg.DialTimeout)
|
||||||
|
defer cancel() // TODO: Is this right for cases where grpc.WithBlock() is not set on the dial options?
|
||||||
|
}
|
||||||
|
|
||||||
|
conn, err := grpc.DialContext(dctx, target, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return conn, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithRequireLeader requires client requests to only succeed
|
||||||
|
// when the cluster has a leader.
|
||||||
|
func WithRequireLeader(ctx context.Context) context.Context {
|
||||||
|
md := metadata.Pairs(rpctypes.MetadataRequireLeaderKey, rpctypes.MetadataHasLeader)
|
||||||
|
return metadata.NewOutgoingContext(ctx, md)
|
||||||
|
}
|
||||||
|
|
||||||
|
func newClient(cfg *Config) (*Client, error) {
|
||||||
|
if cfg == nil {
|
||||||
|
cfg = &Config{}
|
||||||
|
}
|
||||||
|
var creds *credentials.TransportCredentials
|
||||||
|
if cfg.TLS != nil {
|
||||||
|
c := credentials.NewTLS(cfg.TLS)
|
||||||
|
creds = &c
|
||||||
|
}
|
||||||
|
|
||||||
|
// use a temporary skeleton client to bootstrap first connection
|
||||||
|
baseCtx := context.TODO()
|
||||||
|
if cfg.Context != nil {
|
||||||
|
baseCtx = cfg.Context
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithCancel(baseCtx)
|
||||||
|
client := &Client{
|
||||||
|
conn: nil,
|
||||||
|
cfg: *cfg,
|
||||||
|
creds: creds,
|
||||||
|
ctx: ctx,
|
||||||
|
cancel: cancel,
|
||||||
|
mu: new(sync.Mutex),
|
||||||
|
callOpts: defaultCallOpts,
|
||||||
|
}
|
||||||
|
|
||||||
|
lcfg := DefaultLogConfig
|
||||||
|
if cfg.LogConfig != nil {
|
||||||
|
lcfg = *cfg.LogConfig
|
||||||
|
}
|
||||||
|
var err error
|
||||||
|
client.lg, err = lcfg.Build()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.Username != "" && cfg.Password != "" {
|
||||||
|
client.Username = cfg.Username
|
||||||
|
client.Password = cfg.Password
|
||||||
|
}
|
||||||
|
if cfg.MaxCallSendMsgSize > 0 || cfg.MaxCallRecvMsgSize > 0 {
|
||||||
|
if cfg.MaxCallRecvMsgSize > 0 && cfg.MaxCallSendMsgSize > cfg.MaxCallRecvMsgSize {
|
||||||
|
return nil, fmt.Errorf("gRPC message recv limit (%d bytes) must be greater than send limit (%d bytes)", cfg.MaxCallRecvMsgSize, cfg.MaxCallSendMsgSize)
|
||||||
|
}
|
||||||
|
callOpts := []grpc.CallOption{
|
||||||
|
defaultFailFast,
|
||||||
|
defaultMaxCallSendMsgSize,
|
||||||
|
defaultMaxCallRecvMsgSize,
|
||||||
|
}
|
||||||
|
if cfg.MaxCallSendMsgSize > 0 {
|
||||||
|
callOpts[1] = grpc.MaxCallSendMsgSize(cfg.MaxCallSendMsgSize)
|
||||||
|
}
|
||||||
|
if cfg.MaxCallRecvMsgSize > 0 {
|
||||||
|
callOpts[2] = grpc.MaxCallRecvMsgSize(cfg.MaxCallRecvMsgSize)
|
||||||
|
}
|
||||||
|
client.callOpts = callOpts
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prepare a 'endpoint://<unique-client-id>/' resolver for the client and create a endpoint target to pass
|
||||||
|
// to dial so the client knows to use this resolver.
|
||||||
|
client.resolverGroup, err = endpoint.NewResolverGroup(fmt.Sprintf("client-%s", strconv.FormatInt(time.Now().UnixNano(), 36)))
|
||||||
|
if err != nil {
|
||||||
|
client.cancel()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
client.resolverGroup.SetEndpoints(cfg.Endpoints)
|
||||||
|
|
||||||
|
if len(cfg.Endpoints) < 1 {
|
||||||
|
return nil, fmt.Errorf("at least one Endpoint must is required in client config")
|
||||||
|
}
|
||||||
|
dialEndpoint := cfg.Endpoints[0]
|
||||||
|
|
||||||
|
// Use an provided endpoint target so that for https:// without any tls config given, then
|
||||||
|
// grpc will assume the certificate server name is the endpoint host.
|
||||||
|
conn, err := client.dial(dialEndpoint, grpc.WithBalancerName(roundRobinBalancerName))
|
||||||
|
if err != nil {
|
||||||
|
client.cancel()
|
||||||
|
client.resolverGroup.Close()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// TODO: With the old grpc balancer interface, we waited until the dial timeout
|
||||||
|
// for the balancer to be ready. Is there an equivalent wait we should do with the new grpc balancer interface?
|
||||||
|
client.conn = conn
|
||||||
|
|
||||||
|
client.Cluster = NewCluster(client)
|
||||||
|
client.KV = NewKV(client)
|
||||||
|
client.Lease = NewLease(client)
|
||||||
|
client.Watcher = NewWatcher(client)
|
||||||
|
client.Auth = NewAuth(client)
|
||||||
|
client.Maintenance = NewMaintenance(client)
|
||||||
|
|
||||||
|
if cfg.RejectOldCluster {
|
||||||
|
if err := client.checkVersion(); err != nil {
|
||||||
|
client.Close()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
go client.autoSync()
|
||||||
|
return client, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// roundRobinQuorumBackoff retries against quorum between each backoff.
|
||||||
|
// This is intended for use with a round robin load balancer.
|
||||||
|
func (c *Client) roundRobinQuorumBackoff(waitBetween time.Duration, jitterFraction float64) backoffFunc {
|
||||||
|
return func(attempt uint) time.Duration {
|
||||||
|
// after each round robin across quorum, backoff for our wait between duration
|
||||||
|
n := uint(len(c.Endpoints()))
|
||||||
|
quorum := (n/2 + 1)
|
||||||
|
if attempt%quorum == 0 {
|
||||||
|
c.lg.Info("backoff", zap.Uint("attempt", attempt), zap.Uint("quorum", quorum), zap.Duration("waitBetween", waitBetween), zap.Float64("jitterFraction", jitterFraction))
|
||||||
|
return backoffutils.JitterUp(waitBetween, jitterFraction)
|
||||||
|
}
|
||||||
|
c.lg.Info("backoff skipped", zap.Uint("attempt", attempt), zap.Uint("quorum", quorum))
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) checkVersion() (err error) {
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
errc := make(chan error, len(c.cfg.Endpoints))
|
||||||
|
ctx, cancel := context.WithCancel(c.ctx)
|
||||||
|
if c.cfg.DialTimeout > 0 {
|
||||||
|
ctx, cancel = context.WithTimeout(ctx, c.cfg.DialTimeout)
|
||||||
|
}
|
||||||
|
wg.Add(len(c.cfg.Endpoints))
|
||||||
|
for _, ep := range c.cfg.Endpoints {
|
||||||
|
// if cluster is current, any endpoint gives a recent version
|
||||||
|
go func(e string) {
|
||||||
|
defer wg.Done()
|
||||||
|
resp, rerr := c.Status(ctx, e)
|
||||||
|
if rerr != nil {
|
||||||
|
errc <- rerr
|
||||||
|
return
|
||||||
|
}
|
||||||
|
vs := strings.Split(resp.Version, ".")
|
||||||
|
maj, min := 0, 0
|
||||||
|
if len(vs) >= 2 {
|
||||||
|
maj, _ = strconv.Atoi(vs[0])
|
||||||
|
min, rerr = strconv.Atoi(vs[1])
|
||||||
|
}
|
||||||
|
if maj < 3 || (maj == 3 && min < 2) {
|
||||||
|
rerr = ErrOldCluster
|
||||||
|
}
|
||||||
|
errc <- rerr
|
||||||
|
}(ep)
|
||||||
|
}
|
||||||
|
// wait for success
|
||||||
|
for i := 0; i < len(c.cfg.Endpoints); i++ {
|
||||||
|
if err = <-errc; err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cancel()
|
||||||
|
wg.Wait()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// ActiveConnection returns the current in-use connection
|
||||||
|
func (c *Client) ActiveConnection() *grpc.ClientConn { return c.conn }
|
||||||
|
|
||||||
|
// isHaltErr returns true if the given error and context indicate no forward
|
||||||
|
// progress can be made, even after reconnecting.
|
||||||
|
func isHaltErr(ctx context.Context, err error) bool {
|
||||||
|
if ctx != nil && ctx.Err() != nil {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
ev, _ := status.FromError(err)
|
||||||
|
// Unavailable codes mean the system will be right back.
|
||||||
|
// (e.g., can't connect, lost leader)
|
||||||
|
// Treat Internal codes as if something failed, leaving the
|
||||||
|
// system in an inconsistent state, but retrying could make progress.
|
||||||
|
// (e.g., failed in middle of send, corrupted frame)
|
||||||
|
// TODO: are permanent Internal errors possible from grpc?
|
||||||
|
return ev.Code() != codes.Unavailable && ev.Code() != codes.Internal
|
||||||
|
}
|
||||||
|
|
||||||
|
// isUnavailableErr returns true if the given error is an unavailable error
|
||||||
|
func isUnavailableErr(ctx context.Context, err error) bool {
|
||||||
|
if ctx != nil && ctx.Err() != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
ev, _ := status.FromError(err)
|
||||||
|
// Unavailable codes mean the system will be right back.
|
||||||
|
// (e.g., can't connect, lost leader)
|
||||||
|
return ev.Code() == codes.Unavailable
|
||||||
|
}
|
||||||
|
|
||||||
|
func toErr(ctx context.Context, err error) error {
|
||||||
|
if err == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
err = rpctypes.Error(err)
|
||||||
|
if _, ok := err.(rpctypes.EtcdError); ok {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ev, ok := status.FromError(err); ok {
|
||||||
|
code := ev.Code()
|
||||||
|
switch code {
|
||||||
|
case codes.DeadlineExceeded:
|
||||||
|
fallthrough
|
||||||
|
case codes.Canceled:
|
||||||
|
if ctx.Err() != nil {
|
||||||
|
err = ctx.Err()
|
||||||
|
}
|
||||||
|
case codes.Unavailable:
|
||||||
|
case codes.FailedPrecondition:
|
||||||
|
err = grpc.ErrClientConnClosing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func canceledByCaller(stopCtx context.Context, err error) bool {
|
||||||
|
if stopCtx.Err() == nil || err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return err == context.Canceled || err == context.DeadlineExceeded
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsConnCanceled returns true, if error is from a closed gRPC connection.
|
||||||
|
// ref. https://github.com/grpc/grpc-go/pull/1854
|
||||||
|
func IsConnCanceled(err error) bool {
|
||||||
|
if err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// >= gRPC v1.10.x
|
||||||
|
s, ok := status.FromError(err)
|
||||||
|
if ok {
|
||||||
|
// connection is canceled or server has already closed the connection
|
||||||
|
return s.Code() == codes.Canceled || s.Message() == "transport is closing"
|
||||||
|
}
|
||||||
|
// >= gRPC v1.10.x
|
||||||
|
if err == context.Canceled {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
// <= gRPC v1.7.x returns 'errors.New("grpc: the client connection is closing")'
|
||||||
|
return strings.Contains(err.Error(), "grpc: the client connection is closing")
|
||||||
|
}
|
||||||
|
|
||||||
|
func getHost(ep string) string {
|
||||||
|
url, uerr := url.Parse(ep)
|
||||||
|
if uerr != nil || !strings.Contains(ep, "://") {
|
||||||
|
return ep
|
||||||
|
}
|
||||||
|
return url.Host
|
||||||
|
}
|
114
vendor/github.com/coreos/etcd/clientv3/cluster.go
generated
vendored
Normal file
114
vendor/github.com/coreos/etcd/clientv3/cluster.go
generated
vendored
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
"github.com/coreos/etcd/pkg/types"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
Member pb.Member
|
||||||
|
MemberListResponse pb.MemberListResponse
|
||||||
|
MemberAddResponse pb.MemberAddResponse
|
||||||
|
MemberRemoveResponse pb.MemberRemoveResponse
|
||||||
|
MemberUpdateResponse pb.MemberUpdateResponse
|
||||||
|
)
|
||||||
|
|
||||||
|
type Cluster interface {
|
||||||
|
// MemberList lists the current cluster membership.
|
||||||
|
MemberList(ctx context.Context) (*MemberListResponse, error)
|
||||||
|
|
||||||
|
// MemberAdd adds a new member into the cluster.
|
||||||
|
MemberAdd(ctx context.Context, peerAddrs []string) (*MemberAddResponse, error)
|
||||||
|
|
||||||
|
// MemberRemove removes an existing member from the cluster.
|
||||||
|
MemberRemove(ctx context.Context, id uint64) (*MemberRemoveResponse, error)
|
||||||
|
|
||||||
|
// MemberUpdate updates the peer addresses of the member.
|
||||||
|
MemberUpdate(ctx context.Context, id uint64, peerAddrs []string) (*MemberUpdateResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type cluster struct {
|
||||||
|
remote pb.ClusterClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCluster(c *Client) Cluster {
|
||||||
|
api := &cluster{remote: RetryClusterClient(c)}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClusterFromClusterClient(remote pb.ClusterClient, c *Client) Cluster {
|
||||||
|
api := &cluster{remote: remote}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cluster) MemberAdd(ctx context.Context, peerAddrs []string) (*MemberAddResponse, error) {
|
||||||
|
// fail-fast before panic in rafthttp
|
||||||
|
if _, err := types.NewURLs(peerAddrs); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
r := &pb.MemberAddRequest{PeerURLs: peerAddrs}
|
||||||
|
resp, err := c.remote.MemberAdd(ctx, r, c.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
return (*MemberAddResponse)(resp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cluster) MemberRemove(ctx context.Context, id uint64) (*MemberRemoveResponse, error) {
|
||||||
|
r := &pb.MemberRemoveRequest{ID: id}
|
||||||
|
resp, err := c.remote.MemberRemove(ctx, r, c.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
return (*MemberRemoveResponse)(resp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cluster) MemberUpdate(ctx context.Context, id uint64, peerAddrs []string) (*MemberUpdateResponse, error) {
|
||||||
|
// fail-fast before panic in rafthttp
|
||||||
|
if _, err := types.NewURLs(peerAddrs); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// it is safe to retry on update.
|
||||||
|
r := &pb.MemberUpdateRequest{ID: id, PeerURLs: peerAddrs}
|
||||||
|
resp, err := c.remote.MemberUpdate(ctx, r, c.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return (*MemberUpdateResponse)(resp), nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cluster) MemberList(ctx context.Context) (*MemberListResponse, error) {
|
||||||
|
// it is safe to retry on list.
|
||||||
|
resp, err := c.remote.MemberList(ctx, &pb.MemberListRequest{}, c.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return (*MemberListResponse)(resp), nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
51
vendor/github.com/coreos/etcd/clientv3/compact_op.go
generated
vendored
Normal file
51
vendor/github.com/coreos/etcd/clientv3/compact_op.go
generated
vendored
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CompactOp represents a compact operation.
|
||||||
|
type CompactOp struct {
|
||||||
|
revision int64
|
||||||
|
physical bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// CompactOption configures compact operation.
|
||||||
|
type CompactOption func(*CompactOp)
|
||||||
|
|
||||||
|
func (op *CompactOp) applyCompactOpts(opts []CompactOption) {
|
||||||
|
for _, opt := range opts {
|
||||||
|
opt(op)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpCompact wraps slice CompactOption to create a CompactOp.
|
||||||
|
func OpCompact(rev int64, opts ...CompactOption) CompactOp {
|
||||||
|
ret := CompactOp{revision: rev}
|
||||||
|
ret.applyCompactOpts(opts)
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
func (op CompactOp) toRequest() *pb.CompactionRequest {
|
||||||
|
return &pb.CompactionRequest{Revision: op.revision, Physical: op.physical}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithCompactPhysical makes Compact wait until all compacted entries are
|
||||||
|
// removed from the etcd server's storage.
|
||||||
|
func WithCompactPhysical() CompactOption {
|
||||||
|
return func(op *CompactOp) { op.physical = true }
|
||||||
|
}
|
140
vendor/github.com/coreos/etcd/clientv3/compare.go
generated
vendored
Normal file
140
vendor/github.com/coreos/etcd/clientv3/compare.go
generated
vendored
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CompareTarget int
|
||||||
|
type CompareResult int
|
||||||
|
|
||||||
|
const (
|
||||||
|
CompareVersion CompareTarget = iota
|
||||||
|
CompareCreated
|
||||||
|
CompareModified
|
||||||
|
CompareValue
|
||||||
|
)
|
||||||
|
|
||||||
|
type Cmp pb.Compare
|
||||||
|
|
||||||
|
func Compare(cmp Cmp, result string, v interface{}) Cmp {
|
||||||
|
var r pb.Compare_CompareResult
|
||||||
|
|
||||||
|
switch result {
|
||||||
|
case "=":
|
||||||
|
r = pb.Compare_EQUAL
|
||||||
|
case "!=":
|
||||||
|
r = pb.Compare_NOT_EQUAL
|
||||||
|
case ">":
|
||||||
|
r = pb.Compare_GREATER
|
||||||
|
case "<":
|
||||||
|
r = pb.Compare_LESS
|
||||||
|
default:
|
||||||
|
panic("Unknown result op")
|
||||||
|
}
|
||||||
|
|
||||||
|
cmp.Result = r
|
||||||
|
switch cmp.Target {
|
||||||
|
case pb.Compare_VALUE:
|
||||||
|
val, ok := v.(string)
|
||||||
|
if !ok {
|
||||||
|
panic("bad compare value")
|
||||||
|
}
|
||||||
|
cmp.TargetUnion = &pb.Compare_Value{Value: []byte(val)}
|
||||||
|
case pb.Compare_VERSION:
|
||||||
|
cmp.TargetUnion = &pb.Compare_Version{Version: mustInt64(v)}
|
||||||
|
case pb.Compare_CREATE:
|
||||||
|
cmp.TargetUnion = &pb.Compare_CreateRevision{CreateRevision: mustInt64(v)}
|
||||||
|
case pb.Compare_MOD:
|
||||||
|
cmp.TargetUnion = &pb.Compare_ModRevision{ModRevision: mustInt64(v)}
|
||||||
|
case pb.Compare_LEASE:
|
||||||
|
cmp.TargetUnion = &pb.Compare_Lease{Lease: mustInt64orLeaseID(v)}
|
||||||
|
default:
|
||||||
|
panic("Unknown compare type")
|
||||||
|
}
|
||||||
|
return cmp
|
||||||
|
}
|
||||||
|
|
||||||
|
func Value(key string) Cmp {
|
||||||
|
return Cmp{Key: []byte(key), Target: pb.Compare_VALUE}
|
||||||
|
}
|
||||||
|
|
||||||
|
func Version(key string) Cmp {
|
||||||
|
return Cmp{Key: []byte(key), Target: pb.Compare_VERSION}
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateRevision(key string) Cmp {
|
||||||
|
return Cmp{Key: []byte(key), Target: pb.Compare_CREATE}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ModRevision(key string) Cmp {
|
||||||
|
return Cmp{Key: []byte(key), Target: pb.Compare_MOD}
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseValue compares a key's LeaseID to a value of your choosing. The empty
|
||||||
|
// LeaseID is 0, otherwise known as `NoLease`.
|
||||||
|
func LeaseValue(key string) Cmp {
|
||||||
|
return Cmp{Key: []byte(key), Target: pb.Compare_LEASE}
|
||||||
|
}
|
||||||
|
|
||||||
|
// KeyBytes returns the byte slice holding with the comparison key.
|
||||||
|
func (cmp *Cmp) KeyBytes() []byte { return cmp.Key }
|
||||||
|
|
||||||
|
// WithKeyBytes sets the byte slice for the comparison key.
|
||||||
|
func (cmp *Cmp) WithKeyBytes(key []byte) { cmp.Key = key }
|
||||||
|
|
||||||
|
// ValueBytes returns the byte slice holding the comparison value, if any.
|
||||||
|
func (cmp *Cmp) ValueBytes() []byte {
|
||||||
|
if tu, ok := cmp.TargetUnion.(*pb.Compare_Value); ok {
|
||||||
|
return tu.Value
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithValueBytes sets the byte slice for the comparison's value.
|
||||||
|
func (cmp *Cmp) WithValueBytes(v []byte) { cmp.TargetUnion.(*pb.Compare_Value).Value = v }
|
||||||
|
|
||||||
|
// WithRange sets the comparison to scan the range [key, end).
|
||||||
|
func (cmp Cmp) WithRange(end string) Cmp {
|
||||||
|
cmp.RangeEnd = []byte(end)
|
||||||
|
return cmp
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithPrefix sets the comparison to scan all keys prefixed by the key.
|
||||||
|
func (cmp Cmp) WithPrefix() Cmp {
|
||||||
|
cmp.RangeEnd = getPrefix(cmp.Key)
|
||||||
|
return cmp
|
||||||
|
}
|
||||||
|
|
||||||
|
// mustInt64 panics if val isn't an int or int64. It returns an int64 otherwise.
|
||||||
|
func mustInt64(val interface{}) int64 {
|
||||||
|
if v, ok := val.(int64); ok {
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
if v, ok := val.(int); ok {
|
||||||
|
return int64(v)
|
||||||
|
}
|
||||||
|
panic("bad value")
|
||||||
|
}
|
||||||
|
|
||||||
|
// mustInt64orLeaseID panics if val isn't a LeaseID, int or int64. It returns an
|
||||||
|
// int64 otherwise.
|
||||||
|
func mustInt64orLeaseID(val interface{}) int64 {
|
||||||
|
if v, ok := val.(LeaseID); ok {
|
||||||
|
return int64(v)
|
||||||
|
}
|
||||||
|
return mustInt64(val)
|
||||||
|
}
|
99
vendor/github.com/coreos/etcd/clientv3/config.go
generated
vendored
Normal file
99
vendor/github.com/coreos/etcd/clientv3/config.go
generated
vendored
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/tls"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
// Endpoints is a list of URLs.
|
||||||
|
Endpoints []string `json:"endpoints"`
|
||||||
|
|
||||||
|
// AutoSyncInterval is the interval to update endpoints with its latest members.
|
||||||
|
// 0 disables auto-sync. By default auto-sync is disabled.
|
||||||
|
AutoSyncInterval time.Duration `json:"auto-sync-interval"`
|
||||||
|
|
||||||
|
// DialTimeout is the timeout for failing to establish a connection.
|
||||||
|
DialTimeout time.Duration `json:"dial-timeout"`
|
||||||
|
|
||||||
|
// DialKeepAliveTime is the time after which client pings the server to see if
|
||||||
|
// transport is alive.
|
||||||
|
DialKeepAliveTime time.Duration `json:"dial-keep-alive-time"`
|
||||||
|
|
||||||
|
// DialKeepAliveTimeout is the time that the client waits for a response for the
|
||||||
|
// keep-alive probe. If the response is not received in this time, the connection is closed.
|
||||||
|
DialKeepAliveTimeout time.Duration `json:"dial-keep-alive-timeout"`
|
||||||
|
|
||||||
|
// MaxCallSendMsgSize is the client-side request send limit in bytes.
|
||||||
|
// If 0, it defaults to 2.0 MiB (2 * 1024 * 1024).
|
||||||
|
// Make sure that "MaxCallSendMsgSize" < server-side default send/recv limit.
|
||||||
|
// ("--max-request-bytes" flag to etcd or "embed.Config.MaxRequestBytes").
|
||||||
|
MaxCallSendMsgSize int
|
||||||
|
|
||||||
|
// MaxCallRecvMsgSize is the client-side response receive limit.
|
||||||
|
// If 0, it defaults to "math.MaxInt32", because range response can
|
||||||
|
// easily exceed request send limits.
|
||||||
|
// Make sure that "MaxCallRecvMsgSize" >= server-side default send/recv limit.
|
||||||
|
// ("--max-request-bytes" flag to etcd or "embed.Config.MaxRequestBytes").
|
||||||
|
MaxCallRecvMsgSize int
|
||||||
|
|
||||||
|
// TLS holds the client secure credentials, if any.
|
||||||
|
TLS *tls.Config
|
||||||
|
|
||||||
|
// Username is a user name for authentication.
|
||||||
|
Username string `json:"username"`
|
||||||
|
|
||||||
|
// Password is a password for authentication.
|
||||||
|
Password string `json:"password"`
|
||||||
|
|
||||||
|
// RejectOldCluster when set will refuse to create a client against an outdated cluster.
|
||||||
|
RejectOldCluster bool `json:"reject-old-cluster"`
|
||||||
|
|
||||||
|
// DialOptions is a list of dial options for the grpc client (e.g., for interceptors).
|
||||||
|
DialOptions []grpc.DialOption
|
||||||
|
|
||||||
|
// Context is the default client context; it can be used to cancel grpc dial out and
|
||||||
|
// other operations that do not have an explicit context.
|
||||||
|
Context context.Context
|
||||||
|
|
||||||
|
// LogConfig configures client-side logger.
|
||||||
|
// If nil, use the default logger.
|
||||||
|
// TODO: configure gRPC logger
|
||||||
|
LogConfig *zap.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultLogConfig is the default client logging configuration.
|
||||||
|
// Default log level is "Warn". Use "zap.InfoLevel" for debugging.
|
||||||
|
// Use "/dev/null" for output paths, to discard all logs.
|
||||||
|
var DefaultLogConfig = zap.Config{
|
||||||
|
Level: zap.NewAtomicLevelAt(zap.WarnLevel),
|
||||||
|
Development: false,
|
||||||
|
Sampling: &zap.SamplingConfig{
|
||||||
|
Initial: 100,
|
||||||
|
Thereafter: 100,
|
||||||
|
},
|
||||||
|
Encoding: "json",
|
||||||
|
EncoderConfig: zap.NewProductionEncoderConfig(),
|
||||||
|
|
||||||
|
// Use "/dev/null" to discard all
|
||||||
|
OutputPaths: []string{"stderr"},
|
||||||
|
ErrorOutputPaths: []string{"stderr"},
|
||||||
|
}
|
106
vendor/github.com/coreos/etcd/clientv3/doc.go
generated
vendored
Normal file
106
vendor/github.com/coreos/etcd/clientv3/doc.go
generated
vendored
Normal file
@ -0,0 +1,106 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package clientv3 implements the official Go etcd client for v3.
|
||||||
|
//
|
||||||
|
// Create client using `clientv3.New`:
|
||||||
|
//
|
||||||
|
// // expect dial time-out on ipv4 blackhole
|
||||||
|
// _, err := clientv3.New(clientv3.Config{
|
||||||
|
// Endpoints: []string{"http://254.0.0.1:12345"},
|
||||||
|
// DialTimeout: 2 * time.Second
|
||||||
|
// })
|
||||||
|
//
|
||||||
|
// // etcd clientv3 >= v3.2.10, grpc/grpc-go >= v1.7.3
|
||||||
|
// if err == context.DeadlineExceeded {
|
||||||
|
// // handle errors
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// // etcd clientv3 <= v3.2.9, grpc/grpc-go <= v1.2.1
|
||||||
|
// if err == grpc.ErrClientConnTimeout {
|
||||||
|
// // handle errors
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// cli, err := clientv3.New(clientv3.Config{
|
||||||
|
// Endpoints: []string{"localhost:2379", "localhost:22379", "localhost:32379"},
|
||||||
|
// DialTimeout: 5 * time.Second,
|
||||||
|
// })
|
||||||
|
// if err != nil {
|
||||||
|
// // handle error!
|
||||||
|
// }
|
||||||
|
// defer cli.Close()
|
||||||
|
//
|
||||||
|
// Make sure to close the client after using it. If the client is not closed, the
|
||||||
|
// connection will have leaky goroutines.
|
||||||
|
//
|
||||||
|
// To specify a client request timeout, wrap the context with context.WithTimeout:
|
||||||
|
//
|
||||||
|
// ctx, cancel := context.WithTimeout(context.Background(), timeout)
|
||||||
|
// resp, err := kvc.Put(ctx, "sample_key", "sample_value")
|
||||||
|
// cancel()
|
||||||
|
// if err != nil {
|
||||||
|
// // handle error!
|
||||||
|
// }
|
||||||
|
// // use the response
|
||||||
|
//
|
||||||
|
// The Client has internal state (watchers and leases), so Clients should be reused instead of created as needed.
|
||||||
|
// Clients are safe for concurrent use by multiple goroutines.
|
||||||
|
//
|
||||||
|
// etcd client returns 3 types of errors:
|
||||||
|
//
|
||||||
|
// 1. context error: canceled or deadline exceeded.
|
||||||
|
// 2. gRPC status error: e.g. when clock drifts in server-side before client's context deadline exceeded.
|
||||||
|
// 3. gRPC error: see https://github.com/coreos/etcd/blob/master/etcdserver/api/v3rpc/rpctypes/error.go
|
||||||
|
//
|
||||||
|
// Here is the example code to handle client errors:
|
||||||
|
//
|
||||||
|
// resp, err := kvc.Put(ctx, "", "")
|
||||||
|
// if err != nil {
|
||||||
|
// if err == context.Canceled {
|
||||||
|
// // ctx is canceled by another routine
|
||||||
|
// } else if err == context.DeadlineExceeded {
|
||||||
|
// // ctx is attached with a deadline and it exceeded
|
||||||
|
// } else if ev, ok := status.FromError(err); ok {
|
||||||
|
// code := ev.Code()
|
||||||
|
// if code == codes.DeadlineExceeded {
|
||||||
|
// // server-side context might have timed-out first (due to clock skew)
|
||||||
|
// // while original client-side context is not timed-out yet
|
||||||
|
// }
|
||||||
|
// } else if verr, ok := err.(*v3rpc.ErrEmptyKey); ok {
|
||||||
|
// // process (verr.Errors)
|
||||||
|
// } else {
|
||||||
|
// // bad cluster endpoints, which are not etcd servers
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// go func() { cli.Close() }()
|
||||||
|
// _, err := kvc.Get(ctx, "a")
|
||||||
|
// if err != nil {
|
||||||
|
// // with etcd clientv3 <= v3.3
|
||||||
|
// if err == context.Canceled {
|
||||||
|
// // grpc balancer calls 'Get' with an inflight client.Close
|
||||||
|
// } else if err == grpc.ErrClientConnClosing {
|
||||||
|
// // grpc balancer calls 'Get' after client.Close.
|
||||||
|
// }
|
||||||
|
// // with etcd clientv3 >= v3.4
|
||||||
|
// if clientv3.IsConnCanceled(err) {
|
||||||
|
// // gRPC client connection is closed
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// The grpc load balancer is registered statically and is shared across etcd clients.
|
||||||
|
// To enable detailed load balancer logging, set the ETCD_CLIENT_DEBUG environment
|
||||||
|
// variable. E.g. "ETCD_CLIENT_DEBUG=1".
|
||||||
|
//
|
||||||
|
package clientv3
|
177
vendor/github.com/coreos/etcd/clientv3/kv.go
generated
vendored
Normal file
177
vendor/github.com/coreos/etcd/clientv3/kv.go
generated
vendored
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
CompactResponse pb.CompactionResponse
|
||||||
|
PutResponse pb.PutResponse
|
||||||
|
GetResponse pb.RangeResponse
|
||||||
|
DeleteResponse pb.DeleteRangeResponse
|
||||||
|
TxnResponse pb.TxnResponse
|
||||||
|
)
|
||||||
|
|
||||||
|
type KV interface {
|
||||||
|
// Put puts a key-value pair into etcd.
|
||||||
|
// Note that key,value can be plain bytes array and string is
|
||||||
|
// an immutable representation of that bytes array.
|
||||||
|
// To get a string of bytes, do string([]byte{0x10, 0x20}).
|
||||||
|
Put(ctx context.Context, key, val string, opts ...OpOption) (*PutResponse, error)
|
||||||
|
|
||||||
|
// Get retrieves keys.
|
||||||
|
// By default, Get will return the value for "key", if any.
|
||||||
|
// When passed WithRange(end), Get will return the keys in the range [key, end).
|
||||||
|
// When passed WithFromKey(), Get returns keys greater than or equal to key.
|
||||||
|
// When passed WithRev(rev) with rev > 0, Get retrieves keys at the given revision;
|
||||||
|
// if the required revision is compacted, the request will fail with ErrCompacted .
|
||||||
|
// When passed WithLimit(limit), the number of returned keys is bounded by limit.
|
||||||
|
// When passed WithSort(), the keys will be sorted.
|
||||||
|
Get(ctx context.Context, key string, opts ...OpOption) (*GetResponse, error)
|
||||||
|
|
||||||
|
// Delete deletes a key, or optionally using WithRange(end), [key, end).
|
||||||
|
Delete(ctx context.Context, key string, opts ...OpOption) (*DeleteResponse, error)
|
||||||
|
|
||||||
|
// Compact compacts etcd KV history before the given rev.
|
||||||
|
Compact(ctx context.Context, rev int64, opts ...CompactOption) (*CompactResponse, error)
|
||||||
|
|
||||||
|
// Do applies a single Op on KV without a transaction.
|
||||||
|
// Do is useful when creating arbitrary operations to be issued at a
|
||||||
|
// later time; the user can range over the operations, calling Do to
|
||||||
|
// execute them. Get/Put/Delete, on the other hand, are best suited
|
||||||
|
// for when the operation should be issued at the time of declaration.
|
||||||
|
Do(ctx context.Context, op Op) (OpResponse, error)
|
||||||
|
|
||||||
|
// Txn creates a transaction.
|
||||||
|
Txn(ctx context.Context) Txn
|
||||||
|
}
|
||||||
|
|
||||||
|
type OpResponse struct {
|
||||||
|
put *PutResponse
|
||||||
|
get *GetResponse
|
||||||
|
del *DeleteResponse
|
||||||
|
txn *TxnResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
func (op OpResponse) Put() *PutResponse { return op.put }
|
||||||
|
func (op OpResponse) Get() *GetResponse { return op.get }
|
||||||
|
func (op OpResponse) Del() *DeleteResponse { return op.del }
|
||||||
|
func (op OpResponse) Txn() *TxnResponse { return op.txn }
|
||||||
|
|
||||||
|
func (resp *PutResponse) OpResponse() OpResponse {
|
||||||
|
return OpResponse{put: resp}
|
||||||
|
}
|
||||||
|
func (resp *GetResponse) OpResponse() OpResponse {
|
||||||
|
return OpResponse{get: resp}
|
||||||
|
}
|
||||||
|
func (resp *DeleteResponse) OpResponse() OpResponse {
|
||||||
|
return OpResponse{del: resp}
|
||||||
|
}
|
||||||
|
func (resp *TxnResponse) OpResponse() OpResponse {
|
||||||
|
return OpResponse{txn: resp}
|
||||||
|
}
|
||||||
|
|
||||||
|
type kv struct {
|
||||||
|
remote pb.KVClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewKV(c *Client) KV {
|
||||||
|
api := &kv{remote: RetryKVClient(c)}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewKVFromKVClient(remote pb.KVClient, c *Client) KV {
|
||||||
|
api := &kv{remote: remote}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func (kv *kv) Put(ctx context.Context, key, val string, opts ...OpOption) (*PutResponse, error) {
|
||||||
|
r, err := kv.Do(ctx, OpPut(key, val, opts...))
|
||||||
|
return r.put, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (kv *kv) Get(ctx context.Context, key string, opts ...OpOption) (*GetResponse, error) {
|
||||||
|
r, err := kv.Do(ctx, OpGet(key, opts...))
|
||||||
|
return r.get, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (kv *kv) Delete(ctx context.Context, key string, opts ...OpOption) (*DeleteResponse, error) {
|
||||||
|
r, err := kv.Do(ctx, OpDelete(key, opts...))
|
||||||
|
return r.del, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (kv *kv) Compact(ctx context.Context, rev int64, opts ...CompactOption) (*CompactResponse, error) {
|
||||||
|
resp, err := kv.remote.Compact(ctx, OpCompact(rev, opts...).toRequest(), kv.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
return (*CompactResponse)(resp), err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (kv *kv) Txn(ctx context.Context) Txn {
|
||||||
|
return &txn{
|
||||||
|
kv: kv,
|
||||||
|
ctx: ctx,
|
||||||
|
callOpts: kv.callOpts,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (kv *kv) Do(ctx context.Context, op Op) (OpResponse, error) {
|
||||||
|
var err error
|
||||||
|
switch op.t {
|
||||||
|
case tRange:
|
||||||
|
var resp *pb.RangeResponse
|
||||||
|
resp, err = kv.remote.Range(ctx, op.toRangeRequest(), kv.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return OpResponse{get: (*GetResponse)(resp)}, nil
|
||||||
|
}
|
||||||
|
case tPut:
|
||||||
|
var resp *pb.PutResponse
|
||||||
|
r := &pb.PutRequest{Key: op.key, Value: op.val, Lease: int64(op.leaseID), PrevKv: op.prevKV, IgnoreValue: op.ignoreValue, IgnoreLease: op.ignoreLease}
|
||||||
|
resp, err = kv.remote.Put(ctx, r, kv.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return OpResponse{put: (*PutResponse)(resp)}, nil
|
||||||
|
}
|
||||||
|
case tDeleteRange:
|
||||||
|
var resp *pb.DeleteRangeResponse
|
||||||
|
r := &pb.DeleteRangeRequest{Key: op.key, RangeEnd: op.end, PrevKv: op.prevKV}
|
||||||
|
resp, err = kv.remote.DeleteRange(ctx, r, kv.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return OpResponse{del: (*DeleteResponse)(resp)}, nil
|
||||||
|
}
|
||||||
|
case tTxn:
|
||||||
|
var resp *pb.TxnResponse
|
||||||
|
resp, err = kv.remote.Txn(ctx, op.toTxnRequest(), kv.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return OpResponse{txn: (*TxnResponse)(resp)}, nil
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
panic("Unknown op")
|
||||||
|
}
|
||||||
|
return OpResponse{}, toErr(ctx, err)
|
||||||
|
}
|
584
vendor/github.com/coreos/etcd/clientv3/lease.go
generated
vendored
Normal file
584
vendor/github.com/coreos/etcd/clientv3/lease.go
generated
vendored
Normal file
@ -0,0 +1,584 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/metadata"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
LeaseRevokeResponse pb.LeaseRevokeResponse
|
||||||
|
LeaseID int64
|
||||||
|
)
|
||||||
|
|
||||||
|
// LeaseGrantResponse wraps the protobuf message LeaseGrantResponse.
|
||||||
|
type LeaseGrantResponse struct {
|
||||||
|
*pb.ResponseHeader
|
||||||
|
ID LeaseID
|
||||||
|
TTL int64
|
||||||
|
Error string
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseKeepAliveResponse wraps the protobuf message LeaseKeepAliveResponse.
|
||||||
|
type LeaseKeepAliveResponse struct {
|
||||||
|
*pb.ResponseHeader
|
||||||
|
ID LeaseID
|
||||||
|
TTL int64
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseTimeToLiveResponse wraps the protobuf message LeaseTimeToLiveResponse.
|
||||||
|
type LeaseTimeToLiveResponse struct {
|
||||||
|
*pb.ResponseHeader
|
||||||
|
ID LeaseID `json:"id"`
|
||||||
|
|
||||||
|
// TTL is the remaining TTL in seconds for the lease; the lease will expire in under TTL+1 seconds. Expired lease will return -1.
|
||||||
|
TTL int64 `json:"ttl"`
|
||||||
|
|
||||||
|
// GrantedTTL is the initial granted time in seconds upon lease creation/renewal.
|
||||||
|
GrantedTTL int64 `json:"granted-ttl"`
|
||||||
|
|
||||||
|
// Keys is the list of keys attached to this lease.
|
||||||
|
Keys [][]byte `json:"keys"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseStatus represents a lease status.
|
||||||
|
type LeaseStatus struct {
|
||||||
|
ID LeaseID `json:"id"`
|
||||||
|
// TODO: TTL int64
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseLeasesResponse wraps the protobuf message LeaseLeasesResponse.
|
||||||
|
type LeaseLeasesResponse struct {
|
||||||
|
*pb.ResponseHeader
|
||||||
|
Leases []LeaseStatus `json:"leases"`
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// defaultTTL is the assumed lease TTL used for the first keepalive
|
||||||
|
// deadline before the actual TTL is known to the client.
|
||||||
|
defaultTTL = 5 * time.Second
|
||||||
|
// a small buffer to store unsent lease responses.
|
||||||
|
leaseResponseChSize = 16
|
||||||
|
// NoLease is a lease ID for the absence of a lease.
|
||||||
|
NoLease LeaseID = 0
|
||||||
|
|
||||||
|
// retryConnWait is how long to wait before retrying request due to an error
|
||||||
|
retryConnWait = 500 * time.Millisecond
|
||||||
|
)
|
||||||
|
|
||||||
|
// ErrKeepAliveHalted is returned if client keep alive loop halts with an unexpected error.
|
||||||
|
//
|
||||||
|
// This usually means that automatic lease renewal via KeepAlive is broken, but KeepAliveOnce will still work as expected.
|
||||||
|
type ErrKeepAliveHalted struct {
|
||||||
|
Reason error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e ErrKeepAliveHalted) Error() string {
|
||||||
|
s := "etcdclient: leases keep alive halted"
|
||||||
|
if e.Reason != nil {
|
||||||
|
s += ": " + e.Reason.Error()
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
type Lease interface {
|
||||||
|
// Grant creates a new lease.
|
||||||
|
Grant(ctx context.Context, ttl int64) (*LeaseGrantResponse, error)
|
||||||
|
|
||||||
|
// Revoke revokes the given lease.
|
||||||
|
Revoke(ctx context.Context, id LeaseID) (*LeaseRevokeResponse, error)
|
||||||
|
|
||||||
|
// TimeToLive retrieves the lease information of the given lease ID.
|
||||||
|
TimeToLive(ctx context.Context, id LeaseID, opts ...LeaseOption) (*LeaseTimeToLiveResponse, error)
|
||||||
|
|
||||||
|
// Leases retrieves all leases.
|
||||||
|
Leases(ctx context.Context) (*LeaseLeasesResponse, error)
|
||||||
|
|
||||||
|
// KeepAlive keeps the given lease alive forever. If the keepalive response
|
||||||
|
// posted to the channel is not consumed immediately, the lease client will
|
||||||
|
// continue sending keep alive requests to the etcd server at least every
|
||||||
|
// second until latest response is consumed.
|
||||||
|
//
|
||||||
|
// The returned "LeaseKeepAliveResponse" channel closes if underlying keep
|
||||||
|
// alive stream is interrupted in some way the client cannot handle itself;
|
||||||
|
// given context "ctx" is canceled or timed out. "LeaseKeepAliveResponse"
|
||||||
|
// from this closed channel is nil.
|
||||||
|
//
|
||||||
|
// If client keep alive loop halts with an unexpected error (e.g. "etcdserver:
|
||||||
|
// no leader") or canceled by the caller (e.g. context.Canceled), the error
|
||||||
|
// is returned. Otherwise, it retries.
|
||||||
|
//
|
||||||
|
// TODO(v4.0): post errors to last keep alive message before closing
|
||||||
|
// (see https://github.com/coreos/etcd/pull/7866)
|
||||||
|
KeepAlive(ctx context.Context, id LeaseID) (<-chan *LeaseKeepAliveResponse, error)
|
||||||
|
|
||||||
|
// KeepAliveOnce renews the lease once. The response corresponds to the
|
||||||
|
// first message from calling KeepAlive. If the response has a recoverable
|
||||||
|
// error, KeepAliveOnce will retry the RPC with a new keep alive message.
|
||||||
|
//
|
||||||
|
// In most of the cases, Keepalive should be used instead of KeepAliveOnce.
|
||||||
|
KeepAliveOnce(ctx context.Context, id LeaseID) (*LeaseKeepAliveResponse, error)
|
||||||
|
|
||||||
|
// Close releases all resources Lease keeps for efficient communication
|
||||||
|
// with the etcd server.
|
||||||
|
Close() error
|
||||||
|
}
|
||||||
|
|
||||||
|
type lessor struct {
|
||||||
|
mu sync.Mutex // guards all fields
|
||||||
|
|
||||||
|
// donec is closed and loopErr is set when recvKeepAliveLoop stops
|
||||||
|
donec chan struct{}
|
||||||
|
loopErr error
|
||||||
|
|
||||||
|
remote pb.LeaseClient
|
||||||
|
|
||||||
|
stream pb.Lease_LeaseKeepAliveClient
|
||||||
|
streamCancel context.CancelFunc
|
||||||
|
|
||||||
|
stopCtx context.Context
|
||||||
|
stopCancel context.CancelFunc
|
||||||
|
|
||||||
|
keepAlives map[LeaseID]*keepAlive
|
||||||
|
|
||||||
|
// firstKeepAliveTimeout is the timeout for the first keepalive request
|
||||||
|
// before the actual TTL is known to the lease client
|
||||||
|
firstKeepAliveTimeout time.Duration
|
||||||
|
|
||||||
|
// firstKeepAliveOnce ensures stream starts after first KeepAlive call.
|
||||||
|
firstKeepAliveOnce sync.Once
|
||||||
|
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
// keepAlive multiplexes a keepalive for a lease over multiple channels
|
||||||
|
type keepAlive struct {
|
||||||
|
chs []chan<- *LeaseKeepAliveResponse
|
||||||
|
ctxs []context.Context
|
||||||
|
// deadline is the time the keep alive channels close if no response
|
||||||
|
deadline time.Time
|
||||||
|
// nextKeepAlive is when to send the next keep alive message
|
||||||
|
nextKeepAlive time.Time
|
||||||
|
// donec is closed on lease revoke, expiration, or cancel.
|
||||||
|
donec chan struct{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewLease(c *Client) Lease {
|
||||||
|
return NewLeaseFromLeaseClient(RetryLeaseClient(c), c, c.cfg.DialTimeout+time.Second)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewLeaseFromLeaseClient(remote pb.LeaseClient, c *Client, keepAliveTimeout time.Duration) Lease {
|
||||||
|
l := &lessor{
|
||||||
|
donec: make(chan struct{}),
|
||||||
|
keepAlives: make(map[LeaseID]*keepAlive),
|
||||||
|
remote: remote,
|
||||||
|
firstKeepAliveTimeout: keepAliveTimeout,
|
||||||
|
}
|
||||||
|
if l.firstKeepAliveTimeout == time.Second {
|
||||||
|
l.firstKeepAliveTimeout = defaultTTL
|
||||||
|
}
|
||||||
|
if c != nil {
|
||||||
|
l.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
reqLeaderCtx := WithRequireLeader(context.Background())
|
||||||
|
l.stopCtx, l.stopCancel = context.WithCancel(reqLeaderCtx)
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) Grant(ctx context.Context, ttl int64) (*LeaseGrantResponse, error) {
|
||||||
|
r := &pb.LeaseGrantRequest{TTL: ttl}
|
||||||
|
resp, err := l.remote.LeaseGrant(ctx, r, l.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
gresp := &LeaseGrantResponse{
|
||||||
|
ResponseHeader: resp.GetHeader(),
|
||||||
|
ID: LeaseID(resp.ID),
|
||||||
|
TTL: resp.TTL,
|
||||||
|
Error: resp.Error,
|
||||||
|
}
|
||||||
|
return gresp, nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) Revoke(ctx context.Context, id LeaseID) (*LeaseRevokeResponse, error) {
|
||||||
|
r := &pb.LeaseRevokeRequest{ID: int64(id)}
|
||||||
|
resp, err := l.remote.LeaseRevoke(ctx, r, l.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return (*LeaseRevokeResponse)(resp), nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) TimeToLive(ctx context.Context, id LeaseID, opts ...LeaseOption) (*LeaseTimeToLiveResponse, error) {
|
||||||
|
r := toLeaseTimeToLiveRequest(id, opts...)
|
||||||
|
resp, err := l.remote.LeaseTimeToLive(ctx, r, l.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
gresp := &LeaseTimeToLiveResponse{
|
||||||
|
ResponseHeader: resp.GetHeader(),
|
||||||
|
ID: LeaseID(resp.ID),
|
||||||
|
TTL: resp.TTL,
|
||||||
|
GrantedTTL: resp.GrantedTTL,
|
||||||
|
Keys: resp.Keys,
|
||||||
|
}
|
||||||
|
return gresp, nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) Leases(ctx context.Context) (*LeaseLeasesResponse, error) {
|
||||||
|
resp, err := l.remote.LeaseLeases(ctx, &pb.LeaseLeasesRequest{}, l.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
leases := make([]LeaseStatus, len(resp.Leases))
|
||||||
|
for i := range resp.Leases {
|
||||||
|
leases[i] = LeaseStatus{ID: LeaseID(resp.Leases[i].ID)}
|
||||||
|
}
|
||||||
|
return &LeaseLeasesResponse{ResponseHeader: resp.GetHeader(), Leases: leases}, nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) KeepAlive(ctx context.Context, id LeaseID) (<-chan *LeaseKeepAliveResponse, error) {
|
||||||
|
ch := make(chan *LeaseKeepAliveResponse, leaseResponseChSize)
|
||||||
|
|
||||||
|
l.mu.Lock()
|
||||||
|
// ensure that recvKeepAliveLoop is still running
|
||||||
|
select {
|
||||||
|
case <-l.donec:
|
||||||
|
err := l.loopErr
|
||||||
|
l.mu.Unlock()
|
||||||
|
close(ch)
|
||||||
|
return ch, ErrKeepAliveHalted{Reason: err}
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
ka, ok := l.keepAlives[id]
|
||||||
|
if !ok {
|
||||||
|
// create fresh keep alive
|
||||||
|
ka = &keepAlive{
|
||||||
|
chs: []chan<- *LeaseKeepAliveResponse{ch},
|
||||||
|
ctxs: []context.Context{ctx},
|
||||||
|
deadline: time.Now().Add(l.firstKeepAliveTimeout),
|
||||||
|
nextKeepAlive: time.Now(),
|
||||||
|
donec: make(chan struct{}),
|
||||||
|
}
|
||||||
|
l.keepAlives[id] = ka
|
||||||
|
} else {
|
||||||
|
// add channel and context to existing keep alive
|
||||||
|
ka.ctxs = append(ka.ctxs, ctx)
|
||||||
|
ka.chs = append(ka.chs, ch)
|
||||||
|
}
|
||||||
|
l.mu.Unlock()
|
||||||
|
|
||||||
|
go l.keepAliveCtxCloser(id, ctx, ka.donec)
|
||||||
|
l.firstKeepAliveOnce.Do(func() {
|
||||||
|
go l.recvKeepAliveLoop()
|
||||||
|
go l.deadlineLoop()
|
||||||
|
})
|
||||||
|
|
||||||
|
return ch, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) KeepAliveOnce(ctx context.Context, id LeaseID) (*LeaseKeepAliveResponse, error) {
|
||||||
|
for {
|
||||||
|
resp, err := l.keepAliveOnce(ctx, id)
|
||||||
|
if err == nil {
|
||||||
|
if resp.TTL <= 0 {
|
||||||
|
err = rpctypes.ErrLeaseNotFound
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
if isHaltErr(ctx, err) {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) Close() error {
|
||||||
|
l.stopCancel()
|
||||||
|
// close for synchronous teardown if stream goroutines never launched
|
||||||
|
l.firstKeepAliveOnce.Do(func() { close(l.donec) })
|
||||||
|
<-l.donec
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) keepAliveCtxCloser(id LeaseID, ctx context.Context, donec <-chan struct{}) {
|
||||||
|
select {
|
||||||
|
case <-donec:
|
||||||
|
return
|
||||||
|
case <-l.donec:
|
||||||
|
return
|
||||||
|
case <-ctx.Done():
|
||||||
|
}
|
||||||
|
|
||||||
|
l.mu.Lock()
|
||||||
|
defer l.mu.Unlock()
|
||||||
|
|
||||||
|
ka, ok := l.keepAlives[id]
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// close channel and remove context if still associated with keep alive
|
||||||
|
for i, c := range ka.ctxs {
|
||||||
|
if c == ctx {
|
||||||
|
close(ka.chs[i])
|
||||||
|
ka.ctxs = append(ka.ctxs[:i], ka.ctxs[i+1:]...)
|
||||||
|
ka.chs = append(ka.chs[:i], ka.chs[i+1:]...)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// remove if no one more listeners
|
||||||
|
if len(ka.chs) == 0 {
|
||||||
|
delete(l.keepAlives, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// closeRequireLeader scans keepAlives for ctxs that have require leader
|
||||||
|
// and closes the associated channels.
|
||||||
|
func (l *lessor) closeRequireLeader() {
|
||||||
|
l.mu.Lock()
|
||||||
|
defer l.mu.Unlock()
|
||||||
|
for _, ka := range l.keepAlives {
|
||||||
|
reqIdxs := 0
|
||||||
|
// find all required leader channels, close, mark as nil
|
||||||
|
for i, ctx := range ka.ctxs {
|
||||||
|
md, ok := metadata.FromOutgoingContext(ctx)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
ks := md[rpctypes.MetadataRequireLeaderKey]
|
||||||
|
if len(ks) < 1 || ks[0] != rpctypes.MetadataHasLeader {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
close(ka.chs[i])
|
||||||
|
ka.chs[i] = nil
|
||||||
|
reqIdxs++
|
||||||
|
}
|
||||||
|
if reqIdxs == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// remove all channels that required a leader from keepalive
|
||||||
|
newChs := make([]chan<- *LeaseKeepAliveResponse, len(ka.chs)-reqIdxs)
|
||||||
|
newCtxs := make([]context.Context, len(newChs))
|
||||||
|
newIdx := 0
|
||||||
|
for i := range ka.chs {
|
||||||
|
if ka.chs[i] == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
newChs[newIdx], newCtxs[newIdx] = ka.chs[i], ka.ctxs[newIdx]
|
||||||
|
newIdx++
|
||||||
|
}
|
||||||
|
ka.chs, ka.ctxs = newChs, newCtxs
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) keepAliveOnce(ctx context.Context, id LeaseID) (*LeaseKeepAliveResponse, error) {
|
||||||
|
cctx, cancel := context.WithCancel(ctx)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
stream, err := l.remote.LeaseKeepAlive(cctx, l.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = stream.Send(&pb.LeaseKeepAliveRequest{ID: int64(id)})
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, rerr := stream.Recv()
|
||||||
|
if rerr != nil {
|
||||||
|
return nil, toErr(ctx, rerr)
|
||||||
|
}
|
||||||
|
|
||||||
|
karesp := &LeaseKeepAliveResponse{
|
||||||
|
ResponseHeader: resp.GetHeader(),
|
||||||
|
ID: LeaseID(resp.ID),
|
||||||
|
TTL: resp.TTL,
|
||||||
|
}
|
||||||
|
return karesp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *lessor) recvKeepAliveLoop() (gerr error) {
|
||||||
|
defer func() {
|
||||||
|
l.mu.Lock()
|
||||||
|
close(l.donec)
|
||||||
|
l.loopErr = gerr
|
||||||
|
for _, ka := range l.keepAlives {
|
||||||
|
ka.close()
|
||||||
|
}
|
||||||
|
l.keepAlives = make(map[LeaseID]*keepAlive)
|
||||||
|
l.mu.Unlock()
|
||||||
|
}()
|
||||||
|
|
||||||
|
for {
|
||||||
|
stream, err := l.resetRecv()
|
||||||
|
if err != nil {
|
||||||
|
if canceledByCaller(l.stopCtx, err) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for {
|
||||||
|
resp, err := stream.Recv()
|
||||||
|
if err != nil {
|
||||||
|
if canceledByCaller(l.stopCtx, err) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if toErr(l.stopCtx, err) == rpctypes.ErrNoLeader {
|
||||||
|
l.closeRequireLeader()
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
l.recvKeepAlive(resp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-time.After(retryConnWait):
|
||||||
|
continue
|
||||||
|
case <-l.stopCtx.Done():
|
||||||
|
return l.stopCtx.Err()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// resetRecv opens a new lease stream and starts sending keep alive requests.
|
||||||
|
func (l *lessor) resetRecv() (pb.Lease_LeaseKeepAliveClient, error) {
|
||||||
|
sctx, cancel := context.WithCancel(l.stopCtx)
|
||||||
|
stream, err := l.remote.LeaseKeepAlive(sctx, append(l.callOpts, withMax(0))...)
|
||||||
|
if err != nil {
|
||||||
|
cancel()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
l.mu.Lock()
|
||||||
|
defer l.mu.Unlock()
|
||||||
|
if l.stream != nil && l.streamCancel != nil {
|
||||||
|
l.streamCancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
l.streamCancel = cancel
|
||||||
|
l.stream = stream
|
||||||
|
|
||||||
|
go l.sendKeepAliveLoop(stream)
|
||||||
|
return stream, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// recvKeepAlive updates a lease based on its LeaseKeepAliveResponse
|
||||||
|
func (l *lessor) recvKeepAlive(resp *pb.LeaseKeepAliveResponse) {
|
||||||
|
karesp := &LeaseKeepAliveResponse{
|
||||||
|
ResponseHeader: resp.GetHeader(),
|
||||||
|
ID: LeaseID(resp.ID),
|
||||||
|
TTL: resp.TTL,
|
||||||
|
}
|
||||||
|
|
||||||
|
l.mu.Lock()
|
||||||
|
defer l.mu.Unlock()
|
||||||
|
|
||||||
|
ka, ok := l.keepAlives[karesp.ID]
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if karesp.TTL <= 0 {
|
||||||
|
// lease expired; close all keep alive channels
|
||||||
|
delete(l.keepAlives, karesp.ID)
|
||||||
|
ka.close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// send update to all channels
|
||||||
|
nextKeepAlive := time.Now().Add((time.Duration(karesp.TTL) * time.Second) / 3.0)
|
||||||
|
ka.deadline = time.Now().Add(time.Duration(karesp.TTL) * time.Second)
|
||||||
|
for _, ch := range ka.chs {
|
||||||
|
select {
|
||||||
|
case ch <- karesp:
|
||||||
|
ka.nextKeepAlive = nextKeepAlive
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// deadlineLoop reaps any keep alive channels that have not received a response
|
||||||
|
// within the lease TTL
|
||||||
|
func (l *lessor) deadlineLoop() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-time.After(time.Second):
|
||||||
|
case <-l.donec:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
now := time.Now()
|
||||||
|
l.mu.Lock()
|
||||||
|
for id, ka := range l.keepAlives {
|
||||||
|
if ka.deadline.Before(now) {
|
||||||
|
// waited too long for response; lease may be expired
|
||||||
|
ka.close()
|
||||||
|
delete(l.keepAlives, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
l.mu.Unlock()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// sendKeepAliveLoop sends keep alive requests for the lifetime of the given stream.
|
||||||
|
func (l *lessor) sendKeepAliveLoop(stream pb.Lease_LeaseKeepAliveClient) {
|
||||||
|
for {
|
||||||
|
var tosend []LeaseID
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
l.mu.Lock()
|
||||||
|
for id, ka := range l.keepAlives {
|
||||||
|
if ka.nextKeepAlive.Before(now) {
|
||||||
|
tosend = append(tosend, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
l.mu.Unlock()
|
||||||
|
|
||||||
|
for _, id := range tosend {
|
||||||
|
r := &pb.LeaseKeepAliveRequest{ID: int64(id)}
|
||||||
|
if err := stream.Send(r); err != nil {
|
||||||
|
// TODO do something with this error?
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-time.After(500 * time.Millisecond):
|
||||||
|
case <-stream.Context().Done():
|
||||||
|
return
|
||||||
|
case <-l.donec:
|
||||||
|
return
|
||||||
|
case <-l.stopCtx.Done():
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ka *keepAlive) close() {
|
||||||
|
close(ka.donec)
|
||||||
|
for _, ch := range ka.chs {
|
||||||
|
close(ch)
|
||||||
|
}
|
||||||
|
}
|
101
vendor/github.com/coreos/etcd/clientv3/logger.go
generated
vendored
Normal file
101
vendor/github.com/coreos/etcd/clientv3/logger.go
generated
vendored
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/pkg/logutil"
|
||||||
|
|
||||||
|
"google.golang.org/grpc/grpclog"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
lgMu sync.RWMutex
|
||||||
|
lg logutil.Logger
|
||||||
|
)
|
||||||
|
|
||||||
|
type settableLogger struct {
|
||||||
|
l grpclog.LoggerV2
|
||||||
|
mu sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
// disable client side logs by default
|
||||||
|
lg = &settableLogger{}
|
||||||
|
SetLogger(grpclog.NewLoggerV2(ioutil.Discard, ioutil.Discard, ioutil.Discard))
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetLogger sets client-side Logger.
|
||||||
|
func SetLogger(l grpclog.LoggerV2) {
|
||||||
|
lgMu.Lock()
|
||||||
|
lg = logutil.NewLogger(l)
|
||||||
|
// override grpclog so that any changes happen with locking
|
||||||
|
grpclog.SetLoggerV2(lg)
|
||||||
|
lgMu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetLogger returns the current logutil.Logger.
|
||||||
|
func GetLogger() logutil.Logger {
|
||||||
|
lgMu.RLock()
|
||||||
|
l := lg
|
||||||
|
lgMu.RUnlock()
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewLogger returns a new Logger with logutil.Logger.
|
||||||
|
func NewLogger(gl grpclog.LoggerV2) logutil.Logger {
|
||||||
|
return &settableLogger{l: gl}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *settableLogger) get() grpclog.LoggerV2 {
|
||||||
|
s.mu.RLock()
|
||||||
|
l := s.l
|
||||||
|
s.mu.RUnlock()
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
|
||||||
|
// implement the grpclog.LoggerV2 interface
|
||||||
|
|
||||||
|
func (s *settableLogger) Info(args ...interface{}) { s.get().Info(args...) }
|
||||||
|
func (s *settableLogger) Infof(format string, args ...interface{}) { s.get().Infof(format, args...) }
|
||||||
|
func (s *settableLogger) Infoln(args ...interface{}) { s.get().Infoln(args...) }
|
||||||
|
func (s *settableLogger) Warning(args ...interface{}) { s.get().Warning(args...) }
|
||||||
|
func (s *settableLogger) Warningf(format string, args ...interface{}) {
|
||||||
|
s.get().Warningf(format, args...)
|
||||||
|
}
|
||||||
|
func (s *settableLogger) Warningln(args ...interface{}) { s.get().Warningln(args...) }
|
||||||
|
func (s *settableLogger) Error(args ...interface{}) { s.get().Error(args...) }
|
||||||
|
func (s *settableLogger) Errorf(format string, args ...interface{}) {
|
||||||
|
s.get().Errorf(format, args...)
|
||||||
|
}
|
||||||
|
func (s *settableLogger) Errorln(args ...interface{}) { s.get().Errorln(args...) }
|
||||||
|
func (s *settableLogger) Fatal(args ...interface{}) { s.get().Fatal(args...) }
|
||||||
|
func (s *settableLogger) Fatalf(format string, args ...interface{}) { s.get().Fatalf(format, args...) }
|
||||||
|
func (s *settableLogger) Fatalln(args ...interface{}) { s.get().Fatalln(args...) }
|
||||||
|
func (s *settableLogger) Print(args ...interface{}) { s.get().Info(args...) }
|
||||||
|
func (s *settableLogger) Printf(format string, args ...interface{}) { s.get().Infof(format, args...) }
|
||||||
|
func (s *settableLogger) Println(args ...interface{}) { s.get().Infoln(args...) }
|
||||||
|
func (s *settableLogger) V(l int) bool { return s.get().V(l) }
|
||||||
|
func (s *settableLogger) Lvl(lvl int) grpclog.LoggerV2 {
|
||||||
|
s.mu.RLock()
|
||||||
|
l := s.l
|
||||||
|
s.mu.RUnlock()
|
||||||
|
if l.V(lvl) {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
return logutil.NewDiscardLogger()
|
||||||
|
}
|
230
vendor/github.com/coreos/etcd/clientv3/maintenance.go
generated
vendored
Normal file
230
vendor/github.com/coreos/etcd/clientv3/maintenance.go
generated
vendored
Normal file
@ -0,0 +1,230 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
DefragmentResponse pb.DefragmentResponse
|
||||||
|
AlarmResponse pb.AlarmResponse
|
||||||
|
AlarmMember pb.AlarmMember
|
||||||
|
StatusResponse pb.StatusResponse
|
||||||
|
HashKVResponse pb.HashKVResponse
|
||||||
|
MoveLeaderResponse pb.MoveLeaderResponse
|
||||||
|
)
|
||||||
|
|
||||||
|
type Maintenance interface {
|
||||||
|
// AlarmList gets all active alarms.
|
||||||
|
AlarmList(ctx context.Context) (*AlarmResponse, error)
|
||||||
|
|
||||||
|
// AlarmDisarm disarms a given alarm.
|
||||||
|
AlarmDisarm(ctx context.Context, m *AlarmMember) (*AlarmResponse, error)
|
||||||
|
|
||||||
|
// Defragment releases wasted space from internal fragmentation on a given etcd member.
|
||||||
|
// Defragment is only needed when deleting a large number of keys and want to reclaim
|
||||||
|
// the resources.
|
||||||
|
// Defragment is an expensive operation. User should avoid defragmenting multiple members
|
||||||
|
// at the same time.
|
||||||
|
// To defragment multiple members in the cluster, user need to call defragment multiple
|
||||||
|
// times with different endpoints.
|
||||||
|
Defragment(ctx context.Context, endpoint string) (*DefragmentResponse, error)
|
||||||
|
|
||||||
|
// Status gets the status of the endpoint.
|
||||||
|
Status(ctx context.Context, endpoint string) (*StatusResponse, error)
|
||||||
|
|
||||||
|
// HashKV returns a hash of the KV state at the time of the RPC.
|
||||||
|
// If revision is zero, the hash is computed on all keys. If the revision
|
||||||
|
// is non-zero, the hash is computed on all keys at or below the given revision.
|
||||||
|
HashKV(ctx context.Context, endpoint string, rev int64) (*HashKVResponse, error)
|
||||||
|
|
||||||
|
// Snapshot provides a reader for a point-in-time snapshot of etcd.
|
||||||
|
// If the context "ctx" is canceled or timed out, reading from returned
|
||||||
|
// "io.ReadCloser" would error out (e.g. context.Canceled, context.DeadlineExceeded).
|
||||||
|
Snapshot(ctx context.Context) (io.ReadCloser, error)
|
||||||
|
|
||||||
|
// MoveLeader requests current leader to transfer its leadership to the transferee.
|
||||||
|
// Request must be made to the leader.
|
||||||
|
MoveLeader(ctx context.Context, transfereeID uint64) (*MoveLeaderResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type maintenance struct {
|
||||||
|
dial func(endpoint string) (pb.MaintenanceClient, func(), error)
|
||||||
|
remote pb.MaintenanceClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMaintenance(c *Client) Maintenance {
|
||||||
|
api := &maintenance{
|
||||||
|
dial: func(endpoint string) (pb.MaintenanceClient, func(), error) {
|
||||||
|
conn, err := c.dial(endpoint)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, fmt.Errorf("failed to dial endpoint %s with maintenance client: %v", endpoint, err)
|
||||||
|
}
|
||||||
|
cancel := func() { conn.Close() }
|
||||||
|
return RetryMaintenanceClient(c, conn), cancel, nil
|
||||||
|
},
|
||||||
|
remote: RetryMaintenanceClient(c, c.conn),
|
||||||
|
}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMaintenanceFromMaintenanceClient(remote pb.MaintenanceClient, c *Client) Maintenance {
|
||||||
|
api := &maintenance{
|
||||||
|
dial: func(string) (pb.MaintenanceClient, func(), error) {
|
||||||
|
return remote, func() {}, nil
|
||||||
|
},
|
||||||
|
remote: remote,
|
||||||
|
}
|
||||||
|
if c != nil {
|
||||||
|
api.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return api
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) AlarmList(ctx context.Context) (*AlarmResponse, error) {
|
||||||
|
req := &pb.AlarmRequest{
|
||||||
|
Action: pb.AlarmRequest_GET,
|
||||||
|
MemberID: 0, // all
|
||||||
|
Alarm: pb.AlarmType_NONE, // all
|
||||||
|
}
|
||||||
|
resp, err := m.remote.Alarm(ctx, req, m.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return (*AlarmResponse)(resp), nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) AlarmDisarm(ctx context.Context, am *AlarmMember) (*AlarmResponse, error) {
|
||||||
|
req := &pb.AlarmRequest{
|
||||||
|
Action: pb.AlarmRequest_DEACTIVATE,
|
||||||
|
MemberID: am.MemberID,
|
||||||
|
Alarm: am.Alarm,
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.MemberID == 0 && req.Alarm == pb.AlarmType_NONE {
|
||||||
|
ar, err := m.AlarmList(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
ret := AlarmResponse{}
|
||||||
|
for _, am := range ar.Alarms {
|
||||||
|
dresp, derr := m.AlarmDisarm(ctx, (*AlarmMember)(am))
|
||||||
|
if derr != nil {
|
||||||
|
return nil, toErr(ctx, derr)
|
||||||
|
}
|
||||||
|
ret.Alarms = append(ret.Alarms, dresp.Alarms...)
|
||||||
|
}
|
||||||
|
return &ret, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := m.remote.Alarm(ctx, req, m.callOpts...)
|
||||||
|
if err == nil {
|
||||||
|
return (*AlarmResponse)(resp), nil
|
||||||
|
}
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) Defragment(ctx context.Context, endpoint string) (*DefragmentResponse, error) {
|
||||||
|
remote, cancel, err := m.dial(endpoint)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
defer cancel()
|
||||||
|
resp, err := remote.Defragment(ctx, &pb.DefragmentRequest{}, m.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
return (*DefragmentResponse)(resp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) Status(ctx context.Context, endpoint string) (*StatusResponse, error) {
|
||||||
|
remote, cancel, err := m.dial(endpoint)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
defer cancel()
|
||||||
|
resp, err := remote.Status(ctx, &pb.StatusRequest{}, m.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
return (*StatusResponse)(resp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) HashKV(ctx context.Context, endpoint string, rev int64) (*HashKVResponse, error) {
|
||||||
|
remote, cancel, err := m.dial(endpoint)
|
||||||
|
if err != nil {
|
||||||
|
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
defer cancel()
|
||||||
|
resp, err := remote.HashKV(ctx, &pb.HashKVRequest{Revision: rev}, m.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
return (*HashKVResponse)(resp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) Snapshot(ctx context.Context) (io.ReadCloser, error) {
|
||||||
|
ss, err := m.remote.Snapshot(ctx, &pb.SnapshotRequest{}, append(m.callOpts, withMax(defaultStreamMaxRetries))...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
pr, pw := io.Pipe()
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
resp, err := ss.Recv()
|
||||||
|
if err != nil {
|
||||||
|
pw.CloseWithError(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if resp == nil && err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if _, werr := pw.Write(resp.Blob); werr != nil {
|
||||||
|
pw.CloseWithError(werr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pw.Close()
|
||||||
|
}()
|
||||||
|
return &snapshotReadCloser{ctx: ctx, ReadCloser: pr}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type snapshotReadCloser struct {
|
||||||
|
ctx context.Context
|
||||||
|
io.ReadCloser
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rc *snapshotReadCloser) Read(p []byte) (n int, err error) {
|
||||||
|
n, err = rc.ReadCloser.Read(p)
|
||||||
|
return n, toErr(rc.ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *maintenance) MoveLeader(ctx context.Context, transfereeID uint64) (*MoveLeaderResponse, error) {
|
||||||
|
resp, err := m.remote.MoveLeader(ctx, &pb.MoveLeaderRequest{TargetID: transfereeID}, m.callOpts...)
|
||||||
|
return (*MoveLeaderResponse)(resp), toErr(ctx, err)
|
||||||
|
}
|
539
vendor/github.com/coreos/etcd/clientv3/op.go
generated
vendored
Normal file
539
vendor/github.com/coreos/etcd/clientv3/op.go
generated
vendored
Normal file
@ -0,0 +1,539 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
|
||||||
|
type opType int
|
||||||
|
|
||||||
|
const (
|
||||||
|
// A default Op has opType 0, which is invalid.
|
||||||
|
tRange opType = iota + 1
|
||||||
|
tPut
|
||||||
|
tDeleteRange
|
||||||
|
tTxn
|
||||||
|
)
|
||||||
|
|
||||||
|
var noPrefixEnd = []byte{0}
|
||||||
|
|
||||||
|
// Op represents an Operation that kv can execute.
|
||||||
|
type Op struct {
|
||||||
|
t opType
|
||||||
|
key []byte
|
||||||
|
end []byte
|
||||||
|
|
||||||
|
// for range
|
||||||
|
limit int64
|
||||||
|
sort *SortOption
|
||||||
|
serializable bool
|
||||||
|
keysOnly bool
|
||||||
|
countOnly bool
|
||||||
|
minModRev int64
|
||||||
|
maxModRev int64
|
||||||
|
minCreateRev int64
|
||||||
|
maxCreateRev int64
|
||||||
|
|
||||||
|
// for range, watch
|
||||||
|
rev int64
|
||||||
|
|
||||||
|
// for watch, put, delete
|
||||||
|
prevKV bool
|
||||||
|
|
||||||
|
// for watch
|
||||||
|
// fragmentation should be disabled by default
|
||||||
|
// if true, split watch events when total exceeds
|
||||||
|
// "--max-request-bytes" flag value + 512-byte
|
||||||
|
fragment bool
|
||||||
|
|
||||||
|
// for put
|
||||||
|
ignoreValue bool
|
||||||
|
ignoreLease bool
|
||||||
|
|
||||||
|
// progressNotify is for progress updates.
|
||||||
|
progressNotify bool
|
||||||
|
// createdNotify is for created event
|
||||||
|
createdNotify bool
|
||||||
|
// filters for watchers
|
||||||
|
filterPut bool
|
||||||
|
filterDelete bool
|
||||||
|
|
||||||
|
// for put
|
||||||
|
val []byte
|
||||||
|
leaseID LeaseID
|
||||||
|
|
||||||
|
// txn
|
||||||
|
cmps []Cmp
|
||||||
|
thenOps []Op
|
||||||
|
elseOps []Op
|
||||||
|
}
|
||||||
|
|
||||||
|
// accessors / mutators
|
||||||
|
|
||||||
|
// IsTxn returns true if the "Op" type is transaction.
|
||||||
|
func (op Op) IsTxn() bool {
|
||||||
|
return op.t == tTxn
|
||||||
|
}
|
||||||
|
|
||||||
|
// Txn returns the comparison(if) operations, "then" operations, and "else" operations.
|
||||||
|
func (op Op) Txn() ([]Cmp, []Op, []Op) {
|
||||||
|
return op.cmps, op.thenOps, op.elseOps
|
||||||
|
}
|
||||||
|
|
||||||
|
// KeyBytes returns the byte slice holding the Op's key.
|
||||||
|
func (op Op) KeyBytes() []byte { return op.key }
|
||||||
|
|
||||||
|
// WithKeyBytes sets the byte slice for the Op's key.
|
||||||
|
func (op *Op) WithKeyBytes(key []byte) { op.key = key }
|
||||||
|
|
||||||
|
// RangeBytes returns the byte slice holding with the Op's range end, if any.
|
||||||
|
func (op Op) RangeBytes() []byte { return op.end }
|
||||||
|
|
||||||
|
// Rev returns the requested revision, if any.
|
||||||
|
func (op Op) Rev() int64 { return op.rev }
|
||||||
|
|
||||||
|
// IsPut returns true iff the operation is a Put.
|
||||||
|
func (op Op) IsPut() bool { return op.t == tPut }
|
||||||
|
|
||||||
|
// IsGet returns true iff the operation is a Get.
|
||||||
|
func (op Op) IsGet() bool { return op.t == tRange }
|
||||||
|
|
||||||
|
// IsDelete returns true iff the operation is a Delete.
|
||||||
|
func (op Op) IsDelete() bool { return op.t == tDeleteRange }
|
||||||
|
|
||||||
|
// IsSerializable returns true if the serializable field is true.
|
||||||
|
func (op Op) IsSerializable() bool { return op.serializable == true }
|
||||||
|
|
||||||
|
// IsKeysOnly returns whether keysOnly is set.
|
||||||
|
func (op Op) IsKeysOnly() bool { return op.keysOnly == true }
|
||||||
|
|
||||||
|
// IsCountOnly returns whether countOnly is set.
|
||||||
|
func (op Op) IsCountOnly() bool { return op.countOnly == true }
|
||||||
|
|
||||||
|
// MinModRev returns the operation's minimum modify revision.
|
||||||
|
func (op Op) MinModRev() int64 { return op.minModRev }
|
||||||
|
|
||||||
|
// MaxModRev returns the operation's maximum modify revision.
|
||||||
|
func (op Op) MaxModRev() int64 { return op.maxModRev }
|
||||||
|
|
||||||
|
// MinCreateRev returns the operation's minimum create revision.
|
||||||
|
func (op Op) MinCreateRev() int64 { return op.minCreateRev }
|
||||||
|
|
||||||
|
// MaxCreateRev returns the operation's maximum create revision.
|
||||||
|
func (op Op) MaxCreateRev() int64 { return op.maxCreateRev }
|
||||||
|
|
||||||
|
// WithRangeBytes sets the byte slice for the Op's range end.
|
||||||
|
func (op *Op) WithRangeBytes(end []byte) { op.end = end }
|
||||||
|
|
||||||
|
// ValueBytes returns the byte slice holding the Op's value, if any.
|
||||||
|
func (op Op) ValueBytes() []byte { return op.val }
|
||||||
|
|
||||||
|
// WithValueBytes sets the byte slice for the Op's value.
|
||||||
|
func (op *Op) WithValueBytes(v []byte) { op.val = v }
|
||||||
|
|
||||||
|
func (op Op) toRangeRequest() *pb.RangeRequest {
|
||||||
|
if op.t != tRange {
|
||||||
|
panic("op.t != tRange")
|
||||||
|
}
|
||||||
|
r := &pb.RangeRequest{
|
||||||
|
Key: op.key,
|
||||||
|
RangeEnd: op.end,
|
||||||
|
Limit: op.limit,
|
||||||
|
Revision: op.rev,
|
||||||
|
Serializable: op.serializable,
|
||||||
|
KeysOnly: op.keysOnly,
|
||||||
|
CountOnly: op.countOnly,
|
||||||
|
MinModRevision: op.minModRev,
|
||||||
|
MaxModRevision: op.maxModRev,
|
||||||
|
MinCreateRevision: op.minCreateRev,
|
||||||
|
MaxCreateRevision: op.maxCreateRev,
|
||||||
|
}
|
||||||
|
if op.sort != nil {
|
||||||
|
r.SortOrder = pb.RangeRequest_SortOrder(op.sort.Order)
|
||||||
|
r.SortTarget = pb.RangeRequest_SortTarget(op.sort.Target)
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (op Op) toTxnRequest() *pb.TxnRequest {
|
||||||
|
thenOps := make([]*pb.RequestOp, len(op.thenOps))
|
||||||
|
for i, tOp := range op.thenOps {
|
||||||
|
thenOps[i] = tOp.toRequestOp()
|
||||||
|
}
|
||||||
|
elseOps := make([]*pb.RequestOp, len(op.elseOps))
|
||||||
|
for i, eOp := range op.elseOps {
|
||||||
|
elseOps[i] = eOp.toRequestOp()
|
||||||
|
}
|
||||||
|
cmps := make([]*pb.Compare, len(op.cmps))
|
||||||
|
for i := range op.cmps {
|
||||||
|
cmps[i] = (*pb.Compare)(&op.cmps[i])
|
||||||
|
}
|
||||||
|
return &pb.TxnRequest{Compare: cmps, Success: thenOps, Failure: elseOps}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (op Op) toRequestOp() *pb.RequestOp {
|
||||||
|
switch op.t {
|
||||||
|
case tRange:
|
||||||
|
return &pb.RequestOp{Request: &pb.RequestOp_RequestRange{RequestRange: op.toRangeRequest()}}
|
||||||
|
case tPut:
|
||||||
|
r := &pb.PutRequest{Key: op.key, Value: op.val, Lease: int64(op.leaseID), PrevKv: op.prevKV, IgnoreValue: op.ignoreValue, IgnoreLease: op.ignoreLease}
|
||||||
|
return &pb.RequestOp{Request: &pb.RequestOp_RequestPut{RequestPut: r}}
|
||||||
|
case tDeleteRange:
|
||||||
|
r := &pb.DeleteRangeRequest{Key: op.key, RangeEnd: op.end, PrevKv: op.prevKV}
|
||||||
|
return &pb.RequestOp{Request: &pb.RequestOp_RequestDeleteRange{RequestDeleteRange: r}}
|
||||||
|
case tTxn:
|
||||||
|
return &pb.RequestOp{Request: &pb.RequestOp_RequestTxn{RequestTxn: op.toTxnRequest()}}
|
||||||
|
default:
|
||||||
|
panic("Unknown Op")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (op Op) isWrite() bool {
|
||||||
|
if op.t == tTxn {
|
||||||
|
for _, tOp := range op.thenOps {
|
||||||
|
if tOp.isWrite() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, tOp := range op.elseOps {
|
||||||
|
if tOp.isWrite() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return op.t != tRange
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpGet returns "get" operation based on given key and operation options.
|
||||||
|
func OpGet(key string, opts ...OpOption) Op {
|
||||||
|
ret := Op{t: tRange, key: []byte(key)}
|
||||||
|
ret.applyOpts(opts)
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpDelete returns "delete" operation based on given key and operation options.
|
||||||
|
func OpDelete(key string, opts ...OpOption) Op {
|
||||||
|
ret := Op{t: tDeleteRange, key: []byte(key)}
|
||||||
|
ret.applyOpts(opts)
|
||||||
|
switch {
|
||||||
|
case ret.leaseID != 0:
|
||||||
|
panic("unexpected lease in delete")
|
||||||
|
case ret.limit != 0:
|
||||||
|
panic("unexpected limit in delete")
|
||||||
|
case ret.rev != 0:
|
||||||
|
panic("unexpected revision in delete")
|
||||||
|
case ret.sort != nil:
|
||||||
|
panic("unexpected sort in delete")
|
||||||
|
case ret.serializable:
|
||||||
|
panic("unexpected serializable in delete")
|
||||||
|
case ret.countOnly:
|
||||||
|
panic("unexpected countOnly in delete")
|
||||||
|
case ret.minModRev != 0, ret.maxModRev != 0:
|
||||||
|
panic("unexpected mod revision filter in delete")
|
||||||
|
case ret.minCreateRev != 0, ret.maxCreateRev != 0:
|
||||||
|
panic("unexpected create revision filter in delete")
|
||||||
|
case ret.filterDelete, ret.filterPut:
|
||||||
|
panic("unexpected filter in delete")
|
||||||
|
case ret.createdNotify:
|
||||||
|
panic("unexpected createdNotify in delete")
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpPut returns "put" operation based on given key-value and operation options.
|
||||||
|
func OpPut(key, val string, opts ...OpOption) Op {
|
||||||
|
ret := Op{t: tPut, key: []byte(key), val: []byte(val)}
|
||||||
|
ret.applyOpts(opts)
|
||||||
|
switch {
|
||||||
|
case ret.end != nil:
|
||||||
|
panic("unexpected range in put")
|
||||||
|
case ret.limit != 0:
|
||||||
|
panic("unexpected limit in put")
|
||||||
|
case ret.rev != 0:
|
||||||
|
panic("unexpected revision in put")
|
||||||
|
case ret.sort != nil:
|
||||||
|
panic("unexpected sort in put")
|
||||||
|
case ret.serializable:
|
||||||
|
panic("unexpected serializable in put")
|
||||||
|
case ret.countOnly:
|
||||||
|
panic("unexpected countOnly in put")
|
||||||
|
case ret.minModRev != 0, ret.maxModRev != 0:
|
||||||
|
panic("unexpected mod revision filter in put")
|
||||||
|
case ret.minCreateRev != 0, ret.maxCreateRev != 0:
|
||||||
|
panic("unexpected create revision filter in put")
|
||||||
|
case ret.filterDelete, ret.filterPut:
|
||||||
|
panic("unexpected filter in put")
|
||||||
|
case ret.createdNotify:
|
||||||
|
panic("unexpected createdNotify in put")
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpTxn returns "txn" operation based on given transaction conditions.
|
||||||
|
func OpTxn(cmps []Cmp, thenOps []Op, elseOps []Op) Op {
|
||||||
|
return Op{t: tTxn, cmps: cmps, thenOps: thenOps, elseOps: elseOps}
|
||||||
|
}
|
||||||
|
|
||||||
|
func opWatch(key string, opts ...OpOption) Op {
|
||||||
|
ret := Op{t: tRange, key: []byte(key)}
|
||||||
|
ret.applyOpts(opts)
|
||||||
|
switch {
|
||||||
|
case ret.leaseID != 0:
|
||||||
|
panic("unexpected lease in watch")
|
||||||
|
case ret.limit != 0:
|
||||||
|
panic("unexpected limit in watch")
|
||||||
|
case ret.sort != nil:
|
||||||
|
panic("unexpected sort in watch")
|
||||||
|
case ret.serializable:
|
||||||
|
panic("unexpected serializable in watch")
|
||||||
|
case ret.countOnly:
|
||||||
|
panic("unexpected countOnly in watch")
|
||||||
|
case ret.minModRev != 0, ret.maxModRev != 0:
|
||||||
|
panic("unexpected mod revision filter in watch")
|
||||||
|
case ret.minCreateRev != 0, ret.maxCreateRev != 0:
|
||||||
|
panic("unexpected create revision filter in watch")
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
func (op *Op) applyOpts(opts []OpOption) {
|
||||||
|
for _, opt := range opts {
|
||||||
|
opt(op)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpOption configures Operations like Get, Put, Delete.
|
||||||
|
type OpOption func(*Op)
|
||||||
|
|
||||||
|
// WithLease attaches a lease ID to a key in 'Put' request.
|
||||||
|
func WithLease(leaseID LeaseID) OpOption {
|
||||||
|
return func(op *Op) { op.leaseID = leaseID }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithLimit limits the number of results to return from 'Get' request.
|
||||||
|
// If WithLimit is given a 0 limit, it is treated as no limit.
|
||||||
|
func WithLimit(n int64) OpOption { return func(op *Op) { op.limit = n } }
|
||||||
|
|
||||||
|
// WithRev specifies the store revision for 'Get' request.
|
||||||
|
// Or the start revision of 'Watch' request.
|
||||||
|
func WithRev(rev int64) OpOption { return func(op *Op) { op.rev = rev } }
|
||||||
|
|
||||||
|
// WithSort specifies the ordering in 'Get' request. It requires
|
||||||
|
// 'WithRange' and/or 'WithPrefix' to be specified too.
|
||||||
|
// 'target' specifies the target to sort by: key, version, revisions, value.
|
||||||
|
// 'order' can be either 'SortNone', 'SortAscend', 'SortDescend'.
|
||||||
|
func WithSort(target SortTarget, order SortOrder) OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
if target == SortByKey && order == SortAscend {
|
||||||
|
// If order != SortNone, server fetches the entire key-space,
|
||||||
|
// and then applies the sort and limit, if provided.
|
||||||
|
// Since by default the server returns results sorted by keys
|
||||||
|
// in lexicographically ascending order, the client should ignore
|
||||||
|
// SortOrder if the target is SortByKey.
|
||||||
|
order = SortNone
|
||||||
|
}
|
||||||
|
op.sort = &SortOption{target, order}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetPrefixRangeEnd gets the range end of the prefix.
|
||||||
|
// 'Get(foo, WithPrefix())' is equal to 'Get(foo, WithRange(GetPrefixRangeEnd(foo))'.
|
||||||
|
func GetPrefixRangeEnd(prefix string) string {
|
||||||
|
return string(getPrefix([]byte(prefix)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func getPrefix(key []byte) []byte {
|
||||||
|
end := make([]byte, len(key))
|
||||||
|
copy(end, key)
|
||||||
|
for i := len(end) - 1; i >= 0; i-- {
|
||||||
|
if end[i] < 0xff {
|
||||||
|
end[i] = end[i] + 1
|
||||||
|
end = end[:i+1]
|
||||||
|
return end
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// next prefix does not exist (e.g., 0xffff);
|
||||||
|
// default to WithFromKey policy
|
||||||
|
return noPrefixEnd
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithPrefix enables 'Get', 'Delete', or 'Watch' requests to operate
|
||||||
|
// on the keys with matching prefix. For example, 'Get(foo, WithPrefix())'
|
||||||
|
// can return 'foo1', 'foo2', and so on.
|
||||||
|
func WithPrefix() OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
if len(op.key) == 0 {
|
||||||
|
op.key, op.end = []byte{0}, []byte{0}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
op.end = getPrefix(op.key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithRange specifies the range of 'Get', 'Delete', 'Watch' requests.
|
||||||
|
// For example, 'Get' requests with 'WithRange(end)' returns
|
||||||
|
// the keys in the range [key, end).
|
||||||
|
// endKey must be lexicographically greater than start key.
|
||||||
|
func WithRange(endKey string) OpOption {
|
||||||
|
return func(op *Op) { op.end = []byte(endKey) }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFromKey specifies the range of 'Get', 'Delete', 'Watch' requests
|
||||||
|
// to be equal or greater than the key in the argument.
|
||||||
|
func WithFromKey() OpOption { return WithRange("\x00") }
|
||||||
|
|
||||||
|
// WithSerializable makes 'Get' request serializable. By default,
|
||||||
|
// it's linearizable. Serializable requests are better for lower latency
|
||||||
|
// requirement.
|
||||||
|
func WithSerializable() OpOption {
|
||||||
|
return func(op *Op) { op.serializable = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithKeysOnly makes the 'Get' request return only the keys and the corresponding
|
||||||
|
// values will be omitted.
|
||||||
|
func WithKeysOnly() OpOption {
|
||||||
|
return func(op *Op) { op.keysOnly = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithCountOnly makes the 'Get' request return only the count of keys.
|
||||||
|
func WithCountOnly() OpOption {
|
||||||
|
return func(op *Op) { op.countOnly = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithMinModRev filters out keys for Get with modification revisions less than the given revision.
|
||||||
|
func WithMinModRev(rev int64) OpOption { return func(op *Op) { op.minModRev = rev } }
|
||||||
|
|
||||||
|
// WithMaxModRev filters out keys for Get with modification revisions greater than the given revision.
|
||||||
|
func WithMaxModRev(rev int64) OpOption { return func(op *Op) { op.maxModRev = rev } }
|
||||||
|
|
||||||
|
// WithMinCreateRev filters out keys for Get with creation revisions less than the given revision.
|
||||||
|
func WithMinCreateRev(rev int64) OpOption { return func(op *Op) { op.minCreateRev = rev } }
|
||||||
|
|
||||||
|
// WithMaxCreateRev filters out keys for Get with creation revisions greater than the given revision.
|
||||||
|
func WithMaxCreateRev(rev int64) OpOption { return func(op *Op) { op.maxCreateRev = rev } }
|
||||||
|
|
||||||
|
// WithFirstCreate gets the key with the oldest creation revision in the request range.
|
||||||
|
func WithFirstCreate() []OpOption { return withTop(SortByCreateRevision, SortAscend) }
|
||||||
|
|
||||||
|
// WithLastCreate gets the key with the latest creation revision in the request range.
|
||||||
|
func WithLastCreate() []OpOption { return withTop(SortByCreateRevision, SortDescend) }
|
||||||
|
|
||||||
|
// WithFirstKey gets the lexically first key in the request range.
|
||||||
|
func WithFirstKey() []OpOption { return withTop(SortByKey, SortAscend) }
|
||||||
|
|
||||||
|
// WithLastKey gets the lexically last key in the request range.
|
||||||
|
func WithLastKey() []OpOption { return withTop(SortByKey, SortDescend) }
|
||||||
|
|
||||||
|
// WithFirstRev gets the key with the oldest modification revision in the request range.
|
||||||
|
func WithFirstRev() []OpOption { return withTop(SortByModRevision, SortAscend) }
|
||||||
|
|
||||||
|
// WithLastRev gets the key with the latest modification revision in the request range.
|
||||||
|
func WithLastRev() []OpOption { return withTop(SortByModRevision, SortDescend) }
|
||||||
|
|
||||||
|
// withTop gets the first key over the get's prefix given a sort order
|
||||||
|
func withTop(target SortTarget, order SortOrder) []OpOption {
|
||||||
|
return []OpOption{WithPrefix(), WithSort(target, order), WithLimit(1)}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithProgressNotify makes watch server send periodic progress updates
|
||||||
|
// every 10 minutes when there is no incoming events.
|
||||||
|
// Progress updates have zero events in WatchResponse.
|
||||||
|
func WithProgressNotify() OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
op.progressNotify = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithCreatedNotify makes watch server sends the created event.
|
||||||
|
func WithCreatedNotify() OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
op.createdNotify = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFilterPut discards PUT events from the watcher.
|
||||||
|
func WithFilterPut() OpOption {
|
||||||
|
return func(op *Op) { op.filterPut = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFilterDelete discards DELETE events from the watcher.
|
||||||
|
func WithFilterDelete() OpOption {
|
||||||
|
return func(op *Op) { op.filterDelete = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithPrevKV gets the previous key-value pair before the event happens. If the previous KV is already compacted,
|
||||||
|
// nothing will be returned.
|
||||||
|
func WithPrevKV() OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
op.prevKV = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFragment to receive raw watch response with fragmentation.
|
||||||
|
// Fragmentation is disabled by default. If fragmentation is enabled,
|
||||||
|
// etcd watch server will split watch response before sending to clients
|
||||||
|
// when the total size of watch events exceed server-side request limit.
|
||||||
|
// The default server-side request limit is 1.5 MiB, which can be configured
|
||||||
|
// as "--max-request-bytes" flag value + gRPC-overhead 512 bytes.
|
||||||
|
// See "etcdserver/api/v3rpc/watch.go" for more details.
|
||||||
|
func WithFragment() OpOption {
|
||||||
|
return func(op *Op) { op.fragment = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithIgnoreValue updates the key using its current value.
|
||||||
|
// This option can not be combined with non-empty values.
|
||||||
|
// Returns an error if the key does not exist.
|
||||||
|
func WithIgnoreValue() OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
op.ignoreValue = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithIgnoreLease updates the key using its current lease.
|
||||||
|
// This option can not be combined with WithLease.
|
||||||
|
// Returns an error if the key does not exist.
|
||||||
|
func WithIgnoreLease() OpOption {
|
||||||
|
return func(op *Op) {
|
||||||
|
op.ignoreLease = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseOp represents an Operation that lease can execute.
|
||||||
|
type LeaseOp struct {
|
||||||
|
id LeaseID
|
||||||
|
|
||||||
|
// for TimeToLive
|
||||||
|
attachedKeys bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// LeaseOption configures lease operations.
|
||||||
|
type LeaseOption func(*LeaseOp)
|
||||||
|
|
||||||
|
func (op *LeaseOp) applyOpts(opts []LeaseOption) {
|
||||||
|
for _, opt := range opts {
|
||||||
|
opt(op)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithAttachedKeys makes TimeToLive list the keys attached to the given lease ID.
|
||||||
|
func WithAttachedKeys() LeaseOption {
|
||||||
|
return func(op *LeaseOp) { op.attachedKeys = true }
|
||||||
|
}
|
||||||
|
|
||||||
|
func toLeaseTimeToLiveRequest(id LeaseID, opts ...LeaseOption) *pb.LeaseTimeToLiveRequest {
|
||||||
|
ret := &LeaseOp{id: id}
|
||||||
|
ret.applyOpts(opts)
|
||||||
|
return &pb.LeaseTimeToLiveRequest{ID: int64(id), Keys: ret.attachedKeys}
|
||||||
|
}
|
65
vendor/github.com/coreos/etcd/clientv3/options.go
generated
vendored
Normal file
65
vendor/github.com/coreos/etcd/clientv3/options.go
generated
vendored
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
// Copyright 2017 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// client-side handling retrying of request failures where data was not written to the wire or
|
||||||
|
// where server indicates it did not process the data. gRPC default is default is "FailFast(true)"
|
||||||
|
// but for etcd we default to "FailFast(false)" to minimize client request error responses due to
|
||||||
|
// transient failures.
|
||||||
|
defaultFailFast = grpc.FailFast(false)
|
||||||
|
|
||||||
|
// client-side request send limit, gRPC default is math.MaxInt32
|
||||||
|
// Make sure that "client-side send limit < server-side default send/recv limit"
|
||||||
|
// Same value as "embed.DefaultMaxRequestBytes" plus gRPC overhead bytes
|
||||||
|
defaultMaxCallSendMsgSize = grpc.MaxCallSendMsgSize(2 * 1024 * 1024)
|
||||||
|
|
||||||
|
// client-side response receive limit, gRPC default is 4MB
|
||||||
|
// Make sure that "client-side receive limit >= server-side default send/recv limit"
|
||||||
|
// because range response can easily exceed request send limits
|
||||||
|
// Default to math.MaxInt32; writes exceeding server-side send limit fails anyway
|
||||||
|
defaultMaxCallRecvMsgSize = grpc.MaxCallRecvMsgSize(math.MaxInt32)
|
||||||
|
|
||||||
|
// client-side non-streaming retry limit, only applied to requests where server responds with
|
||||||
|
// a error code clearly indicating it was unable to process the request such as codes.Unavailable.
|
||||||
|
// If set to 0, retry is disabled.
|
||||||
|
defaultUnaryMaxRetries uint = 100
|
||||||
|
|
||||||
|
// client-side streaming retry limit, only applied to requests where server responds with
|
||||||
|
// a error code clearly indicating it was unable to process the request such as codes.Unavailable.
|
||||||
|
// If set to 0, retry is disabled.
|
||||||
|
defaultStreamMaxRetries uint = ^uint(0) // max uint
|
||||||
|
|
||||||
|
// client-side retry backoff wait between requests.
|
||||||
|
defaultBackoffWaitBetween = 25 * time.Millisecond
|
||||||
|
|
||||||
|
// client-side retry backoff default jitter fraction.
|
||||||
|
defaultBackoffJitterFraction = 0.10
|
||||||
|
)
|
||||||
|
|
||||||
|
// defaultCallOpts defines a list of default "gRPC.CallOption".
|
||||||
|
// Some options are exposed to "clientv3.Config".
|
||||||
|
// Defaults will be overridden by the settings in "clientv3.Config".
|
||||||
|
var defaultCallOpts = []grpc.CallOption{defaultFailFast, defaultMaxCallSendMsgSize, defaultMaxCallRecvMsgSize}
|
||||||
|
|
||||||
|
// MaxLeaseTTL is the maximum lease TTL value
|
||||||
|
const MaxLeaseTTL = 9000000000
|
30
vendor/github.com/coreos/etcd/clientv3/ready_wait.go
generated
vendored
Normal file
30
vendor/github.com/coreos/etcd/clientv3/ready_wait.go
generated
vendored
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
// Copyright 2017 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import "context"
|
||||||
|
|
||||||
|
// TODO: remove this when "FailFast=false" is fixed.
|
||||||
|
// See https://github.com/grpc/grpc-go/issues/1532.
|
||||||
|
func readyWait(rpcCtx, clientCtx context.Context, ready <-chan struct{}) error {
|
||||||
|
select {
|
||||||
|
case <-ready:
|
||||||
|
return nil
|
||||||
|
case <-rpcCtx.Done():
|
||||||
|
return rpcCtx.Err()
|
||||||
|
case <-clientCtx.Done():
|
||||||
|
return clientCtx.Err()
|
||||||
|
}
|
||||||
|
}
|
298
vendor/github.com/coreos/etcd/clientv3/retry.go
generated
vendored
Normal file
298
vendor/github.com/coreos/etcd/clientv3/retry.go
generated
vendored
Normal file
@ -0,0 +1,298 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
type retryPolicy uint8
|
||||||
|
|
||||||
|
const (
|
||||||
|
repeatable retryPolicy = iota
|
||||||
|
nonRepeatable
|
||||||
|
)
|
||||||
|
|
||||||
|
func (rp retryPolicy) String() string {
|
||||||
|
switch rp {
|
||||||
|
case repeatable:
|
||||||
|
return "repeatable"
|
||||||
|
case nonRepeatable:
|
||||||
|
return "nonRepeatable"
|
||||||
|
default:
|
||||||
|
return "UNKNOWN"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type rpcFunc func(ctx context.Context) error
|
||||||
|
type retryRPCFunc func(context.Context, rpcFunc, retryPolicy) error
|
||||||
|
type retryStopErrFunc func(error) bool
|
||||||
|
|
||||||
|
// isSafeRetryImmutableRPC returns "true" when an immutable request is safe for retry.
|
||||||
|
//
|
||||||
|
// immutable requests (e.g. Get) should be retried unless it's
|
||||||
|
// an obvious server-side error (e.g. rpctypes.ErrRequestTooLarge).
|
||||||
|
//
|
||||||
|
// Returning "false" means retry should stop, since client cannot
|
||||||
|
// handle itself even with retries.
|
||||||
|
func isSafeRetryImmutableRPC(err error) bool {
|
||||||
|
eErr := rpctypes.Error(err)
|
||||||
|
if serverErr, ok := eErr.(rpctypes.EtcdError); ok && serverErr.Code() != codes.Unavailable {
|
||||||
|
// interrupted by non-transient server-side or gRPC-side error
|
||||||
|
// client cannot handle itself (e.g. rpctypes.ErrCompacted)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// only retry if unavailable
|
||||||
|
ev, ok := status.FromError(err)
|
||||||
|
if !ok {
|
||||||
|
// all errors from RPC is typed "grpc/status.(*statusError)"
|
||||||
|
// (ref. https://github.com/grpc/grpc-go/pull/1782)
|
||||||
|
//
|
||||||
|
// if the error type is not "grpc/status.(*statusError)",
|
||||||
|
// it could be from "Dial"
|
||||||
|
// TODO: do not retry for now
|
||||||
|
// ref. https://github.com/grpc/grpc-go/issues/1581
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return ev.Code() == codes.Unavailable
|
||||||
|
}
|
||||||
|
|
||||||
|
// isSafeRetryMutableRPC returns "true" when a mutable request is safe for retry.
|
||||||
|
//
|
||||||
|
// mutable requests (e.g. Put, Delete, Txn) should only be retried
|
||||||
|
// when the status code is codes.Unavailable when initial connection
|
||||||
|
// has not been established (no endpoint is up).
|
||||||
|
//
|
||||||
|
// Returning "false" means retry should stop, otherwise it violates
|
||||||
|
// write-at-most-once semantics.
|
||||||
|
func isSafeRetryMutableRPC(err error) bool {
|
||||||
|
if ev, ok := status.FromError(err); ok && ev.Code() != codes.Unavailable {
|
||||||
|
// not safe for mutable RPCs
|
||||||
|
// e.g. interrupted by non-transient error that client cannot handle itself,
|
||||||
|
// or transient error while the connection has already been established
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
desc := rpctypes.ErrorDesc(err)
|
||||||
|
return desc == "there is no address available" || desc == "there is no connection available"
|
||||||
|
}
|
||||||
|
|
||||||
|
type retryKVClient struct {
|
||||||
|
kc pb.KVClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// RetryKVClient implements a KVClient.
|
||||||
|
func RetryKVClient(c *Client) pb.KVClient {
|
||||||
|
return &retryKVClient{
|
||||||
|
kc: pb.NewKVClient(c.conn),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
func (rkv *retryKVClient) Range(ctx context.Context, in *pb.RangeRequest, opts ...grpc.CallOption) (resp *pb.RangeResponse, err error) {
|
||||||
|
return rkv.kc.Range(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rkv *retryKVClient) Put(ctx context.Context, in *pb.PutRequest, opts ...grpc.CallOption) (resp *pb.PutResponse, err error) {
|
||||||
|
return rkv.kc.Put(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rkv *retryKVClient) DeleteRange(ctx context.Context, in *pb.DeleteRangeRequest, opts ...grpc.CallOption) (resp *pb.DeleteRangeResponse, err error) {
|
||||||
|
return rkv.kc.DeleteRange(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rkv *retryKVClient) Txn(ctx context.Context, in *pb.TxnRequest, opts ...grpc.CallOption) (resp *pb.TxnResponse, err error) {
|
||||||
|
return rkv.kc.Txn(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rkv *retryKVClient) Compact(ctx context.Context, in *pb.CompactionRequest, opts ...grpc.CallOption) (resp *pb.CompactionResponse, err error) {
|
||||||
|
return rkv.kc.Compact(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
type retryLeaseClient struct {
|
||||||
|
lc pb.LeaseClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// RetryLeaseClient implements a LeaseClient.
|
||||||
|
func RetryLeaseClient(c *Client) pb.LeaseClient {
|
||||||
|
return &retryLeaseClient{
|
||||||
|
lc: pb.NewLeaseClient(c.conn),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rlc *retryLeaseClient) LeaseTimeToLive(ctx context.Context, in *pb.LeaseTimeToLiveRequest, opts ...grpc.CallOption) (resp *pb.LeaseTimeToLiveResponse, err error) {
|
||||||
|
return rlc.lc.LeaseTimeToLive(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rlc *retryLeaseClient) LeaseLeases(ctx context.Context, in *pb.LeaseLeasesRequest, opts ...grpc.CallOption) (resp *pb.LeaseLeasesResponse, err error) {
|
||||||
|
return rlc.lc.LeaseLeases(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rlc *retryLeaseClient) LeaseGrant(ctx context.Context, in *pb.LeaseGrantRequest, opts ...grpc.CallOption) (resp *pb.LeaseGrantResponse, err error) {
|
||||||
|
return rlc.lc.LeaseGrant(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rlc *retryLeaseClient) LeaseRevoke(ctx context.Context, in *pb.LeaseRevokeRequest, opts ...grpc.CallOption) (resp *pb.LeaseRevokeResponse, err error) {
|
||||||
|
return rlc.lc.LeaseRevoke(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rlc *retryLeaseClient) LeaseKeepAlive(ctx context.Context, opts ...grpc.CallOption) (stream pb.Lease_LeaseKeepAliveClient, err error) {
|
||||||
|
return rlc.lc.LeaseKeepAlive(ctx, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
type retryClusterClient struct {
|
||||||
|
cc pb.ClusterClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// RetryClusterClient implements a ClusterClient.
|
||||||
|
func RetryClusterClient(c *Client) pb.ClusterClient {
|
||||||
|
return &retryClusterClient{
|
||||||
|
cc: pb.NewClusterClient(c.conn),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rcc *retryClusterClient) MemberList(ctx context.Context, in *pb.MemberListRequest, opts ...grpc.CallOption) (resp *pb.MemberListResponse, err error) {
|
||||||
|
return rcc.cc.MemberList(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rcc *retryClusterClient) MemberAdd(ctx context.Context, in *pb.MemberAddRequest, opts ...grpc.CallOption) (resp *pb.MemberAddResponse, err error) {
|
||||||
|
return rcc.cc.MemberAdd(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rcc *retryClusterClient) MemberRemove(ctx context.Context, in *pb.MemberRemoveRequest, opts ...grpc.CallOption) (resp *pb.MemberRemoveResponse, err error) {
|
||||||
|
return rcc.cc.MemberRemove(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rcc *retryClusterClient) MemberUpdate(ctx context.Context, in *pb.MemberUpdateRequest, opts ...grpc.CallOption) (resp *pb.MemberUpdateResponse, err error) {
|
||||||
|
return rcc.cc.MemberUpdate(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
type retryMaintenanceClient struct {
|
||||||
|
mc pb.MaintenanceClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// RetryMaintenanceClient implements a Maintenance.
|
||||||
|
func RetryMaintenanceClient(c *Client, conn *grpc.ClientConn) pb.MaintenanceClient {
|
||||||
|
return &retryMaintenanceClient{
|
||||||
|
mc: pb.NewMaintenanceClient(conn),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) Alarm(ctx context.Context, in *pb.AlarmRequest, opts ...grpc.CallOption) (resp *pb.AlarmResponse, err error) {
|
||||||
|
return rmc.mc.Alarm(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) Status(ctx context.Context, in *pb.StatusRequest, opts ...grpc.CallOption) (resp *pb.StatusResponse, err error) {
|
||||||
|
return rmc.mc.Status(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) Hash(ctx context.Context, in *pb.HashRequest, opts ...grpc.CallOption) (resp *pb.HashResponse, err error) {
|
||||||
|
return rmc.mc.Hash(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) HashKV(ctx context.Context, in *pb.HashKVRequest, opts ...grpc.CallOption) (resp *pb.HashKVResponse, err error) {
|
||||||
|
return rmc.mc.HashKV(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) Snapshot(ctx context.Context, in *pb.SnapshotRequest, opts ...grpc.CallOption) (stream pb.Maintenance_SnapshotClient, err error) {
|
||||||
|
return rmc.mc.Snapshot(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) MoveLeader(ctx context.Context, in *pb.MoveLeaderRequest, opts ...grpc.CallOption) (resp *pb.MoveLeaderResponse, err error) {
|
||||||
|
return rmc.mc.MoveLeader(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rmc *retryMaintenanceClient) Defragment(ctx context.Context, in *pb.DefragmentRequest, opts ...grpc.CallOption) (resp *pb.DefragmentResponse, err error) {
|
||||||
|
return rmc.mc.Defragment(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
type retryAuthClient struct {
|
||||||
|
ac pb.AuthClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// RetryAuthClient implements a AuthClient.
|
||||||
|
func RetryAuthClient(c *Client) pb.AuthClient {
|
||||||
|
return &retryAuthClient{
|
||||||
|
ac: pb.NewAuthClient(c.conn),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserList(ctx context.Context, in *pb.AuthUserListRequest, opts ...grpc.CallOption) (resp *pb.AuthUserListResponse, err error) {
|
||||||
|
return rac.ac.UserList(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserGet(ctx context.Context, in *pb.AuthUserGetRequest, opts ...grpc.CallOption) (resp *pb.AuthUserGetResponse, err error) {
|
||||||
|
return rac.ac.UserGet(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) RoleGet(ctx context.Context, in *pb.AuthRoleGetRequest, opts ...grpc.CallOption) (resp *pb.AuthRoleGetResponse, err error) {
|
||||||
|
return rac.ac.RoleGet(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) RoleList(ctx context.Context, in *pb.AuthRoleListRequest, opts ...grpc.CallOption) (resp *pb.AuthRoleListResponse, err error) {
|
||||||
|
return rac.ac.RoleList(ctx, in, append(opts, withRetryPolicy(repeatable))...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) AuthEnable(ctx context.Context, in *pb.AuthEnableRequest, opts ...grpc.CallOption) (resp *pb.AuthEnableResponse, err error) {
|
||||||
|
return rac.ac.AuthEnable(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) AuthDisable(ctx context.Context, in *pb.AuthDisableRequest, opts ...grpc.CallOption) (resp *pb.AuthDisableResponse, err error) {
|
||||||
|
return rac.ac.AuthDisable(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserAdd(ctx context.Context, in *pb.AuthUserAddRequest, opts ...grpc.CallOption) (resp *pb.AuthUserAddResponse, err error) {
|
||||||
|
return rac.ac.UserAdd(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserDelete(ctx context.Context, in *pb.AuthUserDeleteRequest, opts ...grpc.CallOption) (resp *pb.AuthUserDeleteResponse, err error) {
|
||||||
|
return rac.ac.UserDelete(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserChangePassword(ctx context.Context, in *pb.AuthUserChangePasswordRequest, opts ...grpc.CallOption) (resp *pb.AuthUserChangePasswordResponse, err error) {
|
||||||
|
return rac.ac.UserChangePassword(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserGrantRole(ctx context.Context, in *pb.AuthUserGrantRoleRequest, opts ...grpc.CallOption) (resp *pb.AuthUserGrantRoleResponse, err error) {
|
||||||
|
return rac.ac.UserGrantRole(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) UserRevokeRole(ctx context.Context, in *pb.AuthUserRevokeRoleRequest, opts ...grpc.CallOption) (resp *pb.AuthUserRevokeRoleResponse, err error) {
|
||||||
|
return rac.ac.UserRevokeRole(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) RoleAdd(ctx context.Context, in *pb.AuthRoleAddRequest, opts ...grpc.CallOption) (resp *pb.AuthRoleAddResponse, err error) {
|
||||||
|
return rac.ac.RoleAdd(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) RoleDelete(ctx context.Context, in *pb.AuthRoleDeleteRequest, opts ...grpc.CallOption) (resp *pb.AuthRoleDeleteResponse, err error) {
|
||||||
|
return rac.ac.RoleDelete(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) RoleGrantPermission(ctx context.Context, in *pb.AuthRoleGrantPermissionRequest, opts ...grpc.CallOption) (resp *pb.AuthRoleGrantPermissionResponse, err error) {
|
||||||
|
return rac.ac.RoleGrantPermission(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) RoleRevokePermission(ctx context.Context, in *pb.AuthRoleRevokePermissionRequest, opts ...grpc.CallOption) (resp *pb.AuthRoleRevokePermissionResponse, err error) {
|
||||||
|
return rac.ac.RoleRevokePermission(ctx, in, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rac *retryAuthClient) Authenticate(ctx context.Context, in *pb.AuthenticateRequest, opts ...grpc.CallOption) (resp *pb.AuthenticateResponse, err error) {
|
||||||
|
return rac.ac.Authenticate(ctx, in, opts...)
|
||||||
|
}
|
382
vendor/github.com/coreos/etcd/clientv3/retry_interceptor.go
generated
vendored
Normal file
382
vendor/github.com/coreos/etcd/clientv3/retry_interceptor.go
generated
vendored
Normal file
@ -0,0 +1,382 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Based on github.com/grpc-ecosystem/go-grpc-middleware/retry, but modified to support the more
|
||||||
|
// fine grained error checking required by write-at-most-once retry semantics of etcd.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
|
||||||
|
"github.com/grpc-ecosystem/go-grpc-middleware/util/backoffutils"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/metadata"
|
||||||
|
)
|
||||||
|
|
||||||
|
// unaryClientInterceptor returns a new retrying unary client interceptor.
|
||||||
|
//
|
||||||
|
// The default configuration of the interceptor is to not retry *at all*. This behaviour can be
|
||||||
|
// changed through options (e.g. WithMax) on creation of the interceptor or on call (through grpc.CallOptions).
|
||||||
|
func (c *Client) unaryClientInterceptor(logger *zap.Logger, optFuncs ...retryOption) grpc.UnaryClientInterceptor {
|
||||||
|
intOpts := reuseOrNewWithCallOptions(defaultOptions, optFuncs)
|
||||||
|
return func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
|
||||||
|
grpcOpts, retryOpts := filterCallOptions(opts)
|
||||||
|
callOpts := reuseOrNewWithCallOptions(intOpts, retryOpts)
|
||||||
|
// short circuit for simplicity, and avoiding allocations.
|
||||||
|
if callOpts.max == 0 {
|
||||||
|
return invoker(ctx, method, req, reply, cc, grpcOpts...)
|
||||||
|
}
|
||||||
|
var lastErr error
|
||||||
|
for attempt := uint(0); attempt < callOpts.max; attempt++ {
|
||||||
|
if err := waitRetryBackoff(attempt, ctx, callOpts); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
lastErr = invoker(ctx, method, req, reply, cc, grpcOpts...)
|
||||||
|
logger.Info("retry unary intercept", zap.Uint("attempt", attempt), zap.Error(lastErr))
|
||||||
|
if lastErr == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if isContextError(lastErr) {
|
||||||
|
if ctx.Err() != nil {
|
||||||
|
// its the context deadline or cancellation.
|
||||||
|
return lastErr
|
||||||
|
}
|
||||||
|
// its the callCtx deadline or cancellation, in which case try again.
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if callOpts.retryAuth && rpctypes.Error(lastErr) == rpctypes.ErrInvalidAuthToken {
|
||||||
|
gterr := c.getToken(ctx)
|
||||||
|
if gterr != nil {
|
||||||
|
logger.Info("retry failed to fetch new auth token", zap.Error(gterr))
|
||||||
|
return lastErr // return the original error for simplicity
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if !isSafeRetry(c.lg, lastErr, callOpts) {
|
||||||
|
return lastErr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lastErr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// streamClientInterceptor returns a new retrying stream client interceptor for server side streaming calls.
|
||||||
|
//
|
||||||
|
// The default configuration of the interceptor is to not retry *at all*. This behaviour can be
|
||||||
|
// changed through options (e.g. WithMax) on creation of the interceptor or on call (through grpc.CallOptions).
|
||||||
|
//
|
||||||
|
// Retry logic is available *only for ServerStreams*, i.e. 1:n streams, as the internal logic needs
|
||||||
|
// to buffer the messages sent by the client. If retry is enabled on any other streams (ClientStreams,
|
||||||
|
// BidiStreams), the retry interceptor will fail the call.
|
||||||
|
func (c *Client) streamClientInterceptor(logger *zap.Logger, optFuncs ...retryOption) grpc.StreamClientInterceptor {
|
||||||
|
intOpts := reuseOrNewWithCallOptions(defaultOptions, optFuncs)
|
||||||
|
return func(ctx context.Context, desc *grpc.StreamDesc, cc *grpc.ClientConn, method string, streamer grpc.Streamer, opts ...grpc.CallOption) (grpc.ClientStream, error) {
|
||||||
|
grpcOpts, retryOpts := filterCallOptions(opts)
|
||||||
|
callOpts := reuseOrNewWithCallOptions(intOpts, retryOpts)
|
||||||
|
// short circuit for simplicity, and avoiding allocations.
|
||||||
|
if callOpts.max == 0 {
|
||||||
|
return streamer(ctx, desc, cc, method, grpcOpts...)
|
||||||
|
}
|
||||||
|
if desc.ClientStreams {
|
||||||
|
return nil, grpc.Errorf(codes.Unimplemented, "clientv3/retry_interceptor: cannot retry on ClientStreams, set Disable()")
|
||||||
|
}
|
||||||
|
newStreamer, err := streamer(ctx, desc, cc, method, grpcOpts...)
|
||||||
|
logger.Info("retry stream intercept", zap.Error(err))
|
||||||
|
if err != nil {
|
||||||
|
// TODO(mwitkow): Maybe dial and transport errors should be retriable?
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
retryingStreamer := &serverStreamingRetryingStream{
|
||||||
|
client: c,
|
||||||
|
ClientStream: newStreamer,
|
||||||
|
callOpts: callOpts,
|
||||||
|
ctx: ctx,
|
||||||
|
streamerCall: func(ctx context.Context) (grpc.ClientStream, error) {
|
||||||
|
return streamer(ctx, desc, cc, method, grpcOpts...)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
return retryingStreamer, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// type serverStreamingRetryingStream is the implementation of grpc.ClientStream that acts as a
|
||||||
|
// proxy to the underlying call. If any of the RecvMsg() calls fail, it will try to reestablish
|
||||||
|
// a new ClientStream according to the retry policy.
|
||||||
|
type serverStreamingRetryingStream struct {
|
||||||
|
grpc.ClientStream
|
||||||
|
client *Client
|
||||||
|
bufferedSends []interface{} // single message that the client can sen
|
||||||
|
receivedGood bool // indicates whether any prior receives were successful
|
||||||
|
wasClosedSend bool // indicates that CloseSend was closed
|
||||||
|
ctx context.Context
|
||||||
|
callOpts *options
|
||||||
|
streamerCall func(ctx context.Context) (grpc.ClientStream, error)
|
||||||
|
mu sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) setStream(clientStream grpc.ClientStream) {
|
||||||
|
s.mu.Lock()
|
||||||
|
s.ClientStream = clientStream
|
||||||
|
s.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) getStream() grpc.ClientStream {
|
||||||
|
s.mu.RLock()
|
||||||
|
defer s.mu.RUnlock()
|
||||||
|
return s.ClientStream
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) SendMsg(m interface{}) error {
|
||||||
|
s.mu.Lock()
|
||||||
|
s.bufferedSends = append(s.bufferedSends, m)
|
||||||
|
s.mu.Unlock()
|
||||||
|
return s.getStream().SendMsg(m)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) CloseSend() error {
|
||||||
|
s.mu.Lock()
|
||||||
|
s.wasClosedSend = true
|
||||||
|
s.mu.Unlock()
|
||||||
|
return s.getStream().CloseSend()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) Header() (metadata.MD, error) {
|
||||||
|
return s.getStream().Header()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) Trailer() metadata.MD {
|
||||||
|
return s.getStream().Trailer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) RecvMsg(m interface{}) error {
|
||||||
|
attemptRetry, lastErr := s.receiveMsgAndIndicateRetry(m)
|
||||||
|
if !attemptRetry {
|
||||||
|
return lastErr // success or hard failure
|
||||||
|
}
|
||||||
|
// We start off from attempt 1, because zeroth was already made on normal SendMsg().
|
||||||
|
for attempt := uint(1); attempt < s.callOpts.max; attempt++ {
|
||||||
|
if err := waitRetryBackoff(attempt, s.ctx, s.callOpts); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
newStream, err := s.reestablishStreamAndResendBuffer(s.ctx)
|
||||||
|
if err != nil {
|
||||||
|
// TODO(mwitkow): Maybe dial and transport errors should be retriable?
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
s.setStream(newStream)
|
||||||
|
attemptRetry, lastErr = s.receiveMsgAndIndicateRetry(m)
|
||||||
|
//fmt.Printf("Received message and indicate: %v %v\n", attemptRetry, lastErr)
|
||||||
|
if !attemptRetry {
|
||||||
|
return lastErr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lastErr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) receiveMsgAndIndicateRetry(m interface{}) (bool, error) {
|
||||||
|
s.mu.RLock()
|
||||||
|
wasGood := s.receivedGood
|
||||||
|
s.mu.RUnlock()
|
||||||
|
err := s.getStream().RecvMsg(m)
|
||||||
|
if err == nil || err == io.EOF {
|
||||||
|
s.mu.Lock()
|
||||||
|
s.receivedGood = true
|
||||||
|
s.mu.Unlock()
|
||||||
|
return false, err
|
||||||
|
} else if wasGood {
|
||||||
|
// previous RecvMsg in the stream succeeded, no retry logic should interfere
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
if isContextError(err) {
|
||||||
|
if s.ctx.Err() != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
// its the callCtx deadline or cancellation, in which case try again.
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
if s.callOpts.retryAuth && rpctypes.Error(err) == rpctypes.ErrInvalidAuthToken {
|
||||||
|
gterr := s.client.getToken(s.ctx)
|
||||||
|
if gterr != nil {
|
||||||
|
s.client.lg.Info("retry failed to fetch new auth token", zap.Error(gterr))
|
||||||
|
return false, err // return the original error for simplicity
|
||||||
|
}
|
||||||
|
return true, err
|
||||||
|
|
||||||
|
}
|
||||||
|
return isSafeRetry(s.client.lg, err, s.callOpts), err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *serverStreamingRetryingStream) reestablishStreamAndResendBuffer(callCtx context.Context) (grpc.ClientStream, error) {
|
||||||
|
s.mu.RLock()
|
||||||
|
bufferedSends := s.bufferedSends
|
||||||
|
s.mu.RUnlock()
|
||||||
|
newStream, err := s.streamerCall(callCtx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
for _, msg := range bufferedSends {
|
||||||
|
if err := newStream.SendMsg(msg); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err := newStream.CloseSend(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return newStream, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func waitRetryBackoff(attempt uint, ctx context.Context, callOpts *options) error {
|
||||||
|
var waitTime time.Duration = 0
|
||||||
|
if attempt > 0 {
|
||||||
|
waitTime = callOpts.backoffFunc(attempt)
|
||||||
|
}
|
||||||
|
if waitTime > 0 {
|
||||||
|
timer := time.NewTimer(waitTime)
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
timer.Stop()
|
||||||
|
return contextErrToGrpcErr(ctx.Err())
|
||||||
|
case <-timer.C:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// isSafeRetry returns "true", if request is safe for retry with the given error.
|
||||||
|
func isSafeRetry(lg *zap.Logger, err error, callOpts *options) bool {
|
||||||
|
if isContextError(err) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
switch callOpts.retryPolicy {
|
||||||
|
case repeatable:
|
||||||
|
return isSafeRetryImmutableRPC(err)
|
||||||
|
case nonRepeatable:
|
||||||
|
return isSafeRetryMutableRPC(err)
|
||||||
|
default:
|
||||||
|
lg.Warn("unrecognized retry policy", zap.String("retryPolicy", callOpts.retryPolicy.String()))
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func isContextError(err error) bool {
|
||||||
|
return grpc.Code(err) == codes.DeadlineExceeded || grpc.Code(err) == codes.Canceled
|
||||||
|
}
|
||||||
|
|
||||||
|
func contextErrToGrpcErr(err error) error {
|
||||||
|
switch err {
|
||||||
|
case context.DeadlineExceeded:
|
||||||
|
return grpc.Errorf(codes.DeadlineExceeded, err.Error())
|
||||||
|
case context.Canceled:
|
||||||
|
return grpc.Errorf(codes.Canceled, err.Error())
|
||||||
|
default:
|
||||||
|
return grpc.Errorf(codes.Unknown, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
defaultOptions = &options{
|
||||||
|
retryPolicy: nonRepeatable,
|
||||||
|
max: 0, // disable
|
||||||
|
backoffFunc: backoffLinearWithJitter(50*time.Millisecond /*jitter*/, 0.10),
|
||||||
|
retryAuth: true,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// backoffFunc denotes a family of functions that control the backoff duration between call retries.
|
||||||
|
//
|
||||||
|
// They are called with an identifier of the attempt, and should return a time the system client should
|
||||||
|
// hold off for. If the time returned is longer than the `context.Context.Deadline` of the request
|
||||||
|
// the deadline of the request takes precedence and the wait will be interrupted before proceeding
|
||||||
|
// with the next iteration.
|
||||||
|
type backoffFunc func(attempt uint) time.Duration
|
||||||
|
|
||||||
|
// withRetryPolicy sets the retry policy of this call.
|
||||||
|
func withRetryPolicy(rp retryPolicy) retryOption {
|
||||||
|
return retryOption{applyFunc: func(o *options) {
|
||||||
|
o.retryPolicy = rp
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
|
||||||
|
// withAuthRetry sets enables authentication retries.
|
||||||
|
func withAuthRetry(retryAuth bool) retryOption {
|
||||||
|
return retryOption{applyFunc: func(o *options) {
|
||||||
|
o.retryAuth = retryAuth
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
|
||||||
|
// withMax sets the maximum number of retries on this call, or this interceptor.
|
||||||
|
func withMax(maxRetries uint) retryOption {
|
||||||
|
return retryOption{applyFunc: func(o *options) {
|
||||||
|
o.max = maxRetries
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithBackoff sets the `BackoffFunc `used to control time between retries.
|
||||||
|
func withBackoff(bf backoffFunc) retryOption {
|
||||||
|
return retryOption{applyFunc: func(o *options) {
|
||||||
|
o.backoffFunc = bf
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
|
||||||
|
type options struct {
|
||||||
|
retryPolicy retryPolicy
|
||||||
|
max uint
|
||||||
|
backoffFunc backoffFunc
|
||||||
|
retryAuth bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// retryOption is a grpc.CallOption that is local to clientv3's retry interceptor.
|
||||||
|
type retryOption struct {
|
||||||
|
grpc.EmptyCallOption // make sure we implement private after() and before() fields so we don't panic.
|
||||||
|
applyFunc func(opt *options)
|
||||||
|
}
|
||||||
|
|
||||||
|
func reuseOrNewWithCallOptions(opt *options, retryOptions []retryOption) *options {
|
||||||
|
if len(retryOptions) == 0 {
|
||||||
|
return opt
|
||||||
|
}
|
||||||
|
optCopy := &options{}
|
||||||
|
*optCopy = *opt
|
||||||
|
for _, f := range retryOptions {
|
||||||
|
f.applyFunc(optCopy)
|
||||||
|
}
|
||||||
|
return optCopy
|
||||||
|
}
|
||||||
|
|
||||||
|
func filterCallOptions(callOptions []grpc.CallOption) (grpcOptions []grpc.CallOption, retryOptions []retryOption) {
|
||||||
|
for _, opt := range callOptions {
|
||||||
|
if co, ok := opt.(retryOption); ok {
|
||||||
|
retryOptions = append(retryOptions, co)
|
||||||
|
} else {
|
||||||
|
grpcOptions = append(grpcOptions, opt)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return grpcOptions, retryOptions
|
||||||
|
}
|
||||||
|
|
||||||
|
// BackoffLinearWithJitter waits a set period of time, allowing for jitter (fractional adjustment).
|
||||||
|
//
|
||||||
|
// For example waitBetween=1s and jitter=0.10 can generate waits between 900ms and 1100ms.
|
||||||
|
func backoffLinearWithJitter(waitBetween time.Duration, jitterFraction float64) backoffFunc {
|
||||||
|
return func(attempt uint) time.Duration {
|
||||||
|
return backoffutils.JitterUp(waitBetween, jitterFraction)
|
||||||
|
}
|
||||||
|
}
|
37
vendor/github.com/coreos/etcd/clientv3/sort.go
generated
vendored
Normal file
37
vendor/github.com/coreos/etcd/clientv3/sort.go
generated
vendored
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
type SortTarget int
|
||||||
|
type SortOrder int
|
||||||
|
|
||||||
|
const (
|
||||||
|
SortNone SortOrder = iota
|
||||||
|
SortAscend
|
||||||
|
SortDescend
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
SortByKey SortTarget = iota
|
||||||
|
SortByVersion
|
||||||
|
SortByCreateRevision
|
||||||
|
SortByModRevision
|
||||||
|
SortByValue
|
||||||
|
)
|
||||||
|
|
||||||
|
type SortOption struct {
|
||||||
|
Target SortTarget
|
||||||
|
Order SortOrder
|
||||||
|
}
|
151
vendor/github.com/coreos/etcd/clientv3/txn.go
generated
vendored
Normal file
151
vendor/github.com/coreos/etcd/clientv3/txn.go
generated
vendored
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Txn is the interface that wraps mini-transactions.
|
||||||
|
//
|
||||||
|
// Txn(context.TODO()).If(
|
||||||
|
// Compare(Value(k1), ">", v1),
|
||||||
|
// Compare(Version(k1), "=", 2)
|
||||||
|
// ).Then(
|
||||||
|
// OpPut(k2,v2), OpPut(k3,v3)
|
||||||
|
// ).Else(
|
||||||
|
// OpPut(k4,v4), OpPut(k5,v5)
|
||||||
|
// ).Commit()
|
||||||
|
//
|
||||||
|
type Txn interface {
|
||||||
|
// If takes a list of comparison. If all comparisons passed in succeed,
|
||||||
|
// the operations passed into Then() will be executed. Or the operations
|
||||||
|
// passed into Else() will be executed.
|
||||||
|
If(cs ...Cmp) Txn
|
||||||
|
|
||||||
|
// Then takes a list of operations. The Ops list will be executed, if the
|
||||||
|
// comparisons passed in If() succeed.
|
||||||
|
Then(ops ...Op) Txn
|
||||||
|
|
||||||
|
// Else takes a list of operations. The Ops list will be executed, if the
|
||||||
|
// comparisons passed in If() fail.
|
||||||
|
Else(ops ...Op) Txn
|
||||||
|
|
||||||
|
// Commit tries to commit the transaction.
|
||||||
|
Commit() (*TxnResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type txn struct {
|
||||||
|
kv *kv
|
||||||
|
ctx context.Context
|
||||||
|
|
||||||
|
mu sync.Mutex
|
||||||
|
cif bool
|
||||||
|
cthen bool
|
||||||
|
celse bool
|
||||||
|
|
||||||
|
isWrite bool
|
||||||
|
|
||||||
|
cmps []*pb.Compare
|
||||||
|
|
||||||
|
sus []*pb.RequestOp
|
||||||
|
fas []*pb.RequestOp
|
||||||
|
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
}
|
||||||
|
|
||||||
|
func (txn *txn) If(cs ...Cmp) Txn {
|
||||||
|
txn.mu.Lock()
|
||||||
|
defer txn.mu.Unlock()
|
||||||
|
|
||||||
|
if txn.cif {
|
||||||
|
panic("cannot call If twice!")
|
||||||
|
}
|
||||||
|
|
||||||
|
if txn.cthen {
|
||||||
|
panic("cannot call If after Then!")
|
||||||
|
}
|
||||||
|
|
||||||
|
if txn.celse {
|
||||||
|
panic("cannot call If after Else!")
|
||||||
|
}
|
||||||
|
|
||||||
|
txn.cif = true
|
||||||
|
|
||||||
|
for i := range cs {
|
||||||
|
txn.cmps = append(txn.cmps, (*pb.Compare)(&cs[i]))
|
||||||
|
}
|
||||||
|
|
||||||
|
return txn
|
||||||
|
}
|
||||||
|
|
||||||
|
func (txn *txn) Then(ops ...Op) Txn {
|
||||||
|
txn.mu.Lock()
|
||||||
|
defer txn.mu.Unlock()
|
||||||
|
|
||||||
|
if txn.cthen {
|
||||||
|
panic("cannot call Then twice!")
|
||||||
|
}
|
||||||
|
if txn.celse {
|
||||||
|
panic("cannot call Then after Else!")
|
||||||
|
}
|
||||||
|
|
||||||
|
txn.cthen = true
|
||||||
|
|
||||||
|
for _, op := range ops {
|
||||||
|
txn.isWrite = txn.isWrite || op.isWrite()
|
||||||
|
txn.sus = append(txn.sus, op.toRequestOp())
|
||||||
|
}
|
||||||
|
|
||||||
|
return txn
|
||||||
|
}
|
||||||
|
|
||||||
|
func (txn *txn) Else(ops ...Op) Txn {
|
||||||
|
txn.mu.Lock()
|
||||||
|
defer txn.mu.Unlock()
|
||||||
|
|
||||||
|
if txn.celse {
|
||||||
|
panic("cannot call Else twice!")
|
||||||
|
}
|
||||||
|
|
||||||
|
txn.celse = true
|
||||||
|
|
||||||
|
for _, op := range ops {
|
||||||
|
txn.isWrite = txn.isWrite || op.isWrite()
|
||||||
|
txn.fas = append(txn.fas, op.toRequestOp())
|
||||||
|
}
|
||||||
|
|
||||||
|
return txn
|
||||||
|
}
|
||||||
|
|
||||||
|
func (txn *txn) Commit() (*TxnResponse, error) {
|
||||||
|
txn.mu.Lock()
|
||||||
|
defer txn.mu.Unlock()
|
||||||
|
|
||||||
|
r := &pb.TxnRequest{Compare: txn.cmps, Success: txn.sus, Failure: txn.fas}
|
||||||
|
|
||||||
|
var resp *pb.TxnResponse
|
||||||
|
var err error
|
||||||
|
resp, err = txn.kv.remote.Txn(txn.ctx, r, txn.callOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, toErr(txn.ctx, err)
|
||||||
|
}
|
||||||
|
return (*TxnResponse)(resp), nil
|
||||||
|
}
|
980
vendor/github.com/coreos/etcd/clientv3/watch.go
generated
vendored
Normal file
980
vendor/github.com/coreos/etcd/clientv3/watch.go
generated
vendored
Normal file
@ -0,0 +1,980 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package clientv3
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
v3rpc "github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
|
||||||
|
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
|
||||||
|
mvccpb "github.com/coreos/etcd/mvcc/mvccpb"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/metadata"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
EventTypeDelete = mvccpb.DELETE
|
||||||
|
EventTypePut = mvccpb.PUT
|
||||||
|
|
||||||
|
closeSendErrTimeout = 250 * time.Millisecond
|
||||||
|
)
|
||||||
|
|
||||||
|
type Event mvccpb.Event
|
||||||
|
|
||||||
|
type WatchChan <-chan WatchResponse
|
||||||
|
|
||||||
|
type Watcher interface {
|
||||||
|
// Watch watches on a key or prefix. The watched events will be returned
|
||||||
|
// through the returned channel. If revisions waiting to be sent over the
|
||||||
|
// watch are compacted, then the watch will be canceled by the server, the
|
||||||
|
// client will post a compacted error watch response, and the channel will close.
|
||||||
|
// If the context "ctx" is canceled or timed out, returned "WatchChan" is closed,
|
||||||
|
// and "WatchResponse" from this closed channel has zero events and nil "Err()".
|
||||||
|
// The context "ctx" MUST be canceled, as soon as watcher is no longer being used,
|
||||||
|
// to release the associated resources.
|
||||||
|
//
|
||||||
|
// If the context is "context.Background/TODO", returned "WatchChan" will
|
||||||
|
// not be closed and block until event is triggered, except when server
|
||||||
|
// returns a non-recoverable error (e.g. ErrCompacted).
|
||||||
|
// For example, when context passed with "WithRequireLeader" and the
|
||||||
|
// connected server has no leader (e.g. due to network partition),
|
||||||
|
// error "etcdserver: no leader" (ErrNoLeader) will be returned,
|
||||||
|
// and then "WatchChan" is closed with non-nil "Err()".
|
||||||
|
// In order to prevent a watch stream being stuck in a partitioned node,
|
||||||
|
// make sure to wrap context with "WithRequireLeader".
|
||||||
|
//
|
||||||
|
// Otherwise, as long as the context has not been canceled or timed out,
|
||||||
|
// watch will retry on other recoverable errors forever until reconnected.
|
||||||
|
//
|
||||||
|
// TODO: explicitly set context error in the last "WatchResponse" message and close channel?
|
||||||
|
// Currently, client contexts are overwritten with "valCtx" that never closes.
|
||||||
|
// TODO(v3.4): configure watch retry policy, limit maximum retry number
|
||||||
|
// (see https://github.com/coreos/etcd/issues/8980)
|
||||||
|
Watch(ctx context.Context, key string, opts ...OpOption) WatchChan
|
||||||
|
|
||||||
|
// RequestProgress requests a progress notify response be sent in all watch channels.
|
||||||
|
RequestProgress(ctx context.Context) error
|
||||||
|
|
||||||
|
// Close closes the watcher and cancels all watch requests.
|
||||||
|
Close() error
|
||||||
|
}
|
||||||
|
|
||||||
|
type WatchResponse struct {
|
||||||
|
Header pb.ResponseHeader
|
||||||
|
Events []*Event
|
||||||
|
|
||||||
|
// CompactRevision is the minimum revision the watcher may receive.
|
||||||
|
CompactRevision int64
|
||||||
|
|
||||||
|
// Canceled is used to indicate watch failure.
|
||||||
|
// If the watch failed and the stream was about to close, before the channel is closed,
|
||||||
|
// the channel sends a final response that has Canceled set to true with a non-nil Err().
|
||||||
|
Canceled bool
|
||||||
|
|
||||||
|
// Created is used to indicate the creation of the watcher.
|
||||||
|
Created bool
|
||||||
|
|
||||||
|
closeErr error
|
||||||
|
|
||||||
|
// cancelReason is a reason of canceling watch
|
||||||
|
cancelReason string
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsCreate returns true if the event tells that the key is newly created.
|
||||||
|
func (e *Event) IsCreate() bool {
|
||||||
|
return e.Type == EventTypePut && e.Kv.CreateRevision == e.Kv.ModRevision
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsModify returns true if the event tells that a new value is put on existing key.
|
||||||
|
func (e *Event) IsModify() bool {
|
||||||
|
return e.Type == EventTypePut && e.Kv.CreateRevision != e.Kv.ModRevision
|
||||||
|
}
|
||||||
|
|
||||||
|
// Err is the error value if this WatchResponse holds an error.
|
||||||
|
func (wr *WatchResponse) Err() error {
|
||||||
|
switch {
|
||||||
|
case wr.closeErr != nil:
|
||||||
|
return v3rpc.Error(wr.closeErr)
|
||||||
|
case wr.CompactRevision != 0:
|
||||||
|
return v3rpc.ErrCompacted
|
||||||
|
case wr.Canceled:
|
||||||
|
if len(wr.cancelReason) != 0 {
|
||||||
|
return v3rpc.Error(status.Error(codes.FailedPrecondition, wr.cancelReason))
|
||||||
|
}
|
||||||
|
return v3rpc.ErrFutureRev
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsProgressNotify returns true if the WatchResponse is progress notification.
|
||||||
|
func (wr *WatchResponse) IsProgressNotify() bool {
|
||||||
|
return len(wr.Events) == 0 && !wr.Canceled && !wr.Created && wr.CompactRevision == 0 && wr.Header.Revision != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// watcher implements the Watcher interface
|
||||||
|
type watcher struct {
|
||||||
|
remote pb.WatchClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
|
||||||
|
// mu protects the grpc streams map
|
||||||
|
mu sync.RWMutex
|
||||||
|
|
||||||
|
// streams holds all the active grpc streams keyed by ctx value.
|
||||||
|
streams map[string]*watchGrpcStream
|
||||||
|
}
|
||||||
|
|
||||||
|
// watchGrpcStream tracks all watch resources attached to a single grpc stream.
|
||||||
|
type watchGrpcStream struct {
|
||||||
|
owner *watcher
|
||||||
|
remote pb.WatchClient
|
||||||
|
callOpts []grpc.CallOption
|
||||||
|
|
||||||
|
// ctx controls internal remote.Watch requests
|
||||||
|
ctx context.Context
|
||||||
|
// ctxKey is the key used when looking up this stream's context
|
||||||
|
ctxKey string
|
||||||
|
cancel context.CancelFunc
|
||||||
|
|
||||||
|
// substreams holds all active watchers on this grpc stream
|
||||||
|
substreams map[int64]*watcherStream
|
||||||
|
// resuming holds all resuming watchers on this grpc stream
|
||||||
|
resuming []*watcherStream
|
||||||
|
|
||||||
|
// reqc sends a watch request from Watch() to the main goroutine
|
||||||
|
reqc chan watchStreamRequest
|
||||||
|
// respc receives data from the watch client
|
||||||
|
respc chan *pb.WatchResponse
|
||||||
|
// donec closes to broadcast shutdown
|
||||||
|
donec chan struct{}
|
||||||
|
// errc transmits errors from grpc Recv to the watch stream reconnect logic
|
||||||
|
errc chan error
|
||||||
|
// closingc gets the watcherStream of closing watchers
|
||||||
|
closingc chan *watcherStream
|
||||||
|
// wg is Done when all substream goroutines have exited
|
||||||
|
wg sync.WaitGroup
|
||||||
|
|
||||||
|
// resumec closes to signal that all substreams should begin resuming
|
||||||
|
resumec chan struct{}
|
||||||
|
// closeErr is the error that closed the watch stream
|
||||||
|
closeErr error
|
||||||
|
}
|
||||||
|
|
||||||
|
// watchStreamRequest is a union of the supported watch request operation types
|
||||||
|
type watchStreamRequest interface {
|
||||||
|
toPB() *pb.WatchRequest
|
||||||
|
}
|
||||||
|
|
||||||
|
// watchRequest is issued by the subscriber to start a new watcher
|
||||||
|
type watchRequest struct {
|
||||||
|
ctx context.Context
|
||||||
|
key string
|
||||||
|
end string
|
||||||
|
rev int64
|
||||||
|
|
||||||
|
// send created notification event if this field is true
|
||||||
|
createdNotify bool
|
||||||
|
// progressNotify is for progress updates
|
||||||
|
progressNotify bool
|
||||||
|
// fragmentation should be disabled by default
|
||||||
|
// if true, split watch events when total exceeds
|
||||||
|
// "--max-request-bytes" flag value + 512-byte
|
||||||
|
fragment bool
|
||||||
|
|
||||||
|
// filters is the list of events to filter out
|
||||||
|
filters []pb.WatchCreateRequest_FilterType
|
||||||
|
// get the previous key-value pair before the event happens
|
||||||
|
prevKV bool
|
||||||
|
// retc receives a chan WatchResponse once the watcher is established
|
||||||
|
retc chan chan WatchResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
// progressRequest is issued by the subscriber to request watch progress
|
||||||
|
type progressRequest struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
// watcherStream represents a registered watcher
|
||||||
|
type watcherStream struct {
|
||||||
|
// initReq is the request that initiated this request
|
||||||
|
initReq watchRequest
|
||||||
|
|
||||||
|
// outc publishes watch responses to subscriber
|
||||||
|
outc chan WatchResponse
|
||||||
|
// recvc buffers watch responses before publishing
|
||||||
|
recvc chan *WatchResponse
|
||||||
|
// donec closes when the watcherStream goroutine stops.
|
||||||
|
donec chan struct{}
|
||||||
|
// closing is set to true when stream should be scheduled to shutdown.
|
||||||
|
closing bool
|
||||||
|
// id is the registered watch id on the grpc stream
|
||||||
|
id int64
|
||||||
|
|
||||||
|
// buf holds all events received from etcd but not yet consumed by the client
|
||||||
|
buf []*WatchResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWatcher(c *Client) Watcher {
|
||||||
|
return NewWatchFromWatchClient(pb.NewWatchClient(c.conn), c)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWatchFromWatchClient(wc pb.WatchClient, c *Client) Watcher {
|
||||||
|
w := &watcher{
|
||||||
|
remote: wc,
|
||||||
|
streams: make(map[string]*watchGrpcStream),
|
||||||
|
}
|
||||||
|
if c != nil {
|
||||||
|
w.callOpts = c.callOpts
|
||||||
|
}
|
||||||
|
return w
|
||||||
|
}
|
||||||
|
|
||||||
|
// never closes
|
||||||
|
var valCtxCh = make(chan struct{})
|
||||||
|
var zeroTime = time.Unix(0, 0)
|
||||||
|
|
||||||
|
// ctx with only the values; never Done
|
||||||
|
type valCtx struct{ context.Context }
|
||||||
|
|
||||||
|
func (vc *valCtx) Deadline() (time.Time, bool) { return zeroTime, false }
|
||||||
|
func (vc *valCtx) Done() <-chan struct{} { return valCtxCh }
|
||||||
|
func (vc *valCtx) Err() error { return nil }
|
||||||
|
|
||||||
|
func (w *watcher) newWatcherGrpcStream(inctx context.Context) *watchGrpcStream {
|
||||||
|
ctx, cancel := context.WithCancel(&valCtx{inctx})
|
||||||
|
wgs := &watchGrpcStream{
|
||||||
|
owner: w,
|
||||||
|
remote: w.remote,
|
||||||
|
callOpts: w.callOpts,
|
||||||
|
ctx: ctx,
|
||||||
|
ctxKey: streamKeyFromCtx(inctx),
|
||||||
|
cancel: cancel,
|
||||||
|
substreams: make(map[int64]*watcherStream),
|
||||||
|
respc: make(chan *pb.WatchResponse),
|
||||||
|
reqc: make(chan watchStreamRequest),
|
||||||
|
donec: make(chan struct{}),
|
||||||
|
errc: make(chan error, 1),
|
||||||
|
closingc: make(chan *watcherStream),
|
||||||
|
resumec: make(chan struct{}),
|
||||||
|
}
|
||||||
|
go wgs.run()
|
||||||
|
return wgs
|
||||||
|
}
|
||||||
|
|
||||||
|
// Watch posts a watch request to run() and waits for a new watcher channel
|
||||||
|
func (w *watcher) Watch(ctx context.Context, key string, opts ...OpOption) WatchChan {
|
||||||
|
ow := opWatch(key, opts...)
|
||||||
|
|
||||||
|
var filters []pb.WatchCreateRequest_FilterType
|
||||||
|
if ow.filterPut {
|
||||||
|
filters = append(filters, pb.WatchCreateRequest_NOPUT)
|
||||||
|
}
|
||||||
|
if ow.filterDelete {
|
||||||
|
filters = append(filters, pb.WatchCreateRequest_NODELETE)
|
||||||
|
}
|
||||||
|
|
||||||
|
wr := &watchRequest{
|
||||||
|
ctx: ctx,
|
||||||
|
createdNotify: ow.createdNotify,
|
||||||
|
key: string(ow.key),
|
||||||
|
end: string(ow.end),
|
||||||
|
rev: ow.rev,
|
||||||
|
progressNotify: ow.progressNotify,
|
||||||
|
fragment: ow.fragment,
|
||||||
|
filters: filters,
|
||||||
|
prevKV: ow.prevKV,
|
||||||
|
retc: make(chan chan WatchResponse, 1),
|
||||||
|
}
|
||||||
|
|
||||||
|
ok := false
|
||||||
|
ctxKey := streamKeyFromCtx(ctx)
|
||||||
|
|
||||||
|
// find or allocate appropriate grpc watch stream
|
||||||
|
w.mu.Lock()
|
||||||
|
if w.streams == nil {
|
||||||
|
// closed
|
||||||
|
w.mu.Unlock()
|
||||||
|
ch := make(chan WatchResponse)
|
||||||
|
close(ch)
|
||||||
|
return ch
|
||||||
|
}
|
||||||
|
wgs := w.streams[ctxKey]
|
||||||
|
if wgs == nil {
|
||||||
|
wgs = w.newWatcherGrpcStream(ctx)
|
||||||
|
w.streams[ctxKey] = wgs
|
||||||
|
}
|
||||||
|
donec := wgs.donec
|
||||||
|
reqc := wgs.reqc
|
||||||
|
w.mu.Unlock()
|
||||||
|
|
||||||
|
// couldn't create channel; return closed channel
|
||||||
|
closeCh := make(chan WatchResponse, 1)
|
||||||
|
|
||||||
|
// submit request
|
||||||
|
select {
|
||||||
|
case reqc <- wr:
|
||||||
|
ok = true
|
||||||
|
case <-wr.ctx.Done():
|
||||||
|
case <-donec:
|
||||||
|
if wgs.closeErr != nil {
|
||||||
|
closeCh <- WatchResponse{closeErr: wgs.closeErr}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// retry; may have dropped stream from no ctxs
|
||||||
|
return w.Watch(ctx, key, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// receive channel
|
||||||
|
if ok {
|
||||||
|
select {
|
||||||
|
case ret := <-wr.retc:
|
||||||
|
return ret
|
||||||
|
case <-ctx.Done():
|
||||||
|
case <-donec:
|
||||||
|
if wgs.closeErr != nil {
|
||||||
|
closeCh <- WatchResponse{closeErr: wgs.closeErr}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// retry; may have dropped stream from no ctxs
|
||||||
|
return w.Watch(ctx, key, opts...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
close(closeCh)
|
||||||
|
return closeCh
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watcher) Close() (err error) {
|
||||||
|
w.mu.Lock()
|
||||||
|
streams := w.streams
|
||||||
|
w.streams = nil
|
||||||
|
w.mu.Unlock()
|
||||||
|
for _, wgs := range streams {
|
||||||
|
if werr := wgs.close(); werr != nil {
|
||||||
|
err = werr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// RequestProgress requests a progress notify response be sent in all watch channels.
|
||||||
|
func (w *watcher) RequestProgress(ctx context.Context) (err error) {
|
||||||
|
ctxKey := streamKeyFromCtx(ctx)
|
||||||
|
|
||||||
|
w.mu.Lock()
|
||||||
|
if w.streams == nil {
|
||||||
|
return fmt.Errorf("no stream found for context")
|
||||||
|
}
|
||||||
|
wgs := w.streams[ctxKey]
|
||||||
|
if wgs == nil {
|
||||||
|
wgs = w.newWatcherGrpcStream(ctx)
|
||||||
|
w.streams[ctxKey] = wgs
|
||||||
|
}
|
||||||
|
donec := wgs.donec
|
||||||
|
reqc := wgs.reqc
|
||||||
|
w.mu.Unlock()
|
||||||
|
|
||||||
|
pr := &progressRequest{}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case reqc <- pr:
|
||||||
|
return nil
|
||||||
|
case <-ctx.Done():
|
||||||
|
if err == nil {
|
||||||
|
return ctx.Err()
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
case <-donec:
|
||||||
|
if wgs.closeErr != nil {
|
||||||
|
return wgs.closeErr
|
||||||
|
}
|
||||||
|
// retry; may have dropped stream from no ctxs
|
||||||
|
return w.RequestProgress(ctx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watchGrpcStream) close() (err error) {
|
||||||
|
w.cancel()
|
||||||
|
<-w.donec
|
||||||
|
select {
|
||||||
|
case err = <-w.errc:
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
return toErr(w.ctx, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watcher) closeStream(wgs *watchGrpcStream) {
|
||||||
|
w.mu.Lock()
|
||||||
|
close(wgs.donec)
|
||||||
|
wgs.cancel()
|
||||||
|
if w.streams != nil {
|
||||||
|
delete(w.streams, wgs.ctxKey)
|
||||||
|
}
|
||||||
|
w.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watchGrpcStream) addSubstream(resp *pb.WatchResponse, ws *watcherStream) {
|
||||||
|
// check watch ID for backward compatibility (<= v3.3)
|
||||||
|
if resp.WatchId == -1 || (resp.Canceled && resp.CancelReason != "") {
|
||||||
|
// failed; no channel
|
||||||
|
close(ws.recvc)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ws.id = resp.WatchId
|
||||||
|
w.substreams[ws.id] = ws
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watchGrpcStream) sendCloseSubstream(ws *watcherStream, resp *WatchResponse) {
|
||||||
|
select {
|
||||||
|
case ws.outc <- *resp:
|
||||||
|
case <-ws.initReq.ctx.Done():
|
||||||
|
case <-time.After(closeSendErrTimeout):
|
||||||
|
}
|
||||||
|
close(ws.outc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watchGrpcStream) closeSubstream(ws *watcherStream) {
|
||||||
|
// send channel response in case stream was never established
|
||||||
|
select {
|
||||||
|
case ws.initReq.retc <- ws.outc:
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
// close subscriber's channel
|
||||||
|
if closeErr := w.closeErr; closeErr != nil && ws.initReq.ctx.Err() == nil {
|
||||||
|
go w.sendCloseSubstream(ws, &WatchResponse{closeErr: w.closeErr})
|
||||||
|
} else if ws.outc != nil {
|
||||||
|
close(ws.outc)
|
||||||
|
}
|
||||||
|
if ws.id != -1 {
|
||||||
|
delete(w.substreams, ws.id)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for i := range w.resuming {
|
||||||
|
if w.resuming[i] == ws {
|
||||||
|
w.resuming[i] = nil
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// run is the root of the goroutines for managing a watcher client
|
||||||
|
func (w *watchGrpcStream) run() {
|
||||||
|
var wc pb.Watch_WatchClient
|
||||||
|
var closeErr error
|
||||||
|
|
||||||
|
// substreams marked to close but goroutine still running; needed for
|
||||||
|
// avoiding double-closing recvc on grpc stream teardown
|
||||||
|
closing := make(map[*watcherStream]struct{})
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
w.closeErr = closeErr
|
||||||
|
// shutdown substreams and resuming substreams
|
||||||
|
for _, ws := range w.substreams {
|
||||||
|
if _, ok := closing[ws]; !ok {
|
||||||
|
close(ws.recvc)
|
||||||
|
closing[ws] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, ws := range w.resuming {
|
||||||
|
if _, ok := closing[ws]; ws != nil && !ok {
|
||||||
|
close(ws.recvc)
|
||||||
|
closing[ws] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
w.joinSubstreams()
|
||||||
|
for range closing {
|
||||||
|
w.closeSubstream(<-w.closingc)
|
||||||
|
}
|
||||||
|
w.wg.Wait()
|
||||||
|
w.owner.closeStream(w)
|
||||||
|
}()
|
||||||
|
|
||||||
|
// start a stream with the etcd grpc server
|
||||||
|
if wc, closeErr = w.newWatchClient(); closeErr != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
cancelSet := make(map[int64]struct{})
|
||||||
|
|
||||||
|
var cur *pb.WatchResponse
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
// Watch() requested
|
||||||
|
case req := <-w.reqc:
|
||||||
|
switch wreq := req.(type) {
|
||||||
|
case *watchRequest:
|
||||||
|
outc := make(chan WatchResponse, 1)
|
||||||
|
// TODO: pass custom watch ID?
|
||||||
|
ws := &watcherStream{
|
||||||
|
initReq: *wreq,
|
||||||
|
id: -1,
|
||||||
|
outc: outc,
|
||||||
|
// unbuffered so resumes won't cause repeat events
|
||||||
|
recvc: make(chan *WatchResponse),
|
||||||
|
}
|
||||||
|
|
||||||
|
ws.donec = make(chan struct{})
|
||||||
|
w.wg.Add(1)
|
||||||
|
go w.serveSubstream(ws, w.resumec)
|
||||||
|
|
||||||
|
// queue up for watcher creation/resume
|
||||||
|
w.resuming = append(w.resuming, ws)
|
||||||
|
if len(w.resuming) == 1 {
|
||||||
|
// head of resume queue, can register a new watcher
|
||||||
|
wc.Send(ws.initReq.toPB())
|
||||||
|
}
|
||||||
|
case *progressRequest:
|
||||||
|
wc.Send(wreq.toPB())
|
||||||
|
}
|
||||||
|
|
||||||
|
// new events from the watch client
|
||||||
|
case pbresp := <-w.respc:
|
||||||
|
if cur == nil || pbresp.Created || pbresp.Canceled {
|
||||||
|
cur = pbresp
|
||||||
|
} else if cur != nil && cur.WatchId == pbresp.WatchId {
|
||||||
|
// merge new events
|
||||||
|
cur.Events = append(cur.Events, pbresp.Events...)
|
||||||
|
// update "Fragment" field; last response with "Fragment" == false
|
||||||
|
cur.Fragment = pbresp.Fragment
|
||||||
|
}
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case pbresp.Created:
|
||||||
|
// response to head of queue creation
|
||||||
|
if ws := w.resuming[0]; ws != nil {
|
||||||
|
w.addSubstream(pbresp, ws)
|
||||||
|
w.dispatchEvent(pbresp)
|
||||||
|
w.resuming[0] = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if ws := w.nextResume(); ws != nil {
|
||||||
|
wc.Send(ws.initReq.toPB())
|
||||||
|
}
|
||||||
|
|
||||||
|
// reset for next iteration
|
||||||
|
cur = nil
|
||||||
|
|
||||||
|
case pbresp.Canceled && pbresp.CompactRevision == 0:
|
||||||
|
delete(cancelSet, pbresp.WatchId)
|
||||||
|
if ws, ok := w.substreams[pbresp.WatchId]; ok {
|
||||||
|
// signal to stream goroutine to update closingc
|
||||||
|
close(ws.recvc)
|
||||||
|
closing[ws] = struct{}{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// reset for next iteration
|
||||||
|
cur = nil
|
||||||
|
|
||||||
|
case cur.Fragment:
|
||||||
|
// watch response events are still fragmented
|
||||||
|
// continue to fetch next fragmented event arrival
|
||||||
|
continue
|
||||||
|
|
||||||
|
default:
|
||||||
|
// dispatch to appropriate watch stream
|
||||||
|
ok := w.dispatchEvent(cur)
|
||||||
|
|
||||||
|
// reset for next iteration
|
||||||
|
cur = nil
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// watch response on unexpected watch id; cancel id
|
||||||
|
if _, ok := cancelSet[pbresp.WatchId]; ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
cancelSet[pbresp.WatchId] = struct{}{}
|
||||||
|
cr := &pb.WatchRequest_CancelRequest{
|
||||||
|
CancelRequest: &pb.WatchCancelRequest{
|
||||||
|
WatchId: pbresp.WatchId,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
req := &pb.WatchRequest{RequestUnion: cr}
|
||||||
|
wc.Send(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
// watch client failed on Recv; spawn another if possible
|
||||||
|
case err := <-w.errc:
|
||||||
|
if isHaltErr(w.ctx, err) || toErr(w.ctx, err) == v3rpc.ErrNoLeader {
|
||||||
|
closeErr = err
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if wc, closeErr = w.newWatchClient(); closeErr != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if ws := w.nextResume(); ws != nil {
|
||||||
|
wc.Send(ws.initReq.toPB())
|
||||||
|
}
|
||||||
|
cancelSet = make(map[int64]struct{})
|
||||||
|
|
||||||
|
case <-w.ctx.Done():
|
||||||
|
return
|
||||||
|
|
||||||
|
case ws := <-w.closingc:
|
||||||
|
w.closeSubstream(ws)
|
||||||
|
delete(closing, ws)
|
||||||
|
// no more watchers on this stream, shutdown
|
||||||
|
if len(w.substreams)+len(w.resuming) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// nextResume chooses the next resuming to register with the grpc stream. Abandoned
|
||||||
|
// streams are marked as nil in the queue since the head must wait for its inflight registration.
|
||||||
|
func (w *watchGrpcStream) nextResume() *watcherStream {
|
||||||
|
for len(w.resuming) != 0 {
|
||||||
|
if w.resuming[0] != nil {
|
||||||
|
return w.resuming[0]
|
||||||
|
}
|
||||||
|
w.resuming = w.resuming[1:len(w.resuming)]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// dispatchEvent sends a WatchResponse to the appropriate watcher stream
|
||||||
|
func (w *watchGrpcStream) dispatchEvent(pbresp *pb.WatchResponse) bool {
|
||||||
|
events := make([]*Event, len(pbresp.Events))
|
||||||
|
for i, ev := range pbresp.Events {
|
||||||
|
events[i] = (*Event)(ev)
|
||||||
|
}
|
||||||
|
// TODO: return watch ID?
|
||||||
|
wr := &WatchResponse{
|
||||||
|
Header: *pbresp.Header,
|
||||||
|
Events: events,
|
||||||
|
CompactRevision: pbresp.CompactRevision,
|
||||||
|
Created: pbresp.Created,
|
||||||
|
Canceled: pbresp.Canceled,
|
||||||
|
cancelReason: pbresp.CancelReason,
|
||||||
|
}
|
||||||
|
|
||||||
|
// watch IDs are zero indexed, so request notify watch responses are assigned a watch ID of -1 to
|
||||||
|
// indicate they should be broadcast.
|
||||||
|
if wr.IsProgressNotify() && pbresp.WatchId == -1 {
|
||||||
|
return w.broadcastResponse(wr)
|
||||||
|
}
|
||||||
|
|
||||||
|
return w.unicastResponse(wr, pbresp.WatchId)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// broadcastResponse send a watch response to all watch substreams.
|
||||||
|
func (w *watchGrpcStream) broadcastResponse(wr *WatchResponse) bool {
|
||||||
|
for _, ws := range w.substreams {
|
||||||
|
select {
|
||||||
|
case ws.recvc <- wr:
|
||||||
|
case <-ws.donec:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// unicastResponse sends a watch response to a specific watch substream.
|
||||||
|
func (w *watchGrpcStream) unicastResponse(wr *WatchResponse, watchId int64) bool {
|
||||||
|
ws, ok := w.substreams[watchId]
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case ws.recvc <- wr:
|
||||||
|
case <-ws.donec:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// serveWatchClient forwards messages from the grpc stream to run()
|
||||||
|
func (w *watchGrpcStream) serveWatchClient(wc pb.Watch_WatchClient) {
|
||||||
|
for {
|
||||||
|
resp, err := wc.Recv()
|
||||||
|
if err != nil {
|
||||||
|
select {
|
||||||
|
case w.errc <- err:
|
||||||
|
case <-w.donec:
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case w.respc <- resp:
|
||||||
|
case <-w.donec:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// serveSubstream forwards watch responses from run() to the subscriber
|
||||||
|
func (w *watchGrpcStream) serveSubstream(ws *watcherStream, resumec chan struct{}) {
|
||||||
|
if ws.closing {
|
||||||
|
panic("created substream goroutine but substream is closing")
|
||||||
|
}
|
||||||
|
|
||||||
|
// nextRev is the minimum expected next revision
|
||||||
|
nextRev := ws.initReq.rev
|
||||||
|
resuming := false
|
||||||
|
defer func() {
|
||||||
|
if !resuming {
|
||||||
|
ws.closing = true
|
||||||
|
}
|
||||||
|
close(ws.donec)
|
||||||
|
if !resuming {
|
||||||
|
w.closingc <- ws
|
||||||
|
}
|
||||||
|
w.wg.Done()
|
||||||
|
}()
|
||||||
|
|
||||||
|
emptyWr := &WatchResponse{}
|
||||||
|
for {
|
||||||
|
curWr := emptyWr
|
||||||
|
outc := ws.outc
|
||||||
|
|
||||||
|
if len(ws.buf) > 0 {
|
||||||
|
curWr = ws.buf[0]
|
||||||
|
} else {
|
||||||
|
outc = nil
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case outc <- *curWr:
|
||||||
|
if ws.buf[0].Err() != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ws.buf[0] = nil
|
||||||
|
ws.buf = ws.buf[1:]
|
||||||
|
case wr, ok := <-ws.recvc:
|
||||||
|
if !ok {
|
||||||
|
// shutdown from closeSubstream
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if wr.Created {
|
||||||
|
if ws.initReq.retc != nil {
|
||||||
|
ws.initReq.retc <- ws.outc
|
||||||
|
// to prevent next write from taking the slot in buffered channel
|
||||||
|
// and posting duplicate create events
|
||||||
|
ws.initReq.retc = nil
|
||||||
|
|
||||||
|
// send first creation event only if requested
|
||||||
|
if ws.initReq.createdNotify {
|
||||||
|
ws.outc <- *wr
|
||||||
|
}
|
||||||
|
// once the watch channel is returned, a current revision
|
||||||
|
// watch must resume at the store revision. This is necessary
|
||||||
|
// for the following case to work as expected:
|
||||||
|
// wch := m1.Watch("a")
|
||||||
|
// m2.Put("a", "b")
|
||||||
|
// <-wch
|
||||||
|
// If the revision is only bound on the first observed event,
|
||||||
|
// if wch is disconnected before the Put is issued, then reconnects
|
||||||
|
// after it is committed, it'll miss the Put.
|
||||||
|
if ws.initReq.rev == 0 {
|
||||||
|
nextRev = wr.Header.Revision
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// current progress of watch; <= store revision
|
||||||
|
nextRev = wr.Header.Revision
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(wr.Events) > 0 {
|
||||||
|
nextRev = wr.Events[len(wr.Events)-1].Kv.ModRevision + 1
|
||||||
|
}
|
||||||
|
ws.initReq.rev = nextRev
|
||||||
|
|
||||||
|
// created event is already sent above,
|
||||||
|
// watcher should not post duplicate events
|
||||||
|
if wr.Created {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO pause channel if buffer gets too large
|
||||||
|
ws.buf = append(ws.buf, wr)
|
||||||
|
case <-w.ctx.Done():
|
||||||
|
return
|
||||||
|
case <-ws.initReq.ctx.Done():
|
||||||
|
return
|
||||||
|
case <-resumec:
|
||||||
|
resuming = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// lazily send cancel message if events on missing id
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watchGrpcStream) newWatchClient() (pb.Watch_WatchClient, error) {
|
||||||
|
// mark all substreams as resuming
|
||||||
|
close(w.resumec)
|
||||||
|
w.resumec = make(chan struct{})
|
||||||
|
w.joinSubstreams()
|
||||||
|
for _, ws := range w.substreams {
|
||||||
|
ws.id = -1
|
||||||
|
w.resuming = append(w.resuming, ws)
|
||||||
|
}
|
||||||
|
// strip out nils, if any
|
||||||
|
var resuming []*watcherStream
|
||||||
|
for _, ws := range w.resuming {
|
||||||
|
if ws != nil {
|
||||||
|
resuming = append(resuming, ws)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
w.resuming = resuming
|
||||||
|
w.substreams = make(map[int64]*watcherStream)
|
||||||
|
|
||||||
|
// connect to grpc stream while accepting watcher cancelation
|
||||||
|
stopc := make(chan struct{})
|
||||||
|
donec := w.waitCancelSubstreams(stopc)
|
||||||
|
wc, err := w.openWatchClient()
|
||||||
|
close(stopc)
|
||||||
|
<-donec
|
||||||
|
|
||||||
|
// serve all non-closing streams, even if there's a client error
|
||||||
|
// so that the teardown path can shutdown the streams as expected.
|
||||||
|
for _, ws := range w.resuming {
|
||||||
|
if ws.closing {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
ws.donec = make(chan struct{})
|
||||||
|
w.wg.Add(1)
|
||||||
|
go w.serveSubstream(ws, w.resumec)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, v3rpc.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// receive data from new grpc stream
|
||||||
|
go w.serveWatchClient(wc)
|
||||||
|
return wc, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *watchGrpcStream) waitCancelSubstreams(stopc <-chan struct{}) <-chan struct{} {
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(len(w.resuming))
|
||||||
|
donec := make(chan struct{})
|
||||||
|
for i := range w.resuming {
|
||||||
|
go func(ws *watcherStream) {
|
||||||
|
defer wg.Done()
|
||||||
|
if ws.closing {
|
||||||
|
if ws.initReq.ctx.Err() != nil && ws.outc != nil {
|
||||||
|
close(ws.outc)
|
||||||
|
ws.outc = nil
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case <-ws.initReq.ctx.Done():
|
||||||
|
// closed ws will be removed from resuming
|
||||||
|
ws.closing = true
|
||||||
|
close(ws.outc)
|
||||||
|
ws.outc = nil
|
||||||
|
w.wg.Add(1)
|
||||||
|
go func() {
|
||||||
|
defer w.wg.Done()
|
||||||
|
w.closingc <- ws
|
||||||
|
}()
|
||||||
|
case <-stopc:
|
||||||
|
}
|
||||||
|
}(w.resuming[i])
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
defer close(donec)
|
||||||
|
wg.Wait()
|
||||||
|
}()
|
||||||
|
return donec
|
||||||
|
}
|
||||||
|
|
||||||
|
// joinSubstreams waits for all substream goroutines to complete.
|
||||||
|
func (w *watchGrpcStream) joinSubstreams() {
|
||||||
|
for _, ws := range w.substreams {
|
||||||
|
<-ws.donec
|
||||||
|
}
|
||||||
|
for _, ws := range w.resuming {
|
||||||
|
if ws != nil {
|
||||||
|
<-ws.donec
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var maxBackoff = 100 * time.Millisecond
|
||||||
|
|
||||||
|
// openWatchClient retries opening a watch client until success or halt.
|
||||||
|
// manually retry in case "ws==nil && err==nil"
|
||||||
|
// TODO: remove FailFast=false
|
||||||
|
func (w *watchGrpcStream) openWatchClient() (ws pb.Watch_WatchClient, err error) {
|
||||||
|
backoff := time.Millisecond
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-w.ctx.Done():
|
||||||
|
if err == nil {
|
||||||
|
return nil, w.ctx.Err()
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
if ws, err = w.remote.Watch(w.ctx, w.callOpts...); ws != nil && err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if isHaltErr(w.ctx, err) {
|
||||||
|
return nil, v3rpc.Error(err)
|
||||||
|
}
|
||||||
|
if isUnavailableErr(w.ctx, err) {
|
||||||
|
// retry, but backoff
|
||||||
|
if backoff < maxBackoff {
|
||||||
|
// 25% backoff factor
|
||||||
|
backoff = backoff + backoff/4
|
||||||
|
if backoff > maxBackoff {
|
||||||
|
backoff = maxBackoff
|
||||||
|
}
|
||||||
|
}
|
||||||
|
time.Sleep(backoff)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ws, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// toPB converts an internal watch request structure to its protobuf WatchRequest structure.
|
||||||
|
func (wr *watchRequest) toPB() *pb.WatchRequest {
|
||||||
|
req := &pb.WatchCreateRequest{
|
||||||
|
StartRevision: wr.rev,
|
||||||
|
Key: []byte(wr.key),
|
||||||
|
RangeEnd: []byte(wr.end),
|
||||||
|
ProgressNotify: wr.progressNotify,
|
||||||
|
Filters: wr.filters,
|
||||||
|
PrevKv: wr.prevKV,
|
||||||
|
Fragment: wr.fragment,
|
||||||
|
}
|
||||||
|
cr := &pb.WatchRequest_CreateRequest{CreateRequest: req}
|
||||||
|
return &pb.WatchRequest{RequestUnion: cr}
|
||||||
|
}
|
||||||
|
|
||||||
|
// toPB converts an internal progress request structure to its protobuf WatchRequest structure.
|
||||||
|
func (pr *progressRequest) toPB() *pb.WatchRequest {
|
||||||
|
req := &pb.WatchProgressRequest{}
|
||||||
|
cr := &pb.WatchRequest_ProgressRequest{ProgressRequest: req}
|
||||||
|
return &pb.WatchRequest{RequestUnion: cr}
|
||||||
|
}
|
||||||
|
|
||||||
|
func streamKeyFromCtx(ctx context.Context) string {
|
||||||
|
if md, ok := metadata.FromOutgoingContext(ctx); ok {
|
||||||
|
return fmt.Sprintf("%+v", md)
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
16
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/doc.go
generated
vendored
Normal file
16
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/doc.go
generated
vendored
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package rpctypes has types and values shared by the etcd server and client for v3 RPC interaction.
|
||||||
|
package rpctypes
|
215
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/error.go
generated
vendored
Normal file
215
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/error.go
generated
vendored
Normal file
@ -0,0 +1,215 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package rpctypes
|
||||||
|
|
||||||
|
import (
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// server-side error
|
||||||
|
var (
|
||||||
|
ErrGRPCEmptyKey = status.New(codes.InvalidArgument, "etcdserver: key is not provided").Err()
|
||||||
|
ErrGRPCKeyNotFound = status.New(codes.InvalidArgument, "etcdserver: key not found").Err()
|
||||||
|
ErrGRPCValueProvided = status.New(codes.InvalidArgument, "etcdserver: value is provided").Err()
|
||||||
|
ErrGRPCLeaseProvided = status.New(codes.InvalidArgument, "etcdserver: lease is provided").Err()
|
||||||
|
ErrGRPCTooManyOps = status.New(codes.InvalidArgument, "etcdserver: too many operations in txn request").Err()
|
||||||
|
ErrGRPCDuplicateKey = status.New(codes.InvalidArgument, "etcdserver: duplicate key given in txn request").Err()
|
||||||
|
ErrGRPCCompacted = status.New(codes.OutOfRange, "etcdserver: mvcc: required revision has been compacted").Err()
|
||||||
|
ErrGRPCFutureRev = status.New(codes.OutOfRange, "etcdserver: mvcc: required revision is a future revision").Err()
|
||||||
|
ErrGRPCNoSpace = status.New(codes.ResourceExhausted, "etcdserver: mvcc: database space exceeded").Err()
|
||||||
|
|
||||||
|
ErrGRPCLeaseNotFound = status.New(codes.NotFound, "etcdserver: requested lease not found").Err()
|
||||||
|
ErrGRPCLeaseExist = status.New(codes.FailedPrecondition, "etcdserver: lease already exists").Err()
|
||||||
|
ErrGRPCLeaseTTLTooLarge = status.New(codes.OutOfRange, "etcdserver: too large lease TTL").Err()
|
||||||
|
|
||||||
|
ErrGRPCMemberExist = status.New(codes.FailedPrecondition, "etcdserver: member ID already exist").Err()
|
||||||
|
ErrGRPCPeerURLExist = status.New(codes.FailedPrecondition, "etcdserver: Peer URLs already exists").Err()
|
||||||
|
ErrGRPCMemberNotEnoughStarted = status.New(codes.FailedPrecondition, "etcdserver: re-configuration failed due to not enough started members").Err()
|
||||||
|
ErrGRPCMemberBadURLs = status.New(codes.InvalidArgument, "etcdserver: given member URLs are invalid").Err()
|
||||||
|
ErrGRPCMemberNotFound = status.New(codes.NotFound, "etcdserver: member not found").Err()
|
||||||
|
|
||||||
|
ErrGRPCRequestTooLarge = status.New(codes.InvalidArgument, "etcdserver: request is too large").Err()
|
||||||
|
ErrGRPCRequestTooManyRequests = status.New(codes.ResourceExhausted, "etcdserver: too many requests").Err()
|
||||||
|
|
||||||
|
ErrGRPCRootUserNotExist = status.New(codes.FailedPrecondition, "etcdserver: root user does not exist").Err()
|
||||||
|
ErrGRPCRootRoleNotExist = status.New(codes.FailedPrecondition, "etcdserver: root user does not have root role").Err()
|
||||||
|
ErrGRPCUserAlreadyExist = status.New(codes.FailedPrecondition, "etcdserver: user name already exists").Err()
|
||||||
|
ErrGRPCUserEmpty = status.New(codes.InvalidArgument, "etcdserver: user name is empty").Err()
|
||||||
|
ErrGRPCUserNotFound = status.New(codes.FailedPrecondition, "etcdserver: user name not found").Err()
|
||||||
|
ErrGRPCRoleAlreadyExist = status.New(codes.FailedPrecondition, "etcdserver: role name already exists").Err()
|
||||||
|
ErrGRPCRoleNotFound = status.New(codes.FailedPrecondition, "etcdserver: role name not found").Err()
|
||||||
|
ErrGRPCAuthFailed = status.New(codes.InvalidArgument, "etcdserver: authentication failed, invalid user ID or password").Err()
|
||||||
|
ErrGRPCPermissionDenied = status.New(codes.PermissionDenied, "etcdserver: permission denied").Err()
|
||||||
|
ErrGRPCRoleNotGranted = status.New(codes.FailedPrecondition, "etcdserver: role is not granted to the user").Err()
|
||||||
|
ErrGRPCPermissionNotGranted = status.New(codes.FailedPrecondition, "etcdserver: permission is not granted to the role").Err()
|
||||||
|
ErrGRPCAuthNotEnabled = status.New(codes.FailedPrecondition, "etcdserver: authentication is not enabled").Err()
|
||||||
|
ErrGRPCInvalidAuthToken = status.New(codes.Unauthenticated, "etcdserver: invalid auth token").Err()
|
||||||
|
ErrGRPCInvalidAuthMgmt = status.New(codes.InvalidArgument, "etcdserver: invalid auth management").Err()
|
||||||
|
|
||||||
|
ErrGRPCNoLeader = status.New(codes.Unavailable, "etcdserver: no leader").Err()
|
||||||
|
ErrGRPCNotLeader = status.New(codes.FailedPrecondition, "etcdserver: not leader").Err()
|
||||||
|
ErrGRPCNotCapable = status.New(codes.Unavailable, "etcdserver: not capable").Err()
|
||||||
|
ErrGRPCStopped = status.New(codes.Unavailable, "etcdserver: server stopped").Err()
|
||||||
|
ErrGRPCTimeout = status.New(codes.Unavailable, "etcdserver: request timed out").Err()
|
||||||
|
ErrGRPCTimeoutDueToLeaderFail = status.New(codes.Unavailable, "etcdserver: request timed out, possibly due to previous leader failure").Err()
|
||||||
|
ErrGRPCTimeoutDueToConnectionLost = status.New(codes.Unavailable, "etcdserver: request timed out, possibly due to connection lost").Err()
|
||||||
|
ErrGRPCUnhealthy = status.New(codes.Unavailable, "etcdserver: unhealthy cluster").Err()
|
||||||
|
ErrGRPCCorrupt = status.New(codes.DataLoss, "etcdserver: corrupt cluster").Err()
|
||||||
|
|
||||||
|
errStringToError = map[string]error{
|
||||||
|
ErrorDesc(ErrGRPCEmptyKey): ErrGRPCEmptyKey,
|
||||||
|
ErrorDesc(ErrGRPCKeyNotFound): ErrGRPCKeyNotFound,
|
||||||
|
ErrorDesc(ErrGRPCValueProvided): ErrGRPCValueProvided,
|
||||||
|
ErrorDesc(ErrGRPCLeaseProvided): ErrGRPCLeaseProvided,
|
||||||
|
|
||||||
|
ErrorDesc(ErrGRPCTooManyOps): ErrGRPCTooManyOps,
|
||||||
|
ErrorDesc(ErrGRPCDuplicateKey): ErrGRPCDuplicateKey,
|
||||||
|
ErrorDesc(ErrGRPCCompacted): ErrGRPCCompacted,
|
||||||
|
ErrorDesc(ErrGRPCFutureRev): ErrGRPCFutureRev,
|
||||||
|
ErrorDesc(ErrGRPCNoSpace): ErrGRPCNoSpace,
|
||||||
|
|
||||||
|
ErrorDesc(ErrGRPCLeaseNotFound): ErrGRPCLeaseNotFound,
|
||||||
|
ErrorDesc(ErrGRPCLeaseExist): ErrGRPCLeaseExist,
|
||||||
|
ErrorDesc(ErrGRPCLeaseTTLTooLarge): ErrGRPCLeaseTTLTooLarge,
|
||||||
|
|
||||||
|
ErrorDesc(ErrGRPCMemberExist): ErrGRPCMemberExist,
|
||||||
|
ErrorDesc(ErrGRPCPeerURLExist): ErrGRPCPeerURLExist,
|
||||||
|
ErrorDesc(ErrGRPCMemberNotEnoughStarted): ErrGRPCMemberNotEnoughStarted,
|
||||||
|
ErrorDesc(ErrGRPCMemberBadURLs): ErrGRPCMemberBadURLs,
|
||||||
|
ErrorDesc(ErrGRPCMemberNotFound): ErrGRPCMemberNotFound,
|
||||||
|
|
||||||
|
ErrorDesc(ErrGRPCRequestTooLarge): ErrGRPCRequestTooLarge,
|
||||||
|
ErrorDesc(ErrGRPCRequestTooManyRequests): ErrGRPCRequestTooManyRequests,
|
||||||
|
|
||||||
|
ErrorDesc(ErrGRPCRootUserNotExist): ErrGRPCRootUserNotExist,
|
||||||
|
ErrorDesc(ErrGRPCRootRoleNotExist): ErrGRPCRootRoleNotExist,
|
||||||
|
ErrorDesc(ErrGRPCUserAlreadyExist): ErrGRPCUserAlreadyExist,
|
||||||
|
ErrorDesc(ErrGRPCUserEmpty): ErrGRPCUserEmpty,
|
||||||
|
ErrorDesc(ErrGRPCUserNotFound): ErrGRPCUserNotFound,
|
||||||
|
ErrorDesc(ErrGRPCRoleAlreadyExist): ErrGRPCRoleAlreadyExist,
|
||||||
|
ErrorDesc(ErrGRPCRoleNotFound): ErrGRPCRoleNotFound,
|
||||||
|
ErrorDesc(ErrGRPCAuthFailed): ErrGRPCAuthFailed,
|
||||||
|
ErrorDesc(ErrGRPCPermissionDenied): ErrGRPCPermissionDenied,
|
||||||
|
ErrorDesc(ErrGRPCRoleNotGranted): ErrGRPCRoleNotGranted,
|
||||||
|
ErrorDesc(ErrGRPCPermissionNotGranted): ErrGRPCPermissionNotGranted,
|
||||||
|
ErrorDesc(ErrGRPCAuthNotEnabled): ErrGRPCAuthNotEnabled,
|
||||||
|
ErrorDesc(ErrGRPCInvalidAuthToken): ErrGRPCInvalidAuthToken,
|
||||||
|
ErrorDesc(ErrGRPCInvalidAuthMgmt): ErrGRPCInvalidAuthMgmt,
|
||||||
|
|
||||||
|
ErrorDesc(ErrGRPCNoLeader): ErrGRPCNoLeader,
|
||||||
|
ErrorDesc(ErrGRPCNotLeader): ErrGRPCNotLeader,
|
||||||
|
ErrorDesc(ErrGRPCNotCapable): ErrGRPCNotCapable,
|
||||||
|
ErrorDesc(ErrGRPCStopped): ErrGRPCStopped,
|
||||||
|
ErrorDesc(ErrGRPCTimeout): ErrGRPCTimeout,
|
||||||
|
ErrorDesc(ErrGRPCTimeoutDueToLeaderFail): ErrGRPCTimeoutDueToLeaderFail,
|
||||||
|
ErrorDesc(ErrGRPCTimeoutDueToConnectionLost): ErrGRPCTimeoutDueToConnectionLost,
|
||||||
|
ErrorDesc(ErrGRPCUnhealthy): ErrGRPCUnhealthy,
|
||||||
|
ErrorDesc(ErrGRPCCorrupt): ErrGRPCCorrupt,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// client-side error
|
||||||
|
var (
|
||||||
|
ErrEmptyKey = Error(ErrGRPCEmptyKey)
|
||||||
|
ErrKeyNotFound = Error(ErrGRPCKeyNotFound)
|
||||||
|
ErrValueProvided = Error(ErrGRPCValueProvided)
|
||||||
|
ErrLeaseProvided = Error(ErrGRPCLeaseProvided)
|
||||||
|
ErrTooManyOps = Error(ErrGRPCTooManyOps)
|
||||||
|
ErrDuplicateKey = Error(ErrGRPCDuplicateKey)
|
||||||
|
ErrCompacted = Error(ErrGRPCCompacted)
|
||||||
|
ErrFutureRev = Error(ErrGRPCFutureRev)
|
||||||
|
ErrNoSpace = Error(ErrGRPCNoSpace)
|
||||||
|
|
||||||
|
ErrLeaseNotFound = Error(ErrGRPCLeaseNotFound)
|
||||||
|
ErrLeaseExist = Error(ErrGRPCLeaseExist)
|
||||||
|
ErrLeaseTTLTooLarge = Error(ErrGRPCLeaseTTLTooLarge)
|
||||||
|
|
||||||
|
ErrMemberExist = Error(ErrGRPCMemberExist)
|
||||||
|
ErrPeerURLExist = Error(ErrGRPCPeerURLExist)
|
||||||
|
ErrMemberNotEnoughStarted = Error(ErrGRPCMemberNotEnoughStarted)
|
||||||
|
ErrMemberBadURLs = Error(ErrGRPCMemberBadURLs)
|
||||||
|
ErrMemberNotFound = Error(ErrGRPCMemberNotFound)
|
||||||
|
|
||||||
|
ErrRequestTooLarge = Error(ErrGRPCRequestTooLarge)
|
||||||
|
ErrTooManyRequests = Error(ErrGRPCRequestTooManyRequests)
|
||||||
|
|
||||||
|
ErrRootUserNotExist = Error(ErrGRPCRootUserNotExist)
|
||||||
|
ErrRootRoleNotExist = Error(ErrGRPCRootRoleNotExist)
|
||||||
|
ErrUserAlreadyExist = Error(ErrGRPCUserAlreadyExist)
|
||||||
|
ErrUserEmpty = Error(ErrGRPCUserEmpty)
|
||||||
|
ErrUserNotFound = Error(ErrGRPCUserNotFound)
|
||||||
|
ErrRoleAlreadyExist = Error(ErrGRPCRoleAlreadyExist)
|
||||||
|
ErrRoleNotFound = Error(ErrGRPCRoleNotFound)
|
||||||
|
ErrAuthFailed = Error(ErrGRPCAuthFailed)
|
||||||
|
ErrPermissionDenied = Error(ErrGRPCPermissionDenied)
|
||||||
|
ErrRoleNotGranted = Error(ErrGRPCRoleNotGranted)
|
||||||
|
ErrPermissionNotGranted = Error(ErrGRPCPermissionNotGranted)
|
||||||
|
ErrAuthNotEnabled = Error(ErrGRPCAuthNotEnabled)
|
||||||
|
ErrInvalidAuthToken = Error(ErrGRPCInvalidAuthToken)
|
||||||
|
ErrInvalidAuthMgmt = Error(ErrGRPCInvalidAuthMgmt)
|
||||||
|
|
||||||
|
ErrNoLeader = Error(ErrGRPCNoLeader)
|
||||||
|
ErrNotLeader = Error(ErrGRPCNotLeader)
|
||||||
|
ErrNotCapable = Error(ErrGRPCNotCapable)
|
||||||
|
ErrStopped = Error(ErrGRPCStopped)
|
||||||
|
ErrTimeout = Error(ErrGRPCTimeout)
|
||||||
|
ErrTimeoutDueToLeaderFail = Error(ErrGRPCTimeoutDueToLeaderFail)
|
||||||
|
ErrTimeoutDueToConnectionLost = Error(ErrGRPCTimeoutDueToConnectionLost)
|
||||||
|
ErrUnhealthy = Error(ErrGRPCUnhealthy)
|
||||||
|
ErrCorrupt = Error(ErrGRPCCorrupt)
|
||||||
|
)
|
||||||
|
|
||||||
|
// EtcdError defines gRPC server errors.
|
||||||
|
// (https://github.com/grpc/grpc-go/blob/master/rpc_util.go#L319-L323)
|
||||||
|
type EtcdError struct {
|
||||||
|
code codes.Code
|
||||||
|
desc string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Code returns grpc/codes.Code.
|
||||||
|
// TODO: define clientv3/codes.Code.
|
||||||
|
func (e EtcdError) Code() codes.Code {
|
||||||
|
return e.code
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e EtcdError) Error() string {
|
||||||
|
return e.desc
|
||||||
|
}
|
||||||
|
|
||||||
|
func Error(err error) error {
|
||||||
|
if err == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
verr, ok := errStringToError[ErrorDesc(err)]
|
||||||
|
if !ok { // not gRPC error
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
ev, ok := status.FromError(verr)
|
||||||
|
var desc string
|
||||||
|
if ok {
|
||||||
|
desc = ev.Message()
|
||||||
|
} else {
|
||||||
|
desc = verr.Error()
|
||||||
|
}
|
||||||
|
return EtcdError{code: ev.Code(), desc: desc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ErrorDesc(err error) string {
|
||||||
|
if s, ok := status.FromError(err); ok {
|
||||||
|
return s.Message()
|
||||||
|
}
|
||||||
|
return err.Error()
|
||||||
|
}
|
20
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/md.go
generated
vendored
Normal file
20
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/md.go
generated
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package rpctypes
|
||||||
|
|
||||||
|
var (
|
||||||
|
MetadataRequireLeaderKey = "hasleader"
|
||||||
|
MetadataHasLeader = "true"
|
||||||
|
)
|
20
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/metadatafields.go
generated
vendored
Normal file
20
vendor/github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes/metadatafields.go
generated
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package rpctypes
|
||||||
|
|
||||||
|
var (
|
||||||
|
TokenFieldNameGRPC = "token"
|
||||||
|
TokenFieldNameSwagger = "authorization"
|
||||||
|
)
|
1036
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/etcdserver.pb.go
generated
vendored
Normal file
1036
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/etcdserver.pb.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
34
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/etcdserver.proto
generated
vendored
Normal file
34
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/etcdserver.proto
generated
vendored
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
syntax = "proto2";
|
||||||
|
package etcdserverpb;
|
||||||
|
|
||||||
|
import "gogoproto/gogo.proto";
|
||||||
|
|
||||||
|
option (gogoproto.marshaler_all) = true;
|
||||||
|
option (gogoproto.sizer_all) = true;
|
||||||
|
option (gogoproto.unmarshaler_all) = true;
|
||||||
|
option (gogoproto.goproto_getters_all) = false;
|
||||||
|
|
||||||
|
message Request {
|
||||||
|
optional uint64 ID = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional string Method = 2 [(gogoproto.nullable) = false];
|
||||||
|
optional string Path = 3 [(gogoproto.nullable) = false];
|
||||||
|
optional string Val = 4 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Dir = 5 [(gogoproto.nullable) = false];
|
||||||
|
optional string PrevValue = 6 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 PrevIndex = 7 [(gogoproto.nullable) = false];
|
||||||
|
optional bool PrevExist = 8 [(gogoproto.nullable) = true];
|
||||||
|
optional int64 Expiration = 9 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Wait = 10 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 Since = 11 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Recursive = 12 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Sorted = 13 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Quorum = 14 [(gogoproto.nullable) = false];
|
||||||
|
optional int64 Time = 15 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Stream = 16 [(gogoproto.nullable) = false];
|
||||||
|
optional bool Refresh = 17 [(gogoproto.nullable) = true];
|
||||||
|
}
|
||||||
|
|
||||||
|
message Metadata {
|
||||||
|
optional uint64 NodeID = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 ClusterID = 2 [(gogoproto.nullable) = false];
|
||||||
|
}
|
2077
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/raft_internal.pb.go
generated
vendored
Normal file
2077
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/raft_internal.pb.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
74
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/raft_internal.proto
generated
vendored
Normal file
74
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/raft_internal.proto
generated
vendored
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
package etcdserverpb;
|
||||||
|
|
||||||
|
import "gogoproto/gogo.proto";
|
||||||
|
import "etcdserver.proto";
|
||||||
|
import "rpc.proto";
|
||||||
|
|
||||||
|
option (gogoproto.marshaler_all) = true;
|
||||||
|
option (gogoproto.sizer_all) = true;
|
||||||
|
option (gogoproto.unmarshaler_all) = true;
|
||||||
|
option (gogoproto.goproto_getters_all) = false;
|
||||||
|
|
||||||
|
message RequestHeader {
|
||||||
|
uint64 ID = 1;
|
||||||
|
// username is a username that is associated with an auth token of gRPC connection
|
||||||
|
string username = 2;
|
||||||
|
// auth_revision is a revision number of auth.authStore. It is not related to mvcc
|
||||||
|
uint64 auth_revision = 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
// An InternalRaftRequest is the union of all requests which can be
|
||||||
|
// sent via raft.
|
||||||
|
message InternalRaftRequest {
|
||||||
|
RequestHeader header = 100;
|
||||||
|
uint64 ID = 1;
|
||||||
|
|
||||||
|
Request v2 = 2;
|
||||||
|
|
||||||
|
RangeRequest range = 3;
|
||||||
|
PutRequest put = 4;
|
||||||
|
DeleteRangeRequest delete_range = 5;
|
||||||
|
TxnRequest txn = 6;
|
||||||
|
CompactionRequest compaction = 7;
|
||||||
|
|
||||||
|
LeaseGrantRequest lease_grant = 8;
|
||||||
|
LeaseRevokeRequest lease_revoke = 9;
|
||||||
|
|
||||||
|
AlarmRequest alarm = 10;
|
||||||
|
|
||||||
|
AuthEnableRequest auth_enable = 1000;
|
||||||
|
AuthDisableRequest auth_disable = 1011;
|
||||||
|
|
||||||
|
InternalAuthenticateRequest authenticate = 1012;
|
||||||
|
|
||||||
|
AuthUserAddRequest auth_user_add = 1100;
|
||||||
|
AuthUserDeleteRequest auth_user_delete = 1101;
|
||||||
|
AuthUserGetRequest auth_user_get = 1102;
|
||||||
|
AuthUserChangePasswordRequest auth_user_change_password = 1103;
|
||||||
|
AuthUserGrantRoleRequest auth_user_grant_role = 1104;
|
||||||
|
AuthUserRevokeRoleRequest auth_user_revoke_role = 1105;
|
||||||
|
AuthUserListRequest auth_user_list = 1106;
|
||||||
|
AuthRoleListRequest auth_role_list = 1107;
|
||||||
|
|
||||||
|
AuthRoleAddRequest auth_role_add = 1200;
|
||||||
|
AuthRoleDeleteRequest auth_role_delete = 1201;
|
||||||
|
AuthRoleGetRequest auth_role_get = 1202;
|
||||||
|
AuthRoleGrantPermissionRequest auth_role_grant_permission = 1203;
|
||||||
|
AuthRoleRevokePermissionRequest auth_role_revoke_permission = 1204;
|
||||||
|
}
|
||||||
|
|
||||||
|
message EmptyResponse {
|
||||||
|
}
|
||||||
|
|
||||||
|
// What is the difference between AuthenticateRequest (defined in rpc.proto) and InternalAuthenticateRequest?
|
||||||
|
// InternalAuthenticateRequest has a member that is filled by etcdserver and shouldn't be user-facing.
|
||||||
|
// For avoiding misusage the field, we have an internal version of AuthenticateRequest.
|
||||||
|
message InternalAuthenticateRequest {
|
||||||
|
string name = 1;
|
||||||
|
string password = 2;
|
||||||
|
|
||||||
|
// simple_token is generated in API layer (etcdserver/v3_server.go)
|
||||||
|
string simple_token = 3;
|
||||||
|
}
|
||||||
|
|
183
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/raft_internal_stringer.go
generated
vendored
Normal file
183
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/raft_internal_stringer.go
generated
vendored
Normal file
@ -0,0 +1,183 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package etcdserverpb
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
proto "github.com/golang/protobuf/proto"
|
||||||
|
)
|
||||||
|
|
||||||
|
// InternalRaftStringer implements custom proto Stringer:
|
||||||
|
// redact password, replace value fields with value_size fields.
|
||||||
|
type InternalRaftStringer struct {
|
||||||
|
Request *InternalRaftRequest
|
||||||
|
}
|
||||||
|
|
||||||
|
func (as *InternalRaftStringer) String() string {
|
||||||
|
switch {
|
||||||
|
case as.Request.LeaseGrant != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> lease_grant:<ttl:%d-second id:%016x>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
as.Request.LeaseGrant.TTL,
|
||||||
|
as.Request.LeaseGrant.ID,
|
||||||
|
)
|
||||||
|
case as.Request.LeaseRevoke != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> lease_revoke:<id:%016x>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
as.Request.LeaseRevoke.ID,
|
||||||
|
)
|
||||||
|
case as.Request.Authenticate != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> authenticate:<name:%s simple_token:%s>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
as.Request.Authenticate.Name,
|
||||||
|
as.Request.Authenticate.SimpleToken,
|
||||||
|
)
|
||||||
|
case as.Request.AuthUserAdd != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> auth_user_add:<name:%s>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
as.Request.AuthUserAdd.Name,
|
||||||
|
)
|
||||||
|
case as.Request.AuthUserChangePassword != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> auth_user_change_password:<name:%s>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
as.Request.AuthUserChangePassword.Name,
|
||||||
|
)
|
||||||
|
case as.Request.Put != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> put:<%s>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
newLoggablePutRequest(as.Request.Put).String(),
|
||||||
|
)
|
||||||
|
case as.Request.Txn != nil:
|
||||||
|
return fmt.Sprintf("header:<%s> txn:<%s>",
|
||||||
|
as.Request.Header.String(),
|
||||||
|
NewLoggableTxnRequest(as.Request.Txn).String(),
|
||||||
|
)
|
||||||
|
default:
|
||||||
|
// nothing to redact
|
||||||
|
}
|
||||||
|
return as.Request.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
// txnRequestStringer implements a custom proto String to replace value bytes fields with value size
|
||||||
|
// fields in any nested txn and put operations.
|
||||||
|
type txnRequestStringer struct {
|
||||||
|
Request *TxnRequest
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewLoggableTxnRequest(request *TxnRequest) *txnRequestStringer {
|
||||||
|
return &txnRequestStringer{request}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (as *txnRequestStringer) String() string {
|
||||||
|
var compare []string
|
||||||
|
for _, c := range as.Request.Compare {
|
||||||
|
switch cv := c.TargetUnion.(type) {
|
||||||
|
case *Compare_Value:
|
||||||
|
compare = append(compare, newLoggableValueCompare(c, cv).String())
|
||||||
|
default:
|
||||||
|
// nothing to redact
|
||||||
|
compare = append(compare, c.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var success []string
|
||||||
|
for _, s := range as.Request.Success {
|
||||||
|
success = append(success, newLoggableRequestOp(s).String())
|
||||||
|
}
|
||||||
|
var failure []string
|
||||||
|
for _, f := range as.Request.Failure {
|
||||||
|
failure = append(failure, newLoggableRequestOp(f).String())
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("compare:<%s> success:<%s> failure:<%s>",
|
||||||
|
strings.Join(compare, " "),
|
||||||
|
strings.Join(success, " "),
|
||||||
|
strings.Join(failure, " "),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// requestOpStringer implements a custom proto String to replace value bytes fields with value
|
||||||
|
// size fields in any nested txn and put operations.
|
||||||
|
type requestOpStringer struct {
|
||||||
|
Op *RequestOp
|
||||||
|
}
|
||||||
|
|
||||||
|
func newLoggableRequestOp(op *RequestOp) *requestOpStringer {
|
||||||
|
return &requestOpStringer{op}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (as *requestOpStringer) String() string {
|
||||||
|
switch op := as.Op.Request.(type) {
|
||||||
|
case *RequestOp_RequestPut:
|
||||||
|
return fmt.Sprintf("request_put:<%s>", newLoggablePutRequest(op.RequestPut).String())
|
||||||
|
case *RequestOp_RequestTxn:
|
||||||
|
return fmt.Sprintf("request_txn:<%s>", NewLoggableTxnRequest(op.RequestTxn).String())
|
||||||
|
default:
|
||||||
|
// nothing to redact
|
||||||
|
}
|
||||||
|
return as.Op.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
// loggableValueCompare implements a custom proto String for Compare.Value union member types to
|
||||||
|
// replace the value bytes field with a value size field.
|
||||||
|
// To preserve proto encoding of the key and range_end bytes, a faked out proto type is used here.
|
||||||
|
type loggableValueCompare struct {
|
||||||
|
Result Compare_CompareResult `protobuf:"varint,1,opt,name=result,proto3,enum=etcdserverpb.Compare_CompareResult"`
|
||||||
|
Target Compare_CompareTarget `protobuf:"varint,2,opt,name=target,proto3,enum=etcdserverpb.Compare_CompareTarget"`
|
||||||
|
Key []byte `protobuf:"bytes,3,opt,name=key,proto3"`
|
||||||
|
ValueSize int `protobuf:"bytes,7,opt,name=value_size,proto3"`
|
||||||
|
RangeEnd []byte `protobuf:"bytes,64,opt,name=range_end,proto3"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func newLoggableValueCompare(c *Compare, cv *Compare_Value) *loggableValueCompare {
|
||||||
|
return &loggableValueCompare{
|
||||||
|
c.Result,
|
||||||
|
c.Target,
|
||||||
|
c.Key,
|
||||||
|
len(cv.Value),
|
||||||
|
c.RangeEnd,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *loggableValueCompare) Reset() { *m = loggableValueCompare{} }
|
||||||
|
func (m *loggableValueCompare) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*loggableValueCompare) ProtoMessage() {}
|
||||||
|
|
||||||
|
// loggablePutRequest implements a custom proto String to replace value bytes field with a value
|
||||||
|
// size field.
|
||||||
|
// To preserve proto encoding of the key bytes, a faked out proto type is used here.
|
||||||
|
type loggablePutRequest struct {
|
||||||
|
Key []byte `protobuf:"bytes,1,opt,name=key,proto3"`
|
||||||
|
ValueSize int `protobuf:"varint,2,opt,name=value_size,proto3"`
|
||||||
|
Lease int64 `protobuf:"varint,3,opt,name=lease,proto3"`
|
||||||
|
PrevKv bool `protobuf:"varint,4,opt,name=prev_kv,proto3"`
|
||||||
|
IgnoreValue bool `protobuf:"varint,5,opt,name=ignore_value,proto3"`
|
||||||
|
IgnoreLease bool `protobuf:"varint,6,opt,name=ignore_lease,proto3"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func newLoggablePutRequest(request *PutRequest) *loggablePutRequest {
|
||||||
|
return &loggablePutRequest{
|
||||||
|
request.Key,
|
||||||
|
len(request.Value),
|
||||||
|
request.Lease,
|
||||||
|
request.PrevKv,
|
||||||
|
request.IgnoreValue,
|
||||||
|
request.IgnoreLease,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *loggablePutRequest) Reset() { *m = loggablePutRequest{} }
|
||||||
|
func (m *loggablePutRequest) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*loggablePutRequest) ProtoMessage() {}
|
19112
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/rpc.pb.go
generated
vendored
Normal file
19112
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/rpc.pb.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
1104
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/rpc.proto
generated
vendored
Normal file
1104
vendor/github.com/coreos/etcd/etcdserver/etcdserverpb/rpc.proto
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
718
vendor/github.com/coreos/etcd/mvcc/mvccpb/kv.pb.go
generated
vendored
Normal file
718
vendor/github.com/coreos/etcd/mvcc/mvccpb/kv.pb.go
generated
vendored
Normal file
@ -0,0 +1,718 @@
|
|||||||
|
// Code generated by protoc-gen-gogo. DO NOT EDIT.
|
||||||
|
// source: kv.proto
|
||||||
|
|
||||||
|
/*
|
||||||
|
Package mvccpb is a generated protocol buffer package.
|
||||||
|
|
||||||
|
It is generated from these files:
|
||||||
|
kv.proto
|
||||||
|
|
||||||
|
It has these top-level messages:
|
||||||
|
KeyValue
|
||||||
|
Event
|
||||||
|
*/
|
||||||
|
package mvccpb
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
proto "github.com/golang/protobuf/proto"
|
||||||
|
|
||||||
|
math "math"
|
||||||
|
|
||||||
|
_ "github.com/gogo/protobuf/gogoproto"
|
||||||
|
|
||||||
|
io "io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Reference imports to suppress errors if they are not otherwise used.
|
||||||
|
var _ = proto.Marshal
|
||||||
|
var _ = fmt.Errorf
|
||||||
|
var _ = math.Inf
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the proto package it is being compiled against.
|
||||||
|
// A compilation error at this line likely means your copy of the
|
||||||
|
// proto package needs to be updated.
|
||||||
|
const _ = proto.ProtoPackageIsVersion2 // please upgrade the proto package
|
||||||
|
|
||||||
|
type Event_EventType int32
|
||||||
|
|
||||||
|
const (
|
||||||
|
PUT Event_EventType = 0
|
||||||
|
DELETE Event_EventType = 1
|
||||||
|
)
|
||||||
|
|
||||||
|
var Event_EventType_name = map[int32]string{
|
||||||
|
0: "PUT",
|
||||||
|
1: "DELETE",
|
||||||
|
}
|
||||||
|
var Event_EventType_value = map[string]int32{
|
||||||
|
"PUT": 0,
|
||||||
|
"DELETE": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x Event_EventType) String() string {
|
||||||
|
return proto.EnumName(Event_EventType_name, int32(x))
|
||||||
|
}
|
||||||
|
func (Event_EventType) EnumDescriptor() ([]byte, []int) { return fileDescriptorKv, []int{1, 0} }
|
||||||
|
|
||||||
|
type KeyValue struct {
|
||||||
|
// key is the key in bytes. An empty key is not allowed.
|
||||||
|
Key []byte `protobuf:"bytes,1,opt,name=key,proto3" json:"key,omitempty"`
|
||||||
|
// create_revision is the revision of last creation on this key.
|
||||||
|
CreateRevision int64 `protobuf:"varint,2,opt,name=create_revision,json=createRevision,proto3" json:"create_revision,omitempty"`
|
||||||
|
// mod_revision is the revision of last modification on this key.
|
||||||
|
ModRevision int64 `protobuf:"varint,3,opt,name=mod_revision,json=modRevision,proto3" json:"mod_revision,omitempty"`
|
||||||
|
// version is the version of the key. A deletion resets
|
||||||
|
// the version to zero and any modification of the key
|
||||||
|
// increases its version.
|
||||||
|
Version int64 `protobuf:"varint,4,opt,name=version,proto3" json:"version,omitempty"`
|
||||||
|
// value is the value held by the key, in bytes.
|
||||||
|
Value []byte `protobuf:"bytes,5,opt,name=value,proto3" json:"value,omitempty"`
|
||||||
|
// lease is the ID of the lease that attached to key.
|
||||||
|
// When the attached lease expires, the key will be deleted.
|
||||||
|
// If lease is 0, then no lease is attached to the key.
|
||||||
|
Lease int64 `protobuf:"varint,6,opt,name=lease,proto3" json:"lease,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *KeyValue) Reset() { *m = KeyValue{} }
|
||||||
|
func (m *KeyValue) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*KeyValue) ProtoMessage() {}
|
||||||
|
func (*KeyValue) Descriptor() ([]byte, []int) { return fileDescriptorKv, []int{0} }
|
||||||
|
|
||||||
|
type Event struct {
|
||||||
|
// type is the kind of event. If type is a PUT, it indicates
|
||||||
|
// new data has been stored to the key. If type is a DELETE,
|
||||||
|
// it indicates the key was deleted.
|
||||||
|
Type Event_EventType `protobuf:"varint,1,opt,name=type,proto3,enum=mvccpb.Event_EventType" json:"type,omitempty"`
|
||||||
|
// kv holds the KeyValue for the event.
|
||||||
|
// A PUT event contains current kv pair.
|
||||||
|
// A PUT event with kv.Version=1 indicates the creation of a key.
|
||||||
|
// A DELETE/EXPIRE event contains the deleted key with
|
||||||
|
// its modification revision set to the revision of deletion.
|
||||||
|
Kv *KeyValue `protobuf:"bytes,2,opt,name=kv" json:"kv,omitempty"`
|
||||||
|
// prev_kv holds the key-value pair before the event happens.
|
||||||
|
PrevKv *KeyValue `protobuf:"bytes,3,opt,name=prev_kv,json=prevKv" json:"prev_kv,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Event) Reset() { *m = Event{} }
|
||||||
|
func (m *Event) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*Event) ProtoMessage() {}
|
||||||
|
func (*Event) Descriptor() ([]byte, []int) { return fileDescriptorKv, []int{1} }
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
proto.RegisterType((*KeyValue)(nil), "mvccpb.KeyValue")
|
||||||
|
proto.RegisterType((*Event)(nil), "mvccpb.Event")
|
||||||
|
proto.RegisterEnum("mvccpb.Event_EventType", Event_EventType_name, Event_EventType_value)
|
||||||
|
}
|
||||||
|
func (m *KeyValue) Marshal() (dAtA []byte, err error) {
|
||||||
|
size := m.Size()
|
||||||
|
dAtA = make([]byte, size)
|
||||||
|
n, err := m.MarshalTo(dAtA)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return dAtA[:n], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *KeyValue) MarshalTo(dAtA []byte) (int, error) {
|
||||||
|
var i int
|
||||||
|
_ = i
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if len(m.Key) > 0 {
|
||||||
|
dAtA[i] = 0xa
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(len(m.Key)))
|
||||||
|
i += copy(dAtA[i:], m.Key)
|
||||||
|
}
|
||||||
|
if m.CreateRevision != 0 {
|
||||||
|
dAtA[i] = 0x10
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.CreateRevision))
|
||||||
|
}
|
||||||
|
if m.ModRevision != 0 {
|
||||||
|
dAtA[i] = 0x18
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.ModRevision))
|
||||||
|
}
|
||||||
|
if m.Version != 0 {
|
||||||
|
dAtA[i] = 0x20
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.Version))
|
||||||
|
}
|
||||||
|
if len(m.Value) > 0 {
|
||||||
|
dAtA[i] = 0x2a
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(len(m.Value)))
|
||||||
|
i += copy(dAtA[i:], m.Value)
|
||||||
|
}
|
||||||
|
if m.Lease != 0 {
|
||||||
|
dAtA[i] = 0x30
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.Lease))
|
||||||
|
}
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Event) Marshal() (dAtA []byte, err error) {
|
||||||
|
size := m.Size()
|
||||||
|
dAtA = make([]byte, size)
|
||||||
|
n, err := m.MarshalTo(dAtA)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return dAtA[:n], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Event) MarshalTo(dAtA []byte) (int, error) {
|
||||||
|
var i int
|
||||||
|
_ = i
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if m.Type != 0 {
|
||||||
|
dAtA[i] = 0x8
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.Type))
|
||||||
|
}
|
||||||
|
if m.Kv != nil {
|
||||||
|
dAtA[i] = 0x12
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.Kv.Size()))
|
||||||
|
n1, err := m.Kv.MarshalTo(dAtA[i:])
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
i += n1
|
||||||
|
}
|
||||||
|
if m.PrevKv != nil {
|
||||||
|
dAtA[i] = 0x1a
|
||||||
|
i++
|
||||||
|
i = encodeVarintKv(dAtA, i, uint64(m.PrevKv.Size()))
|
||||||
|
n2, err := m.PrevKv.MarshalTo(dAtA[i:])
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
i += n2
|
||||||
|
}
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func encodeVarintKv(dAtA []byte, offset int, v uint64) int {
|
||||||
|
for v >= 1<<7 {
|
||||||
|
dAtA[offset] = uint8(v&0x7f | 0x80)
|
||||||
|
v >>= 7
|
||||||
|
offset++
|
||||||
|
}
|
||||||
|
dAtA[offset] = uint8(v)
|
||||||
|
return offset + 1
|
||||||
|
}
|
||||||
|
func (m *KeyValue) Size() (n int) {
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
l = len(m.Key)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovKv(uint64(l))
|
||||||
|
}
|
||||||
|
if m.CreateRevision != 0 {
|
||||||
|
n += 1 + sovKv(uint64(m.CreateRevision))
|
||||||
|
}
|
||||||
|
if m.ModRevision != 0 {
|
||||||
|
n += 1 + sovKv(uint64(m.ModRevision))
|
||||||
|
}
|
||||||
|
if m.Version != 0 {
|
||||||
|
n += 1 + sovKv(uint64(m.Version))
|
||||||
|
}
|
||||||
|
l = len(m.Value)
|
||||||
|
if l > 0 {
|
||||||
|
n += 1 + l + sovKv(uint64(l))
|
||||||
|
}
|
||||||
|
if m.Lease != 0 {
|
||||||
|
n += 1 + sovKv(uint64(m.Lease))
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Event) Size() (n int) {
|
||||||
|
var l int
|
||||||
|
_ = l
|
||||||
|
if m.Type != 0 {
|
||||||
|
n += 1 + sovKv(uint64(m.Type))
|
||||||
|
}
|
||||||
|
if m.Kv != nil {
|
||||||
|
l = m.Kv.Size()
|
||||||
|
n += 1 + l + sovKv(uint64(l))
|
||||||
|
}
|
||||||
|
if m.PrevKv != nil {
|
||||||
|
l = m.PrevKv.Size()
|
||||||
|
n += 1 + l + sovKv(uint64(l))
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func sovKv(x uint64) (n int) {
|
||||||
|
for {
|
||||||
|
n++
|
||||||
|
x >>= 7
|
||||||
|
if x == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
func sozKv(x uint64) (n int) {
|
||||||
|
return sovKv(uint64((x << 1) ^ uint64((int64(x) >> 63))))
|
||||||
|
}
|
||||||
|
func (m *KeyValue) Unmarshal(dAtA []byte) error {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
preIndex := iNdEx
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fieldNum := int32(wire >> 3)
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
if wireType == 4 {
|
||||||
|
return fmt.Errorf("proto: KeyValue: wiretype end group for non-group")
|
||||||
|
}
|
||||||
|
if fieldNum <= 0 {
|
||||||
|
return fmt.Errorf("proto: KeyValue: illegal tag %d (wire type %d)", fieldNum, wire)
|
||||||
|
}
|
||||||
|
switch fieldNum {
|
||||||
|
case 1:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Key", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Key = append(m.Key[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.Key == nil {
|
||||||
|
m.Key = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 2:
|
||||||
|
if wireType != 0 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field CreateRevision", wireType)
|
||||||
|
}
|
||||||
|
m.CreateRevision = 0
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
m.CreateRevision |= (int64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case 3:
|
||||||
|
if wireType != 0 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field ModRevision", wireType)
|
||||||
|
}
|
||||||
|
m.ModRevision = 0
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
m.ModRevision |= (int64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case 4:
|
||||||
|
if wireType != 0 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Version", wireType)
|
||||||
|
}
|
||||||
|
m.Version = 0
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
m.Version |= (int64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case 5:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Value", wireType)
|
||||||
|
}
|
||||||
|
var byteLen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
byteLen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if byteLen < 0 {
|
||||||
|
return ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + byteLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.Value = append(m.Value[:0], dAtA[iNdEx:postIndex]...)
|
||||||
|
if m.Value == nil {
|
||||||
|
m.Value = []byte{}
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 6:
|
||||||
|
if wireType != 0 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Lease", wireType)
|
||||||
|
}
|
||||||
|
m.Lease = 0
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
m.Lease |= (int64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
iNdEx = preIndex
|
||||||
|
skippy, err := skipKv(dAtA[iNdEx:])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if skippy < 0 {
|
||||||
|
return ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
if (iNdEx + skippy) > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx += skippy
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if iNdEx > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func (m *Event) Unmarshal(dAtA []byte) error {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
preIndex := iNdEx
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fieldNum := int32(wire >> 3)
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
if wireType == 4 {
|
||||||
|
return fmt.Errorf("proto: Event: wiretype end group for non-group")
|
||||||
|
}
|
||||||
|
if fieldNum <= 0 {
|
||||||
|
return fmt.Errorf("proto: Event: illegal tag %d (wire type %d)", fieldNum, wire)
|
||||||
|
}
|
||||||
|
switch fieldNum {
|
||||||
|
case 1:
|
||||||
|
if wireType != 0 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Type", wireType)
|
||||||
|
}
|
||||||
|
m.Type = 0
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
m.Type |= (Event_EventType(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case 2:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field Kv", wireType)
|
||||||
|
}
|
||||||
|
var msglen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
msglen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if msglen < 0 {
|
||||||
|
return ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + msglen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
if m.Kv == nil {
|
||||||
|
m.Kv = &KeyValue{}
|
||||||
|
}
|
||||||
|
if err := m.Kv.Unmarshal(dAtA[iNdEx:postIndex]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
case 3:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field PrevKv", wireType)
|
||||||
|
}
|
||||||
|
var msglen int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
msglen |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if msglen < 0 {
|
||||||
|
return ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + msglen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
if m.PrevKv == nil {
|
||||||
|
m.PrevKv = &KeyValue{}
|
||||||
|
}
|
||||||
|
if err := m.PrevKv.Unmarshal(dAtA[iNdEx:postIndex]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
iNdEx = postIndex
|
||||||
|
default:
|
||||||
|
iNdEx = preIndex
|
||||||
|
skippy, err := skipKv(dAtA[iNdEx:])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if skippy < 0 {
|
||||||
|
return ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
if (iNdEx + skippy) > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx += skippy
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if iNdEx > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func skipKv(dAtA []byte) (n int, err error) {
|
||||||
|
l := len(dAtA)
|
||||||
|
iNdEx := 0
|
||||||
|
for iNdEx < l {
|
||||||
|
var wire uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
wire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
wireType := int(wire & 0x7)
|
||||||
|
switch wireType {
|
||||||
|
case 0:
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
iNdEx++
|
||||||
|
if dAtA[iNdEx-1] < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return iNdEx, nil
|
||||||
|
case 1:
|
||||||
|
iNdEx += 8
|
||||||
|
return iNdEx, nil
|
||||||
|
case 2:
|
||||||
|
var length int
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
length |= (int(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
iNdEx += length
|
||||||
|
if length < 0 {
|
||||||
|
return 0, ErrInvalidLengthKv
|
||||||
|
}
|
||||||
|
return iNdEx, nil
|
||||||
|
case 3:
|
||||||
|
for {
|
||||||
|
var innerWire uint64
|
||||||
|
var start int = iNdEx
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return 0, ErrIntOverflowKv
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
innerWire |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
innerWireType := int(innerWire & 0x7)
|
||||||
|
if innerWireType == 4 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
next, err := skipKv(dAtA[start:])
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
iNdEx = start + next
|
||||||
|
}
|
||||||
|
return iNdEx, nil
|
||||||
|
case 4:
|
||||||
|
return iNdEx, nil
|
||||||
|
case 5:
|
||||||
|
iNdEx += 4
|
||||||
|
return iNdEx, nil
|
||||||
|
default:
|
||||||
|
return 0, fmt.Errorf("proto: illegal wireType %d", wireType)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panic("unreachable")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrInvalidLengthKv = fmt.Errorf("proto: negative length found during unmarshaling")
|
||||||
|
ErrIntOverflowKv = fmt.Errorf("proto: integer overflow")
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() { proto.RegisterFile("kv.proto", fileDescriptorKv) }
|
||||||
|
|
||||||
|
var fileDescriptorKv = []byte{
|
||||||
|
// 303 bytes of a gzipped FileDescriptorProto
|
||||||
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x6c, 0x90, 0x41, 0x4e, 0xc2, 0x40,
|
||||||
|
0x14, 0x86, 0x3b, 0x14, 0x0a, 0x3e, 0x08, 0x36, 0x13, 0x12, 0x27, 0x2e, 0x26, 0x95, 0x8d, 0x18,
|
||||||
|
0x13, 0x4c, 0xf0, 0x06, 0xc6, 0xae, 0x70, 0x61, 0x1a, 0x74, 0x4b, 0x4a, 0x79, 0x21, 0xa4, 0x94,
|
||||||
|
0x69, 0x4a, 0x9d, 0xa4, 0x37, 0x71, 0xef, 0xde, 0x73, 0xb0, 0xe4, 0x08, 0x52, 0x2f, 0x62, 0xfa,
|
||||||
|
0xc6, 0xe2, 0xc6, 0xcd, 0xe4, 0xfd, 0xff, 0xff, 0x65, 0xe6, 0x7f, 0x03, 0x9d, 0x58, 0x8f, 0xd3,
|
||||||
|
0x4c, 0xe5, 0x8a, 0x3b, 0x89, 0x8e, 0xa2, 0x74, 0x71, 0x39, 0x58, 0xa9, 0x95, 0x22, 0xeb, 0xae,
|
||||||
|
0x9a, 0x4c, 0x3a, 0xfc, 0x64, 0xd0, 0x99, 0x62, 0xf1, 0x1a, 0x6e, 0xde, 0x90, 0xbb, 0x60, 0xc7,
|
||||||
|
0x58, 0x08, 0xe6, 0xb1, 0x51, 0x2f, 0xa8, 0x46, 0x7e, 0x0d, 0xe7, 0x51, 0x86, 0x61, 0x8e, 0xf3,
|
||||||
|
0x0c, 0xf5, 0x7a, 0xb7, 0x56, 0x5b, 0xd1, 0xf0, 0xd8, 0xc8, 0x0e, 0xfa, 0xc6, 0x0e, 0x7e, 0x5d,
|
||||||
|
0x7e, 0x05, 0xbd, 0x44, 0x2d, 0xff, 0x28, 0x9b, 0xa8, 0x6e, 0xa2, 0x96, 0x27, 0x44, 0x40, 0x5b,
|
||||||
|
0x63, 0x46, 0x69, 0x93, 0xd2, 0x5a, 0xf2, 0x01, 0xb4, 0x74, 0x55, 0x40, 0xb4, 0xe8, 0x65, 0x23,
|
||||||
|
0x2a, 0x77, 0x83, 0xe1, 0x0e, 0x85, 0x43, 0xb4, 0x11, 0xc3, 0x0f, 0x06, 0x2d, 0x5f, 0xe3, 0x36,
|
||||||
|
0xe7, 0xb7, 0xd0, 0xcc, 0x8b, 0x14, 0xa9, 0x6e, 0x7f, 0x72, 0x31, 0x36, 0x7b, 0x8e, 0x29, 0x34,
|
||||||
|
0xe7, 0xac, 0x48, 0x31, 0x20, 0x88, 0x7b, 0xd0, 0x88, 0x35, 0x75, 0xef, 0x4e, 0xdc, 0x1a, 0xad,
|
||||||
|
0x17, 0x0f, 0x1a, 0xb1, 0xe6, 0x37, 0xd0, 0x4e, 0x33, 0xd4, 0xf3, 0x58, 0x53, 0xf9, 0xff, 0x30,
|
||||||
|
0xa7, 0x02, 0xa6, 0x7a, 0xe8, 0xc1, 0xd9, 0xe9, 0x7e, 0xde, 0x06, 0xfb, 0xf9, 0x65, 0xe6, 0x5a,
|
||||||
|
0x1c, 0xc0, 0x79, 0xf4, 0x9f, 0xfc, 0x99, 0xef, 0xb2, 0x07, 0xb1, 0x3f, 0x4a, 0xeb, 0x70, 0x94,
|
||||||
|
0xd6, 0xbe, 0x94, 0xec, 0x50, 0x4a, 0xf6, 0x55, 0x4a, 0xf6, 0xfe, 0x2d, 0xad, 0x85, 0x43, 0xff,
|
||||||
|
0x7e, 0xff, 0x13, 0x00, 0x00, 0xff, 0xff, 0xb5, 0x45, 0x92, 0x5d, 0xa1, 0x01, 0x00, 0x00,
|
||||||
|
}
|
49
vendor/github.com/coreos/etcd/mvcc/mvccpb/kv.proto
generated
vendored
Normal file
49
vendor/github.com/coreos/etcd/mvcc/mvccpb/kv.proto
generated
vendored
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
package mvccpb;
|
||||||
|
|
||||||
|
import "gogoproto/gogo.proto";
|
||||||
|
|
||||||
|
option (gogoproto.marshaler_all) = true;
|
||||||
|
option (gogoproto.sizer_all) = true;
|
||||||
|
option (gogoproto.unmarshaler_all) = true;
|
||||||
|
option (gogoproto.goproto_getters_all) = false;
|
||||||
|
option (gogoproto.goproto_enum_prefix_all) = false;
|
||||||
|
|
||||||
|
message KeyValue {
|
||||||
|
// key is the key in bytes. An empty key is not allowed.
|
||||||
|
bytes key = 1;
|
||||||
|
// create_revision is the revision of last creation on this key.
|
||||||
|
int64 create_revision = 2;
|
||||||
|
// mod_revision is the revision of last modification on this key.
|
||||||
|
int64 mod_revision = 3;
|
||||||
|
// version is the version of the key. A deletion resets
|
||||||
|
// the version to zero and any modification of the key
|
||||||
|
// increases its version.
|
||||||
|
int64 version = 4;
|
||||||
|
// value is the value held by the key, in bytes.
|
||||||
|
bytes value = 5;
|
||||||
|
// lease is the ID of the lease that attached to key.
|
||||||
|
// When the attached lease expires, the key will be deleted.
|
||||||
|
// If lease is 0, then no lease is attached to the key.
|
||||||
|
int64 lease = 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
message Event {
|
||||||
|
enum EventType {
|
||||||
|
PUT = 0;
|
||||||
|
DELETE = 1;
|
||||||
|
}
|
||||||
|
// type is the kind of event. If type is a PUT, it indicates
|
||||||
|
// new data has been stored to the key. If type is a DELETE,
|
||||||
|
// it indicates the key was deleted.
|
||||||
|
EventType type = 1;
|
||||||
|
// kv holds the KeyValue for the event.
|
||||||
|
// A PUT event contains current kv pair.
|
||||||
|
// A PUT event with kv.Version=1 indicates the creation of a key.
|
||||||
|
// A DELETE/EXPIRE event contains the deleted key with
|
||||||
|
// its modification revision set to the revision of deletion.
|
||||||
|
KeyValue kv = 2;
|
||||||
|
|
||||||
|
// prev_kv holds the key-value pair before the event happens.
|
||||||
|
KeyValue prev_kv = 3;
|
||||||
|
}
|
46
vendor/github.com/coreos/etcd/pkg/logutil/discard_logger.go
generated
vendored
Normal file
46
vendor/github.com/coreos/etcd/pkg/logutil/discard_logger.go
generated
vendored
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
|
||||||
|
"google.golang.org/grpc/grpclog"
|
||||||
|
)
|
||||||
|
|
||||||
|
// assert that "discardLogger" satisfy "Logger" interface
|
||||||
|
var _ Logger = &discardLogger{}
|
||||||
|
|
||||||
|
// NewDiscardLogger returns a new Logger that discards everything except "fatal".
|
||||||
|
func NewDiscardLogger() Logger { return &discardLogger{} }
|
||||||
|
|
||||||
|
type discardLogger struct{}
|
||||||
|
|
||||||
|
func (l *discardLogger) Info(args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Infoln(args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Infof(format string, args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Warning(args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Warningln(args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Warningf(format string, args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Error(args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Errorln(args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Errorf(format string, args ...interface{}) {}
|
||||||
|
func (l *discardLogger) Fatal(args ...interface{}) { log.Fatal(args...) }
|
||||||
|
func (l *discardLogger) Fatalln(args ...interface{}) { log.Fatalln(args...) }
|
||||||
|
func (l *discardLogger) Fatalf(format string, args ...interface{}) { log.Fatalf(format, args...) }
|
||||||
|
func (l *discardLogger) V(lvl int) bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
func (l *discardLogger) Lvl(lvl int) grpclog.LoggerV2 { return l }
|
16
vendor/github.com/coreos/etcd/pkg/logutil/doc.go
generated
vendored
Normal file
16
vendor/github.com/coreos/etcd/pkg/logutil/doc.go
generated
vendored
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package logutil includes utilities to facilitate logging.
|
||||||
|
package logutil
|
64
vendor/github.com/coreos/etcd/pkg/logutil/logger.go
generated
vendored
Normal file
64
vendor/github.com/coreos/etcd/pkg/logutil/logger.go
generated
vendored
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import "google.golang.org/grpc/grpclog"
|
||||||
|
|
||||||
|
// Logger defines logging interface.
|
||||||
|
// TODO: deprecate in v3.5.
|
||||||
|
type Logger interface {
|
||||||
|
grpclog.LoggerV2
|
||||||
|
|
||||||
|
// Lvl returns logger if logger's verbosity level >= "lvl".
|
||||||
|
// Otherwise, logger that discards everything.
|
||||||
|
Lvl(lvl int) grpclog.LoggerV2
|
||||||
|
}
|
||||||
|
|
||||||
|
// assert that "defaultLogger" satisfy "Logger" interface
|
||||||
|
var _ Logger = &defaultLogger{}
|
||||||
|
|
||||||
|
// NewLogger wraps "grpclog.LoggerV2" that implements "Logger" interface.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
//
|
||||||
|
// var defaultLogger Logger
|
||||||
|
// g := grpclog.NewLoggerV2WithVerbosity(os.Stderr, os.Stderr, os.Stderr, 4)
|
||||||
|
// defaultLogger = NewLogger(g)
|
||||||
|
//
|
||||||
|
func NewLogger(g grpclog.LoggerV2) Logger { return &defaultLogger{g: g} }
|
||||||
|
|
||||||
|
type defaultLogger struct {
|
||||||
|
g grpclog.LoggerV2
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *defaultLogger) Info(args ...interface{}) { l.g.Info(args...) }
|
||||||
|
func (l *defaultLogger) Infoln(args ...interface{}) { l.g.Info(args...) }
|
||||||
|
func (l *defaultLogger) Infof(format string, args ...interface{}) { l.g.Infof(format, args...) }
|
||||||
|
func (l *defaultLogger) Warning(args ...interface{}) { l.g.Warning(args...) }
|
||||||
|
func (l *defaultLogger) Warningln(args ...interface{}) { l.g.Warning(args...) }
|
||||||
|
func (l *defaultLogger) Warningf(format string, args ...interface{}) { l.g.Warningf(format, args...) }
|
||||||
|
func (l *defaultLogger) Error(args ...interface{}) { l.g.Error(args...) }
|
||||||
|
func (l *defaultLogger) Errorln(args ...interface{}) { l.g.Error(args...) }
|
||||||
|
func (l *defaultLogger) Errorf(format string, args ...interface{}) { l.g.Errorf(format, args...) }
|
||||||
|
func (l *defaultLogger) Fatal(args ...interface{}) { l.g.Fatal(args...) }
|
||||||
|
func (l *defaultLogger) Fatalln(args ...interface{}) { l.g.Fatal(args...) }
|
||||||
|
func (l *defaultLogger) Fatalf(format string, args ...interface{}) { l.g.Fatalf(format, args...) }
|
||||||
|
func (l *defaultLogger) V(lvl int) bool { return l.g.V(lvl) }
|
||||||
|
func (l *defaultLogger) Lvl(lvl int) grpclog.LoggerV2 {
|
||||||
|
if l.g.V(lvl) {
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
return &discardLogger{}
|
||||||
|
}
|
194
vendor/github.com/coreos/etcd/pkg/logutil/merge_logger.go
generated
vendored
Normal file
194
vendor/github.com/coreos/etcd/pkg/logutil/merge_logger.go
generated
vendored
Normal file
@ -0,0 +1,194 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/coreos/pkg/capnslog"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
defaultMergePeriod = time.Second
|
||||||
|
defaultTimeOutputScale = 10 * time.Millisecond
|
||||||
|
|
||||||
|
outputInterval = time.Second
|
||||||
|
)
|
||||||
|
|
||||||
|
// line represents a log line that can be printed out
|
||||||
|
// through capnslog.PackageLogger.
|
||||||
|
type line struct {
|
||||||
|
level capnslog.LogLevel
|
||||||
|
str string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l line) append(s string) line {
|
||||||
|
return line{
|
||||||
|
level: l.level,
|
||||||
|
str: l.str + " " + s,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// status represents the merge status of a line.
|
||||||
|
type status struct {
|
||||||
|
period time.Duration
|
||||||
|
|
||||||
|
start time.Time // start time of latest merge period
|
||||||
|
count int // number of merged lines from starting
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *status) isInMergePeriod(now time.Time) bool {
|
||||||
|
return s.period == 0 || s.start.Add(s.period).After(now)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *status) isEmpty() bool { return s.count == 0 }
|
||||||
|
|
||||||
|
func (s *status) summary(now time.Time) string {
|
||||||
|
ts := s.start.Round(defaultTimeOutputScale)
|
||||||
|
took := now.Round(defaultTimeOutputScale).Sub(ts)
|
||||||
|
return fmt.Sprintf("[merged %d repeated lines in %s]", s.count, took)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *status) reset(now time.Time) {
|
||||||
|
s.start = now
|
||||||
|
s.count = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// MergeLogger supports merge logging, which merges repeated log lines
|
||||||
|
// and prints summary log lines instead.
|
||||||
|
//
|
||||||
|
// For merge logging, MergeLogger prints out the line when the line appears
|
||||||
|
// at the first time. MergeLogger holds the same log line printed within
|
||||||
|
// defaultMergePeriod, and prints out summary log line at the end of defaultMergePeriod.
|
||||||
|
// It stops merging when the line doesn't appear within the
|
||||||
|
// defaultMergePeriod.
|
||||||
|
type MergeLogger struct {
|
||||||
|
*capnslog.PackageLogger
|
||||||
|
|
||||||
|
mu sync.Mutex // protect statusm
|
||||||
|
statusm map[line]*status
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMergeLogger(logger *capnslog.PackageLogger) *MergeLogger {
|
||||||
|
l := &MergeLogger{
|
||||||
|
PackageLogger: logger,
|
||||||
|
statusm: make(map[line]*status),
|
||||||
|
}
|
||||||
|
go l.outputLoop()
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeInfo(entries ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.INFO,
|
||||||
|
str: fmt.Sprint(entries...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeInfof(format string, args ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.INFO,
|
||||||
|
str: fmt.Sprintf(format, args...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeNotice(entries ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.NOTICE,
|
||||||
|
str: fmt.Sprint(entries...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeNoticef(format string, args ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.NOTICE,
|
||||||
|
str: fmt.Sprintf(format, args...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeWarning(entries ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.WARNING,
|
||||||
|
str: fmt.Sprint(entries...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeWarningf(format string, args ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.WARNING,
|
||||||
|
str: fmt.Sprintf(format, args...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeError(entries ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.ERROR,
|
||||||
|
str: fmt.Sprint(entries...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) MergeErrorf(format string, args ...interface{}) {
|
||||||
|
l.merge(line{
|
||||||
|
level: capnslog.ERROR,
|
||||||
|
str: fmt.Sprintf(format, args...),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) merge(ln line) {
|
||||||
|
l.mu.Lock()
|
||||||
|
|
||||||
|
// increase count if the logger is merging the line
|
||||||
|
if status, ok := l.statusm[ln]; ok {
|
||||||
|
status.count++
|
||||||
|
l.mu.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// initialize status of the line
|
||||||
|
l.statusm[ln] = &status{
|
||||||
|
period: defaultMergePeriod,
|
||||||
|
start: time.Now(),
|
||||||
|
}
|
||||||
|
// release the lock before IO operation
|
||||||
|
l.mu.Unlock()
|
||||||
|
// print out the line at its first time
|
||||||
|
l.PackageLogger.Logf(ln.level, ln.str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *MergeLogger) outputLoop() {
|
||||||
|
for now := range time.Tick(outputInterval) {
|
||||||
|
var outputs []line
|
||||||
|
|
||||||
|
l.mu.Lock()
|
||||||
|
for ln, status := range l.statusm {
|
||||||
|
if status.isInMergePeriod(now) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if status.isEmpty() {
|
||||||
|
delete(l.statusm, ln)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
outputs = append(outputs, ln.append(status.summary(now)))
|
||||||
|
status.reset(now)
|
||||||
|
}
|
||||||
|
l.mu.Unlock()
|
||||||
|
|
||||||
|
for _, o := range outputs {
|
||||||
|
l.PackageLogger.Logf(o.level, o.str)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
60
vendor/github.com/coreos/etcd/pkg/logutil/package_logger.go
generated
vendored
Normal file
60
vendor/github.com/coreos/etcd/pkg/logutil/package_logger.go
generated
vendored
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/coreos/pkg/capnslog"
|
||||||
|
"google.golang.org/grpc/grpclog"
|
||||||
|
)
|
||||||
|
|
||||||
|
// assert that "packageLogger" satisfy "Logger" interface
|
||||||
|
var _ Logger = &packageLogger{}
|
||||||
|
|
||||||
|
// NewPackageLogger wraps "*capnslog.PackageLogger" that implements "Logger" interface.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
//
|
||||||
|
// var defaultLogger Logger
|
||||||
|
// defaultLogger = NewPackageLogger("github.com/coreos/etcd", "snapshot")
|
||||||
|
//
|
||||||
|
func NewPackageLogger(repo, pkg string) Logger {
|
||||||
|
return &packageLogger{p: capnslog.NewPackageLogger(repo, pkg)}
|
||||||
|
}
|
||||||
|
|
||||||
|
type packageLogger struct {
|
||||||
|
p *capnslog.PackageLogger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *packageLogger) Info(args ...interface{}) { l.p.Info(args...) }
|
||||||
|
func (l *packageLogger) Infoln(args ...interface{}) { l.p.Info(args...) }
|
||||||
|
func (l *packageLogger) Infof(format string, args ...interface{}) { l.p.Infof(format, args...) }
|
||||||
|
func (l *packageLogger) Warning(args ...interface{}) { l.p.Warning(args...) }
|
||||||
|
func (l *packageLogger) Warningln(args ...interface{}) { l.p.Warning(args...) }
|
||||||
|
func (l *packageLogger) Warningf(format string, args ...interface{}) { l.p.Warningf(format, args...) }
|
||||||
|
func (l *packageLogger) Error(args ...interface{}) { l.p.Error(args...) }
|
||||||
|
func (l *packageLogger) Errorln(args ...interface{}) { l.p.Error(args...) }
|
||||||
|
func (l *packageLogger) Errorf(format string, args ...interface{}) { l.p.Errorf(format, args...) }
|
||||||
|
func (l *packageLogger) Fatal(args ...interface{}) { l.p.Fatal(args...) }
|
||||||
|
func (l *packageLogger) Fatalln(args ...interface{}) { l.p.Fatal(args...) }
|
||||||
|
func (l *packageLogger) Fatalf(format string, args ...interface{}) { l.p.Fatalf(format, args...) }
|
||||||
|
func (l *packageLogger) V(lvl int) bool {
|
||||||
|
return l.p.LevelAt(capnslog.LogLevel(lvl))
|
||||||
|
}
|
||||||
|
func (l *packageLogger) Lvl(lvl int) grpclog.LoggerV2 {
|
||||||
|
if l.p.LevelAt(capnslog.LogLevel(lvl)) {
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
return &discardLogger{}
|
||||||
|
}
|
111
vendor/github.com/coreos/etcd/pkg/logutil/zap_grpc.go
generated
vendored
Normal file
111
vendor/github.com/coreos/etcd/pkg/logutil/zap_grpc.go
generated
vendored
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"go.uber.org/zap/zapcore"
|
||||||
|
"google.golang.org/grpc/grpclog"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewGRPCLoggerV2 converts "*zap.Logger" to "grpclog.LoggerV2".
|
||||||
|
// It discards all INFO level logging in gRPC, if debug level
|
||||||
|
// is not enabled in "*zap.Logger".
|
||||||
|
func NewGRPCLoggerV2(lcfg zap.Config) (grpclog.LoggerV2, error) {
|
||||||
|
lg, err := lcfg.Build(zap.AddCallerSkip(1)) // to annotate caller outside of "logutil"
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &zapGRPCLogger{lg: lg, sugar: lg.Sugar()}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGRPCLoggerV2FromZapCore creates "grpclog.LoggerV2" from "zap.Core"
|
||||||
|
// and "zapcore.WriteSyncer". It discards all INFO level logging in gRPC,
|
||||||
|
// if debug level is not enabled in "*zap.Logger".
|
||||||
|
func NewGRPCLoggerV2FromZapCore(cr zapcore.Core, syncer zapcore.WriteSyncer) grpclog.LoggerV2 {
|
||||||
|
// "AddCallerSkip" to annotate caller outside of "logutil"
|
||||||
|
lg := zap.New(cr, zap.AddCaller(), zap.AddCallerSkip(1), zap.ErrorOutput(syncer))
|
||||||
|
return &zapGRPCLogger{lg: lg, sugar: lg.Sugar()}
|
||||||
|
}
|
||||||
|
|
||||||
|
type zapGRPCLogger struct {
|
||||||
|
lg *zap.Logger
|
||||||
|
sugar *zap.SugaredLogger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Info(args ...interface{}) {
|
||||||
|
if !zl.lg.Core().Enabled(zapcore.DebugLevel) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
zl.sugar.Info(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Infoln(args ...interface{}) {
|
||||||
|
if !zl.lg.Core().Enabled(zapcore.DebugLevel) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
zl.sugar.Info(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Infof(format string, args ...interface{}) {
|
||||||
|
if !zl.lg.Core().Enabled(zapcore.DebugLevel) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
zl.sugar.Infof(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Warning(args ...interface{}) {
|
||||||
|
zl.sugar.Warn(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Warningln(args ...interface{}) {
|
||||||
|
zl.sugar.Warn(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Warningf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Warnf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Error(args ...interface{}) {
|
||||||
|
zl.sugar.Error(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Errorln(args ...interface{}) {
|
||||||
|
zl.sugar.Error(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Errorf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Errorf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Fatal(args ...interface{}) {
|
||||||
|
zl.sugar.Fatal(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Fatalln(args ...interface{}) {
|
||||||
|
zl.sugar.Fatal(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) Fatalf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Fatalf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapGRPCLogger) V(l int) bool {
|
||||||
|
// infoLog == 0
|
||||||
|
if l <= 0 { // debug level, then we ignore info level in gRPC
|
||||||
|
return !zl.lg.Core().Enabled(zapcore.DebugLevel)
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
92
vendor/github.com/coreos/etcd/pkg/logutil/zap_journal.go
generated
vendored
Normal file
92
vendor/github.com/coreos/etcd/pkg/logutil/zap_journal.go
generated
vendored
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/pkg/systemd"
|
||||||
|
|
||||||
|
"github.com/coreos/go-systemd/journal"
|
||||||
|
"go.uber.org/zap/zapcore"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewJournalWriter wraps "io.Writer" to redirect log output
|
||||||
|
// to the local systemd journal. If journald send fails, it fails
|
||||||
|
// back to writing to the original writer.
|
||||||
|
// The decode overhead is only <30µs per write.
|
||||||
|
// Reference: https://github.com/coreos/pkg/blob/master/capnslog/journald_formatter.go
|
||||||
|
func NewJournalWriter(wr io.Writer) (io.Writer, error) {
|
||||||
|
return &journalWriter{Writer: wr}, systemd.DialJournal()
|
||||||
|
}
|
||||||
|
|
||||||
|
type journalWriter struct {
|
||||||
|
io.Writer
|
||||||
|
}
|
||||||
|
|
||||||
|
// WARN: assume that etcd uses default field names in zap encoder config
|
||||||
|
// make sure to keep this up-to-date!
|
||||||
|
type logLine struct {
|
||||||
|
Level string `json:"level"`
|
||||||
|
Caller string `json:"caller"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *journalWriter) Write(p []byte) (int, error) {
|
||||||
|
line := &logLine{}
|
||||||
|
if err := json.NewDecoder(bytes.NewReader(p)).Decode(line); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var pri journal.Priority
|
||||||
|
switch line.Level {
|
||||||
|
case zapcore.DebugLevel.String():
|
||||||
|
pri = journal.PriDebug
|
||||||
|
case zapcore.InfoLevel.String():
|
||||||
|
pri = journal.PriInfo
|
||||||
|
|
||||||
|
case zapcore.WarnLevel.String():
|
||||||
|
pri = journal.PriWarning
|
||||||
|
case zapcore.ErrorLevel.String():
|
||||||
|
pri = journal.PriErr
|
||||||
|
|
||||||
|
case zapcore.DPanicLevel.String():
|
||||||
|
pri = journal.PriCrit
|
||||||
|
case zapcore.PanicLevel.String():
|
||||||
|
pri = journal.PriCrit
|
||||||
|
case zapcore.FatalLevel.String():
|
||||||
|
pri = journal.PriCrit
|
||||||
|
|
||||||
|
default:
|
||||||
|
panic(fmt.Errorf("unknown log level: %q", line.Level))
|
||||||
|
}
|
||||||
|
|
||||||
|
err := journal.Send(string(p), pri, map[string]string{
|
||||||
|
"PACKAGE": filepath.Dir(line.Caller),
|
||||||
|
"SYSLOG_IDENTIFIER": filepath.Base(os.Args[0]),
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
// "journal" also falls back to stderr
|
||||||
|
// "fmt.Fprintln(os.Stderr, s)"
|
||||||
|
return w.Writer.Write(p)
|
||||||
|
}
|
||||||
|
return 0, nil
|
||||||
|
}
|
97
vendor/github.com/coreos/etcd/pkg/logutil/zap_raft.go
generated
vendored
Normal file
97
vendor/github.com/coreos/etcd/pkg/logutil/zap_raft.go
generated
vendored
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package logutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/coreos/etcd/raft"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"go.uber.org/zap/zapcore"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewRaftLogger converts "*zap.Logger" to "raft.Logger".
|
||||||
|
func NewRaftLogger(lcfg *zap.Config) (raft.Logger, error) {
|
||||||
|
if lcfg == nil {
|
||||||
|
return nil, errors.New("nil zap.Config")
|
||||||
|
}
|
||||||
|
lg, err := lcfg.Build(zap.AddCallerSkip(1)) // to annotate caller outside of "logutil"
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &zapRaftLogger{lg: lg, sugar: lg.Sugar()}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRaftLoggerFromZapCore creates "raft.Logger" from "zap.Core"
|
||||||
|
// and "zapcore.WriteSyncer".
|
||||||
|
func NewRaftLoggerFromZapCore(cr zapcore.Core, syncer zapcore.WriteSyncer) raft.Logger {
|
||||||
|
// "AddCallerSkip" to annotate caller outside of "logutil"
|
||||||
|
lg := zap.New(cr, zap.AddCaller(), zap.AddCallerSkip(1), zap.ErrorOutput(syncer))
|
||||||
|
return &zapRaftLogger{lg: lg, sugar: lg.Sugar()}
|
||||||
|
}
|
||||||
|
|
||||||
|
type zapRaftLogger struct {
|
||||||
|
lg *zap.Logger
|
||||||
|
sugar *zap.SugaredLogger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Debug(args ...interface{}) {
|
||||||
|
zl.sugar.Debug(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Debugf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Debugf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Error(args ...interface{}) {
|
||||||
|
zl.sugar.Error(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Errorf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Errorf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Info(args ...interface{}) {
|
||||||
|
zl.sugar.Info(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Infof(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Infof(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Warning(args ...interface{}) {
|
||||||
|
zl.sugar.Warn(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Warningf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Warnf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Fatal(args ...interface{}) {
|
||||||
|
zl.sugar.Fatal(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Fatalf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Fatalf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Panic(args ...interface{}) {
|
||||||
|
zl.sugar.Panic(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (zl *zapRaftLogger) Panicf(format string, args ...interface{}) {
|
||||||
|
zl.sugar.Panicf(format, args...)
|
||||||
|
}
|
31
vendor/github.com/coreos/etcd/pkg/pathutil/path.go
generated
vendored
31
vendor/github.com/coreos/etcd/pkg/pathutil/path.go
generated
vendored
@ -1,31 +0,0 @@
|
|||||||
// Copyright 2009 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 pathutil implements utility functions for handling slash-separated
|
|
||||||
// paths.
|
|
||||||
package pathutil
|
|
||||||
|
|
||||||
import "path"
|
|
||||||
|
|
||||||
// CanonicalURLPath returns the canonical url path for p, which follows the rules:
|
|
||||||
// 1. the path always starts with "/"
|
|
||||||
// 2. replace multiple slashes with a single slash
|
|
||||||
// 3. replace each '.' '..' path name element with equivalent one
|
|
||||||
// 4. keep the trailing slash
|
|
||||||
// The function is borrowed from stdlib http.cleanPath in server.go.
|
|
||||||
func CanonicalURLPath(p string) string {
|
|
||||||
if p == "" {
|
|
||||||
return "/"
|
|
||||||
}
|
|
||||||
if p[0] != '/' {
|
|
||||||
p = "/" + p
|
|
||||||
}
|
|
||||||
np := path.Clean(p)
|
|
||||||
// path.Clean removes trailing slash except for root,
|
|
||||||
// put the trailing slash back if necessary.
|
|
||||||
if p[len(p)-1] == '/' && np != "/" {
|
|
||||||
np += "/"
|
|
||||||
}
|
|
||||||
return np
|
|
||||||
}
|
|
140
vendor/github.com/coreos/etcd/pkg/srv/srv.go
generated
vendored
140
vendor/github.com/coreos/etcd/pkg/srv/srv.go
generated
vendored
@ -1,140 +0,0 @@
|
|||||||
// Copyright 2015 The etcd Authors
|
|
||||||
//
|
|
||||||
// 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.
|
|
||||||
|
|
||||||
// Package srv looks up DNS SRV records.
|
|
||||||
package srv
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"net/url"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/coreos/etcd/pkg/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
// indirection for testing
|
|
||||||
lookupSRV = net.LookupSRV // net.DefaultResolver.LookupSRV when ctxs don't conflict
|
|
||||||
resolveTCPAddr = net.ResolveTCPAddr
|
|
||||||
)
|
|
||||||
|
|
||||||
// GetCluster gets the cluster information via DNS discovery.
|
|
||||||
// Also sees each entry as a separate instance.
|
|
||||||
func GetCluster(service, name, dns string, apurls types.URLs) ([]string, error) {
|
|
||||||
tempName := int(0)
|
|
||||||
tcp2ap := make(map[string]url.URL)
|
|
||||||
|
|
||||||
// First, resolve the apurls
|
|
||||||
for _, url := range apurls {
|
|
||||||
tcpAddr, err := resolveTCPAddr("tcp", url.Host)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
tcp2ap[tcpAddr.String()] = url
|
|
||||||
}
|
|
||||||
|
|
||||||
stringParts := []string{}
|
|
||||||
updateNodeMap := func(service, scheme string) error {
|
|
||||||
_, addrs, err := lookupSRV(service, "tcp", dns)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
for _, srv := range addrs {
|
|
||||||
port := fmt.Sprintf("%d", srv.Port)
|
|
||||||
host := net.JoinHostPort(srv.Target, port)
|
|
||||||
tcpAddr, terr := resolveTCPAddr("tcp", host)
|
|
||||||
if terr != nil {
|
|
||||||
err = terr
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
n := ""
|
|
||||||
url, ok := tcp2ap[tcpAddr.String()]
|
|
||||||
if ok {
|
|
||||||
n = name
|
|
||||||
}
|
|
||||||
if n == "" {
|
|
||||||
n = fmt.Sprintf("%d", tempName)
|
|
||||||
tempName++
|
|
||||||
}
|
|
||||||
// SRV records have a trailing dot but URL shouldn't.
|
|
||||||
shortHost := strings.TrimSuffix(srv.Target, ".")
|
|
||||||
urlHost := net.JoinHostPort(shortHost, port)
|
|
||||||
stringParts = append(stringParts, fmt.Sprintf("%s=%s://%s", n, scheme, urlHost))
|
|
||||||
if ok && url.Scheme != scheme {
|
|
||||||
err = fmt.Errorf("bootstrap at %s from DNS for %s has scheme mismatch with expected peer %s", scheme+"://"+urlHost, service, url.String())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if len(stringParts) == 0 {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
failCount := 0
|
|
||||||
err := updateNodeMap(service+"-ssl", "https")
|
|
||||||
srvErr := make([]string, 2)
|
|
||||||
if err != nil {
|
|
||||||
srvErr[0] = fmt.Sprintf("error querying DNS SRV records for _%s-ssl %s", service, err)
|
|
||||||
failCount++
|
|
||||||
}
|
|
||||||
err = updateNodeMap(service, "http")
|
|
||||||
if err != nil {
|
|
||||||
srvErr[1] = fmt.Sprintf("error querying DNS SRV records for _%s %s", service, err)
|
|
||||||
failCount++
|
|
||||||
}
|
|
||||||
if failCount == 2 {
|
|
||||||
return nil, fmt.Errorf("srv: too many errors querying DNS SRV records (%q, %q)", srvErr[0], srvErr[1])
|
|
||||||
}
|
|
||||||
return stringParts, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type SRVClients struct {
|
|
||||||
Endpoints []string
|
|
||||||
SRVs []*net.SRV
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetClient looks up the client endpoints for a service and domain.
|
|
||||||
func GetClient(service, domain string) (*SRVClients, error) {
|
|
||||||
var urls []*url.URL
|
|
||||||
var srvs []*net.SRV
|
|
||||||
|
|
||||||
updateURLs := func(service, scheme string) error {
|
|
||||||
_, addrs, err := lookupSRV(service, "tcp", domain)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
for _, srv := range addrs {
|
|
||||||
urls = append(urls, &url.URL{
|
|
||||||
Scheme: scheme,
|
|
||||||
Host: net.JoinHostPort(srv.Target, fmt.Sprintf("%d", srv.Port)),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
srvs = append(srvs, addrs...)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
errHTTPS := updateURLs(service+"-ssl", "https")
|
|
||||||
errHTTP := updateURLs(service, "http")
|
|
||||||
|
|
||||||
if errHTTPS != nil && errHTTP != nil {
|
|
||||||
return nil, fmt.Errorf("dns lookup errors: %s and %s", errHTTPS, errHTTP)
|
|
||||||
}
|
|
||||||
|
|
||||||
endpoints := make([]string, len(urls))
|
|
||||||
for i := range urls {
|
|
||||||
endpoints[i] = urls[i].String()
|
|
||||||
}
|
|
||||||
return &SRVClients{Endpoints: endpoints, SRVs: srvs}, nil
|
|
||||||
}
|
|
16
vendor/github.com/coreos/etcd/pkg/systemd/doc.go
generated
vendored
Normal file
16
vendor/github.com/coreos/etcd/pkg/systemd/doc.go
generated
vendored
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
// Package systemd provides utility functions for systemd.
|
||||||
|
package systemd
|
29
vendor/github.com/coreos/etcd/pkg/systemd/journal.go
generated
vendored
Normal file
29
vendor/github.com/coreos/etcd/pkg/systemd/journal.go
generated
vendored
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
// Copyright 2018 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package systemd
|
||||||
|
|
||||||
|
import "net"
|
||||||
|
|
||||||
|
// DialJournal returns no error if the process can dial journal socket.
|
||||||
|
// Returns an error if dial failed, whichi indicates journald is not available
|
||||||
|
// (e.g. run embedded etcd as docker daemon).
|
||||||
|
// Reference: https://github.com/coreos/go-systemd/blob/master/journal/journal.go.
|
||||||
|
func DialJournal() error {
|
||||||
|
conn, err := net.Dial("unixgram", "/run/systemd/journal/socket")
|
||||||
|
if conn != nil {
|
||||||
|
defer conn.Close()
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
4
vendor/github.com/coreos/etcd/pkg/types/id.go
generated
vendored
4
vendor/github.com/coreos/etcd/pkg/types/id.go
generated
vendored
@ -14,9 +14,7 @@
|
|||||||
|
|
||||||
package types
|
package types
|
||||||
|
|
||||||
import (
|
import "strconv"
|
||||||
"strconv"
|
|
||||||
)
|
|
||||||
|
|
||||||
// ID represents a generic identifier which is canonically
|
// ID represents a generic identifier which is canonically
|
||||||
// stored as a uint64 but is typically represented as a
|
// stored as a uint64 but is typically represented as a
|
||||||
|
4
vendor/github.com/coreos/etcd/pkg/types/set.go
generated
vendored
4
vendor/github.com/coreos/etcd/pkg/types/set.go
generated
vendored
@ -61,7 +61,7 @@ func (us *unsafeSet) Remove(value string) {
|
|||||||
// Contains returns whether the set contains the given value
|
// Contains returns whether the set contains the given value
|
||||||
func (us *unsafeSet) Contains(value string) (exists bool) {
|
func (us *unsafeSet) Contains(value string) (exists bool) {
|
||||||
_, exists = us.d[value]
|
_, exists = us.d[value]
|
||||||
return
|
return exists
|
||||||
}
|
}
|
||||||
|
|
||||||
// ContainsAll returns whether the set contains all given values
|
// ContainsAll returns whether the set contains all given values
|
||||||
@ -94,7 +94,7 @@ func (us *unsafeSet) Values() (values []string) {
|
|||||||
for val := range us.d {
|
for val := range us.d {
|
||||||
values = append(values, val)
|
values = append(values, val)
|
||||||
}
|
}
|
||||||
return
|
return values
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy creates a new Set containing the values of the first
|
// Copy creates a new Set containing the values of the first
|
||||||
|
196
vendor/github.com/coreos/etcd/raft/README.md
generated
vendored
Normal file
196
vendor/github.com/coreos/etcd/raft/README.md
generated
vendored
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
# Raft library
|
||||||
|
|
||||||
|
Raft is a protocol with which a cluster of nodes can maintain a replicated state machine.
|
||||||
|
The state machine is kept in sync through the use of a replicated log.
|
||||||
|
For more details on Raft, see "In Search of an Understandable Consensus Algorithm"
|
||||||
|
(https://ramcloud.stanford.edu/raft.pdf) by Diego Ongaro and John Ousterhout.
|
||||||
|
|
||||||
|
This Raft library is stable and feature complete. As of 2016, it is **the most widely used** Raft library in production, serving tens of thousands clusters each day. It powers distributed systems such as etcd, Kubernetes, Docker Swarm, Cloud Foundry Diego, CockroachDB, TiDB, Project Calico, Flannel, and more.
|
||||||
|
|
||||||
|
Most Raft implementations have a monolithic design, including storage handling, messaging serialization, and network transport. This library instead follows a minimalistic design philosophy by only implementing the core raft algorithm. This minimalism buys flexibility, determinism, and performance.
|
||||||
|
|
||||||
|
To keep the codebase small as well as provide flexibility, the library only implements the Raft algorithm; both network and disk IO are left to the user. Library users must implement their own transportation layer for message passing between Raft peers over the wire. Similarly, users must implement their own storage layer to persist the Raft log and state.
|
||||||
|
|
||||||
|
In order to easily test the Raft library, its behavior should be deterministic. To achieve this determinism, the library models Raft as a state machine. The state machine takes a `Message` as input. A message can either be a local timer update or a network message sent from a remote peer. The state machine's output is a 3-tuple `{[]Messages, []LogEntries, NextState}` consisting of an array of `Messages`, `log entries`, and `Raft state changes`. For state machines with the same state, the same state machine input should always generate the same state machine output.
|
||||||
|
|
||||||
|
A simple example application, _raftexample_, is also available to help illustrate how to use this package in practice: https://github.com/coreos/etcd/tree/master/contrib/raftexample
|
||||||
|
|
||||||
|
# Features
|
||||||
|
|
||||||
|
This raft implementation is a full feature implementation of Raft protocol. Features includes:
|
||||||
|
|
||||||
|
- Leader election
|
||||||
|
- Log replication
|
||||||
|
- Log compaction
|
||||||
|
- Membership changes
|
||||||
|
- Leadership transfer extension
|
||||||
|
- Efficient linearizable read-only queries served by both the leader and followers
|
||||||
|
- leader checks with quorum and bypasses Raft log before processing read-only queries
|
||||||
|
- followers asks leader to get a safe read index before processing read-only queries
|
||||||
|
- More efficient lease-based linearizable read-only queries served by both the leader and followers
|
||||||
|
- leader bypasses Raft log and processing read-only queries locally
|
||||||
|
- followers asks leader to get a safe read index before processing read-only queries
|
||||||
|
- this approach relies on the clock of the all the machines in raft group
|
||||||
|
|
||||||
|
This raft implementation also includes a few optional enhancements:
|
||||||
|
|
||||||
|
- Optimistic pipelining to reduce log replication latency
|
||||||
|
- Flow control for log replication
|
||||||
|
- Batching Raft messages to reduce synchronized network I/O calls
|
||||||
|
- Batching log entries to reduce disk synchronized I/O
|
||||||
|
- Writing to leader's disk in parallel
|
||||||
|
- Internal proposal redirection from followers to leader
|
||||||
|
- Automatic stepping down when the leader loses quorum
|
||||||
|
|
||||||
|
## Notable Users
|
||||||
|
|
||||||
|
- [cockroachdb](https://github.com/cockroachdb/cockroach) A Scalable, Survivable, Strongly-Consistent SQL Database
|
||||||
|
- [dgraph](https://github.com/dgraph-io/dgraph) A Scalable, Distributed, Low Latency, High Throughput Graph Database
|
||||||
|
- [etcd](https://github.com/coreos/etcd) A distributed reliable key-value store
|
||||||
|
- [tikv](https://github.com/pingcap/tikv) A Distributed transactional key value database powered by Rust and Raft
|
||||||
|
- [swarmkit](https://github.com/docker/swarmkit) A toolkit for orchestrating distributed systems at any scale.
|
||||||
|
- [chain core](https://github.com/chain/chain) Software for operating permissioned, multi-asset blockchain networks
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
The primary object in raft is a Node. Either start a Node from scratch using raft.StartNode or start a Node from some initial state using raft.RestartNode.
|
||||||
|
|
||||||
|
To start a three-node cluster
|
||||||
|
```go
|
||||||
|
storage := raft.NewMemoryStorage()
|
||||||
|
c := &Config{
|
||||||
|
ID: 0x01,
|
||||||
|
ElectionTick: 10,
|
||||||
|
HeartbeatTick: 1,
|
||||||
|
Storage: storage,
|
||||||
|
MaxSizePerMsg: 4096,
|
||||||
|
MaxInflightMsgs: 256,
|
||||||
|
}
|
||||||
|
// Set peer list to the other nodes in the cluster.
|
||||||
|
// Note that they need to be started separately as well.
|
||||||
|
n := raft.StartNode(c, []raft.Peer{{ID: 0x02}, {ID: 0x03}})
|
||||||
|
```
|
||||||
|
|
||||||
|
Start a single node cluster, like so:
|
||||||
|
```go
|
||||||
|
// Create storage and config as shown above.
|
||||||
|
// Set peer list to itself, so this node can become the leader of this single-node cluster.
|
||||||
|
peers := []raft.Peer{{ID: 0x01}}
|
||||||
|
n := raft.StartNode(c, peers)
|
||||||
|
```
|
||||||
|
|
||||||
|
To allow a new node to join this cluster, do not pass in any peers. First, add the node to the existing cluster by calling `ProposeConfChange` on any existing node inside the cluster. Then, start the node with an empty peer list, like so:
|
||||||
|
```go
|
||||||
|
// Create storage and config as shown above.
|
||||||
|
n := raft.StartNode(c, nil)
|
||||||
|
```
|
||||||
|
|
||||||
|
To restart a node from previous state:
|
||||||
|
```go
|
||||||
|
storage := raft.NewMemoryStorage()
|
||||||
|
|
||||||
|
// Recover the in-memory storage from persistent snapshot, state and entries.
|
||||||
|
storage.ApplySnapshot(snapshot)
|
||||||
|
storage.SetHardState(state)
|
||||||
|
storage.Append(entries)
|
||||||
|
|
||||||
|
c := &Config{
|
||||||
|
ID: 0x01,
|
||||||
|
ElectionTick: 10,
|
||||||
|
HeartbeatTick: 1,
|
||||||
|
Storage: storage,
|
||||||
|
MaxSizePerMsg: 4096,
|
||||||
|
MaxInflightMsgs: 256,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restart raft without peer information.
|
||||||
|
// Peer information is already included in the storage.
|
||||||
|
n := raft.RestartNode(c)
|
||||||
|
```
|
||||||
|
|
||||||
|
After creating a Node, the user has a few responsibilities:
|
||||||
|
|
||||||
|
First, read from the Node.Ready() channel and process the updates it contains. These steps may be performed in parallel, except as noted in step 2.
|
||||||
|
|
||||||
|
1. Write Entries, HardState and Snapshot to persistent storage in order, i.e. Entries first, then HardState and Snapshot if they are not empty. If persistent storage supports atomic writes then all of them can be written together. Note that when writing an Entry with Index i, any previously-persisted entries with Index >= i must be discarded.
|
||||||
|
|
||||||
|
2. Send all Messages to the nodes named in the To field. It is important that no messages be sent until the latest HardState has been persisted to disk, and all Entries written by any previous Ready batch (Messages may be sent while entries from the same batch are being persisted). To reduce the I/O latency, an optimization can be applied to make leader write to disk in parallel with its followers (as explained at section 10.2.1 in Raft thesis). If any Message has type MsgSnap, call Node.ReportSnapshot() after it has been sent (these messages may be large). Note: Marshalling messages is not thread-safe; it is important to make sure that no new entries are persisted while marshalling. The easiest way to achieve this is to serialise the messages directly inside the main raft loop.
|
||||||
|
|
||||||
|
3. Apply Snapshot (if any) and CommittedEntries to the state machine. If any committed Entry has Type EntryConfChange, call Node.ApplyConfChange() to apply it to the node. The configuration change may be cancelled at this point by setting the NodeID field to zero before calling ApplyConfChange (but ApplyConfChange must be called one way or the other, and the decision to cancel must be based solely on the state machine and not external information such as the observed health of the node).
|
||||||
|
|
||||||
|
4. Call Node.Advance() to signal readiness for the next batch of updates. This may be done at any time after step 1, although all updates must be processed in the order they were returned by Ready.
|
||||||
|
|
||||||
|
Second, all persisted log entries must be made available via an implementation of the Storage interface. The provided MemoryStorage type can be used for this (if repopulating its state upon a restart), or a custom disk-backed implementation can be supplied.
|
||||||
|
|
||||||
|
Third, after receiving a message from another node, pass it to Node.Step:
|
||||||
|
|
||||||
|
```go
|
||||||
|
func recvRaftRPC(ctx context.Context, m raftpb.Message) {
|
||||||
|
n.Step(ctx, m)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Finally, call `Node.Tick()` at regular intervals (probably via a `time.Ticker`). Raft has two important timeouts: heartbeat and the election timeout. However, internally to the raft package time is represented by an abstract "tick".
|
||||||
|
|
||||||
|
The total state machine handling loop will look something like this:
|
||||||
|
|
||||||
|
```go
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-s.Ticker:
|
||||||
|
n.Tick()
|
||||||
|
case rd := <-s.Node.Ready():
|
||||||
|
saveToStorage(rd.HardState, rd.Entries, rd.Snapshot)
|
||||||
|
send(rd.Messages)
|
||||||
|
if !raft.IsEmptySnap(rd.Snapshot) {
|
||||||
|
processSnapshot(rd.Snapshot)
|
||||||
|
}
|
||||||
|
for _, entry := range rd.CommittedEntries {
|
||||||
|
process(entry)
|
||||||
|
if entry.Type == raftpb.EntryConfChange {
|
||||||
|
var cc raftpb.ConfChange
|
||||||
|
cc.Unmarshal(entry.Data)
|
||||||
|
s.Node.ApplyConfChange(cc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
s.Node.Advance()
|
||||||
|
case <-s.done:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
To propose changes to the state machine from the node to take application data, serialize it into a byte slice and call:
|
||||||
|
|
||||||
|
```go
|
||||||
|
n.Propose(ctx, data)
|
||||||
|
```
|
||||||
|
|
||||||
|
If the proposal is committed, data will appear in committed entries with type raftpb.EntryNormal. There is no guarantee that a proposed command will be committed; the command may have to be reproposed after a timeout.
|
||||||
|
|
||||||
|
To add or remove node in a cluster, build ConfChange struct 'cc' and call:
|
||||||
|
|
||||||
|
```go
|
||||||
|
n.ProposeConfChange(ctx, cc)
|
||||||
|
```
|
||||||
|
|
||||||
|
After config change is committed, some committed entry with type raftpb.EntryConfChange will be returned. This must be applied to node through:
|
||||||
|
|
||||||
|
```go
|
||||||
|
var cc raftpb.ConfChange
|
||||||
|
cc.Unmarshal(data)
|
||||||
|
n.ApplyConfChange(cc)
|
||||||
|
```
|
||||||
|
|
||||||
|
Note: An ID represents a unique node in a cluster for all time. A
|
||||||
|
given ID MUST be used only once even if the old node has been removed.
|
||||||
|
This means that for example IP addresses make poor node IDs since they
|
||||||
|
may be reused. Node IDs must be non-zero.
|
||||||
|
|
||||||
|
## Implementation notes
|
||||||
|
|
||||||
|
This implementation is up to date with the final Raft thesis (https://ramcloud.stanford.edu/~ongaro/thesis.pdf), although this implementation of the membership change protocol differs somewhat from that described in chapter 4. The key invariant that membership changes happen one node at a time is preserved, but in our implementation the membership change takes effect when its entry is applied, not when it is added to the log (so the entry is committed under the old membership instead of the new). This is equivalent in terms of safety, since the old and new configurations are guaranteed to overlap.
|
||||||
|
|
||||||
|
To ensure there is no attempt to commit two membership changes at once by matching log positions (which would be unsafe since they should have different quorum requirements), any proposed membership change is simply disallowed while any uncommitted change appears in the leader's log.
|
||||||
|
|
||||||
|
This approach introduces a problem when removing a member from a two-member cluster: If one of the members dies before the other one receives the commit of the confchange entry, then the member cannot be removed any more since the cluster cannot make progress. For this reason it is highly recommended to use three or more nodes in every cluster.
|
57
vendor/github.com/coreos/etcd/raft/design.md
generated
vendored
Normal file
57
vendor/github.com/coreos/etcd/raft/design.md
generated
vendored
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
## Progress
|
||||||
|
|
||||||
|
Progress represents a follower’s progress in the view of the leader. Leader maintains progresses of all followers, and sends `replication message` to the follower based on its progress.
|
||||||
|
|
||||||
|
`replication message` is a `msgApp` with log entries.
|
||||||
|
|
||||||
|
A progress has two attribute: `match` and `next`. `match` is the index of the highest known matched entry. If leader knows nothing about follower’s replication status, `match` is set to zero. `next` is the index of the first entry that will be replicated to the follower. Leader puts entries from `next` to its latest one in next `replication message`.
|
||||||
|
|
||||||
|
A progress is in one of the three state: `probe`, `replicate`, `snapshot`.
|
||||||
|
|
||||||
|
```
|
||||||
|
+--------------------------------------------------------+
|
||||||
|
| send snapshot |
|
||||||
|
| |
|
||||||
|
+---------+----------+ +----------v---------+
|
||||||
|
+---> probe | | snapshot |
|
||||||
|
| | max inflight = 1 <----------------------------------+ max inflight = 0 |
|
||||||
|
| +---------+----------+ +--------------------+
|
||||||
|
| | 1. snapshot success
|
||||||
|
| | (next=snapshot.index + 1)
|
||||||
|
| | 2. snapshot failure
|
||||||
|
| | (no change)
|
||||||
|
| | 3. receives msgAppResp(rej=false&&index>lastsnap.index)
|
||||||
|
| | (match=m.index,next=match+1)
|
||||||
|
receives msgAppResp(rej=true)
|
||||||
|
(next=match+1)| |
|
||||||
|
| |
|
||||||
|
| |
|
||||||
|
| | receives msgAppResp(rej=false&&index>match)
|
||||||
|
| | (match=m.index,next=match+1)
|
||||||
|
| |
|
||||||
|
| |
|
||||||
|
| |
|
||||||
|
| +---------v----------+
|
||||||
|
| | replicate |
|
||||||
|
+---+ max inflight = n |
|
||||||
|
+--------------------+
|
||||||
|
```
|
||||||
|
|
||||||
|
When the progress of a follower is in `probe` state, leader sends at most one `replication message` per heartbeat interval. The leader sends `replication message` slowly and probing the actual progress of the follower. A `msgHeartbeatResp` or a `msgAppResp` with reject might trigger the sending of the next `replication message`.
|
||||||
|
|
||||||
|
When the progress of a follower is in `replicate` state, leader sends `replication message`, then optimistically increases `next` to the latest entry sent. This is an optimized state for fast replicating log entries to the follower.
|
||||||
|
|
||||||
|
When the progress of a follower is in `snapshot` state, leader stops sending any `replication message`.
|
||||||
|
|
||||||
|
A newly elected leader sets the progresses of all the followers to `probe` state with `match` = 0 and `next` = last index. The leader slowly (at most once per heartbeat) sends `replication message` to the follower and probes its progress.
|
||||||
|
|
||||||
|
A progress changes to `replicate` when the follower replies with a non-rejection `msgAppResp`, which implies that it has matched the index sent. At this point, leader starts to stream log entries to the follower fast. The progress will fall back to `probe` when the follower replies a rejection `msgAppResp` or the link layer reports the follower is unreachable. We aggressively reset `next` to `match`+1 since if we receive any `msgAppResp` soon, both `match` and `next` will increase directly to the `index` in `msgAppResp`. (We might end up with sending some duplicate entries when aggressively reset `next` too low. see open question)
|
||||||
|
|
||||||
|
A progress changes from `probe` to `snapshot` when the follower falls very far behind and requires a snapshot. After sending `msgSnap`, the leader waits until the success, failure or abortion of the previous snapshot sent. The progress will go back to `probe` after the sending result is applied.
|
||||||
|
|
||||||
|
### Flow Control
|
||||||
|
|
||||||
|
1. limit the max size of message sent per message. Max should be configurable.
|
||||||
|
Lower the cost at probing state as we limit the size per message; lower the penalty when aggressively decreased to a too low `next`
|
||||||
|
|
||||||
|
2. limit the # of in flight messages < N when in `replicate` state. N should be configurable. Most implementation will have a sending buffer on top of its actual network transport layer (not blocking raft node). We want to make sure raft does not overflow that buffer, which can cause message dropping and triggering a bunch of unnecessary resending repeatedly.
|
300
vendor/github.com/coreos/etcd/raft/doc.go
generated
vendored
Normal file
300
vendor/github.com/coreos/etcd/raft/doc.go
generated
vendored
Normal file
@ -0,0 +1,300 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
/*
|
||||||
|
Package raft sends and receives messages in the Protocol Buffer format
|
||||||
|
defined in the raftpb package.
|
||||||
|
|
||||||
|
Raft is a protocol with which a cluster of nodes can maintain a replicated state machine.
|
||||||
|
The state machine is kept in sync through the use of a replicated log.
|
||||||
|
For more details on Raft, see "In Search of an Understandable Consensus Algorithm"
|
||||||
|
(https://ramcloud.stanford.edu/raft.pdf) by Diego Ongaro and John Ousterhout.
|
||||||
|
|
||||||
|
A simple example application, _raftexample_, is also available to help illustrate
|
||||||
|
how to use this package in practice:
|
||||||
|
https://github.com/coreos/etcd/tree/master/contrib/raftexample
|
||||||
|
|
||||||
|
Usage
|
||||||
|
|
||||||
|
The primary object in raft is a Node. You either start a Node from scratch
|
||||||
|
using raft.StartNode or start a Node from some initial state using raft.RestartNode.
|
||||||
|
|
||||||
|
To start a node from scratch:
|
||||||
|
|
||||||
|
storage := raft.NewMemoryStorage()
|
||||||
|
c := &Config{
|
||||||
|
ID: 0x01,
|
||||||
|
ElectionTick: 10,
|
||||||
|
HeartbeatTick: 1,
|
||||||
|
Storage: storage,
|
||||||
|
MaxSizePerMsg: 4096,
|
||||||
|
MaxInflightMsgs: 256,
|
||||||
|
}
|
||||||
|
n := raft.StartNode(c, []raft.Peer{{ID: 0x02}, {ID: 0x03}})
|
||||||
|
|
||||||
|
To restart a node from previous state:
|
||||||
|
|
||||||
|
storage := raft.NewMemoryStorage()
|
||||||
|
|
||||||
|
// recover the in-memory storage from persistent
|
||||||
|
// snapshot, state and entries.
|
||||||
|
storage.ApplySnapshot(snapshot)
|
||||||
|
storage.SetHardState(state)
|
||||||
|
storage.Append(entries)
|
||||||
|
|
||||||
|
c := &Config{
|
||||||
|
ID: 0x01,
|
||||||
|
ElectionTick: 10,
|
||||||
|
HeartbeatTick: 1,
|
||||||
|
Storage: storage,
|
||||||
|
MaxSizePerMsg: 4096,
|
||||||
|
MaxInflightMsgs: 256,
|
||||||
|
}
|
||||||
|
|
||||||
|
// restart raft without peer information.
|
||||||
|
// peer information is already included in the storage.
|
||||||
|
n := raft.RestartNode(c)
|
||||||
|
|
||||||
|
Now that you are holding onto a Node you have a few responsibilities:
|
||||||
|
|
||||||
|
First, you must read from the Node.Ready() channel and process the updates
|
||||||
|
it contains. These steps may be performed in parallel, except as noted in step
|
||||||
|
2.
|
||||||
|
|
||||||
|
1. Write HardState, Entries, and Snapshot to persistent storage if they are
|
||||||
|
not empty. Note that when writing an Entry with Index i, any
|
||||||
|
previously-persisted entries with Index >= i must be discarded.
|
||||||
|
|
||||||
|
2. Send all Messages to the nodes named in the To field. It is important that
|
||||||
|
no messages be sent until the latest HardState has been persisted to disk,
|
||||||
|
and all Entries written by any previous Ready batch (Messages may be sent while
|
||||||
|
entries from the same batch are being persisted). To reduce the I/O latency, an
|
||||||
|
optimization can be applied to make leader write to disk in parallel with its
|
||||||
|
followers (as explained at section 10.2.1 in Raft thesis). If any Message has type
|
||||||
|
MsgSnap, call Node.ReportSnapshot() after it has been sent (these messages may be
|
||||||
|
large).
|
||||||
|
|
||||||
|
Note: Marshalling messages is not thread-safe; it is important that you
|
||||||
|
make sure that no new entries are persisted while marshalling.
|
||||||
|
The easiest way to achieve this is to serialise the messages directly inside
|
||||||
|
your main raft loop.
|
||||||
|
|
||||||
|
3. Apply Snapshot (if any) and CommittedEntries to the state machine.
|
||||||
|
If any committed Entry has Type EntryConfChange, call Node.ApplyConfChange()
|
||||||
|
to apply it to the node. The configuration change may be cancelled at this point
|
||||||
|
by setting the NodeID field to zero before calling ApplyConfChange
|
||||||
|
(but ApplyConfChange must be called one way or the other, and the decision to cancel
|
||||||
|
must be based solely on the state machine and not external information such as
|
||||||
|
the observed health of the node).
|
||||||
|
|
||||||
|
4. Call Node.Advance() to signal readiness for the next batch of updates.
|
||||||
|
This may be done at any time after step 1, although all updates must be processed
|
||||||
|
in the order they were returned by Ready.
|
||||||
|
|
||||||
|
Second, all persisted log entries must be made available via an
|
||||||
|
implementation of the Storage interface. The provided MemoryStorage
|
||||||
|
type can be used for this (if you repopulate its state upon a
|
||||||
|
restart), or you can supply your own disk-backed implementation.
|
||||||
|
|
||||||
|
Third, when you receive a message from another node, pass it to Node.Step:
|
||||||
|
|
||||||
|
func recvRaftRPC(ctx context.Context, m raftpb.Message) {
|
||||||
|
n.Step(ctx, m)
|
||||||
|
}
|
||||||
|
|
||||||
|
Finally, you need to call Node.Tick() at regular intervals (probably
|
||||||
|
via a time.Ticker). Raft has two important timeouts: heartbeat and the
|
||||||
|
election timeout. However, internally to the raft package time is
|
||||||
|
represented by an abstract "tick".
|
||||||
|
|
||||||
|
The total state machine handling loop will look something like this:
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-s.Ticker:
|
||||||
|
n.Tick()
|
||||||
|
case rd := <-s.Node.Ready():
|
||||||
|
saveToStorage(rd.State, rd.Entries, rd.Snapshot)
|
||||||
|
send(rd.Messages)
|
||||||
|
if !raft.IsEmptySnap(rd.Snapshot) {
|
||||||
|
processSnapshot(rd.Snapshot)
|
||||||
|
}
|
||||||
|
for _, entry := range rd.CommittedEntries {
|
||||||
|
process(entry)
|
||||||
|
if entry.Type == raftpb.EntryConfChange {
|
||||||
|
var cc raftpb.ConfChange
|
||||||
|
cc.Unmarshal(entry.Data)
|
||||||
|
s.Node.ApplyConfChange(cc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
s.Node.Advance()
|
||||||
|
case <-s.done:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
To propose changes to the state machine from your node take your application
|
||||||
|
data, serialize it into a byte slice and call:
|
||||||
|
|
||||||
|
n.Propose(ctx, data)
|
||||||
|
|
||||||
|
If the proposal is committed, data will appear in committed entries with type
|
||||||
|
raftpb.EntryNormal. There is no guarantee that a proposed command will be
|
||||||
|
committed; you may have to re-propose after a timeout.
|
||||||
|
|
||||||
|
To add or remove node in a cluster, build ConfChange struct 'cc' and call:
|
||||||
|
|
||||||
|
n.ProposeConfChange(ctx, cc)
|
||||||
|
|
||||||
|
After config change is committed, some committed entry with type
|
||||||
|
raftpb.EntryConfChange will be returned. You must apply it to node through:
|
||||||
|
|
||||||
|
var cc raftpb.ConfChange
|
||||||
|
cc.Unmarshal(data)
|
||||||
|
n.ApplyConfChange(cc)
|
||||||
|
|
||||||
|
Note: An ID represents a unique node in a cluster for all time. A
|
||||||
|
given ID MUST be used only once even if the old node has been removed.
|
||||||
|
This means that for example IP addresses make poor node IDs since they
|
||||||
|
may be reused. Node IDs must be non-zero.
|
||||||
|
|
||||||
|
Implementation notes
|
||||||
|
|
||||||
|
This implementation is up to date with the final Raft thesis
|
||||||
|
(https://ramcloud.stanford.edu/~ongaro/thesis.pdf), although our
|
||||||
|
implementation of the membership change protocol differs somewhat from
|
||||||
|
that described in chapter 4. The key invariant that membership changes
|
||||||
|
happen one node at a time is preserved, but in our implementation the
|
||||||
|
membership change takes effect when its entry is applied, not when it
|
||||||
|
is added to the log (so the entry is committed under the old
|
||||||
|
membership instead of the new). This is equivalent in terms of safety,
|
||||||
|
since the old and new configurations are guaranteed to overlap.
|
||||||
|
|
||||||
|
To ensure that we do not attempt to commit two membership changes at
|
||||||
|
once by matching log positions (which would be unsafe since they
|
||||||
|
should have different quorum requirements), we simply disallow any
|
||||||
|
proposed membership change while any uncommitted change appears in
|
||||||
|
the leader's log.
|
||||||
|
|
||||||
|
This approach introduces a problem when you try to remove a member
|
||||||
|
from a two-member cluster: If one of the members dies before the
|
||||||
|
other one receives the commit of the confchange entry, then the member
|
||||||
|
cannot be removed any more since the cluster cannot make progress.
|
||||||
|
For this reason it is highly recommended to use three or more nodes in
|
||||||
|
every cluster.
|
||||||
|
|
||||||
|
MessageType
|
||||||
|
|
||||||
|
Package raft sends and receives message in Protocol Buffer format (defined
|
||||||
|
in raftpb package). Each state (follower, candidate, leader) implements its
|
||||||
|
own 'step' method ('stepFollower', 'stepCandidate', 'stepLeader') when
|
||||||
|
advancing with the given raftpb.Message. Each step is determined by its
|
||||||
|
raftpb.MessageType. Note that every step is checked by one common method
|
||||||
|
'Step' that safety-checks the terms of node and incoming message to prevent
|
||||||
|
stale log entries:
|
||||||
|
|
||||||
|
'MsgHup' is used for election. If a node is a follower or candidate, the
|
||||||
|
'tick' function in 'raft' struct is set as 'tickElection'. If a follower or
|
||||||
|
candidate has not received any heartbeat before the election timeout, it
|
||||||
|
passes 'MsgHup' to its Step method and becomes (or remains) a candidate to
|
||||||
|
start a new election.
|
||||||
|
|
||||||
|
'MsgBeat' is an internal type that signals the leader to send a heartbeat of
|
||||||
|
the 'MsgHeartbeat' type. If a node is a leader, the 'tick' function in
|
||||||
|
the 'raft' struct is set as 'tickHeartbeat', and triggers the leader to
|
||||||
|
send periodic 'MsgHeartbeat' messages to its followers.
|
||||||
|
|
||||||
|
'MsgProp' proposes to append data to its log entries. This is a special
|
||||||
|
type to redirect proposals to leader. Therefore, send method overwrites
|
||||||
|
raftpb.Message's term with its HardState's term to avoid attaching its
|
||||||
|
local term to 'MsgProp'. When 'MsgProp' is passed to the leader's 'Step'
|
||||||
|
method, the leader first calls the 'appendEntry' method to append entries
|
||||||
|
to its log, and then calls 'bcastAppend' method to send those entries to
|
||||||
|
its peers. When passed to candidate, 'MsgProp' is dropped. When passed to
|
||||||
|
follower, 'MsgProp' is stored in follower's mailbox(msgs) by the send
|
||||||
|
method. It is stored with sender's ID and later forwarded to leader by
|
||||||
|
rafthttp package.
|
||||||
|
|
||||||
|
'MsgApp' contains log entries to replicate. A leader calls bcastAppend,
|
||||||
|
which calls sendAppend, which sends soon-to-be-replicated logs in 'MsgApp'
|
||||||
|
type. When 'MsgApp' is passed to candidate's Step method, candidate reverts
|
||||||
|
back to follower, because it indicates that there is a valid leader sending
|
||||||
|
'MsgApp' messages. Candidate and follower respond to this message in
|
||||||
|
'MsgAppResp' type.
|
||||||
|
|
||||||
|
'MsgAppResp' is response to log replication request('MsgApp'). When
|
||||||
|
'MsgApp' is passed to candidate or follower's Step method, it responds by
|
||||||
|
calling 'handleAppendEntries' method, which sends 'MsgAppResp' to raft
|
||||||
|
mailbox.
|
||||||
|
|
||||||
|
'MsgVote' requests votes for election. When a node is a follower or
|
||||||
|
candidate and 'MsgHup' is passed to its Step method, then the node calls
|
||||||
|
'campaign' method to campaign itself to become a leader. Once 'campaign'
|
||||||
|
method is called, the node becomes candidate and sends 'MsgVote' to peers
|
||||||
|
in cluster to request votes. When passed to leader or candidate's Step
|
||||||
|
method and the message's Term is lower than leader's or candidate's,
|
||||||
|
'MsgVote' will be rejected ('MsgVoteResp' is returned with Reject true).
|
||||||
|
If leader or candidate receives 'MsgVote' with higher term, it will revert
|
||||||
|
back to follower. When 'MsgVote' is passed to follower, it votes for the
|
||||||
|
sender only when sender's last term is greater than MsgVote's term or
|
||||||
|
sender's last term is equal to MsgVote's term but sender's last committed
|
||||||
|
index is greater than or equal to follower's.
|
||||||
|
|
||||||
|
'MsgVoteResp' contains responses from voting request. When 'MsgVoteResp' is
|
||||||
|
passed to candidate, the candidate calculates how many votes it has won. If
|
||||||
|
it's more than majority (quorum), it becomes leader and calls 'bcastAppend'.
|
||||||
|
If candidate receives majority of votes of denials, it reverts back to
|
||||||
|
follower.
|
||||||
|
|
||||||
|
'MsgPreVote' and 'MsgPreVoteResp' are used in an optional two-phase election
|
||||||
|
protocol. When Config.PreVote is true, a pre-election is carried out first
|
||||||
|
(using the same rules as a regular election), and no node increases its term
|
||||||
|
number unless the pre-election indicates that the campaigining node would win.
|
||||||
|
This minimizes disruption when a partitioned node rejoins the cluster.
|
||||||
|
|
||||||
|
'MsgSnap' requests to install a snapshot message. When a node has just
|
||||||
|
become a leader or the leader receives 'MsgProp' message, it calls
|
||||||
|
'bcastAppend' method, which then calls 'sendAppend' method to each
|
||||||
|
follower. In 'sendAppend', if a leader fails to get term or entries,
|
||||||
|
the leader requests snapshot by sending 'MsgSnap' type message.
|
||||||
|
|
||||||
|
'MsgSnapStatus' tells the result of snapshot install message. When a
|
||||||
|
follower rejected 'MsgSnap', it indicates the snapshot request with
|
||||||
|
'MsgSnap' had failed from network issues which causes the network layer
|
||||||
|
to fail to send out snapshots to its followers. Then leader considers
|
||||||
|
follower's progress as probe. When 'MsgSnap' were not rejected, it
|
||||||
|
indicates that the snapshot succeeded and the leader sets follower's
|
||||||
|
progress to probe and resumes its log replication.
|
||||||
|
|
||||||
|
'MsgHeartbeat' sends heartbeat from leader. When 'MsgHeartbeat' is passed
|
||||||
|
to candidate and message's term is higher than candidate's, the candidate
|
||||||
|
reverts back to follower and updates its committed index from the one in
|
||||||
|
this heartbeat. And it sends the message to its mailbox. When
|
||||||
|
'MsgHeartbeat' is passed to follower's Step method and message's term is
|
||||||
|
higher than follower's, the follower updates its leaderID with the ID
|
||||||
|
from the message.
|
||||||
|
|
||||||
|
'MsgHeartbeatResp' is a response to 'MsgHeartbeat'. When 'MsgHeartbeatResp'
|
||||||
|
is passed to leader's Step method, the leader knows which follower
|
||||||
|
responded. And only when the leader's last committed index is greater than
|
||||||
|
follower's Match index, the leader runs 'sendAppend` method.
|
||||||
|
|
||||||
|
'MsgUnreachable' tells that request(message) wasn't delivered. When
|
||||||
|
'MsgUnreachable' is passed to leader's Step method, the leader discovers
|
||||||
|
that the follower that sent this 'MsgUnreachable' is not reachable, often
|
||||||
|
indicating 'MsgApp' is lost. When follower's progress state is replicate,
|
||||||
|
the leader sets it back to probe.
|
||||||
|
|
||||||
|
*/
|
||||||
|
package raft
|
358
vendor/github.com/coreos/etcd/raft/log.go
generated
vendored
Normal file
358
vendor/github.com/coreos/etcd/raft/log.go
generated
vendored
Normal file
@ -0,0 +1,358 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/raft/raftpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
type raftLog struct {
|
||||||
|
// storage contains all stable entries since the last snapshot.
|
||||||
|
storage Storage
|
||||||
|
|
||||||
|
// unstable contains all unstable entries and snapshot.
|
||||||
|
// they will be saved into storage.
|
||||||
|
unstable unstable
|
||||||
|
|
||||||
|
// committed is the highest log position that is known to be in
|
||||||
|
// stable storage on a quorum of nodes.
|
||||||
|
committed uint64
|
||||||
|
// applied is the highest log position that the application has
|
||||||
|
// been instructed to apply to its state machine.
|
||||||
|
// Invariant: applied <= committed
|
||||||
|
applied uint64
|
||||||
|
|
||||||
|
logger Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// newLog returns log using the given storage. It recovers the log to the state
|
||||||
|
// that it just commits and applies the latest snapshot.
|
||||||
|
func newLog(storage Storage, logger Logger) *raftLog {
|
||||||
|
if storage == nil {
|
||||||
|
log.Panic("storage must not be nil")
|
||||||
|
}
|
||||||
|
log := &raftLog{
|
||||||
|
storage: storage,
|
||||||
|
logger: logger,
|
||||||
|
}
|
||||||
|
firstIndex, err := storage.FirstIndex()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
lastIndex, err := storage.LastIndex()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
log.unstable.offset = lastIndex + 1
|
||||||
|
log.unstable.logger = logger
|
||||||
|
// Initialize our committed and applied pointers to the time of the last compaction.
|
||||||
|
log.committed = firstIndex - 1
|
||||||
|
log.applied = firstIndex - 1
|
||||||
|
|
||||||
|
return log
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) String() string {
|
||||||
|
return fmt.Sprintf("committed=%d, applied=%d, unstable.offset=%d, len(unstable.Entries)=%d", l.committed, l.applied, l.unstable.offset, len(l.unstable.entries))
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybeAppend returns (0, false) if the entries cannot be appended. Otherwise,
|
||||||
|
// it returns (last index of new entries, true).
|
||||||
|
func (l *raftLog) maybeAppend(index, logTerm, committed uint64, ents ...pb.Entry) (lastnewi uint64, ok bool) {
|
||||||
|
if l.matchTerm(index, logTerm) {
|
||||||
|
lastnewi = index + uint64(len(ents))
|
||||||
|
ci := l.findConflict(ents)
|
||||||
|
switch {
|
||||||
|
case ci == 0:
|
||||||
|
case ci <= l.committed:
|
||||||
|
l.logger.Panicf("entry %d conflict with committed entry [committed(%d)]", ci, l.committed)
|
||||||
|
default:
|
||||||
|
offset := index + 1
|
||||||
|
l.append(ents[ci-offset:]...)
|
||||||
|
}
|
||||||
|
l.commitTo(min(committed, lastnewi))
|
||||||
|
return lastnewi, true
|
||||||
|
}
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) append(ents ...pb.Entry) uint64 {
|
||||||
|
if len(ents) == 0 {
|
||||||
|
return l.lastIndex()
|
||||||
|
}
|
||||||
|
if after := ents[0].Index - 1; after < l.committed {
|
||||||
|
l.logger.Panicf("after(%d) is out of range [committed(%d)]", after, l.committed)
|
||||||
|
}
|
||||||
|
l.unstable.truncateAndAppend(ents)
|
||||||
|
return l.lastIndex()
|
||||||
|
}
|
||||||
|
|
||||||
|
// findConflict finds the index of the conflict.
|
||||||
|
// It returns the first pair of conflicting entries between the existing
|
||||||
|
// entries and the given entries, if there are any.
|
||||||
|
// If there is no conflicting entries, and the existing entries contains
|
||||||
|
// all the given entries, zero will be returned.
|
||||||
|
// If there is no conflicting entries, but the given entries contains new
|
||||||
|
// entries, the index of the first new entry will be returned.
|
||||||
|
// An entry is considered to be conflicting if it has the same index but
|
||||||
|
// a different term.
|
||||||
|
// The first entry MUST have an index equal to the argument 'from'.
|
||||||
|
// The index of the given entries MUST be continuously increasing.
|
||||||
|
func (l *raftLog) findConflict(ents []pb.Entry) uint64 {
|
||||||
|
for _, ne := range ents {
|
||||||
|
if !l.matchTerm(ne.Index, ne.Term) {
|
||||||
|
if ne.Index <= l.lastIndex() {
|
||||||
|
l.logger.Infof("found conflict at index %d [existing term: %d, conflicting term: %d]",
|
||||||
|
ne.Index, l.zeroTermOnErrCompacted(l.term(ne.Index)), ne.Term)
|
||||||
|
}
|
||||||
|
return ne.Index
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) unstableEntries() []pb.Entry {
|
||||||
|
if len(l.unstable.entries) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return l.unstable.entries
|
||||||
|
}
|
||||||
|
|
||||||
|
// nextEnts returns all the available entries for execution.
|
||||||
|
// If applied is smaller than the index of snapshot, it returns all committed
|
||||||
|
// entries after the index of snapshot.
|
||||||
|
func (l *raftLog) nextEnts() (ents []pb.Entry) {
|
||||||
|
off := max(l.applied+1, l.firstIndex())
|
||||||
|
if l.committed+1 > off {
|
||||||
|
ents, err := l.slice(off, l.committed+1, noLimit)
|
||||||
|
if err != nil {
|
||||||
|
l.logger.Panicf("unexpected error when getting unapplied entries (%v)", err)
|
||||||
|
}
|
||||||
|
return ents
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// hasNextEnts returns if there is any available entries for execution. This
|
||||||
|
// is a fast check without heavy raftLog.slice() in raftLog.nextEnts().
|
||||||
|
func (l *raftLog) hasNextEnts() bool {
|
||||||
|
off := max(l.applied+1, l.firstIndex())
|
||||||
|
return l.committed+1 > off
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) snapshot() (pb.Snapshot, error) {
|
||||||
|
if l.unstable.snapshot != nil {
|
||||||
|
return *l.unstable.snapshot, nil
|
||||||
|
}
|
||||||
|
return l.storage.Snapshot()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) firstIndex() uint64 {
|
||||||
|
if i, ok := l.unstable.maybeFirstIndex(); ok {
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
index, err := l.storage.FirstIndex()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
return index
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) lastIndex() uint64 {
|
||||||
|
if i, ok := l.unstable.maybeLastIndex(); ok {
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
i, err := l.storage.LastIndex()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) commitTo(tocommit uint64) {
|
||||||
|
// never decrease commit
|
||||||
|
if l.committed < tocommit {
|
||||||
|
if l.lastIndex() < tocommit {
|
||||||
|
l.logger.Panicf("tocommit(%d) is out of range [lastIndex(%d)]. Was the raft log corrupted, truncated, or lost?", tocommit, l.lastIndex())
|
||||||
|
}
|
||||||
|
l.committed = tocommit
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) appliedTo(i uint64) {
|
||||||
|
if i == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if l.committed < i || i < l.applied {
|
||||||
|
l.logger.Panicf("applied(%d) is out of range [prevApplied(%d), committed(%d)]", i, l.applied, l.committed)
|
||||||
|
}
|
||||||
|
l.applied = i
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) stableTo(i, t uint64) { l.unstable.stableTo(i, t) }
|
||||||
|
|
||||||
|
func (l *raftLog) stableSnapTo(i uint64) { l.unstable.stableSnapTo(i) }
|
||||||
|
|
||||||
|
func (l *raftLog) lastTerm() uint64 {
|
||||||
|
t, err := l.term(l.lastIndex())
|
||||||
|
if err != nil {
|
||||||
|
l.logger.Panicf("unexpected error when getting the last term (%v)", err)
|
||||||
|
}
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) term(i uint64) (uint64, error) {
|
||||||
|
// the valid term range is [index of dummy entry, last index]
|
||||||
|
dummyIndex := l.firstIndex() - 1
|
||||||
|
if i < dummyIndex || i > l.lastIndex() {
|
||||||
|
// TODO: return an error instead?
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if t, ok := l.unstable.maybeTerm(i); ok {
|
||||||
|
return t, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
t, err := l.storage.Term(i)
|
||||||
|
if err == nil {
|
||||||
|
return t, nil
|
||||||
|
}
|
||||||
|
if err == ErrCompacted || err == ErrUnavailable {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) entries(i, maxsize uint64) ([]pb.Entry, error) {
|
||||||
|
if i > l.lastIndex() {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return l.slice(i, l.lastIndex()+1, maxsize)
|
||||||
|
}
|
||||||
|
|
||||||
|
// allEntries returns all entries in the log.
|
||||||
|
func (l *raftLog) allEntries() []pb.Entry {
|
||||||
|
ents, err := l.entries(l.firstIndex(), noLimit)
|
||||||
|
if err == nil {
|
||||||
|
return ents
|
||||||
|
}
|
||||||
|
if err == ErrCompacted { // try again if there was a racing compaction
|
||||||
|
return l.allEntries()
|
||||||
|
}
|
||||||
|
// TODO (xiangli): handle error?
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// isUpToDate determines if the given (lastIndex,term) log is more up-to-date
|
||||||
|
// by comparing the index and term of the last entries in the existing logs.
|
||||||
|
// If the logs have last entries with different terms, then the log with the
|
||||||
|
// later term is more up-to-date. If the logs end with the same term, then
|
||||||
|
// whichever log has the larger lastIndex is more up-to-date. If the logs are
|
||||||
|
// the same, the given log is up-to-date.
|
||||||
|
func (l *raftLog) isUpToDate(lasti, term uint64) bool {
|
||||||
|
return term > l.lastTerm() || (term == l.lastTerm() && lasti >= l.lastIndex())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) matchTerm(i, term uint64) bool {
|
||||||
|
t, err := l.term(i)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return t == term
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) maybeCommit(maxIndex, term uint64) bool {
|
||||||
|
if maxIndex > l.committed && l.zeroTermOnErrCompacted(l.term(maxIndex)) == term {
|
||||||
|
l.commitTo(maxIndex)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) restore(s pb.Snapshot) {
|
||||||
|
l.logger.Infof("log [%s] starts to restore snapshot [index: %d, term: %d]", l, s.Metadata.Index, s.Metadata.Term)
|
||||||
|
l.committed = s.Metadata.Index
|
||||||
|
l.unstable.restore(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// slice returns a slice of log entries from lo through hi-1, inclusive.
|
||||||
|
func (l *raftLog) slice(lo, hi, maxSize uint64) ([]pb.Entry, error) {
|
||||||
|
err := l.mustCheckOutOfBounds(lo, hi)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if lo == hi {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
var ents []pb.Entry
|
||||||
|
if lo < l.unstable.offset {
|
||||||
|
storedEnts, err := l.storage.Entries(lo, min(hi, l.unstable.offset), maxSize)
|
||||||
|
if err == ErrCompacted {
|
||||||
|
return nil, err
|
||||||
|
} else if err == ErrUnavailable {
|
||||||
|
l.logger.Panicf("entries[%d:%d) is unavailable from storage", lo, min(hi, l.unstable.offset))
|
||||||
|
} else if err != nil {
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if ents has reached the size limitation
|
||||||
|
if uint64(len(storedEnts)) < min(hi, l.unstable.offset)-lo {
|
||||||
|
return storedEnts, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
ents = storedEnts
|
||||||
|
}
|
||||||
|
if hi > l.unstable.offset {
|
||||||
|
unstable := l.unstable.slice(max(lo, l.unstable.offset), hi)
|
||||||
|
if len(ents) > 0 {
|
||||||
|
ents = append([]pb.Entry{}, ents...)
|
||||||
|
ents = append(ents, unstable...)
|
||||||
|
} else {
|
||||||
|
ents = unstable
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return limitSize(ents, maxSize), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// l.firstIndex <= lo <= hi <= l.firstIndex + len(l.entries)
|
||||||
|
func (l *raftLog) mustCheckOutOfBounds(lo, hi uint64) error {
|
||||||
|
if lo > hi {
|
||||||
|
l.logger.Panicf("invalid slice %d > %d", lo, hi)
|
||||||
|
}
|
||||||
|
fi := l.firstIndex()
|
||||||
|
if lo < fi {
|
||||||
|
return ErrCompacted
|
||||||
|
}
|
||||||
|
|
||||||
|
length := l.lastIndex() + 1 - fi
|
||||||
|
if lo < fi || hi > fi+length {
|
||||||
|
l.logger.Panicf("slice[%d,%d) out of bound [%d,%d]", lo, hi, fi, l.lastIndex())
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *raftLog) zeroTermOnErrCompacted(t uint64, err error) uint64 {
|
||||||
|
if err == nil {
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
if err == ErrCompacted {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
l.logger.Panicf("unexpected error (%v)", err)
|
||||||
|
return 0
|
||||||
|
}
|
159
vendor/github.com/coreos/etcd/raft/log_unstable.go
generated
vendored
Normal file
159
vendor/github.com/coreos/etcd/raft/log_unstable.go
generated
vendored
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import pb "github.com/coreos/etcd/raft/raftpb"
|
||||||
|
|
||||||
|
// unstable.entries[i] has raft log position i+unstable.offset.
|
||||||
|
// Note that unstable.offset may be less than the highest log
|
||||||
|
// position in storage; this means that the next write to storage
|
||||||
|
// might need to truncate the log before persisting unstable.entries.
|
||||||
|
type unstable struct {
|
||||||
|
// the incoming unstable snapshot, if any.
|
||||||
|
snapshot *pb.Snapshot
|
||||||
|
// all entries that have not yet been written to storage.
|
||||||
|
entries []pb.Entry
|
||||||
|
offset uint64
|
||||||
|
|
||||||
|
logger Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybeFirstIndex returns the index of the first possible entry in entries
|
||||||
|
// if it has a snapshot.
|
||||||
|
func (u *unstable) maybeFirstIndex() (uint64, bool) {
|
||||||
|
if u.snapshot != nil {
|
||||||
|
return u.snapshot.Metadata.Index + 1, true
|
||||||
|
}
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybeLastIndex returns the last index if it has at least one
|
||||||
|
// unstable entry or snapshot.
|
||||||
|
func (u *unstable) maybeLastIndex() (uint64, bool) {
|
||||||
|
if l := len(u.entries); l != 0 {
|
||||||
|
return u.offset + uint64(l) - 1, true
|
||||||
|
}
|
||||||
|
if u.snapshot != nil {
|
||||||
|
return u.snapshot.Metadata.Index, true
|
||||||
|
}
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybeTerm returns the term of the entry at index i, if there
|
||||||
|
// is any.
|
||||||
|
func (u *unstable) maybeTerm(i uint64) (uint64, bool) {
|
||||||
|
if i < u.offset {
|
||||||
|
if u.snapshot == nil {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
if u.snapshot.Metadata.Index == i {
|
||||||
|
return u.snapshot.Metadata.Term, true
|
||||||
|
}
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
last, ok := u.maybeLastIndex()
|
||||||
|
if !ok {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
if i > last {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
return u.entries[i-u.offset].Term, true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *unstable) stableTo(i, t uint64) {
|
||||||
|
gt, ok := u.maybeTerm(i)
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// if i < offset, term is matched with the snapshot
|
||||||
|
// only update the unstable entries if term is matched with
|
||||||
|
// an unstable entry.
|
||||||
|
if gt == t && i >= u.offset {
|
||||||
|
u.entries = u.entries[i+1-u.offset:]
|
||||||
|
u.offset = i + 1
|
||||||
|
u.shrinkEntriesArray()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// shrinkEntriesArray discards the underlying array used by the entries slice
|
||||||
|
// if most of it isn't being used. This avoids holding references to a bunch of
|
||||||
|
// potentially large entries that aren't needed anymore. Simply clearing the
|
||||||
|
// entries wouldn't be safe because clients might still be using them.
|
||||||
|
func (u *unstable) shrinkEntriesArray() {
|
||||||
|
// We replace the array if we're using less than half of the space in
|
||||||
|
// it. This number is fairly arbitrary, chosen as an attempt to balance
|
||||||
|
// memory usage vs number of allocations. It could probably be improved
|
||||||
|
// with some focused tuning.
|
||||||
|
const lenMultiple = 2
|
||||||
|
if len(u.entries) == 0 {
|
||||||
|
u.entries = nil
|
||||||
|
} else if len(u.entries)*lenMultiple < cap(u.entries) {
|
||||||
|
newEntries := make([]pb.Entry, len(u.entries))
|
||||||
|
copy(newEntries, u.entries)
|
||||||
|
u.entries = newEntries
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *unstable) stableSnapTo(i uint64) {
|
||||||
|
if u.snapshot != nil && u.snapshot.Metadata.Index == i {
|
||||||
|
u.snapshot = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *unstable) restore(s pb.Snapshot) {
|
||||||
|
u.offset = s.Metadata.Index + 1
|
||||||
|
u.entries = nil
|
||||||
|
u.snapshot = &s
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *unstable) truncateAndAppend(ents []pb.Entry) {
|
||||||
|
after := ents[0].Index
|
||||||
|
switch {
|
||||||
|
case after == u.offset+uint64(len(u.entries)):
|
||||||
|
// after is the next index in the u.entries
|
||||||
|
// directly append
|
||||||
|
u.entries = append(u.entries, ents...)
|
||||||
|
case after <= u.offset:
|
||||||
|
u.logger.Infof("replace the unstable entries from index %d", after)
|
||||||
|
// The log is being truncated to before our current offset
|
||||||
|
// portion, so set the offset and replace the entries
|
||||||
|
u.offset = after
|
||||||
|
u.entries = ents
|
||||||
|
default:
|
||||||
|
// truncate to after and copy to u.entries
|
||||||
|
// then append
|
||||||
|
u.logger.Infof("truncate the unstable entries before index %d", after)
|
||||||
|
u.entries = append([]pb.Entry{}, u.slice(u.offset, after)...)
|
||||||
|
u.entries = append(u.entries, ents...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *unstable) slice(lo uint64, hi uint64) []pb.Entry {
|
||||||
|
u.mustCheckOutOfBounds(lo, hi)
|
||||||
|
return u.entries[lo-u.offset : hi-u.offset]
|
||||||
|
}
|
||||||
|
|
||||||
|
// u.offset <= lo <= hi <= u.offset+len(u.entries)
|
||||||
|
func (u *unstable) mustCheckOutOfBounds(lo, hi uint64) {
|
||||||
|
if lo > hi {
|
||||||
|
u.logger.Panicf("invalid unstable.slice %d > %d", lo, hi)
|
||||||
|
}
|
||||||
|
upper := u.offset + uint64(len(u.entries))
|
||||||
|
if lo < u.offset || hi > upper {
|
||||||
|
u.logger.Panicf("unstable.slice[%d,%d) out of bound [%d,%d]", lo, hi, u.offset, upper)
|
||||||
|
}
|
||||||
|
}
|
126
vendor/github.com/coreos/etcd/raft/logger.go
generated
vendored
Normal file
126
vendor/github.com/coreos/etcd/raft/logger.go
generated
vendored
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Logger interface {
|
||||||
|
Debug(v ...interface{})
|
||||||
|
Debugf(format string, v ...interface{})
|
||||||
|
|
||||||
|
Error(v ...interface{})
|
||||||
|
Errorf(format string, v ...interface{})
|
||||||
|
|
||||||
|
Info(v ...interface{})
|
||||||
|
Infof(format string, v ...interface{})
|
||||||
|
|
||||||
|
Warning(v ...interface{})
|
||||||
|
Warningf(format string, v ...interface{})
|
||||||
|
|
||||||
|
Fatal(v ...interface{})
|
||||||
|
Fatalf(format string, v ...interface{})
|
||||||
|
|
||||||
|
Panic(v ...interface{})
|
||||||
|
Panicf(format string, v ...interface{})
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetLogger(l Logger) { raftLogger = l }
|
||||||
|
|
||||||
|
var (
|
||||||
|
defaultLogger = &DefaultLogger{Logger: log.New(os.Stderr, "raft", log.LstdFlags)}
|
||||||
|
discardLogger = &DefaultLogger{Logger: log.New(ioutil.Discard, "", 0)}
|
||||||
|
raftLogger = Logger(defaultLogger)
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
calldepth = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
// DefaultLogger is a default implementation of the Logger interface.
|
||||||
|
type DefaultLogger struct {
|
||||||
|
*log.Logger
|
||||||
|
debug bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) EnableTimestamps() {
|
||||||
|
l.SetFlags(l.Flags() | log.Ldate | log.Ltime)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) EnableDebug() {
|
||||||
|
l.debug = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Debug(v ...interface{}) {
|
||||||
|
if l.debug {
|
||||||
|
l.Output(calldepth, header("DEBUG", fmt.Sprint(v...)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Debugf(format string, v ...interface{}) {
|
||||||
|
if l.debug {
|
||||||
|
l.Output(calldepth, header("DEBUG", fmt.Sprintf(format, v...)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Info(v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("INFO", fmt.Sprint(v...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Infof(format string, v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("INFO", fmt.Sprintf(format, v...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Error(v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("ERROR", fmt.Sprint(v...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Errorf(format string, v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("ERROR", fmt.Sprintf(format, v...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Warning(v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("WARN", fmt.Sprint(v...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Warningf(format string, v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("WARN", fmt.Sprintf(format, v...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Fatal(v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("FATAL", fmt.Sprint(v...)))
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Fatalf(format string, v ...interface{}) {
|
||||||
|
l.Output(calldepth, header("FATAL", fmt.Sprintf(format, v...)))
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Panic(v ...interface{}) {
|
||||||
|
l.Logger.Panic(v...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *DefaultLogger) Panicf(format string, v ...interface{}) {
|
||||||
|
l.Logger.Panicf(format, v...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func header(lvl, msg string) string {
|
||||||
|
return fmt.Sprintf("%s: %s", lvl, msg)
|
||||||
|
}
|
582
vendor/github.com/coreos/etcd/raft/node.go
generated
vendored
Normal file
582
vendor/github.com/coreos/etcd/raft/node.go
generated
vendored
Normal file
@ -0,0 +1,582 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/raft/raftpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
type SnapshotStatus int
|
||||||
|
|
||||||
|
const (
|
||||||
|
SnapshotFinish SnapshotStatus = 1
|
||||||
|
SnapshotFailure SnapshotStatus = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
emptyState = pb.HardState{}
|
||||||
|
|
||||||
|
// ErrStopped is returned by methods on Nodes that have been stopped.
|
||||||
|
ErrStopped = errors.New("raft: stopped")
|
||||||
|
)
|
||||||
|
|
||||||
|
// SoftState provides state that is useful for logging and debugging.
|
||||||
|
// The state is volatile and does not need to be persisted to the WAL.
|
||||||
|
type SoftState struct {
|
||||||
|
Lead uint64 // must use atomic operations to access; keep 64-bit aligned.
|
||||||
|
RaftState StateType
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *SoftState) equal(b *SoftState) bool {
|
||||||
|
return a.Lead == b.Lead && a.RaftState == b.RaftState
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ready encapsulates the entries and messages that are ready to read,
|
||||||
|
// be saved to stable storage, committed or sent to other peers.
|
||||||
|
// All fields in Ready are read-only.
|
||||||
|
type Ready struct {
|
||||||
|
// The current volatile state of a Node.
|
||||||
|
// SoftState will be nil if there is no update.
|
||||||
|
// It is not required to consume or store SoftState.
|
||||||
|
*SoftState
|
||||||
|
|
||||||
|
// The current state of a Node to be saved to stable storage BEFORE
|
||||||
|
// Messages are sent.
|
||||||
|
// HardState will be equal to empty state if there is no update.
|
||||||
|
pb.HardState
|
||||||
|
|
||||||
|
// ReadStates can be used for node to serve linearizable read requests locally
|
||||||
|
// when its applied index is greater than the index in ReadState.
|
||||||
|
// Note that the readState will be returned when raft receives msgReadIndex.
|
||||||
|
// The returned is only valid for the request that requested to read.
|
||||||
|
ReadStates []ReadState
|
||||||
|
|
||||||
|
// Entries specifies entries to be saved to stable storage BEFORE
|
||||||
|
// Messages are sent.
|
||||||
|
Entries []pb.Entry
|
||||||
|
|
||||||
|
// Snapshot specifies the snapshot to be saved to stable storage.
|
||||||
|
Snapshot pb.Snapshot
|
||||||
|
|
||||||
|
// CommittedEntries specifies entries to be committed to a
|
||||||
|
// store/state-machine. These have previously been committed to stable
|
||||||
|
// store.
|
||||||
|
CommittedEntries []pb.Entry
|
||||||
|
|
||||||
|
// Messages specifies outbound messages to be sent AFTER Entries are
|
||||||
|
// committed to stable storage.
|
||||||
|
// If it contains a MsgSnap message, the application MUST report back to raft
|
||||||
|
// when the snapshot has been received or has failed by calling ReportSnapshot.
|
||||||
|
Messages []pb.Message
|
||||||
|
|
||||||
|
// MustSync indicates whether the HardState and Entries must be synchronously
|
||||||
|
// written to disk or if an asynchronous write is permissible.
|
||||||
|
MustSync bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func isHardStateEqual(a, b pb.HardState) bool {
|
||||||
|
return a.Term == b.Term && a.Vote == b.Vote && a.Commit == b.Commit
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsEmptyHardState returns true if the given HardState is empty.
|
||||||
|
func IsEmptyHardState(st pb.HardState) bool {
|
||||||
|
return isHardStateEqual(st, emptyState)
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsEmptySnap returns true if the given Snapshot is empty.
|
||||||
|
func IsEmptySnap(sp pb.Snapshot) bool {
|
||||||
|
return sp.Metadata.Index == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rd Ready) containsUpdates() bool {
|
||||||
|
return rd.SoftState != nil || !IsEmptyHardState(rd.HardState) ||
|
||||||
|
!IsEmptySnap(rd.Snapshot) || len(rd.Entries) > 0 ||
|
||||||
|
len(rd.CommittedEntries) > 0 || len(rd.Messages) > 0 || len(rd.ReadStates) != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Node represents a node in a raft cluster.
|
||||||
|
type Node interface {
|
||||||
|
// Tick increments the internal logical clock for the Node by a single tick. Election
|
||||||
|
// timeouts and heartbeat timeouts are in units of ticks.
|
||||||
|
Tick()
|
||||||
|
// Campaign causes the Node to transition to candidate state and start campaigning to become leader.
|
||||||
|
Campaign(ctx context.Context) error
|
||||||
|
// Propose proposes that data be appended to the log.
|
||||||
|
Propose(ctx context.Context, data []byte) error
|
||||||
|
// ProposeConfChange proposes config change.
|
||||||
|
// At most one ConfChange can be in the process of going through consensus.
|
||||||
|
// Application needs to call ApplyConfChange when applying EntryConfChange type entry.
|
||||||
|
ProposeConfChange(ctx context.Context, cc pb.ConfChange) error
|
||||||
|
// Step advances the state machine using the given message. ctx.Err() will be returned, if any.
|
||||||
|
Step(ctx context.Context, msg pb.Message) error
|
||||||
|
|
||||||
|
// Ready returns a channel that returns the current point-in-time state.
|
||||||
|
// Users of the Node must call Advance after retrieving the state returned by Ready.
|
||||||
|
//
|
||||||
|
// NOTE: No committed entries from the next Ready may be applied until all committed entries
|
||||||
|
// and snapshots from the previous one have finished.
|
||||||
|
Ready() <-chan Ready
|
||||||
|
|
||||||
|
// Advance notifies the Node that the application has saved progress up to the last Ready.
|
||||||
|
// It prepares the node to return the next available Ready.
|
||||||
|
//
|
||||||
|
// The application should generally call Advance after it applies the entries in last Ready.
|
||||||
|
//
|
||||||
|
// However, as an optimization, the application may call Advance while it is applying the
|
||||||
|
// commands. For example. when the last Ready contains a snapshot, the application might take
|
||||||
|
// a long time to apply the snapshot data. To continue receiving Ready without blocking raft
|
||||||
|
// progress, it can call Advance before finishing applying the last ready.
|
||||||
|
Advance()
|
||||||
|
// ApplyConfChange applies config change to the local node.
|
||||||
|
// Returns an opaque ConfState protobuf which must be recorded
|
||||||
|
// in snapshots. Will never return nil; it returns a pointer only
|
||||||
|
// to match MemoryStorage.Compact.
|
||||||
|
ApplyConfChange(cc pb.ConfChange) *pb.ConfState
|
||||||
|
|
||||||
|
// TransferLeadership attempts to transfer leadership to the given transferee.
|
||||||
|
TransferLeadership(ctx context.Context, lead, transferee uint64)
|
||||||
|
|
||||||
|
// ReadIndex request a read state. The read state will be set in the ready.
|
||||||
|
// Read state has a read index. Once the application advances further than the read
|
||||||
|
// index, any linearizable read requests issued before the read request can be
|
||||||
|
// processed safely. The read state will have the same rctx attached.
|
||||||
|
ReadIndex(ctx context.Context, rctx []byte) error
|
||||||
|
|
||||||
|
// Status returns the current status of the raft state machine.
|
||||||
|
Status() Status
|
||||||
|
// ReportUnreachable reports the given node is not reachable for the last send.
|
||||||
|
ReportUnreachable(id uint64)
|
||||||
|
// ReportSnapshot reports the status of the sent snapshot.
|
||||||
|
ReportSnapshot(id uint64, status SnapshotStatus)
|
||||||
|
// Stop performs any necessary termination of the Node.
|
||||||
|
Stop()
|
||||||
|
}
|
||||||
|
|
||||||
|
type Peer struct {
|
||||||
|
ID uint64
|
||||||
|
Context []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// StartNode returns a new Node given configuration and a list of raft peers.
|
||||||
|
// It appends a ConfChangeAddNode entry for each given peer to the initial log.
|
||||||
|
func StartNode(c *Config, peers []Peer) Node {
|
||||||
|
r := newRaft(c)
|
||||||
|
// become the follower at term 1 and apply initial configuration
|
||||||
|
// entries of term 1
|
||||||
|
r.becomeFollower(1, None)
|
||||||
|
for _, peer := range peers {
|
||||||
|
cc := pb.ConfChange{Type: pb.ConfChangeAddNode, NodeID: peer.ID, Context: peer.Context}
|
||||||
|
d, err := cc.Marshal()
|
||||||
|
if err != nil {
|
||||||
|
panic("unexpected marshal error")
|
||||||
|
}
|
||||||
|
e := pb.Entry{Type: pb.EntryConfChange, Term: 1, Index: r.raftLog.lastIndex() + 1, Data: d}
|
||||||
|
r.raftLog.append(e)
|
||||||
|
}
|
||||||
|
// Mark these initial entries as committed.
|
||||||
|
// TODO(bdarnell): These entries are still unstable; do we need to preserve
|
||||||
|
// the invariant that committed < unstable?
|
||||||
|
r.raftLog.committed = r.raftLog.lastIndex()
|
||||||
|
// Now apply them, mainly so that the application can call Campaign
|
||||||
|
// immediately after StartNode in tests. Note that these nodes will
|
||||||
|
// be added to raft twice: here and when the application's Ready
|
||||||
|
// loop calls ApplyConfChange. The calls to addNode must come after
|
||||||
|
// all calls to raftLog.append so progress.next is set after these
|
||||||
|
// bootstrapping entries (it is an error if we try to append these
|
||||||
|
// entries since they have already been committed).
|
||||||
|
// We do not set raftLog.applied so the application will be able
|
||||||
|
// to observe all conf changes via Ready.CommittedEntries.
|
||||||
|
for _, peer := range peers {
|
||||||
|
r.addNode(peer.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
n := newNode()
|
||||||
|
n.logger = c.Logger
|
||||||
|
go n.run(r)
|
||||||
|
return &n
|
||||||
|
}
|
||||||
|
|
||||||
|
// RestartNode is similar to StartNode but does not take a list of peers.
|
||||||
|
// The current membership of the cluster will be restored from the Storage.
|
||||||
|
// If the caller has an existing state machine, pass in the last log index that
|
||||||
|
// has been applied to it; otherwise use zero.
|
||||||
|
func RestartNode(c *Config) Node {
|
||||||
|
r := newRaft(c)
|
||||||
|
|
||||||
|
n := newNode()
|
||||||
|
n.logger = c.Logger
|
||||||
|
go n.run(r)
|
||||||
|
return &n
|
||||||
|
}
|
||||||
|
|
||||||
|
type msgWithResult struct {
|
||||||
|
m pb.Message
|
||||||
|
result chan error
|
||||||
|
}
|
||||||
|
|
||||||
|
// node is the canonical implementation of the Node interface
|
||||||
|
type node struct {
|
||||||
|
propc chan msgWithResult
|
||||||
|
recvc chan pb.Message
|
||||||
|
confc chan pb.ConfChange
|
||||||
|
confstatec chan pb.ConfState
|
||||||
|
readyc chan Ready
|
||||||
|
advancec chan struct{}
|
||||||
|
tickc chan struct{}
|
||||||
|
done chan struct{}
|
||||||
|
stop chan struct{}
|
||||||
|
status chan chan Status
|
||||||
|
|
||||||
|
logger Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func newNode() node {
|
||||||
|
return node{
|
||||||
|
propc: make(chan msgWithResult),
|
||||||
|
recvc: make(chan pb.Message),
|
||||||
|
confc: make(chan pb.ConfChange),
|
||||||
|
confstatec: make(chan pb.ConfState),
|
||||||
|
readyc: make(chan Ready),
|
||||||
|
advancec: make(chan struct{}),
|
||||||
|
// make tickc a buffered chan, so raft node can buffer some ticks when the node
|
||||||
|
// is busy processing raft messages. Raft node will resume process buffered
|
||||||
|
// ticks when it becomes idle.
|
||||||
|
tickc: make(chan struct{}, 128),
|
||||||
|
done: make(chan struct{}),
|
||||||
|
stop: make(chan struct{}),
|
||||||
|
status: make(chan chan Status),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) Stop() {
|
||||||
|
select {
|
||||||
|
case n.stop <- struct{}{}:
|
||||||
|
// Not already stopped, so trigger it
|
||||||
|
case <-n.done:
|
||||||
|
// Node has already been stopped - no need to do anything
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// Block until the stop has been acknowledged by run()
|
||||||
|
<-n.done
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) run(r *raft) {
|
||||||
|
var propc chan msgWithResult
|
||||||
|
var readyc chan Ready
|
||||||
|
var advancec chan struct{}
|
||||||
|
var prevLastUnstablei, prevLastUnstablet uint64
|
||||||
|
var havePrevLastUnstablei bool
|
||||||
|
var prevSnapi uint64
|
||||||
|
var rd Ready
|
||||||
|
|
||||||
|
lead := None
|
||||||
|
prevSoftSt := r.softState()
|
||||||
|
prevHardSt := emptyState
|
||||||
|
|
||||||
|
for {
|
||||||
|
if advancec != nil {
|
||||||
|
readyc = nil
|
||||||
|
} else {
|
||||||
|
rd = newReady(r, prevSoftSt, prevHardSt)
|
||||||
|
if rd.containsUpdates() {
|
||||||
|
readyc = n.readyc
|
||||||
|
} else {
|
||||||
|
readyc = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if lead != r.lead {
|
||||||
|
if r.hasLeader() {
|
||||||
|
if lead == None {
|
||||||
|
r.logger.Infof("raft.node: %x elected leader %x at term %d", r.id, r.lead, r.Term)
|
||||||
|
} else {
|
||||||
|
r.logger.Infof("raft.node: %x changed leader from %x to %x at term %d", r.id, lead, r.lead, r.Term)
|
||||||
|
}
|
||||||
|
propc = n.propc
|
||||||
|
} else {
|
||||||
|
r.logger.Infof("raft.node: %x lost leader %x at term %d", r.id, lead, r.Term)
|
||||||
|
propc = nil
|
||||||
|
}
|
||||||
|
lead = r.lead
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
// TODO: maybe buffer the config propose if there exists one (the way
|
||||||
|
// described in raft dissertation)
|
||||||
|
// Currently it is dropped in Step silently.
|
||||||
|
case pm := <-propc:
|
||||||
|
m := pm.m
|
||||||
|
m.From = r.id
|
||||||
|
err := r.Step(m)
|
||||||
|
if pm.result != nil {
|
||||||
|
pm.result <- err
|
||||||
|
close(pm.result)
|
||||||
|
}
|
||||||
|
case m := <-n.recvc:
|
||||||
|
// filter out response message from unknown From.
|
||||||
|
if pr := r.getProgress(m.From); pr != nil || !IsResponseMsg(m.Type) {
|
||||||
|
r.Step(m)
|
||||||
|
}
|
||||||
|
case cc := <-n.confc:
|
||||||
|
if cc.NodeID == None {
|
||||||
|
select {
|
||||||
|
case n.confstatec <- pb.ConfState{
|
||||||
|
Nodes: r.nodes(),
|
||||||
|
Learners: r.learnerNodes()}:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
switch cc.Type {
|
||||||
|
case pb.ConfChangeAddNode:
|
||||||
|
r.addNode(cc.NodeID)
|
||||||
|
case pb.ConfChangeAddLearnerNode:
|
||||||
|
r.addLearner(cc.NodeID)
|
||||||
|
case pb.ConfChangeRemoveNode:
|
||||||
|
// block incoming proposal when local node is
|
||||||
|
// removed
|
||||||
|
if cc.NodeID == r.id {
|
||||||
|
propc = nil
|
||||||
|
}
|
||||||
|
r.removeNode(cc.NodeID)
|
||||||
|
case pb.ConfChangeUpdateNode:
|
||||||
|
default:
|
||||||
|
panic("unexpected conf type")
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case n.confstatec <- pb.ConfState{
|
||||||
|
Nodes: r.nodes(),
|
||||||
|
Learners: r.learnerNodes()}:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
case <-n.tickc:
|
||||||
|
r.tick()
|
||||||
|
case readyc <- rd:
|
||||||
|
if rd.SoftState != nil {
|
||||||
|
prevSoftSt = rd.SoftState
|
||||||
|
}
|
||||||
|
if len(rd.Entries) > 0 {
|
||||||
|
prevLastUnstablei = rd.Entries[len(rd.Entries)-1].Index
|
||||||
|
prevLastUnstablet = rd.Entries[len(rd.Entries)-1].Term
|
||||||
|
havePrevLastUnstablei = true
|
||||||
|
}
|
||||||
|
if !IsEmptyHardState(rd.HardState) {
|
||||||
|
prevHardSt = rd.HardState
|
||||||
|
}
|
||||||
|
if !IsEmptySnap(rd.Snapshot) {
|
||||||
|
prevSnapi = rd.Snapshot.Metadata.Index
|
||||||
|
}
|
||||||
|
|
||||||
|
r.msgs = nil
|
||||||
|
r.readStates = nil
|
||||||
|
advancec = n.advancec
|
||||||
|
case <-advancec:
|
||||||
|
if prevHardSt.Commit != 0 {
|
||||||
|
r.raftLog.appliedTo(prevHardSt.Commit)
|
||||||
|
}
|
||||||
|
if havePrevLastUnstablei {
|
||||||
|
r.raftLog.stableTo(prevLastUnstablei, prevLastUnstablet)
|
||||||
|
havePrevLastUnstablei = false
|
||||||
|
}
|
||||||
|
r.raftLog.stableSnapTo(prevSnapi)
|
||||||
|
advancec = nil
|
||||||
|
case c := <-n.status:
|
||||||
|
c <- getStatus(r)
|
||||||
|
case <-n.stop:
|
||||||
|
close(n.done)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tick increments the internal logical clock for this Node. Election timeouts
|
||||||
|
// and heartbeat timeouts are in units of ticks.
|
||||||
|
func (n *node) Tick() {
|
||||||
|
select {
|
||||||
|
case n.tickc <- struct{}{}:
|
||||||
|
case <-n.done:
|
||||||
|
default:
|
||||||
|
n.logger.Warningf("A tick missed to fire. Node blocks too long!")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) Campaign(ctx context.Context) error { return n.step(ctx, pb.Message{Type: pb.MsgHup}) }
|
||||||
|
|
||||||
|
func (n *node) Propose(ctx context.Context, data []byte) error {
|
||||||
|
return n.stepWait(ctx, pb.Message{Type: pb.MsgProp, Entries: []pb.Entry{{Data: data}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) Step(ctx context.Context, m pb.Message) error {
|
||||||
|
// ignore unexpected local messages receiving over network
|
||||||
|
if IsLocalMsg(m.Type) {
|
||||||
|
// TODO: return an error?
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return n.step(ctx, m)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) ProposeConfChange(ctx context.Context, cc pb.ConfChange) error {
|
||||||
|
data, err := cc.Marshal()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return n.Step(ctx, pb.Message{Type: pb.MsgProp, Entries: []pb.Entry{{Type: pb.EntryConfChange, Data: data}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) step(ctx context.Context, m pb.Message) error {
|
||||||
|
return n.stepWithWaitOption(ctx, m, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) stepWait(ctx context.Context, m pb.Message) error {
|
||||||
|
return n.stepWithWaitOption(ctx, m, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step advances the state machine using msgs. The ctx.Err() will be returned,
|
||||||
|
// if any.
|
||||||
|
func (n *node) stepWithWaitOption(ctx context.Context, m pb.Message, wait bool) error {
|
||||||
|
if m.Type != pb.MsgProp {
|
||||||
|
select {
|
||||||
|
case n.recvc <- m:
|
||||||
|
return nil
|
||||||
|
case <-ctx.Done():
|
||||||
|
return ctx.Err()
|
||||||
|
case <-n.done:
|
||||||
|
return ErrStopped
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ch := n.propc
|
||||||
|
pm := msgWithResult{m: m}
|
||||||
|
if wait {
|
||||||
|
pm.result = make(chan error, 1)
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case ch <- pm:
|
||||||
|
if !wait {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
case <-ctx.Done():
|
||||||
|
return ctx.Err()
|
||||||
|
case <-n.done:
|
||||||
|
return ErrStopped
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case rsp := <-pm.result:
|
||||||
|
if rsp != nil {
|
||||||
|
return rsp
|
||||||
|
}
|
||||||
|
case <-ctx.Done():
|
||||||
|
return ctx.Err()
|
||||||
|
case <-n.done:
|
||||||
|
return ErrStopped
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) Ready() <-chan Ready { return n.readyc }
|
||||||
|
|
||||||
|
func (n *node) Advance() {
|
||||||
|
select {
|
||||||
|
case n.advancec <- struct{}{}:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) ApplyConfChange(cc pb.ConfChange) *pb.ConfState {
|
||||||
|
var cs pb.ConfState
|
||||||
|
select {
|
||||||
|
case n.confc <- cc:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case cs = <-n.confstatec:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
return &cs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) Status() Status {
|
||||||
|
c := make(chan Status)
|
||||||
|
select {
|
||||||
|
case n.status <- c:
|
||||||
|
return <-c
|
||||||
|
case <-n.done:
|
||||||
|
return Status{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) ReportUnreachable(id uint64) {
|
||||||
|
select {
|
||||||
|
case n.recvc <- pb.Message{Type: pb.MsgUnreachable, From: id}:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) ReportSnapshot(id uint64, status SnapshotStatus) {
|
||||||
|
rej := status == SnapshotFailure
|
||||||
|
|
||||||
|
select {
|
||||||
|
case n.recvc <- pb.Message{Type: pb.MsgSnapStatus, From: id, Reject: rej}:
|
||||||
|
case <-n.done:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) TransferLeadership(ctx context.Context, lead, transferee uint64) {
|
||||||
|
select {
|
||||||
|
// manually set 'from' and 'to', so that leader can voluntarily transfers its leadership
|
||||||
|
case n.recvc <- pb.Message{Type: pb.MsgTransferLeader, From: transferee, To: lead}:
|
||||||
|
case <-n.done:
|
||||||
|
case <-ctx.Done():
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *node) ReadIndex(ctx context.Context, rctx []byte) error {
|
||||||
|
return n.step(ctx, pb.Message{Type: pb.MsgReadIndex, Entries: []pb.Entry{{Data: rctx}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
func newReady(r *raft, prevSoftSt *SoftState, prevHardSt pb.HardState) Ready {
|
||||||
|
rd := Ready{
|
||||||
|
Entries: r.raftLog.unstableEntries(),
|
||||||
|
CommittedEntries: r.raftLog.nextEnts(),
|
||||||
|
Messages: r.msgs,
|
||||||
|
}
|
||||||
|
if softSt := r.softState(); !softSt.equal(prevSoftSt) {
|
||||||
|
rd.SoftState = softSt
|
||||||
|
}
|
||||||
|
if hardSt := r.hardState(); !isHardStateEqual(hardSt, prevHardSt) {
|
||||||
|
rd.HardState = hardSt
|
||||||
|
}
|
||||||
|
if r.raftLog.unstable.snapshot != nil {
|
||||||
|
rd.Snapshot = *r.raftLog.unstable.snapshot
|
||||||
|
}
|
||||||
|
if len(r.readStates) != 0 {
|
||||||
|
rd.ReadStates = r.readStates
|
||||||
|
}
|
||||||
|
rd.MustSync = MustSync(rd.HardState, prevHardSt, len(rd.Entries))
|
||||||
|
return rd
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustSync returns true if the hard state and count of Raft entries indicate
|
||||||
|
// that a synchronous write to persistent storage is required.
|
||||||
|
func MustSync(st, prevst pb.HardState, entsnum int) bool {
|
||||||
|
// Persistent state on all servers:
|
||||||
|
// (Updated on stable storage before responding to RPCs)
|
||||||
|
// currentTerm
|
||||||
|
// votedFor
|
||||||
|
// log entries[]
|
||||||
|
return entsnum != 0 || st.Vote != prevst.Vote || st.Term != prevst.Term
|
||||||
|
}
|
284
vendor/github.com/coreos/etcd/raft/progress.go
generated
vendored
Normal file
284
vendor/github.com/coreos/etcd/raft/progress.go
generated
vendored
Normal file
@ -0,0 +1,284 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
const (
|
||||||
|
ProgressStateProbe ProgressStateType = iota
|
||||||
|
ProgressStateReplicate
|
||||||
|
ProgressStateSnapshot
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProgressStateType uint64
|
||||||
|
|
||||||
|
var prstmap = [...]string{
|
||||||
|
"ProgressStateProbe",
|
||||||
|
"ProgressStateReplicate",
|
||||||
|
"ProgressStateSnapshot",
|
||||||
|
}
|
||||||
|
|
||||||
|
func (st ProgressStateType) String() string { return prstmap[uint64(st)] }
|
||||||
|
|
||||||
|
// Progress represents a follower’s progress in the view of the leader. Leader maintains
|
||||||
|
// progresses of all followers, and sends entries to the follower based on its progress.
|
||||||
|
type Progress struct {
|
||||||
|
Match, Next uint64
|
||||||
|
// State defines how the leader should interact with the follower.
|
||||||
|
//
|
||||||
|
// When in ProgressStateProbe, leader sends at most one replication message
|
||||||
|
// per heartbeat interval. It also probes actual progress of the follower.
|
||||||
|
//
|
||||||
|
// When in ProgressStateReplicate, leader optimistically increases next
|
||||||
|
// to the latest entry sent after sending replication message. This is
|
||||||
|
// an optimized state for fast replicating log entries to the follower.
|
||||||
|
//
|
||||||
|
// When in ProgressStateSnapshot, leader should have sent out snapshot
|
||||||
|
// before and stops sending any replication message.
|
||||||
|
State ProgressStateType
|
||||||
|
|
||||||
|
// Paused is used in ProgressStateProbe.
|
||||||
|
// When Paused is true, raft should pause sending replication message to this peer.
|
||||||
|
Paused bool
|
||||||
|
// PendingSnapshot is used in ProgressStateSnapshot.
|
||||||
|
// If there is a pending snapshot, the pendingSnapshot will be set to the
|
||||||
|
// index of the snapshot. If pendingSnapshot is set, the replication process of
|
||||||
|
// this Progress will be paused. raft will not resend snapshot until the pending one
|
||||||
|
// is reported to be failed.
|
||||||
|
PendingSnapshot uint64
|
||||||
|
|
||||||
|
// RecentActive is true if the progress is recently active. Receiving any messages
|
||||||
|
// from the corresponding follower indicates the progress is active.
|
||||||
|
// RecentActive can be reset to false after an election timeout.
|
||||||
|
RecentActive bool
|
||||||
|
|
||||||
|
// inflights is a sliding window for the inflight messages.
|
||||||
|
// Each inflight message contains one or more log entries.
|
||||||
|
// The max number of entries per message is defined in raft config as MaxSizePerMsg.
|
||||||
|
// Thus inflight effectively limits both the number of inflight messages
|
||||||
|
// and the bandwidth each Progress can use.
|
||||||
|
// When inflights is full, no more message should be sent.
|
||||||
|
// When a leader sends out a message, the index of the last
|
||||||
|
// entry should be added to inflights. The index MUST be added
|
||||||
|
// into inflights in order.
|
||||||
|
// When a leader receives a reply, the previous inflights should
|
||||||
|
// be freed by calling inflights.freeTo with the index of the last
|
||||||
|
// received entry.
|
||||||
|
ins *inflights
|
||||||
|
|
||||||
|
// IsLearner is true if this progress is tracked for a learner.
|
||||||
|
IsLearner bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) resetState(state ProgressStateType) {
|
||||||
|
pr.Paused = false
|
||||||
|
pr.PendingSnapshot = 0
|
||||||
|
pr.State = state
|
||||||
|
pr.ins.reset()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) becomeProbe() {
|
||||||
|
// If the original state is ProgressStateSnapshot, progress knows that
|
||||||
|
// the pending snapshot has been sent to this peer successfully, then
|
||||||
|
// probes from pendingSnapshot + 1.
|
||||||
|
if pr.State == ProgressStateSnapshot {
|
||||||
|
pendingSnapshot := pr.PendingSnapshot
|
||||||
|
pr.resetState(ProgressStateProbe)
|
||||||
|
pr.Next = max(pr.Match+1, pendingSnapshot+1)
|
||||||
|
} else {
|
||||||
|
pr.resetState(ProgressStateProbe)
|
||||||
|
pr.Next = pr.Match + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) becomeReplicate() {
|
||||||
|
pr.resetState(ProgressStateReplicate)
|
||||||
|
pr.Next = pr.Match + 1
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) becomeSnapshot(snapshoti uint64) {
|
||||||
|
pr.resetState(ProgressStateSnapshot)
|
||||||
|
pr.PendingSnapshot = snapshoti
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybeUpdate returns false if the given n index comes from an outdated message.
|
||||||
|
// Otherwise it updates the progress and returns true.
|
||||||
|
func (pr *Progress) maybeUpdate(n uint64) bool {
|
||||||
|
var updated bool
|
||||||
|
if pr.Match < n {
|
||||||
|
pr.Match = n
|
||||||
|
updated = true
|
||||||
|
pr.resume()
|
||||||
|
}
|
||||||
|
if pr.Next < n+1 {
|
||||||
|
pr.Next = n + 1
|
||||||
|
}
|
||||||
|
return updated
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) optimisticUpdate(n uint64) { pr.Next = n + 1 }
|
||||||
|
|
||||||
|
// maybeDecrTo returns false if the given to index comes from an out of order message.
|
||||||
|
// Otherwise it decreases the progress next index to min(rejected, last) and returns true.
|
||||||
|
func (pr *Progress) maybeDecrTo(rejected, last uint64) bool {
|
||||||
|
if pr.State == ProgressStateReplicate {
|
||||||
|
// the rejection must be stale if the progress has matched and "rejected"
|
||||||
|
// is smaller than "match".
|
||||||
|
if rejected <= pr.Match {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// directly decrease next to match + 1
|
||||||
|
pr.Next = pr.Match + 1
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// the rejection must be stale if "rejected" does not match next - 1
|
||||||
|
if pr.Next-1 != rejected {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if pr.Next = min(rejected, last+1); pr.Next < 1 {
|
||||||
|
pr.Next = 1
|
||||||
|
}
|
||||||
|
pr.resume()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) pause() { pr.Paused = true }
|
||||||
|
func (pr *Progress) resume() { pr.Paused = false }
|
||||||
|
|
||||||
|
// IsPaused returns whether sending log entries to this node has been
|
||||||
|
// paused. A node may be paused because it has rejected recent
|
||||||
|
// MsgApps, is currently waiting for a snapshot, or has reached the
|
||||||
|
// MaxInflightMsgs limit.
|
||||||
|
func (pr *Progress) IsPaused() bool {
|
||||||
|
switch pr.State {
|
||||||
|
case ProgressStateProbe:
|
||||||
|
return pr.Paused
|
||||||
|
case ProgressStateReplicate:
|
||||||
|
return pr.ins.full()
|
||||||
|
case ProgressStateSnapshot:
|
||||||
|
return true
|
||||||
|
default:
|
||||||
|
panic("unexpected state")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) snapshotFailure() { pr.PendingSnapshot = 0 }
|
||||||
|
|
||||||
|
// needSnapshotAbort returns true if snapshot progress's Match
|
||||||
|
// is equal or higher than the pendingSnapshot.
|
||||||
|
func (pr *Progress) needSnapshotAbort() bool {
|
||||||
|
return pr.State == ProgressStateSnapshot && pr.Match >= pr.PendingSnapshot
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pr *Progress) String() string {
|
||||||
|
return fmt.Sprintf("next = %d, match = %d, state = %s, waiting = %v, pendingSnapshot = %d", pr.Next, pr.Match, pr.State, pr.IsPaused(), pr.PendingSnapshot)
|
||||||
|
}
|
||||||
|
|
||||||
|
type inflights struct {
|
||||||
|
// the starting index in the buffer
|
||||||
|
start int
|
||||||
|
// number of inflights in the buffer
|
||||||
|
count int
|
||||||
|
|
||||||
|
// the size of the buffer
|
||||||
|
size int
|
||||||
|
|
||||||
|
// buffer contains the index of the last entry
|
||||||
|
// inside one message.
|
||||||
|
buffer []uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func newInflights(size int) *inflights {
|
||||||
|
return &inflights{
|
||||||
|
size: size,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// add adds an inflight into inflights
|
||||||
|
func (in *inflights) add(inflight uint64) {
|
||||||
|
if in.full() {
|
||||||
|
panic("cannot add into a full inflights")
|
||||||
|
}
|
||||||
|
next := in.start + in.count
|
||||||
|
size := in.size
|
||||||
|
if next >= size {
|
||||||
|
next -= size
|
||||||
|
}
|
||||||
|
if next >= len(in.buffer) {
|
||||||
|
in.growBuf()
|
||||||
|
}
|
||||||
|
in.buffer[next] = inflight
|
||||||
|
in.count++
|
||||||
|
}
|
||||||
|
|
||||||
|
// grow the inflight buffer by doubling up to inflights.size. We grow on demand
|
||||||
|
// instead of preallocating to inflights.size to handle systems which have
|
||||||
|
// thousands of Raft groups per process.
|
||||||
|
func (in *inflights) growBuf() {
|
||||||
|
newSize := len(in.buffer) * 2
|
||||||
|
if newSize == 0 {
|
||||||
|
newSize = 1
|
||||||
|
} else if newSize > in.size {
|
||||||
|
newSize = in.size
|
||||||
|
}
|
||||||
|
newBuffer := make([]uint64, newSize)
|
||||||
|
copy(newBuffer, in.buffer)
|
||||||
|
in.buffer = newBuffer
|
||||||
|
}
|
||||||
|
|
||||||
|
// freeTo frees the inflights smaller or equal to the given `to` flight.
|
||||||
|
func (in *inflights) freeTo(to uint64) {
|
||||||
|
if in.count == 0 || to < in.buffer[in.start] {
|
||||||
|
// out of the left side of the window
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
idx := in.start
|
||||||
|
var i int
|
||||||
|
for i = 0; i < in.count; i++ {
|
||||||
|
if to < in.buffer[idx] { // found the first large inflight
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// increase index and maybe rotate
|
||||||
|
size := in.size
|
||||||
|
if idx++; idx >= size {
|
||||||
|
idx -= size
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// free i inflights and set new start index
|
||||||
|
in.count -= i
|
||||||
|
in.start = idx
|
||||||
|
if in.count == 0 {
|
||||||
|
// inflights is empty, reset the start index so that we don't grow the
|
||||||
|
// buffer unnecessarily.
|
||||||
|
in.start = 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (in *inflights) freeFirstOne() { in.freeTo(in.buffer[in.start]) }
|
||||||
|
|
||||||
|
// full returns true if the inflights is full.
|
||||||
|
func (in *inflights) full() bool {
|
||||||
|
return in.count == in.size
|
||||||
|
}
|
||||||
|
|
||||||
|
// resets frees all inflights.
|
||||||
|
func (in *inflights) reset() {
|
||||||
|
in.count = 0
|
||||||
|
in.start = 0
|
||||||
|
}
|
1444
vendor/github.com/coreos/etcd/raft/raft.go
generated
vendored
Normal file
1444
vendor/github.com/coreos/etcd/raft/raft.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
2004
vendor/github.com/coreos/etcd/raft/raftpb/raft.pb.go
generated
vendored
Normal file
2004
vendor/github.com/coreos/etcd/raft/raftpb/raft.pb.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
95
vendor/github.com/coreos/etcd/raft/raftpb/raft.proto
generated
vendored
Normal file
95
vendor/github.com/coreos/etcd/raft/raftpb/raft.proto
generated
vendored
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
syntax = "proto2";
|
||||||
|
package raftpb;
|
||||||
|
|
||||||
|
import "gogoproto/gogo.proto";
|
||||||
|
|
||||||
|
option (gogoproto.marshaler_all) = true;
|
||||||
|
option (gogoproto.sizer_all) = true;
|
||||||
|
option (gogoproto.unmarshaler_all) = true;
|
||||||
|
option (gogoproto.goproto_getters_all) = false;
|
||||||
|
option (gogoproto.goproto_enum_prefix_all) = false;
|
||||||
|
|
||||||
|
enum EntryType {
|
||||||
|
EntryNormal = 0;
|
||||||
|
EntryConfChange = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
message Entry {
|
||||||
|
optional uint64 Term = 2 [(gogoproto.nullable) = false]; // must be 64-bit aligned for atomic operations
|
||||||
|
optional uint64 Index = 3 [(gogoproto.nullable) = false]; // must be 64-bit aligned for atomic operations
|
||||||
|
optional EntryType Type = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional bytes Data = 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
message SnapshotMetadata {
|
||||||
|
optional ConfState conf_state = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 index = 2 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 term = 3 [(gogoproto.nullable) = false];
|
||||||
|
}
|
||||||
|
|
||||||
|
message Snapshot {
|
||||||
|
optional bytes data = 1;
|
||||||
|
optional SnapshotMetadata metadata = 2 [(gogoproto.nullable) = false];
|
||||||
|
}
|
||||||
|
|
||||||
|
enum MessageType {
|
||||||
|
MsgHup = 0;
|
||||||
|
MsgBeat = 1;
|
||||||
|
MsgProp = 2;
|
||||||
|
MsgApp = 3;
|
||||||
|
MsgAppResp = 4;
|
||||||
|
MsgVote = 5;
|
||||||
|
MsgVoteResp = 6;
|
||||||
|
MsgSnap = 7;
|
||||||
|
MsgHeartbeat = 8;
|
||||||
|
MsgHeartbeatResp = 9;
|
||||||
|
MsgUnreachable = 10;
|
||||||
|
MsgSnapStatus = 11;
|
||||||
|
MsgCheckQuorum = 12;
|
||||||
|
MsgTransferLeader = 13;
|
||||||
|
MsgTimeoutNow = 14;
|
||||||
|
MsgReadIndex = 15;
|
||||||
|
MsgReadIndexResp = 16;
|
||||||
|
MsgPreVote = 17;
|
||||||
|
MsgPreVoteResp = 18;
|
||||||
|
}
|
||||||
|
|
||||||
|
message Message {
|
||||||
|
optional MessageType type = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 to = 2 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 from = 3 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 term = 4 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 logTerm = 5 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 index = 6 [(gogoproto.nullable) = false];
|
||||||
|
repeated Entry entries = 7 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 commit = 8 [(gogoproto.nullable) = false];
|
||||||
|
optional Snapshot snapshot = 9 [(gogoproto.nullable) = false];
|
||||||
|
optional bool reject = 10 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 rejectHint = 11 [(gogoproto.nullable) = false];
|
||||||
|
optional bytes context = 12;
|
||||||
|
}
|
||||||
|
|
||||||
|
message HardState {
|
||||||
|
optional uint64 term = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 vote = 2 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 commit = 3 [(gogoproto.nullable) = false];
|
||||||
|
}
|
||||||
|
|
||||||
|
message ConfState {
|
||||||
|
repeated uint64 nodes = 1;
|
||||||
|
repeated uint64 learners = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ConfChangeType {
|
||||||
|
ConfChangeAddNode = 0;
|
||||||
|
ConfChangeRemoveNode = 1;
|
||||||
|
ConfChangeUpdateNode = 2;
|
||||||
|
ConfChangeAddLearnerNode = 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
message ConfChange {
|
||||||
|
optional uint64 ID = 1 [(gogoproto.nullable) = false];
|
||||||
|
optional ConfChangeType Type = 2 [(gogoproto.nullable) = false];
|
||||||
|
optional uint64 NodeID = 3 [(gogoproto.nullable) = false];
|
||||||
|
optional bytes Context = 4;
|
||||||
|
}
|
264
vendor/github.com/coreos/etcd/raft/rawnode.go
generated
vendored
Normal file
264
vendor/github.com/coreos/etcd/raft/rawnode.go
generated
vendored
Normal file
@ -0,0 +1,264 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/raft/raftpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ErrStepLocalMsg is returned when try to step a local raft message
|
||||||
|
var ErrStepLocalMsg = errors.New("raft: cannot step raft local message")
|
||||||
|
|
||||||
|
// ErrStepPeerNotFound is returned when try to step a response message
|
||||||
|
// but there is no peer found in raft.prs for that node.
|
||||||
|
var ErrStepPeerNotFound = errors.New("raft: cannot step as peer not found")
|
||||||
|
|
||||||
|
// RawNode is a thread-unsafe Node.
|
||||||
|
// The methods of this struct correspond to the methods of Node and are described
|
||||||
|
// more fully there.
|
||||||
|
type RawNode struct {
|
||||||
|
raft *raft
|
||||||
|
prevSoftSt *SoftState
|
||||||
|
prevHardSt pb.HardState
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rn *RawNode) newReady() Ready {
|
||||||
|
return newReady(rn.raft, rn.prevSoftSt, rn.prevHardSt)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rn *RawNode) commitReady(rd Ready) {
|
||||||
|
if rd.SoftState != nil {
|
||||||
|
rn.prevSoftSt = rd.SoftState
|
||||||
|
}
|
||||||
|
if !IsEmptyHardState(rd.HardState) {
|
||||||
|
rn.prevHardSt = rd.HardState
|
||||||
|
}
|
||||||
|
if rn.prevHardSt.Commit != 0 {
|
||||||
|
// In most cases, prevHardSt and rd.HardState will be the same
|
||||||
|
// because when there are new entries to apply we just sent a
|
||||||
|
// HardState with an updated Commit value. However, on initial
|
||||||
|
// startup the two are different because we don't send a HardState
|
||||||
|
// until something changes, but we do send any un-applied but
|
||||||
|
// committed entries (and previously-committed entries may be
|
||||||
|
// incorporated into the snapshot, even if rd.CommittedEntries is
|
||||||
|
// empty). Therefore we mark all committed entries as applied
|
||||||
|
// whether they were included in rd.HardState or not.
|
||||||
|
rn.raft.raftLog.appliedTo(rn.prevHardSt.Commit)
|
||||||
|
}
|
||||||
|
if len(rd.Entries) > 0 {
|
||||||
|
e := rd.Entries[len(rd.Entries)-1]
|
||||||
|
rn.raft.raftLog.stableTo(e.Index, e.Term)
|
||||||
|
}
|
||||||
|
if !IsEmptySnap(rd.Snapshot) {
|
||||||
|
rn.raft.raftLog.stableSnapTo(rd.Snapshot.Metadata.Index)
|
||||||
|
}
|
||||||
|
if len(rd.ReadStates) != 0 {
|
||||||
|
rn.raft.readStates = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRawNode returns a new RawNode given configuration and a list of raft peers.
|
||||||
|
func NewRawNode(config *Config, peers []Peer) (*RawNode, error) {
|
||||||
|
if config.ID == 0 {
|
||||||
|
panic("config.ID must not be zero")
|
||||||
|
}
|
||||||
|
r := newRaft(config)
|
||||||
|
rn := &RawNode{
|
||||||
|
raft: r,
|
||||||
|
}
|
||||||
|
lastIndex, err := config.Storage.LastIndex()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // TODO(bdarnell)
|
||||||
|
}
|
||||||
|
// If the log is empty, this is a new RawNode (like StartNode); otherwise it's
|
||||||
|
// restoring an existing RawNode (like RestartNode).
|
||||||
|
// TODO(bdarnell): rethink RawNode initialization and whether the application needs
|
||||||
|
// to be able to tell us when it expects the RawNode to exist.
|
||||||
|
if lastIndex == 0 {
|
||||||
|
r.becomeFollower(1, None)
|
||||||
|
ents := make([]pb.Entry, len(peers))
|
||||||
|
for i, peer := range peers {
|
||||||
|
cc := pb.ConfChange{Type: pb.ConfChangeAddNode, NodeID: peer.ID, Context: peer.Context}
|
||||||
|
data, err := cc.Marshal()
|
||||||
|
if err != nil {
|
||||||
|
panic("unexpected marshal error")
|
||||||
|
}
|
||||||
|
|
||||||
|
ents[i] = pb.Entry{Type: pb.EntryConfChange, Term: 1, Index: uint64(i + 1), Data: data}
|
||||||
|
}
|
||||||
|
r.raftLog.append(ents...)
|
||||||
|
r.raftLog.committed = uint64(len(ents))
|
||||||
|
for _, peer := range peers {
|
||||||
|
r.addNode(peer.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the initial hard and soft states after performing all initialization.
|
||||||
|
rn.prevSoftSt = r.softState()
|
||||||
|
if lastIndex == 0 {
|
||||||
|
rn.prevHardSt = emptyState
|
||||||
|
} else {
|
||||||
|
rn.prevHardSt = r.hardState()
|
||||||
|
}
|
||||||
|
|
||||||
|
return rn, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tick advances the internal logical clock by a single tick.
|
||||||
|
func (rn *RawNode) Tick() {
|
||||||
|
rn.raft.tick()
|
||||||
|
}
|
||||||
|
|
||||||
|
// TickQuiesced advances the internal logical clock by a single tick without
|
||||||
|
// performing any other state machine processing. It allows the caller to avoid
|
||||||
|
// periodic heartbeats and elections when all of the peers in a Raft group are
|
||||||
|
// known to be at the same state. Expected usage is to periodically invoke Tick
|
||||||
|
// or TickQuiesced depending on whether the group is "active" or "quiesced".
|
||||||
|
//
|
||||||
|
// WARNING: Be very careful about using this method as it subverts the Raft
|
||||||
|
// state machine. You should probably be using Tick instead.
|
||||||
|
func (rn *RawNode) TickQuiesced() {
|
||||||
|
rn.raft.electionElapsed++
|
||||||
|
}
|
||||||
|
|
||||||
|
// Campaign causes this RawNode to transition to candidate state.
|
||||||
|
func (rn *RawNode) Campaign() error {
|
||||||
|
return rn.raft.Step(pb.Message{
|
||||||
|
Type: pb.MsgHup,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Propose proposes data be appended to the raft log.
|
||||||
|
func (rn *RawNode) Propose(data []byte) error {
|
||||||
|
return rn.raft.Step(pb.Message{
|
||||||
|
Type: pb.MsgProp,
|
||||||
|
From: rn.raft.id,
|
||||||
|
Entries: []pb.Entry{
|
||||||
|
{Data: data},
|
||||||
|
}})
|
||||||
|
}
|
||||||
|
|
||||||
|
// ProposeConfChange proposes a config change.
|
||||||
|
func (rn *RawNode) ProposeConfChange(cc pb.ConfChange) error {
|
||||||
|
data, err := cc.Marshal()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return rn.raft.Step(pb.Message{
|
||||||
|
Type: pb.MsgProp,
|
||||||
|
Entries: []pb.Entry{
|
||||||
|
{Type: pb.EntryConfChange, Data: data},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// ApplyConfChange applies a config change to the local node.
|
||||||
|
func (rn *RawNode) ApplyConfChange(cc pb.ConfChange) *pb.ConfState {
|
||||||
|
if cc.NodeID == None {
|
||||||
|
return &pb.ConfState{Nodes: rn.raft.nodes(), Learners: rn.raft.learnerNodes()}
|
||||||
|
}
|
||||||
|
switch cc.Type {
|
||||||
|
case pb.ConfChangeAddNode:
|
||||||
|
rn.raft.addNode(cc.NodeID)
|
||||||
|
case pb.ConfChangeAddLearnerNode:
|
||||||
|
rn.raft.addLearner(cc.NodeID)
|
||||||
|
case pb.ConfChangeRemoveNode:
|
||||||
|
rn.raft.removeNode(cc.NodeID)
|
||||||
|
case pb.ConfChangeUpdateNode:
|
||||||
|
default:
|
||||||
|
panic("unexpected conf type")
|
||||||
|
}
|
||||||
|
return &pb.ConfState{Nodes: rn.raft.nodes(), Learners: rn.raft.learnerNodes()}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step advances the state machine using the given message.
|
||||||
|
func (rn *RawNode) Step(m pb.Message) error {
|
||||||
|
// ignore unexpected local messages receiving over network
|
||||||
|
if IsLocalMsg(m.Type) {
|
||||||
|
return ErrStepLocalMsg
|
||||||
|
}
|
||||||
|
if pr := rn.raft.getProgress(m.From); pr != nil || !IsResponseMsg(m.Type) {
|
||||||
|
return rn.raft.Step(m)
|
||||||
|
}
|
||||||
|
return ErrStepPeerNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ready returns the current point-in-time state of this RawNode.
|
||||||
|
func (rn *RawNode) Ready() Ready {
|
||||||
|
rd := rn.newReady()
|
||||||
|
rn.raft.msgs = nil
|
||||||
|
return rd
|
||||||
|
}
|
||||||
|
|
||||||
|
// HasReady called when RawNode user need to check if any Ready pending.
|
||||||
|
// Checking logic in this method should be consistent with Ready.containsUpdates().
|
||||||
|
func (rn *RawNode) HasReady() bool {
|
||||||
|
r := rn.raft
|
||||||
|
if !r.softState().equal(rn.prevSoftSt) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if hardSt := r.hardState(); !IsEmptyHardState(hardSt) && !isHardStateEqual(hardSt, rn.prevHardSt) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if r.raftLog.unstable.snapshot != nil && !IsEmptySnap(*r.raftLog.unstable.snapshot) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if len(r.msgs) > 0 || len(r.raftLog.unstableEntries()) > 0 || r.raftLog.hasNextEnts() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if len(r.readStates) != 0 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Advance notifies the RawNode that the application has applied and saved progress in the
|
||||||
|
// last Ready results.
|
||||||
|
func (rn *RawNode) Advance(rd Ready) {
|
||||||
|
rn.commitReady(rd)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Status returns the current status of the given group.
|
||||||
|
func (rn *RawNode) Status() *Status {
|
||||||
|
status := getStatus(rn.raft)
|
||||||
|
return &status
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReportUnreachable reports the given node is not reachable for the last send.
|
||||||
|
func (rn *RawNode) ReportUnreachable(id uint64) {
|
||||||
|
_ = rn.raft.Step(pb.Message{Type: pb.MsgUnreachable, From: id})
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReportSnapshot reports the status of the sent snapshot.
|
||||||
|
func (rn *RawNode) ReportSnapshot(id uint64, status SnapshotStatus) {
|
||||||
|
rej := status == SnapshotFailure
|
||||||
|
|
||||||
|
_ = rn.raft.Step(pb.Message{Type: pb.MsgSnapStatus, From: id, Reject: rej})
|
||||||
|
}
|
||||||
|
|
||||||
|
// TransferLeader tries to transfer leadership to the given transferee.
|
||||||
|
func (rn *RawNode) TransferLeader(transferee uint64) {
|
||||||
|
_ = rn.raft.Step(pb.Message{Type: pb.MsgTransferLeader, From: transferee})
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadIndex requests a read state. The read state will be set in ready.
|
||||||
|
// Read State has a read index. Once the application advances further than the read
|
||||||
|
// index, any linearizable read requests issued before the read request can be
|
||||||
|
// processed safely. The read state will have the same rctx attached.
|
||||||
|
func (rn *RawNode) ReadIndex(rctx []byte) {
|
||||||
|
_ = rn.raft.Step(pb.Message{Type: pb.MsgReadIndex, Entries: []pb.Entry{{Data: rctx}}})
|
||||||
|
}
|
118
vendor/github.com/coreos/etcd/raft/read_only.go
generated
vendored
Normal file
118
vendor/github.com/coreos/etcd/raft/read_only.go
generated
vendored
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
// Copyright 2016 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import pb "github.com/coreos/etcd/raft/raftpb"
|
||||||
|
|
||||||
|
// ReadState provides state for read only query.
|
||||||
|
// It's caller's responsibility to call ReadIndex first before getting
|
||||||
|
// this state from ready, it's also caller's duty to differentiate if this
|
||||||
|
// state is what it requests through RequestCtx, eg. given a unique id as
|
||||||
|
// RequestCtx
|
||||||
|
type ReadState struct {
|
||||||
|
Index uint64
|
||||||
|
RequestCtx []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type readIndexStatus struct {
|
||||||
|
req pb.Message
|
||||||
|
index uint64
|
||||||
|
acks map[uint64]struct{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type readOnly struct {
|
||||||
|
option ReadOnlyOption
|
||||||
|
pendingReadIndex map[string]*readIndexStatus
|
||||||
|
readIndexQueue []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func newReadOnly(option ReadOnlyOption) *readOnly {
|
||||||
|
return &readOnly{
|
||||||
|
option: option,
|
||||||
|
pendingReadIndex: make(map[string]*readIndexStatus),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// addRequest adds a read only reuqest into readonly struct.
|
||||||
|
// `index` is the commit index of the raft state machine when it received
|
||||||
|
// the read only request.
|
||||||
|
// `m` is the original read only request message from the local or remote node.
|
||||||
|
func (ro *readOnly) addRequest(index uint64, m pb.Message) {
|
||||||
|
ctx := string(m.Entries[0].Data)
|
||||||
|
if _, ok := ro.pendingReadIndex[ctx]; ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ro.pendingReadIndex[ctx] = &readIndexStatus{index: index, req: m, acks: make(map[uint64]struct{})}
|
||||||
|
ro.readIndexQueue = append(ro.readIndexQueue, ctx)
|
||||||
|
}
|
||||||
|
|
||||||
|
// recvAck notifies the readonly struct that the raft state machine received
|
||||||
|
// an acknowledgment of the heartbeat that attached with the read only request
|
||||||
|
// context.
|
||||||
|
func (ro *readOnly) recvAck(m pb.Message) int {
|
||||||
|
rs, ok := ro.pendingReadIndex[string(m.Context)]
|
||||||
|
if !ok {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
rs.acks[m.From] = struct{}{}
|
||||||
|
// add one to include an ack from local node
|
||||||
|
return len(rs.acks) + 1
|
||||||
|
}
|
||||||
|
|
||||||
|
// advance advances the read only request queue kept by the readonly struct.
|
||||||
|
// It dequeues the requests until it finds the read only request that has
|
||||||
|
// the same context as the given `m`.
|
||||||
|
func (ro *readOnly) advance(m pb.Message) []*readIndexStatus {
|
||||||
|
var (
|
||||||
|
i int
|
||||||
|
found bool
|
||||||
|
)
|
||||||
|
|
||||||
|
ctx := string(m.Context)
|
||||||
|
rss := []*readIndexStatus{}
|
||||||
|
|
||||||
|
for _, okctx := range ro.readIndexQueue {
|
||||||
|
i++
|
||||||
|
rs, ok := ro.pendingReadIndex[okctx]
|
||||||
|
if !ok {
|
||||||
|
panic("cannot find corresponding read state from pending map")
|
||||||
|
}
|
||||||
|
rss = append(rss, rs)
|
||||||
|
if okctx == ctx {
|
||||||
|
found = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if found {
|
||||||
|
ro.readIndexQueue = ro.readIndexQueue[i:]
|
||||||
|
for _, rs := range rss {
|
||||||
|
delete(ro.pendingReadIndex, string(rs.req.Entries[0].Data))
|
||||||
|
}
|
||||||
|
return rss
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// lastPendingRequestCtx returns the context of the last pending read only
|
||||||
|
// request in readonly struct.
|
||||||
|
func (ro *readOnly) lastPendingRequestCtx() string {
|
||||||
|
if len(ro.readIndexQueue) == 0 {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return ro.readIndexQueue[len(ro.readIndexQueue)-1]
|
||||||
|
}
|
88
vendor/github.com/coreos/etcd/raft/status.go
generated
vendored
Normal file
88
vendor/github.com/coreos/etcd/raft/status.go
generated
vendored
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
// Copyright 2015 The etcd Authors
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package raft
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
pb "github.com/coreos/etcd/raft/raftpb"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Status struct {
|
||||||
|
ID uint64
|
||||||
|
|
||||||
|
pb.HardState
|
||||||
|
SoftState
|
||||||
|
|
||||||
|
Applied uint64
|
||||||
|
Progress map[uint64]Progress
|
||||||
|
|
||||||
|
LeadTransferee uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// getStatus gets a copy of the current raft status.
|
||||||
|
func getStatus(r *raft) Status {
|
||||||
|
s := Status{
|
||||||
|
ID: r.id,
|
||||||
|
LeadTransferee: r.leadTransferee,
|
||||||
|
}
|
||||||
|
|
||||||
|
s.HardState = r.hardState()
|
||||||
|
s.SoftState = *r.softState()
|
||||||
|
|
||||||
|
s.Applied = r.raftLog.applied
|
||||||
|
|
||||||
|
if s.RaftState == StateLeader {
|
||||||
|
s.Progress = make(map[uint64]Progress)
|
||||||
|
for id, p := range r.prs {
|
||||||
|
s.Progress[id] = *p
|
||||||
|
}
|
||||||
|
|
||||||
|
for id, p := range r.learnerPrs {
|
||||||
|
s.Progress[id] = *p
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
// MarshalJSON translates the raft status into JSON.
|
||||||
|
// TODO: try to simplify this by introducing ID type into raft
|
||||||
|
func (s Status) MarshalJSON() ([]byte, error) {
|
||||||
|
j := fmt.Sprintf(`{"id":"%x","term":%d,"vote":"%x","commit":%d,"lead":"%x","raftState":%q,"applied":%d,"progress":{`,
|
||||||
|
s.ID, s.Term, s.Vote, s.Commit, s.Lead, s.RaftState, s.Applied)
|
||||||
|
|
||||||
|
if len(s.Progress) == 0 {
|
||||||
|
j += "},"
|
||||||
|
} else {
|
||||||
|
for k, v := range s.Progress {
|
||||||
|
subj := fmt.Sprintf(`"%x":{"match":%d,"next":%d,"state":%q},`, k, v.Match, v.Next, v.State)
|
||||||
|
j += subj
|
||||||
|
}
|
||||||
|
// remove the trailing ","
|
||||||
|
j = j[:len(j)-1] + "},"
|
||||||
|
}
|
||||||
|
|
||||||
|
j += fmt.Sprintf(`"leadtransferee":"%x"}`, s.LeadTransferee)
|
||||||
|
return []byte(j), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s Status) String() string {
|
||||||
|
b, err := s.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
raftLogger.Panicf("unexpected error: %v", err)
|
||||||
|
}
|
||||||
|
return string(b)
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user