mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00
141 lines
4.5 KiB
Go
141 lines
4.5 KiB
Go
// Copyright 2015 CoreOS, Inc.
|
|
//
|
|
// 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 etcdserver
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"net/http"
|
|
"path"
|
|
"sort"
|
|
|
|
"github.com/coreos/etcd/pkg/netutil"
|
|
"github.com/coreos/etcd/pkg/types"
|
|
"github.com/coreos/etcd/raft"
|
|
)
|
|
|
|
// ServerConfig holds the configuration of etcd as taken from the command line or discovery.
|
|
type ServerConfig struct {
|
|
Name string
|
|
DiscoveryURL string
|
|
DiscoveryProxy string
|
|
ClientURLs types.URLs
|
|
PeerURLs types.URLs
|
|
DataDir string
|
|
SnapCount uint64
|
|
MaxSnapFiles uint
|
|
MaxWALFiles uint
|
|
Cluster *Cluster
|
|
NewCluster bool
|
|
ForceNewCluster bool
|
|
Transport *http.Transport
|
|
|
|
TickMs uint
|
|
ElectionTicks int
|
|
}
|
|
|
|
// VerifyBootstrapConfig sanity-checks the initial config for bootstrap case
|
|
// and returns an error for things that should never happen.
|
|
func (c *ServerConfig) VerifyBootstrap() error {
|
|
if err := c.verifyLocalMember(true); err != nil {
|
|
return err
|
|
}
|
|
if err := c.Cluster.Validate(); err != nil {
|
|
return err
|
|
}
|
|
if c.Cluster.String() == "" && c.DiscoveryURL == "" {
|
|
return fmt.Errorf("initial cluster unset and no discovery URL found")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// VerifyJoinExisting sanity-checks the initial config for join existing cluster
|
|
// case and returns an error for things that should never happen.
|
|
func (c *ServerConfig) VerifyJoinExisting() error {
|
|
// no need for strict checking since the member have announced its
|
|
// peer urls to the cluster before starting and do not have to set
|
|
// it in the configuration again.
|
|
if err := c.verifyLocalMember(false); err != nil {
|
|
return err
|
|
}
|
|
if err := c.Cluster.Validate(); err != nil {
|
|
return err
|
|
}
|
|
if c.DiscoveryURL != "" {
|
|
return fmt.Errorf("discovery URL should not be set when joining existing initial cluster")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// verifyLocalMember verifies the configured member is in configured
|
|
// cluster. If strict is set, it also verifies the configured member
|
|
// has the same peer urls as configured advertised peer urls.
|
|
func (c *ServerConfig) verifyLocalMember(strict bool) error {
|
|
m := c.Cluster.MemberByName(c.Name)
|
|
// Make sure the cluster at least contains the local server.
|
|
if m == nil {
|
|
return fmt.Errorf("couldn't find local name %q in the initial cluster configuration", c.Name)
|
|
}
|
|
if uint64(m.ID) == raft.None {
|
|
return fmt.Errorf("cannot use %x as member id", raft.None)
|
|
}
|
|
|
|
// Advertised peer URLs must match those in the cluster peer list
|
|
// TODO: Remove URLStringsEqual after improvement of using hostnames #2150 #2123
|
|
apurls := c.PeerURLs.StringSlice()
|
|
sort.Strings(apurls)
|
|
if strict {
|
|
if !netutil.URLStringsEqual(apurls, m.PeerURLs) {
|
|
return fmt.Errorf("%s has different advertised URLs in the cluster and advertised peer URLs list", c.Name)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *ServerConfig) MemberDir() string { return path.Join(c.DataDir, "member") }
|
|
|
|
func (c *ServerConfig) WALDir() string { return path.Join(c.MemberDir(), "wal") }
|
|
|
|
func (c *ServerConfig) SnapDir() string { return path.Join(c.MemberDir(), "snap") }
|
|
|
|
func (c *ServerConfig) ShouldDiscover() bool { return c.DiscoveryURL != "" }
|
|
|
|
func (c *ServerConfig) PrintWithInitial() { c.print(true) }
|
|
|
|
func (c *ServerConfig) Print() { c.print(false) }
|
|
|
|
func (c *ServerConfig) print(initial bool) {
|
|
log.Printf("etcdserver: name = %s", c.Name)
|
|
if c.ForceNewCluster {
|
|
log.Println("etcdserver: force new cluster")
|
|
}
|
|
log.Printf("etcdserver: data dir = %s", c.DataDir)
|
|
log.Printf("etcdserver: member dir = %s", c.MemberDir())
|
|
log.Printf("etcdserver: heartbeat = %dms", c.TickMs)
|
|
log.Printf("etcdserver: election = %dms", c.ElectionTicks*int(c.TickMs))
|
|
log.Printf("etcdserver: snapshot count = %d", c.SnapCount)
|
|
if len(c.DiscoveryURL) != 0 {
|
|
log.Printf("etcdserver: discovery URL= %s", c.DiscoveryURL)
|
|
if len(c.DiscoveryProxy) != 0 {
|
|
log.Printf("etcdserver: discovery proxy = %s", c.DiscoveryProxy)
|
|
}
|
|
}
|
|
log.Printf("etcdserver: advertise client URLs = %s", c.ClientURLs)
|
|
if initial {
|
|
log.Printf("etcdserver: initial advertise peer URLs = %s", c.PeerURLs)
|
|
log.Printf("etcdserver: initial cluster = %s", c.Cluster)
|
|
}
|
|
}
|