Left: | ||
Right: |
OLD | NEW |
---|---|
1 // Copyright 2013 Canonical Ltd. | 1 // Copyright 2013 Canonical Ltd. |
2 // Licensed under the AGPLv3, see LICENCE file for details. | 2 // Licensed under the AGPLv3, see LICENCE file for details. |
3 | 3 |
4 package agent | 4 package agent |
5 | 5 |
6 import ( | 6 import ( |
7 "bytes" | |
7 "fmt" | 8 "fmt" |
9 "io/ioutil" | |
10 "os" | |
8 "path" | 11 "path" |
12 "path/filepath" | |
9 "regexp" | 13 "regexp" |
14 "strings" | |
10 "sync" | 15 "sync" |
11 | 16 |
12 "github.com/errgo/errgo" | 17 "github.com/errgo/errgo" |
13 "github.com/loggo/loggo" | 18 "github.com/loggo/loggo" |
14 | 19 |
15 "launchpad.net/juju-core/errors" | 20 "launchpad.net/juju-core/errors" |
16 "launchpad.net/juju-core/state" | 21 "launchpad.net/juju-core/state" |
17 "launchpad.net/juju-core/state/api" | 22 "launchpad.net/juju-core/state/api" |
18 "launchpad.net/juju-core/state/api/params" | 23 "launchpad.net/juju-core/state/api/params" |
19 "launchpad.net/juju-core/utils" | 24 "launchpad.net/juju-core/utils" |
20 "launchpad.net/juju-core/version" | 25 "launchpad.net/juju-core/version" |
21 ) | 26 ) |
22 | 27 |
23 var logger = loggo.GetLogger("juju.agent") | 28 var logger = loggo.GetLogger("juju.agent") |
24 | 29 |
30 // DefaultLogDir defines the default log directory for juju agents. | |
31 const DefaultLogDir = "/var/log/juju" | |
32 | |
25 const ( | 33 const ( |
26 LxcBridge = "LXC_BRIDGE" | 34 LxcBridge = "LXC_BRIDGE" |
27 ProviderType = "PROVIDER_TYPE" | 35 ProviderType = "PROVIDER_TYPE" |
28 ContainerType = "CONTAINER_TYPE" | 36 ContainerType = "CONTAINER_TYPE" |
29 Namespace = "NAMESPACE" | 37 Namespace = "NAMESPACE" |
30 StorageDir = "STORAGE_DIR" | 38 StorageDir = "STORAGE_DIR" |
31 StorageAddr = "STORAGE_ADDR" | 39 StorageAddr = "STORAGE_ADDR" |
32 AgentServiceName = "AGENT_SERVICE_NAME" | 40 AgentServiceName = "AGENT_SERVICE_NAME" |
33 MongoServiceName = "MONGO_SERVICE_NAME" | 41 MongoServiceName = "MONGO_SERVICE_NAME" |
34 BootstrapJobs = "BOOTSTRAP_JOBS" | |
35 ) | 42 ) |
36 | 43 |
37 // The Config interface is the sole way that the agent gets access to the | 44 // The Config interface is the sole way that the agent gets access to the |
38 // configuration information for the machine and unit agents. There should | 45 // configuration information for the machine and unit agents. There should |
39 // only be one instance of a config object for any given agent, and this | 46 // only be one instance of a config object for any given agent, and this |
40 // interface is passed between multiple go routines. The mutable methods are | 47 // interface is passed between multiple go routines. The mutable methods are |
41 // protected by a mutex, and it is expected that the caller doesn't modify any | 48 // protected by a mutex, and it is expected that the caller doesn't modify any |
42 // slice that may be returned. | 49 // slice that may be returned. |
43 // | 50 // |
44 // NOTE: should new mutating methods be added to this interface, consideration | 51 // NOTE: should new mutating methods be added to this interface, consideration |
45 // is needed around the synchronisation as a single instance is used in | 52 // is needed around the synchronisation as a single instance is used in |
46 // multiple go routines. | 53 // multiple go routines. |
47 type Config interface { | 54 type Config interface { |
48 // DataDir returns the data directory. Each agent has a subdirectory | 55 // DataDir returns the data directory. Each agent has a subdirectory |
49 // containing the configuration files. | 56 // containing the configuration files. |
50 DataDir() string | 57 DataDir() string |
51 | 58 |
59 // LogDir returns the log directory. All logs from all agents on | |
60 // the machine are written to this directory. | |
61 LogDir() string | |
62 | |
63 // Jobs returns a list of MachineJobs that need to run. | |
64 Jobs() []params.MachineJob | |
65 | |
52 // Tag returns the tag of the entity on whose behalf the state connectio n | 66 // Tag returns the tag of the entity on whose behalf the state connectio n |
53 // will be made. | 67 // will be made. |
54 Tag() string | 68 Tag() string |
55 | 69 |
56 // Dir returns the agent's directory. | 70 // Dir returns the agent's directory. |
57 Dir() string | 71 Dir() string |
58 | 72 |
59 // Nonce returns the nonce saved when the machine was provisioned | 73 // Nonce returns the nonce saved when the machine was provisioned |
60 // TODO: make this one of the key/value pairs. | 74 // TODO: make this one of the key/value pairs. |
61 Nonce() string | 75 Nonce() string |
(...skipping 61 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
123 // the values map. Retrieving and setting values here are protected by the | 137 // the values map. Retrieving and setting values here are protected by the |
124 // mutex. New mutating methods should also be synchronized using this mutex. | 138 // mutex. New mutating methods should also be synchronized using this mutex. |
125 var configMutex sync.Mutex | 139 var configMutex sync.Mutex |
126 | 140 |
127 type connectionDetails struct { | 141 type connectionDetails struct { |
128 addresses []string | 142 addresses []string |
129 password string | 143 password string |
130 } | 144 } |
131 | 145 |
132 type configInternal struct { | 146 type configInternal struct { |
147 configFilePath string | |
133 dataDir string | 148 dataDir string |
149 logDir string | |
134 tag string | 150 tag string |
151 nonce string | |
152 jobs []params.MachineJob | |
135 upgradedToVersion version.Number | 153 upgradedToVersion version.Number |
136 nonce string | |
137 caCert []byte | 154 caCert []byte |
138 stateDetails *connectionDetails | 155 stateDetails *connectionDetails |
139 apiDetails *connectionDetails | 156 apiDetails *connectionDetails |
140 oldPassword string | 157 oldPassword string |
141 stateServerCert []byte | 158 stateServerCert []byte |
142 stateServerKey []byte | 159 stateServerKey []byte |
143 apiPort int | 160 apiPort int |
144 values map[string]string | 161 values map[string]string |
145 } | 162 } |
146 | 163 |
147 type AgentConfigParams struct { | 164 type AgentConfigParams struct { |
148 DataDir string | 165 DataDir string |
166 LogDir string | |
167 Jobs []params.MachineJob | |
168 UpgradedToVersion version.Number | |
149 Tag string | 169 Tag string |
150 UpgradedToVersion version.Number | |
151 Password string | 170 Password string |
152 Nonce string | 171 Nonce string |
153 StateAddresses []string | 172 StateAddresses []string |
154 APIAddresses []string | 173 APIAddresses []string |
155 CACert []byte | 174 CACert []byte |
156 Values map[string]string | 175 Values map[string]string |
157 } | 176 } |
158 | 177 |
159 // NewAgentConfig returns a new config object suitable for use for a | 178 // NewAgentConfig returns a new config object suitable for use for a |
160 // machine or unit agent. | 179 // machine or unit agent. |
161 func NewAgentConfig(configParams AgentConfigParams) (Config, error) { | 180 func NewAgentConfig(configParams AgentConfigParams) (Config, error) { |
162 if configParams.DataDir == "" { | 181 if configParams.DataDir == "" { |
163 return nil, errgo.Trace(requiredError("data directory")) | 182 return nil, errgo.Trace(requiredError("data directory")) |
164 } | 183 } |
184 logDir := DefaultLogDir | |
185 if configParams.LogDir != "" { | |
186 logDir = configParams.LogDir | |
187 } | |
165 if configParams.Tag == "" { | 188 if configParams.Tag == "" { |
166 return nil, errgo.Trace(requiredError("entity tag")) | 189 return nil, errgo.Trace(requiredError("entity tag")) |
167 } | 190 } |
168 if configParams.UpgradedToVersion == version.Zero { | 191 if configParams.UpgradedToVersion == version.Zero { |
169 return nil, errgo.Trace(requiredError("upgradedToVersion")) | 192 return nil, errgo.Trace(requiredError("upgradedToVersion")) |
170 } | 193 } |
171 if configParams.Password == "" { | 194 if configParams.Password == "" { |
172 return nil, errgo.Trace(requiredError("password")) | 195 return nil, errgo.Trace(requiredError("password")) |
173 } | 196 } |
174 if configParams.CACert == nil { | 197 if configParams.CACert == nil { |
175 return nil, errgo.Trace(requiredError("CA certificate")) | 198 return nil, errgo.Trace(requiredError("CA certificate")) |
176 } | 199 } |
177 // Note that the password parts of the state and api information are | 200 // Note that the password parts of the state and api information are |
178 // blank. This is by design. | 201 // blank. This is by design. |
179 config := &configInternal{ | 202 config := &configInternal{ |
203 logDir: logDir, | |
180 dataDir: configParams.DataDir, | 204 dataDir: configParams.DataDir, |
205 jobs: configParams.Jobs, | |
206 upgradedToVersion: configParams.UpgradedToVersion, | |
181 tag: configParams.Tag, | 207 tag: configParams.Tag, |
182 upgradedToVersion: configParams.UpgradedToVersion, | |
183 nonce: configParams.Nonce, | 208 nonce: configParams.Nonce, |
184 caCert: configParams.CACert, | 209 caCert: configParams.CACert, |
185 oldPassword: configParams.Password, | 210 oldPassword: configParams.Password, |
186 values: configParams.Values, | 211 values: configParams.Values, |
187 } | 212 } |
188 if len(configParams.StateAddresses) > 0 { | 213 if len(configParams.StateAddresses) > 0 { |
189 config.stateDetails = &connectionDetails{ | 214 config.stateDetails = &connectionDetails{ |
190 addresses: configParams.StateAddresses, | 215 addresses: configParams.StateAddresses, |
191 } | 216 } |
192 } | 217 } |
193 if len(configParams.APIAddresses) > 0 { | 218 if len(configParams.APIAddresses) > 0 { |
194 config.apiDetails = &connectionDetails{ | 219 config.apiDetails = &connectionDetails{ |
195 addresses: configParams.APIAddresses, | 220 addresses: configParams.APIAddresses, |
196 } | 221 } |
197 } | 222 } |
198 if err := config.check(); err != nil { | 223 if err := config.check(); err != nil { |
199 return nil, err | 224 return nil, err |
200 } | 225 } |
201 if config.values == nil { | 226 if config.values == nil { |
202 config.values = make(map[string]string) | 227 config.values = make(map[string]string) |
203 } | 228 } |
229 config.configFilePath = ConfigPath(config.dataDir, config.tag) | |
204 return config, nil | 230 return config, nil |
205 } | 231 } |
206 | 232 |
207 type StateMachineConfigParams struct { | 233 type StateMachineConfigParams struct { |
208 AgentConfigParams | 234 AgentConfigParams |
209 StateServerCert []byte | 235 StateServerCert []byte |
210 StateServerKey []byte | 236 StateServerKey []byte |
211 StatePort int | 237 StatePort int |
212 APIPort int | 238 APIPort int |
213 } | 239 } |
(...skipping 13 matching lines...) Expand all Loading... | |
227 } | 253 } |
228 config := config0.(*configInternal) | 254 config := config0.(*configInternal) |
229 config.stateServerCert = configParams.StateServerCert | 255 config.stateServerCert = configParams.StateServerCert |
230 config.stateServerKey = configParams.StateServerKey | 256 config.stateServerKey = configParams.StateServerKey |
231 config.apiPort = configParams.APIPort | 257 config.apiPort = configParams.APIPort |
232 return config, nil | 258 return config, nil |
233 } | 259 } |
234 | 260 |
235 // Dir returns the agent-specific data directory. | 261 // Dir returns the agent-specific data directory. |
236 func Dir(dataDir, agentName string) string { | 262 func Dir(dataDir, agentName string) string { |
237 » return path.Join(dataDir, "agents", agentName) | 263 » return filepath.Join(dataDir, "agents", agentName) |
238 } | 264 } |
239 | 265 |
240 // ReadConf reads configuration data for the given | 266 // ConfigPath returns the full path to the agent config file. |
241 // entity from the given data directory. | 267 // NOTE: Delete this once all agents accept --config instead |
242 func ReadConf(dataDir, tag string) (Config, error) { | 268 // of --data-dir - it won't be needed anymore. |
269 func ConfigPath(dataDir, agentName string) string { | |
270 » return filepath.Join(Dir(dataDir, agentName), agentConfigFilename) | |
271 } | |
272 | |
273 // ReadConf reads configuration data from the given location. | |
274 func ReadConf(configFilePath string) (Config, error) { | |
243 // Even though the ReadConf is done at the start of the agent loading, a nd | 275 // Even though the ReadConf is done at the start of the agent loading, a nd |
244 // that this should not be called more than once by an agent, I feel tha t | 276 // that this should not be called more than once by an agent, I feel tha t |
245 // not locking the mutex that is used to protect writes is wrong. | 277 // not locking the mutex that is used to protect writes is wrong. |
246 configMutex.Lock() | 278 configMutex.Lock() |
247 defer configMutex.Unlock() | 279 defer configMutex.Unlock() |
248 » dir := Dir(dataDir, tag) | 280 » var ( |
249 » format, err := readFormat(dir) | 281 » » format formatter |
282 » » config *configInternal | |
283 » » err error | |
rog
2014/03/06 15:14:07
d
dimitern
2014/03/06 16:10:33
Done.
| |
284 » ) | |
285 » configData, err := ioutil.ReadFile(configFilePath) | |
286 » if err != nil { | |
287 » » return nil, fmt.Errorf("cannot read agent config %q: %v", config FilePath, err) | |
288 » } | |
289 | |
290 » // Try to read the legacy format file. | |
291 » dir := filepath.Dir(configFilePath) | |
292 » legacyFormatPath := filepath.Join(dir, legacyFormatFilename) | |
293 » formatBytes, err := ioutil.ReadFile(legacyFormatPath) | |
rog
2014/03/06 15:14:07
if err != nil && !os.IsNotExist(err) {
return
dimitern
2014/03/06 16:10:33
Done.
| |
294 » formatData := string(formatBytes) | |
295 » if err == nil { | |
296 » » // It exists, so unmarshal with a legacy formatter. | |
297 » » // Drop the format prefix to leave the version only. | |
298 » » if !strings.HasPrefix(formatData, legacyFormatPrefix) { | |
299 » » » return nil, fmt.Errorf("malformed agent config format: % q", formatData) | |
rog
2014/03/06 15:14:07
s/: //
dimitern
2014/03/06 16:10:33
Done.
| |
300 » » } | |
301 » » format, err = getFormatter(strings.TrimPrefix(formatData, legacy FormatPrefix)) | |
302 » » if err != nil { | |
303 » » » return nil, err | |
304 » » } | |
305 » » config, err = format.unmarshal(configData) | |
306 » } else { | |
307 » » // Does not exist, just parse the data. | |
308 » » format, config, err = parseConfigData(configData) | |
309 » } | |
250 if err != nil { | 310 if err != nil { |
251 return nil, err | 311 return nil, err |
252 } | 312 } |
253 » logger.Debugf("Reading agent config, format: %s", format) | 313 » logger.Debugf("read agent config format %q", format.version()) |
rog
2014/03/06 15:14:07
s/config/config, /
?
dimitern
2014/03/06 16:10:33
Done.
| |
254 » formatter, err := newFormatter(format) | |
255 » if err != nil { | |
256 » » return nil, err | |
257 » } | |
258 » config, err := formatter.read(dir) | |
259 » if err != nil { | |
260 » » return nil, err | |
261 » } | |
262 » config.dataDir = dataDir | |
263 » if err := config.check(); err != nil { | |
264 » » return nil, err | |
265 » } | |
266 | |
267 if format != currentFormat { | 314 if format != currentFormat { |
268 » » // Migrate the config to the new format. | 315 » » // Migrate from a legacy format to the new one. |
269 » » currentFormatter.migrate(config) | 316 » » data, err := currentFormat.marshal(config) |
270 » » // Write the content out in the new format. | 317 » » if err != nil { |
271 » » if err := currentFormatter.write(config); err != nil { | 318 » » » return nil, fmt.Errorf("cannot marshal agent config: %v" , err) |
272 » » » logger.Errorf("cannot write the agent config in format % s: %v", currentFormat, err) | 319 » » } |
273 » » » return nil, err | 320 » » err = utils.AtomicWriteFile(configFilePath, data, 0600) |
rog
2014/03/06 15:14:07
This doesn't look quite right - we're not writing
dimitern
2014/03/06 16:10:33
Done.
| |
321 » » if err != nil { | |
322 » » » return nil, fmt.Errorf("cannot upgrade agent config %s t o %s: %v", format.version(), currentFormat.version(), err) | |
323 » » } | |
324 » » logger.Debugf("migrated agent config from %s to %s", format.vers ion(), currentFormat.version()) | |
325 » » err = os.Remove(legacyFormatPath) | |
326 » » if err != nil && !os.IsNotExist(err) { | |
327 » » » return nil, fmt.Errorf("cannot remove legacy format file %q: %v", legacyFormatPath, err) | |
274 } | 328 } |
275 } | 329 } |
276 | 330 » config.configFilePath = configFilePath |
277 return config, nil | 331 return config, nil |
278 } | 332 } |
279 | 333 |
280 func requiredError(what string) error { | 334 func requiredError(what string) error { |
281 return fmt.Errorf("%s not found in configuration", what) | 335 return fmt.Errorf("%s not found in configuration", what) |
282 } | 336 } |
283 | 337 |
284 func (c *configInternal) File(name string) string { | 338 func (c *configInternal) File(name string) string { |
285 return path.Join(c.Dir(), name) | 339 return path.Join(c.Dir(), name) |
286 } | 340 } |
287 | 341 |
288 func (c *configInternal) DataDir() string { | 342 func (c *configInternal) DataDir() string { |
289 return c.dataDir | 343 return c.dataDir |
290 } | 344 } |
291 | 345 |
346 func (c *configInternal) LogDir() string { | |
347 return c.logDir | |
348 } | |
349 | |
350 func (c *configInternal) Jobs() []params.MachineJob { | |
351 return c.jobs | |
352 } | |
353 | |
292 func (c *configInternal) Nonce() string { | 354 func (c *configInternal) Nonce() string { |
293 return c.nonce | 355 return c.nonce |
294 } | 356 } |
295 | 357 |
296 func (c *configInternal) UpgradedToVersion() version.Number { | 358 func (c *configInternal) UpgradedToVersion() version.Number { |
297 return c.upgradedToVersion | 359 return c.upgradedToVersion |
298 } | 360 } |
299 | 361 |
300 func (c *configInternal) CACert() []byte { | 362 func (c *configInternal) CACert() []byte { |
301 // Give the caller their own copy of the cert to avoid any possibility o f | 363 // Give the caller their own copy of the cert to avoid any possibility o f |
(...skipping 90 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
392 other.apiDetails = &apiDetails | 454 other.apiDetails = &apiDetails |
393 } | 455 } |
394 logger.Debugf("writing configuration file") | 456 logger.Debugf("writing configuration file") |
395 if err := other.Write(); err != nil { | 457 if err := other.Write(); err != nil { |
396 return "", err | 458 return "", err |
397 } | 459 } |
398 *c = other | 460 *c = other |
399 return newPassword, nil | 461 return newPassword, nil |
400 } | 462 } |
401 | 463 |
464 func (c *configInternal) fileContents() ([]byte, error) { | |
465 data, err := currentFormat.marshal(c) | |
466 if err != nil { | |
467 return nil, err | |
468 } | |
469 var buf bytes.Buffer | |
470 fmt.Fprintf(&buf, "%s%s\n", formatPrefix, currentFormat.version()) | |
471 buf.Write(data) | |
472 return buf.Bytes(), nil | |
473 } | |
474 | |
402 func (c *configInternal) Write() error { | 475 func (c *configInternal) Write() error { |
403 // Lock is taken prior to generating any content to write. | 476 // Lock is taken prior to generating any content to write. |
404 configMutex.Lock() | 477 configMutex.Lock() |
405 defer configMutex.Unlock() | 478 defer configMutex.Unlock() |
406 » return currentFormatter.write(c) | 479 » data, err := c.fileContents() |
480 » if err != nil { | |
481 » » return err | |
482 » } | |
483 » // Make sure the config dir gets created. | |
484 » configDir := filepath.Dir(c.configFilePath) | |
485 » if err := os.MkdirAll(configDir, 0755); err != nil { | |
486 » » return fmt.Errorf("cannot create agent config dir %q: %v", confi gDir, err) | |
487 » } | |
488 » return utils.AtomicWriteFile(c.configFilePath, data, 0600) | |
407 } | 489 } |
408 | 490 |
409 func (c *configInternal) WriteUpgradedToVersion(newVersion version.Number) error { | 491 func (c *configInternal) WriteUpgradedToVersion(newVersion version.Number) error { |
410 originalVersion := c.upgradedToVersion | 492 originalVersion := c.upgradedToVersion |
411 c.upgradedToVersion = newVersion | 493 c.upgradedToVersion = newVersion |
412 err := c.Write() | 494 err := c.Write() |
413 if err != nil { | 495 if err != nil { |
414 // We don't want to retain the new version if there's been an er ror writing the file. | 496 // We don't want to retain the new version if there's been an er ror writing the file. |
415 c.upgradedToVersion = originalVersion | 497 c.upgradedToVersion = originalVersion |
416 } | 498 } |
417 return err | 499 return err |
418 } | 500 } |
419 | 501 |
420 func (c *configInternal) WriteCommands() ([]string, error) { | 502 func (c *configInternal) WriteCommands() ([]string, error) { |
421 » return currentFormatter.writeCommands(c) | 503 » data, err := c.fileContents() |
504 » if err != nil { | |
505 » » return nil, err | |
506 » } | |
507 » commands := []string{"mkdir -p " + utils.ShQuote(c.Dir())} | |
508 » commands = append(commands, writeFileCommands(c.File(agentConfigFilename ), data, 0600)...) | |
509 » return commands, nil | |
422 } | 510 } |
423 | 511 |
424 func (c *configInternal) OpenAPI(dialOpts api.DialOpts) (st *api.State, newPassw ord string, err error) { | 512 func (c *configInternal) OpenAPI(dialOpts api.DialOpts) (st *api.State, newPassw ord string, err error) { |
425 info := api.Info{ | 513 info := api.Info{ |
426 Addrs: c.apiDetails.addresses, | 514 Addrs: c.apiDetails.addresses, |
427 Password: c.apiDetails.password, | 515 Password: c.apiDetails.password, |
428 CACert: c.caCert, | 516 CACert: c.caCert, |
429 Tag: c.tag, | 517 Tag: c.tag, |
430 Nonce: c.nonce, | 518 Nonce: c.nonce, |
431 } | 519 } |
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
472 } | 560 } |
473 // TODO(rog) remove this fallback behaviour when | 561 // TODO(rog) remove this fallback behaviour when |
474 // all initial connections are via the API. | 562 // all initial connections are via the API. |
475 if !errors.IsUnauthorizedError(err) { | 563 if !errors.IsUnauthorizedError(err) { |
476 return nil, err | 564 return nil, err |
477 } | 565 } |
478 } | 566 } |
479 info.Password = c.oldPassword | 567 info.Password = c.oldPassword |
480 return state.Open(&info, state.DefaultDialOpts(), policy) | 568 return state.Open(&info, state.DefaultDialOpts(), policy) |
481 } | 569 } |
OLD | NEW |