blob: b79b13619b8a3415eac402077ed49780c8f5baf6 [file] [log] [blame]
package impl
import (
"errors"
"strings"
_ "veyron.io/store/veyron/services/store/typeregistryhack"
"veyron/services/mgmt/lib/fs"
"veyron2/ipc"
"veyron2/naming"
"veyron2/services/mgmt/application"
"veyron2/vlog"
)
// invoker holds the state of an application repository invocation.
type invoker struct {
// store is the storage server used for storing application
// metadata.
// All Invokers share a single dispatcher's Memstore.
store *fs.Memstore
// storeRoot is a name in the Store under which all data will be stored.
storeRoot string
// suffix is the suffix of the current invocation that is assumed to
// be used as a relative object name to identify an application.
suffix string
}
var (
errInvalidSuffix = errors.New("invalid suffix")
errOperationFailed = errors.New("operation failed")
errNotFound = errors.New("not found")
)
// NewInvoker is the invoker factory.
func NewInvoker(store *fs.Memstore, storeRoot, suffix string) *invoker {
return &invoker{store: store, storeRoot: storeRoot, suffix: suffix}
}
func parse(suffix string) (string, string, error) {
tokens := strings.Split(suffix, "/")
switch len(tokens) {
case 2:
return tokens[0], tokens[1], nil
case 1:
return tokens[0], "", nil
default:
return "", "", errInvalidSuffix
}
}
// APPLICATION INTERFACE IMPLEMENTATION
func (i *invoker) Match(context ipc.ServerContext, profiles []string) (application.Envelope, error) {
vlog.VI(0).Infof("%v.Match(%v)", i.suffix, profiles)
empty := application.Envelope{}
name, version, err := parse(i.suffix)
if err != nil {
return empty, err
}
if version == "" {
return empty, errInvalidSuffix
}
i.store.Lock()
defer i.store.Unlock()
for _, profile := range profiles {
path := naming.Join("/applications", name, profile, version)
entry, err := i.store.BindObject(path).Get(context)
if err != nil {
continue
}
envelope, ok := entry.Value.(application.Envelope)
if !ok {
continue
}
return envelope, nil
}
return empty, errNotFound
}
func (i *invoker) Put(context ipc.ServerContext, profiles []string, envelope application.Envelope) error {
vlog.VI(0).Infof("%v.Put(%v, %v)", i.suffix, profiles, envelope)
name, version, err := parse(i.suffix)
if err != nil {
return err
}
if version == "" {
return errInvalidSuffix
}
i.store.Lock()
defer i.store.Unlock()
// Transaction is rooted at "", so tname == tid.
tname, err := i.store.BindTransactionRoot("").CreateTransaction(context)
if err != nil {
return err
}
for _, profile := range profiles {
path := naming.Join(tname, "/applications", name, profile, version)
object := i.store.BindObject(path)
_, err := object.Put(context, envelope)
if err != nil {
return errOperationFailed
}
}
if err := i.store.BindTransaction(tname).Commit(context); err != nil {
return errOperationFailed
}
return nil
}
func (i *invoker) Remove(context ipc.ServerContext, profile string) error {
vlog.VI(0).Infof("%v.Remove(%v)", i.suffix, profile)
name, version, err := parse(i.suffix)
if err != nil {
return err
}
i.store.Lock()
defer i.store.Unlock()
// Transaction is rooted at "", so tname == tid.
tname, err := i.store.BindTransactionRoot("").CreateTransaction(context)
if err != nil {
return err
}
path := naming.Join(tname, "/applications", name, profile)
if version != "" {
path += "/" + version
}
object := i.store.BindObject(path)
found, err := object.Exists(context)
if err != nil {
return errOperationFailed
}
if !found {
return errNotFound
}
if err := object.Remove(context); err != nil {
return errOperationFailed
}
if err := i.store.BindTransaction(tname).Commit(context); err != nil {
return errOperationFailed
}
return nil
}