LEFT | RIGHT |
1 package api | 1 package api |
2 | 2 |
3 import ( | 3 import ( |
4 "code.google.com/p/go.net/websocket" | 4 "code.google.com/p/go.net/websocket" |
5 "fmt" | 5 "fmt" |
| 6 "launchpad.net/juju-core/log" |
6 "launchpad.net/juju-core/state" | 7 "launchpad.net/juju-core/state" |
| 8 "launchpad.net/juju-core/state/statecmd" |
7 statewatcher "launchpad.net/juju-core/state/watcher" | 9 statewatcher "launchpad.net/juju-core/state/watcher" |
8 "strconv" | 10 "strconv" |
9 "sync" | 11 "sync" |
10 ) | 12 ) |
11 | 13 |
12 // TODO(rog) remove this when the rest of the system | 14 // TODO(rog) remove this when the rest of the system |
13 // has been updated to set passwords appropriately. | 15 // has been updated to set passwords appropriately. |
14 var AuthenticationEnabled = false | 16 var AuthenticationEnabled = false |
15 | 17 |
16 // srvRoot represents a single client's connection to the state. | 18 // srvRoot represents a single client's connection to the state. |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
62 } | 64 } |
63 r.admin = &srvAdmin{ | 65 r.admin = &srvAdmin{ |
64 root: r, | 66 root: r, |
65 } | 67 } |
66 r.client = &srvClient{ | 68 r.client = &srvClient{ |
67 root: r, | 69 root: r, |
68 } | 70 } |
69 return r | 71 return r |
70 } | 72 } |
71 | 73 |
| 74 // Kill implements rpc.Killer. It cleans up any resources that need |
| 75 // cleaning up to ensure that all outstanding requests return. |
| 76 func (r *srvRoot) Kill() { |
| 77 r.watchers.stopAll() |
| 78 } |
| 79 |
| 80 // Admin returns an object that provides API access |
| 81 // to methods that can be called even when not |
| 82 // authenticated. |
72 func (r *srvRoot) Admin(id string) (*srvAdmin, error) { | 83 func (r *srvRoot) Admin(id string) (*srvAdmin, error) { |
73 if id != "" { | 84 if id != "" { |
74 // Safeguard id for possible future use. | 85 // Safeguard id for possible future use. |
75 return nil, errBadId | 86 return nil, errBadId |
76 } | 87 } |
77 return r.admin, nil | 88 return r.admin, nil |
78 } | 89 } |
79 | 90 |
80 // requireAgent checks whether the current client is an agent and hence | 91 // requireAgent checks whether the current client is an agent and hence |
81 // may access the agent APIs. We filter out non-agents when calling one | 92 // may access the agent APIs. We filter out non-agents when calling one |
(...skipping 16 matching lines...) Expand all Loading... |
98 e := r.user.entity() | 109 e := r.user.entity() |
99 if e == nil { | 110 if e == nil { |
100 return errNotLoggedIn | 111 return errNotLoggedIn |
101 } | 112 } |
102 if isAgent(e) { | 113 if isAgent(e) { |
103 return errPerm | 114 return errPerm |
104 } | 115 } |
105 return nil | 116 return nil |
106 } | 117 } |
107 | 118 |
108 // Machine returns returns an object that provides | 119 // Machine returns an object that provides |
109 // API access to methods on a state.Machine. | 120 // API access to methods on a state.Machine. |
110 func (r *srvRoot) Machine(id string) (*srvMachine, error) { | 121 func (r *srvRoot) Machine(id string) (*srvMachine, error) { |
111 if err := r.requireAgent(); err != nil { | 122 if err := r.requireAgent(); err != nil { |
112 return nil, err | 123 return nil, err |
113 } | 124 } |
114 m, err := r.srv.state.Machine(id) | 125 m, err := r.srv.state.Machine(id) |
115 if err != nil { | 126 if err != nil { |
116 return nil, err | 127 return nil, err |
117 } | 128 } |
118 return &srvMachine{ | 129 return &srvMachine{ |
119 root: r, | 130 root: r, |
120 m: m, | 131 m: m, |
121 }, nil | 132 }, nil |
122 } | 133 } |
123 | 134 |
124 // Unit returns returns an object that provides | 135 // Unit returns an object that provides |
125 // API access to methods on a state.Unit. | 136 // API access to methods on a state.Unit. |
126 func (r *srvRoot) Unit(name string) (*srvUnit, error) { | 137 func (r *srvRoot) Unit(name string) (*srvUnit, error) { |
127 if err := r.requireAgent(); err != nil { | 138 if err := r.requireAgent(); err != nil { |
128 return nil, err | 139 return nil, err |
129 } | 140 } |
130 u, err := r.srv.state.Unit(name) | 141 u, err := r.srv.state.Unit(name) |
131 if err != nil { | 142 if err != nil { |
132 return nil, err | 143 return nil, err |
133 } | 144 } |
134 return &srvUnit{ | 145 return &srvUnit{ |
135 root: r, | 146 root: r, |
136 u: u, | 147 u: u, |
137 }, nil | 148 }, nil |
138 } | 149 } |
139 | 150 |
140 // User returns returns an object that provides | 151 // User returns an object that provides |
141 // API access to methods on a state.User. | 152 // API access to methods on a state.User. |
142 func (r *srvRoot) User(name string) (*srvUser, error) { | 153 func (r *srvRoot) User(name string) (*srvUser, error) { |
143 // Any user is allowed to access their own user object. | 154 // Any user is allowed to access their own user object. |
144 // We check at this level rather than at the operation | 155 // We check at this level rather than at the operation |
145 // level to stop malicious probing for current user names. | 156 // level to stop malicious probing for current user names. |
146 // When we provide support for user administration, | 157 // When we provide support for user administration, |
147 // this will need to be changed to allow access to | 158 // this will need to be changed to allow access to |
148 // the administrator. | 159 // the administrator. |
149 e := r.user.entity() | 160 e := r.user.entity() |
150 if e == nil { | 161 if e == nil { |
(...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
222 } | 233 } |
223 for _, m := range ms { | 234 for _, m := range ms { |
224 instId, _ := m.InstanceId() | 235 instId, _ := m.InstanceId() |
225 status.Machines[m.Id()] = MachineInfo{ | 236 status.Machines[m.Id()] = MachineInfo{ |
226 InstanceId: string(instId), | 237 InstanceId: string(instId), |
227 } | 238 } |
228 } | 239 } |
229 return status, nil | 240 return status, nil |
230 } | 241 } |
231 | 242 |
| 243 // ServiceSet implements the server side of Client.ServerSet. |
| 244 func (c *srvClient) ServiceSet(p statecmd.ServiceSetParams) error { |
| 245 return statecmd.ServiceSet(c.root.srv.state, p) |
| 246 } |
| 247 |
| 248 // ServiceSetYAML implements the server side of Client.ServerSetYAML. |
| 249 func (c *srvClient) ServiceSetYAML(p statecmd.ServiceSetYAMLParams) error { |
| 250 return statecmd.ServiceSetYAML(c.root.srv.state, p) |
| 251 } |
| 252 |
| 253 func (c *srvClient) ServiceGet(args statecmd.ServiceGetParams) (statecmd.Service
GetResults, error) { |
| 254 return statecmd.ServiceGet(c.root.srv.state, args) |
| 255 } |
| 256 |
| 257 // EnvironmentInfo returns information about the current environment (default |
| 258 // series and type). |
| 259 func (c *srvClient) EnvironmentInfo() (EnvironmentInfo, error) { |
| 260 conf, err := c.root.srv.state.EnvironConfig() |
| 261 if err != nil { |
| 262 return EnvironmentInfo{}, err |
| 263 } |
| 264 info := EnvironmentInfo{ |
| 265 DefaultSeries: conf.DefaultSeries(), |
| 266 ProviderType: conf.Type(), |
| 267 } |
| 268 return info, nil |
| 269 } |
| 270 |
232 type rpcCreds struct { | 271 type rpcCreds struct { |
233 EntityName string | 272 EntityName string |
234 Password string | 273 Password string |
235 } | 274 } |
236 | 275 |
237 // Login logs in with the provided credentials. | 276 // Login logs in with the provided credentials. |
238 // All subsequent requests on the connection will | 277 // All subsequent requests on the connection will |
239 // act as the authenticated user. | 278 // act as the authenticated user. |
240 func (a *srvAdmin) Login(c rpcCreds) error { | 279 func (a *srvAdmin) Login(c rpcCreds) error { |
241 return a.root.user.login(a.root.srv.state, c.EntityName, c.Password) | 280 return a.root.user.login(a.root.srv.state, c.EntityName, c.Password) |
(...skipping 213 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
455 defer ws.mu.Unlock() | 494 defer ws.mu.Unlock() |
456 ws.maxId++ | 495 ws.maxId++ |
457 sw := &srvWatcher{ | 496 sw := &srvWatcher{ |
458 ws: ws, | 497 ws: ws, |
459 id: strconv.FormatUint(ws.maxId, 10), | 498 id: strconv.FormatUint(ws.maxId, 10), |
460 w: w, | 499 w: w, |
461 } | 500 } |
462 ws.ws[sw.id] = sw | 501 ws.ws[sw.id] = sw |
463 return sw | 502 return sw |
464 } | 503 } |
| 504 |
| 505 func (ws *watchers) stopAll() { |
| 506 ws.mu.Lock() |
| 507 defer ws.mu.Unlock() |
| 508 for _, w := range ws.ws { |
| 509 if err := w.w.Stop(); err != nil { |
| 510 log.Printf("state/api: error stopping %T watcher: %v", w
, err) |
| 511 } |
| 512 } |
| 513 ws.ws = make(map[string]*srvWatcher) |
| 514 } |
LEFT | RIGHT |