Rietveld Code Review Tool
Help | Bug tracker | Discussion group | Source code | Sign in
(10)

Side by Side Diff: agent/agent.go

Issue 70010045: Agent config format 1.18 (Closed)
Patch Set: Agent config format 1.18 Created 11 years, 1 month ago
Left:
Right:
Use n/p to move between diff chunks; N/P to move between comments. Please Sign in to add in-line comments.
Jump to:
View unified diff | Download patch
OLDNEW
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
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
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
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
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 }
OLDNEW
« no previous file with comments | « [revision details] ('k') | agent/agent_test.go » ('j') | agent/format.go » ('J')

Powered by Google App Engine
RSS Feeds Recent Issues | This issue
This is Rietveld f62528b