forked from TrueCloudLab/rclone
8a46dd1b57
This is implemented as a state machine parser so it can emit sensible error messages. It does not use the connection strings elsewhere in rclone yet - see subsequent commits. An optional fuzzer is implemented for the Parse function.
187 lines
4.3 KiB
Go
187 lines
4.3 KiB
Go
// Package testserver starts and stops test servers if required
|
|
package testserver
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/rclone/rclone/fs"
|
|
"github.com/rclone/rclone/fs/fspath"
|
|
)
|
|
|
|
var (
|
|
once sync.Once
|
|
configDir string // where the config is stored
|
|
// Note of running servers
|
|
runningMu sync.Mutex
|
|
running = map[string]int{}
|
|
errNotFound = errors.New("command not found")
|
|
)
|
|
|
|
// Assume we are run somewhere within the rclone root
|
|
func findConfig() (string, error) {
|
|
dir := filepath.Join("fstest", "testserver", "init.d")
|
|
for i := 0; i < 5; i++ {
|
|
fi, err := os.Stat(dir)
|
|
if err == nil && fi.IsDir() {
|
|
return filepath.Abs(dir)
|
|
} else if !os.IsNotExist(err) {
|
|
return "", err
|
|
}
|
|
dir = filepath.Join("..", dir)
|
|
}
|
|
return "", errors.New("couldn't find testserver config files - run from within rclone source")
|
|
}
|
|
|
|
// run the command returning the output and an error
|
|
func run(name, command string) (out []byte, err error) {
|
|
cmdPath := filepath.Join(configDir, name)
|
|
fi, err := os.Stat(cmdPath)
|
|
if err != nil || fi.IsDir() {
|
|
return nil, errNotFound
|
|
}
|
|
cmd := exec.Command(cmdPath, command)
|
|
out, err = cmd.CombinedOutput()
|
|
if err != nil {
|
|
err = errors.Wrapf(err, "failed to run %s %s\n%s", cmdPath, command, string(out))
|
|
}
|
|
return out, err
|
|
}
|
|
|
|
// Check to see if the server is running
|
|
func isRunning(name string) bool {
|
|
_, err := run(name, "status")
|
|
return err == nil
|
|
}
|
|
|
|
// envKey returns the environment variable name to set name, key
|
|
func envKey(name, key string) string {
|
|
return fmt.Sprintf("RCLONE_CONFIG_%s_%s", strings.ToUpper(name), strings.ToUpper(key))
|
|
}
|
|
|
|
// match a line of config var=value
|
|
var matchLine = regexp.MustCompile(`^([a-zA-Z_]+)=(.*)$`)
|
|
|
|
// Start the server and set its env vars
|
|
// Call with the mutex held
|
|
func start(name string) error {
|
|
out, err := run(name, "start")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fs.Logf(name, "Starting server")
|
|
// parse the output and set environment vars from it
|
|
var connect string
|
|
for _, line := range bytes.Split(out, []byte("\n")) {
|
|
line = bytes.TrimSpace(line)
|
|
part := matchLine.FindSubmatch(line)
|
|
if part != nil {
|
|
key, value := part[1], part[2]
|
|
if string(key) == "_connect" {
|
|
connect = string(value)
|
|
continue
|
|
}
|
|
|
|
// fs.Debugf(name, "key = %q, envKey = %q, value = %q", key, envKey(name, string(key)), value)
|
|
err = os.Setenv(envKey(name, string(key)), string(value))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
if connect == "" {
|
|
return nil
|
|
}
|
|
// If we got a _connect value then try to connect to it
|
|
const maxTries = 30
|
|
for i := 1; i <= maxTries; i++ {
|
|
fs.Debugf(name, "Attempting to connect to %q try %d/%d", connect, i, maxTries)
|
|
conn, err := net.Dial("tcp", connect)
|
|
if err == nil {
|
|
_ = conn.Close()
|
|
return nil
|
|
}
|
|
time.Sleep(time.Second)
|
|
}
|
|
return errors.Errorf("failed to connect to %q on %q", name, connect)
|
|
}
|
|
|
|
// Start starts the named test server which can be stopped by the
|
|
// function returned.
|
|
func Start(remoteName string) (fn func(), err error) {
|
|
if remoteName == "" {
|
|
// don't start the local backend
|
|
return func() {}, nil
|
|
}
|
|
parsed, err := fspath.Parse(remoteName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
name := parsed.ConfigString
|
|
if name == "" {
|
|
// don't start the local backend
|
|
return func() {}, nil
|
|
}
|
|
|
|
// Make sure we know where the config is
|
|
once.Do(func() {
|
|
configDir, err = findConfig()
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
runningMu.Lock()
|
|
defer runningMu.Unlock()
|
|
|
|
if running[name] <= 0 {
|
|
// if server isn't running check to see if this server has
|
|
// been started already but not by us and stop it if so
|
|
if os.Getenv(envKey(name, "type")) == "" && isRunning(name) {
|
|
stop(name)
|
|
}
|
|
if !isRunning(name) {
|
|
err = start(name)
|
|
if err == errNotFound {
|
|
// if no file found then don't start or stop
|
|
return func() {}, nil
|
|
} else if err != nil {
|
|
return nil, err
|
|
}
|
|
running[name] = 0
|
|
} else {
|
|
running[name] = 1
|
|
}
|
|
}
|
|
running[name]++
|
|
|
|
return func() {
|
|
runningMu.Lock()
|
|
defer runningMu.Unlock()
|
|
stop(name)
|
|
}, nil
|
|
|
|
}
|
|
|
|
// Stops the named test server
|
|
// Call with the mutex held
|
|
func stop(name string) {
|
|
running[name]--
|
|
if running[name] <= 0 {
|
|
_, err := run(name, "stop")
|
|
if err != nil {
|
|
fs.Errorf(name, "Failed to stop server: %v", err)
|
|
}
|
|
running[name] = 0
|
|
fs.Logf(name, "Stopped server")
|
|
}
|
|
}
|