Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 1 | package impl |
| 2 | |
| 3 | import ( |
Bogdan Caprita | c87a914 | 2014-07-21 10:38:13 -0700 | [diff] [blame] | 4 | "fmt" |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 5 | "strings" |
Jiri Simsa | 24e87aa | 2014-06-09 09:27:34 -0700 | [diff] [blame] | 6 | |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 7 | inode "veyron/services/mgmt/node" |
Bogdan Caprita | c87a914 | 2014-07-21 10:38:13 -0700 | [diff] [blame] | 8 | "veyron/services/mgmt/node/config" |
Jiri Simsa | 24e87aa | 2014-06-09 09:27:34 -0700 | [diff] [blame] | 9 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 10 | "veyron2/ipc" |
| 11 | "veyron2/security" |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 12 | "veyron2/services/mgmt/node" |
| 13 | "veyron2/verror" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 14 | ) |
| 15 | |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 16 | // internalState wraps state shared between different node manager |
| 17 | // invocations. |
| 18 | type internalState struct { |
| 19 | callback *callbackState |
| 20 | updating *updatingState |
| 21 | } |
| 22 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 23 | // dispatcher holds the state of the node manager dispatcher. |
| 24 | type dispatcher struct { |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 25 | auth security.Authorizer |
| 26 | // internal holds the state that persists across RPC method invocations. |
Bogdan Caprita | c87a914 | 2014-07-21 10:38:13 -0700 | [diff] [blame] | 27 | internal *internalState |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 28 | // config holds the node manager's (immutable) configuration state. |
| 29 | config *config.State |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 30 | } |
| 31 | |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 32 | const ( |
| 33 | appsSuffix = "apps" |
| 34 | nodeSuffix = "nm" |
| 35 | configSuffix = "cfg" |
| 36 | ) |
| 37 | |
| 38 | var ( |
| 39 | errInvalidSuffix = verror.BadArgf("invalid suffix") |
| 40 | errOperationFailed = verror.Internalf("operation failed") |
| 41 | errInProgress = verror.Existsf("operation in progress") |
| 42 | errIncompatibleUpdate = verror.BadArgf("update failed: mismatching app title") |
| 43 | errUpdateNoOp = verror.NotFoundf("no different version available") |
| 44 | errNotExist = verror.NotFoundf("object does not exist") |
| 45 | errInvalidOperation = verror.BadArgf("invalid operation") |
| 46 | ) |
| 47 | |
Jiri Simsa | 24e87aa | 2014-06-09 09:27:34 -0700 | [diff] [blame] | 48 | // NewDispatcher is the node manager dispatcher factory. |
Bogdan Caprita | c87a914 | 2014-07-21 10:38:13 -0700 | [diff] [blame] | 49 | func NewDispatcher(auth security.Authorizer, config *config.State) (*dispatcher, error) { |
| 50 | if err := config.Validate(); err != nil { |
| 51 | return nil, fmt.Errorf("Invalid config %v: %v", config, err) |
| 52 | } |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 53 | return &dispatcher{ |
Jiri Simsa | 70c3205 | 2014-06-18 11:38:21 -0700 | [diff] [blame] | 54 | auth: auth, |
Bogdan Caprita | c87a914 | 2014-07-21 10:38:13 -0700 | [diff] [blame] | 55 | internal: &internalState{ |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 56 | callback: newCallbackState(), |
| 57 | updating: newUpdatingState(), |
Jiri Simsa | 70c3205 | 2014-06-18 11:38:21 -0700 | [diff] [blame] | 58 | }, |
Bogdan Caprita | c87a914 | 2014-07-21 10:38:13 -0700 | [diff] [blame] | 59 | config: config, |
| 60 | }, nil |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 61 | } |
| 62 | |
| 63 | // DISPATCHER INTERFACE IMPLEMENTATION |
| 64 | |
Cosmos Nicolaou | 8bfacf2 | 2014-08-19 11:19:36 -0700 | [diff] [blame^] | 65 | func (d *dispatcher) Lookup(suffix, method string) (ipc.Invoker, security.Authorizer, error) { |
Bogdan Caprita | 4d67c04 | 2014-08-19 10:41:19 -0700 | [diff] [blame] | 66 | components := strings.Split(suffix, "/") |
| 67 | for i := 0; i < len(components); i++ { |
| 68 | if len(components[i]) == 0 { |
| 69 | components = append(components[:i], components[i+1:]...) |
| 70 | i-- |
| 71 | } |
| 72 | } |
| 73 | if len(components) == 0 { |
| 74 | return nil, nil, errInvalidSuffix |
| 75 | } |
| 76 | // The implementation of the node manager is split up into several |
| 77 | // invokers, which are instantiated depending on the receiver name |
| 78 | // prefix. |
| 79 | var receiver interface{} |
| 80 | switch components[0] { |
| 81 | case nodeSuffix: |
| 82 | receiver = node.NewServerNode(&nodeInvoker{ |
| 83 | callback: d.internal.callback, |
| 84 | updating: d.internal.updating, |
| 85 | config: d.config, |
| 86 | }) |
| 87 | case appsSuffix: |
| 88 | receiver = node.NewServerApplication(&appInvoker{ |
| 89 | callback: d.internal.callback, |
| 90 | config: d.config, |
| 91 | suffix: components[1:], |
| 92 | }) |
| 93 | case configSuffix: |
| 94 | if len(components) != 2 { |
| 95 | return nil, nil, errInvalidSuffix |
| 96 | } |
| 97 | receiver = inode.NewServerConfig(&configInvoker{ |
| 98 | callback: d.internal.callback, |
| 99 | suffix: components[1], |
| 100 | }) |
| 101 | default: |
| 102 | return nil, nil, errInvalidSuffix |
| 103 | } |
| 104 | return ipc.ReflectInvoker(receiver), d.auth, nil |
Jiri Simsa | 24e87aa | 2014-06-09 09:27:34 -0700 | [diff] [blame] | 105 | } |