mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00
tests: Add support for lazyfs
Signed-off-by: Marek Siarkowicz <siarkowicz@google.com>
This commit is contained in:
@@ -185,6 +185,7 @@ type EtcdProcessClusterConfig struct {
|
||||
CompactHashCheckEnabled bool
|
||||
CompactHashCheckTime time.Duration
|
||||
GoFailEnabled bool
|
||||
LazyFSEnabled bool
|
||||
CompactionBatchLimit int
|
||||
CompactionSleepInterval time.Duration
|
||||
|
||||
@@ -344,6 +345,10 @@ func WithGoFailEnabled(enabled bool) EPClusterOption {
|
||||
return func(c *EtcdProcessClusterConfig) { c.GoFailEnabled = enabled }
|
||||
}
|
||||
|
||||
func WithLazyFSEnabled(enabled bool) EPClusterOption {
|
||||
return func(c *EtcdProcessClusterConfig) { c.LazyFSEnabled = enabled }
|
||||
}
|
||||
|
||||
func WithWarningUnaryRequestDuration(time time.Duration) EPClusterOption {
|
||||
return func(c *EtcdProcessClusterConfig) { c.WarningUnaryRequestDuration = time }
|
||||
}
|
||||
@@ -407,7 +412,7 @@ func InitEtcdProcessCluster(t testing.TB, cfg *EtcdProcessClusterConfig) (*EtcdP
|
||||
|
||||
// launch etcd processes
|
||||
for i := range etcdCfgs {
|
||||
proc, err := NewEtcdProcess(etcdCfgs[i])
|
||||
proc, err := NewEtcdProcess(t, etcdCfgs[i])
|
||||
if err != nil {
|
||||
epc.Close()
|
||||
return nil, fmt.Errorf("cannot configure: %v", err)
|
||||
@@ -659,6 +664,7 @@ func (cfg *EtcdProcessClusterConfig) EtcdServerProcessConfig(tb testing.TB, i in
|
||||
InitialToken: cfg.InitialToken,
|
||||
GoFailPort: gofailPort,
|
||||
Proxy: proxyCfg,
|
||||
LazyFSEnabled: cfg.LazyFSEnabled,
|
||||
}
|
||||
}
|
||||
|
||||
@@ -826,7 +832,7 @@ func (epc *EtcdProcessCluster) StartNewProc(ctx context.Context, cfg *EtcdProces
|
||||
|
||||
// Then start process
|
||||
tb.Log("start new member")
|
||||
proc, err := NewEtcdProcess(serverCfg)
|
||||
proc, err := NewEtcdProcess(tb, serverCfg)
|
||||
if err != nil {
|
||||
epc.Close()
|
||||
return 0, fmt.Errorf("cannot configure: %v", err)
|
||||
@@ -855,7 +861,7 @@ func (epc *EtcdProcessCluster) UpdateProcOptions(i int, tb testing.TB, opts ...E
|
||||
}
|
||||
epc.Cfg.SetInitialOrDiscovery(serverCfg, initialCluster, "new")
|
||||
|
||||
proc, err := NewEtcdProcess(serverCfg)
|
||||
proc, err := NewEtcdProcess(tb, serverCfg)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
@@ -16,6 +16,8 @@
|
||||
|
||||
package e2e
|
||||
|
||||
func NewEtcdProcess(cfg *EtcdServerProcessConfig) (EtcdProcess, error) {
|
||||
return NewEtcdServerProcess(cfg)
|
||||
import "testing"
|
||||
|
||||
func NewEtcdProcess(t testing.TB, cfg *EtcdServerProcessConfig) (EtcdProcess, error) {
|
||||
return NewEtcdServerProcess(t, cfg)
|
||||
}
|
||||
|
||||
@@ -24,6 +24,7 @@ import (
|
||||
"path"
|
||||
"strconv"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"go.uber.org/zap"
|
||||
|
||||
@@ -38,12 +39,12 @@ type proxyEtcdProcess struct {
|
||||
proxyV3 *proxyV3Proc
|
||||
}
|
||||
|
||||
func NewEtcdProcess(cfg *EtcdServerProcessConfig) (EtcdProcess, error) {
|
||||
return NewProxyEtcdProcess(cfg)
|
||||
func NewEtcdProcess(t testing.TB, cfg *EtcdServerProcessConfig) (EtcdProcess, error) {
|
||||
return NewProxyEtcdProcess(t, cfg)
|
||||
}
|
||||
|
||||
func NewProxyEtcdProcess(cfg *EtcdServerProcessConfig) (*proxyEtcdProcess, error) {
|
||||
ep, err := NewEtcdServerProcess(cfg)
|
||||
func NewProxyEtcdProcess(t testing.TB, cfg *EtcdServerProcessConfig) (*proxyEtcdProcess, error) {
|
||||
ep, err := NewEtcdServerProcess(t, cfg)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
@@ -56,6 +56,7 @@ type EtcdProcess interface {
|
||||
Config() *EtcdServerProcessConfig
|
||||
PeerProxy() proxy.Server
|
||||
Failpoints() *BinaryFailpoints
|
||||
LazyFS() *LazyFS
|
||||
Logs() LogsExpect
|
||||
Kill() error
|
||||
}
|
||||
@@ -70,6 +71,7 @@ type EtcdServerProcess struct {
|
||||
cfg *EtcdServerProcessConfig
|
||||
proc *expect.ExpectProcess
|
||||
proxy proxy.Server
|
||||
lazyfs *LazyFS
|
||||
failpoints *BinaryFailpoints
|
||||
donec chan struct{} // closed when Interact() terminates
|
||||
}
|
||||
@@ -96,10 +98,11 @@ type EtcdServerProcessConfig struct {
|
||||
InitialCluster string
|
||||
GoFailPort int
|
||||
|
||||
Proxy *proxy.ServerConfig
|
||||
LazyFSEnabled bool
|
||||
Proxy *proxy.ServerConfig
|
||||
}
|
||||
|
||||
func NewEtcdServerProcess(cfg *EtcdServerProcessConfig) (*EtcdServerProcess, error) {
|
||||
func NewEtcdServerProcess(t testing.TB, cfg *EtcdServerProcessConfig) (*EtcdServerProcess, error) {
|
||||
if !fileutil.Exist(cfg.ExecPath) {
|
||||
return nil, fmt.Errorf("could not find etcd binary: %s", cfg.ExecPath)
|
||||
}
|
||||
@@ -107,11 +110,17 @@ func NewEtcdServerProcess(cfg *EtcdServerProcessConfig) (*EtcdServerProcess, err
|
||||
if err := os.RemoveAll(cfg.DataDirPath); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if err := os.Mkdir(cfg.DataDirPath, 0700); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
ep := &EtcdServerProcess{cfg: cfg, donec: make(chan struct{})}
|
||||
if cfg.GoFailPort != 0 {
|
||||
ep.failpoints = &BinaryFailpoints{member: ep}
|
||||
}
|
||||
if cfg.LazyFSEnabled {
|
||||
ep.lazyfs = newLazyFS(cfg.lg, cfg.DataDirPath, t)
|
||||
}
|
||||
return ep, nil
|
||||
}
|
||||
|
||||
@@ -146,6 +155,14 @@ func (ep *EtcdServerProcess) Start(ctx context.Context) error {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if ep.lazyfs != nil {
|
||||
ep.cfg.lg.Info("starting lazyfs...", zap.String("name", ep.cfg.Name))
|
||||
err := ep.lazyfs.Start(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
ep.cfg.lg.Info("starting server...", zap.String("name", ep.cfg.Name))
|
||||
proc, err := SpawnCmdWithLogger(ep.cfg.lg, append([]string{ep.cfg.ExecPath}, ep.cfg.Args...), ep.cfg.EnvVars, ep.cfg.Name)
|
||||
if err != nil {
|
||||
@@ -205,6 +222,14 @@ func (ep *EtcdServerProcess) Stop() (err error) {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if ep.lazyfs != nil {
|
||||
ep.cfg.lg.Info("stopping lazyfs...", zap.String("name", ep.cfg.Name))
|
||||
err = ep.lazyfs.Stop()
|
||||
ep.lazyfs = nil
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -298,6 +323,10 @@ func (ep *EtcdServerProcess) PeerProxy() proxy.Server {
|
||||
return ep.proxy
|
||||
}
|
||||
|
||||
func (ep *EtcdServerProcess) LazyFS() *LazyFS {
|
||||
return ep.lazyfs
|
||||
}
|
||||
|
||||
func (ep *EtcdServerProcess) Failpoints() *BinaryFailpoints {
|
||||
return ep.failpoints
|
||||
}
|
||||
|
||||
@@ -48,6 +48,18 @@ type binPath struct {
|
||||
EtcdLastRelease string
|
||||
Etcdctl string
|
||||
Etcdutl string
|
||||
LazyFS string
|
||||
}
|
||||
|
||||
func (bp *binPath) LazyFSAvailable() bool {
|
||||
_, err := os.Stat(bp.LazyFS)
|
||||
if err != nil {
|
||||
if !os.IsNotExist(err) {
|
||||
panic(err)
|
||||
}
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
func InitFlags() {
|
||||
@@ -65,6 +77,7 @@ func InitFlags() {
|
||||
EtcdLastRelease: *binDir + "/etcd-last-release",
|
||||
Etcdctl: *binDir + "/etcdctl",
|
||||
Etcdutl: *binDir + "/etcdutl",
|
||||
LazyFS: *binDir + "/lazyfs",
|
||||
}
|
||||
CertPath = CertDir + "/server.crt"
|
||||
PrivateKeyPath = CertDir + "/server.key.insecure"
|
||||
|
||||
114
tests/framework/e2e/lazyfs.go
Normal file
114
tests/framework/e2e/lazyfs.go
Normal file
@@ -0,0 +1,114 @@
|
||||
// Copyright 2023 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 e2e
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
|
||||
"go.uber.org/zap"
|
||||
|
||||
"go.etcd.io/etcd/pkg/v3/expect"
|
||||
)
|
||||
|
||||
func newLazyFS(lg *zap.Logger, dataDir string, tmp TempDirProvider) *LazyFS {
|
||||
return &LazyFS{
|
||||
lg: lg,
|
||||
DataDir: dataDir,
|
||||
LazyFSDir: tmp.TempDir(),
|
||||
}
|
||||
}
|
||||
|
||||
type TempDirProvider interface {
|
||||
TempDir() string
|
||||
}
|
||||
|
||||
type LazyFS struct {
|
||||
lg *zap.Logger
|
||||
|
||||
DataDir string
|
||||
LazyFSDir string
|
||||
|
||||
ep *expect.ExpectProcess
|
||||
}
|
||||
|
||||
func (fs *LazyFS) Start(ctx context.Context) (err error) {
|
||||
if fs.ep != nil {
|
||||
return nil
|
||||
}
|
||||
err = os.WriteFile(fs.configPath(), fs.config(), 0666)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
dataPath := filepath.Join(fs.LazyFSDir, "data")
|
||||
err = os.Mkdir(dataPath, 0700)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
flags := []string{fs.DataDir, "--config-path", fs.configPath(), "-o", "modules=subdir", "-o", "subdir=" + dataPath, "-f"}
|
||||
fs.lg.Info("Started lazyfs", zap.Strings("flags", flags))
|
||||
fs.ep, err = expect.NewExpect(BinPath.LazyFS, flags...)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
_, err = fs.ep.ExpectWithContext(ctx, "waiting for fault commands")
|
||||
return err
|
||||
}
|
||||
|
||||
func (fs *LazyFS) configPath() string {
|
||||
return filepath.Join(fs.LazyFSDir, "config.toml")
|
||||
}
|
||||
|
||||
func (fs *LazyFS) socketPath() string {
|
||||
return filepath.Join(fs.LazyFSDir, "sock.fifo")
|
||||
}
|
||||
|
||||
func (fs *LazyFS) config() []byte {
|
||||
return []byte(fmt.Sprintf(`[faults]
|
||||
fifo_path=%q
|
||||
[cache]
|
||||
apply_eviction=false
|
||||
[cache.simple]
|
||||
custom_size="1gb"
|
||||
blocks_per_page=1
|
||||
[filesystem]
|
||||
log_all_operations=false
|
||||
`, fs.socketPath()))
|
||||
}
|
||||
|
||||
func (fs *LazyFS) Stop() error {
|
||||
if fs.ep == nil {
|
||||
return nil
|
||||
}
|
||||
defer func() { fs.ep = nil }()
|
||||
err := fs.ep.Stop()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return fs.ep.Close()
|
||||
}
|
||||
|
||||
func (fs *LazyFS) ClearCache(ctx context.Context) error {
|
||||
err := os.WriteFile(fs.socketPath(), []byte("lazyfs::clear-cache\n"), 0666)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
// TODO: Wait for response on socket instead of reading logs to get command completion.
|
||||
// Set `fifo_path_completed` config for LazyFS to create separate socket to write when it has completed command.
|
||||
_, err = fs.ep.ExpectWithContext(ctx, "cache is cleared")
|
||||
return err
|
||||
}
|
||||
Reference in New Issue
Block a user