tendermint/cmd/barak/main.go

247 lines
6.0 KiB
Go
Raw Normal View History

package main
2015-04-14 01:14:55 -07:00
// A note on the origin of the name.
// http://en.wikipedia.org/wiki/Barak
2015-04-13 13:14:37 -07:00
// TODO: Nonrepudiable command log
import (
2015-04-13 13:14:37 -07:00
"errors"
"fmt"
2015-04-10 02:12:17 -07:00
"io"
"io/ioutil"
2015-04-10 02:12:17 -07:00
"net/http"
"os"
2015-04-13 13:14:37 -07:00
"reflect"
"sync"
"github.com/tendermint/tendermint/binary"
2015-04-14 04:14:18 -07:00
. "github.com/tendermint/tendermint/cmd/barak/types"
. "github.com/tendermint/tendermint/common"
pcm "github.com/tendermint/tendermint/process"
2015-04-10 02:12:17 -07:00
"github.com/tendermint/tendermint/rpc"
)
var Routes = map[string]*rpc.RPCFunc{
"status": rpc.NewRPCFunc(Status, []string{}),
"run": rpc.NewRPCFunc(Run, []string{"auth_command"}),
2015-04-13 13:14:37 -07:00
// NOTE: also, two special non-JSONRPC routes called "download" and "upload"
}
type Options struct {
Validators []Validator
ListenAddress string
2015-04-13 13:14:37 -07:00
StartNonce uint64
}
// Global instance
2015-04-14 01:14:55 -07:00
var barak = struct {
2015-04-13 13:14:37 -07:00
mtx sync.Mutex
pid int
nonce uint64
2015-04-13 13:14:37 -07:00
processes map[string]*pcm.Process
validators []Validator
}{
mtx: sync.Mutex{},
pid: os.Getpid(),
nonce: 0,
processes: make(map[string]*pcm.Process),
validators: nil,
}
func main() {
fmt.Printf("New Debora Process (PID: %d)\n", os.Getpid())
// read options from stdin.
var err error
optionsBytes, err := ioutil.ReadAll(os.Stdin)
if err != nil {
panic(Fmt("Error reading input: %v", err))
}
options := binary.ReadJSON(&Options{}, optionsBytes, &err).(*Options)
if err != nil {
panic(Fmt("Error parsing input: %v", err))
}
2015-04-14 01:14:55 -07:00
barak.nonce = options.StartNonce
barak.validators = options.Validators
// Debug.
2015-04-13 13:14:37 -07:00
fmt.Printf("Options: %v\n", options)
2015-04-14 01:14:55 -07:00
fmt.Printf("Barak: %v\n", barak)
2015-04-13 13:14:37 -07:00
// Start rpc server.
2015-04-10 02:12:17 -07:00
mux := http.NewServeMux()
mux.HandleFunc("/download", ServeFile)
// TODO: mux.HandleFunc("/upload", UploadFile)
rpc.RegisterRPCFuncs(mux, Routes)
rpc.StartHTTPServer(options.ListenAddress, mux)
TrapSignal(func() {
2015-04-14 01:14:55 -07:00
fmt.Println("Barak shutting down")
})
}
//------------------------------------------------------------------------------
// RPC functions
func Status() (*ResponseStatus, error) {
barak.mtx.Lock()
pid := barak.pid
nonce := barak.nonce
validators := barak.validators
barak.mtx.Unlock()
return &ResponseStatus{
Pid: pid,
Nonce: nonce,
Validators: validators,
}, nil
}
2015-04-13 13:14:37 -07:00
func Run(authCommand AuthCommand) (interface{}, error) {
command, err := parseValidateCommand(authCommand)
if err != nil {
return nil, err
}
log.Info(Fmt("Run() received command %v", reflect.TypeOf(command)))
// Issue command
switch c := command.(type) {
case CommandRunProcess:
return RunProcess(c.Wait, c.Label, c.ExecPath, c.Args, c.Input)
case CommandStopProcess:
return StopProcess(c.Label, c.Kill)
case CommandListProcesses:
return ListProcesses()
default:
return nil, errors.New("Invalid endpoint for command")
}
}
func parseValidateCommandStr(authCommandStr string) (Command, error) {
var err error
authCommand := binary.ReadJSON(AuthCommand{}, []byte(authCommandStr), &err).(AuthCommand)
if err != nil {
fmt.Printf("Failed to parse auth_command")
return nil, errors.New("AuthCommand parse error")
}
return parseValidateCommand(authCommand)
}
func parseValidateCommand(authCommand AuthCommand) (Command, error) {
commandJSONStr := authCommand.CommandJSONStr
signatures := authCommand.Signatures
// Validate commandJSONStr
2015-04-14 01:14:55 -07:00
if !validate([]byte(commandJSONStr), barak.validators, signatures) {
2015-04-13 13:14:37 -07:00
fmt.Printf("Failed validation attempt")
return nil, errors.New("Validation error")
}
// Parse command
var err error
command := binary.ReadJSON(NoncedCommand{}, []byte(commandJSONStr), &err).(NoncedCommand)
if err != nil {
fmt.Printf("Failed to parse command")
return nil, errors.New("Command parse error")
}
// Prevent replays
2015-04-14 01:14:55 -07:00
if barak.nonce+1 != command.Nonce {
2015-04-13 13:14:37 -07:00
return nil, errors.New("Replay error")
} else {
2015-04-14 01:14:55 -07:00
barak.nonce += 1
2015-04-13 13:14:37 -07:00
}
return command.Command, nil
}
//------------------------------------------------------------------------------
// RPC base commands
// WARNING Not validated, do not export to routes.
2015-04-10 02:12:17 -07:00
func RunProcess(wait bool, label string, execPath string, args []string, input string) (*ResponseRunProcess, error) {
2015-04-14 01:14:55 -07:00
barak.mtx.Lock()
// First, see if there already is a process labeled 'label'
2015-04-14 01:14:55 -07:00
existing := barak.processes[label]
2015-04-16 10:13:50 -07:00
if existing != nil && existing.EndTime.IsZero() {
2015-04-14 01:14:55 -07:00
barak.mtx.Unlock()
2015-04-15 21:49:14 -07:00
return nil, fmt.Errorf("Process already exists: %v", label)
}
// Otherwise, create one.
2015-04-16 09:46:35 -07:00
proc, err := pcm.Create(pcm.ProcessModeDaemon, label, execPath, args, input)
if err == nil {
barak.processes[label] = proc
}
2015-04-14 01:14:55 -07:00
barak.mtx.Unlock()
2015-04-16 09:46:35 -07:00
if err != nil {
return nil, err
}
if wait {
2015-04-16 10:54:07 -07:00
<-proc.WaitCh
if proc.ExitState == nil {
return &ResponseRunProcess{
Success: true,
}, nil
} else {
return &ResponseRunProcess{
Success: proc.ExitState.Success(), // Would be always false?
}, nil
}
} else {
return &ResponseRunProcess{}, nil
}
}
func StopProcess(label string, kill bool) (*ResponseStopProcess, error) {
2015-04-14 01:14:55 -07:00
barak.mtx.Lock()
proc := barak.processes[label]
barak.mtx.Unlock()
if proc == nil {
2015-04-15 21:49:14 -07:00
return nil, fmt.Errorf("Process does not exist: %v", label)
}
err := pcm.Stop(proc, kill)
return &ResponseStopProcess{}, err
}
2015-04-10 02:12:17 -07:00
2015-04-13 13:14:37 -07:00
func ListProcesses() (*ResponseListProcesses, error) {
var procs = []*pcm.Process{}
2015-04-14 01:14:55 -07:00
barak.mtx.Lock()
2015-04-16 09:46:35 -07:00
fmt.Println("Processes: %v", barak.processes)
2015-04-14 01:14:55 -07:00
for _, proc := range barak.processes {
2015-04-13 13:14:37 -07:00
procs = append(procs, proc)
}
2015-04-14 01:14:55 -07:00
barak.mtx.Unlock()
2015-04-13 13:14:37 -07:00
return &ResponseListProcesses{
Processes: procs,
}, nil
}
2015-04-10 02:12:17 -07:00
func ServeFile(w http.ResponseWriter, req *http.Request) {
2015-04-13 13:14:37 -07:00
authCommandStr := req.FormValue("auth_command")
command, err := parseValidateCommandStr(authCommandStr)
if err != nil {
http.Error(w, Fmt("Invalid command: %v", err), 400)
}
serveCommand, ok := command.(CommandServeFile)
if !ok {
http.Error(w, "Invalid command", 400)
}
path := serveCommand.Path
2015-04-10 02:12:17 -07:00
if path == "" {
http.Error(w, "Must specify path", 400)
return
}
file, err := os.Open(path)
if err != nil {
http.Error(w, Fmt("Error opening file: %v. %v", path, err), 400)
return
}
_, err = io.Copy(w, file)
if err != nil {
fmt.Fprintf(os.Stderr, Fmt("Error serving file: %v. %v", path, err))
return
}
}