OLD | NEW |
1 // Copyright 2012, 2013 Canonical Ltd. | 1 // Copyright 2012, 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 // Stub provider for OpenStack, using goose will be implemented here | 4 // Stub provider for OpenStack, using goose will be implemented here |
5 | 5 |
6 package openstack | 6 package openstack |
7 | 7 |
8 import ( | 8 import ( |
9 "errors" | 9 "errors" |
10 "fmt" | 10 "fmt" |
(...skipping 55 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
66 ## https://juju.ubuntu.com/docs/config-openstack.html | 66 ## https://juju.ubuntu.com/docs/config-openstack.html |
67 openstack: | 67 openstack: |
68 type: openstack | 68 type: openstack |
69 # Specifies whether the use of a floating IP address is required to give the n
odes | 69 # Specifies whether the use of a floating IP address is required to give the n
odes |
70 # a public IP address. Some installations assign public IP addresses by defaul
t without | 70 # a public IP address. Some installations assign public IP addresses by defaul
t without |
71 # requiring a floating IP address. | 71 # requiring a floating IP address. |
72 # use-floating-ip: false | 72 # use-floating-ip: false |
73 admin-secret: {{rand}} | 73 admin-secret: {{rand}} |
74 # Globally unique swift bucket name | 74 # Globally unique swift bucket name |
75 control-bucket: juju-{{rand}} | 75 control-bucket: juju-{{rand}} |
| 76 # If set, tools-url specifies from where tools are fetched. |
| 77 # tools-url: https://you-tools-url |
76 # Usually set via the env variable OS_AUTH_URL, but can be specified here | 78 # Usually set via the env variable OS_AUTH_URL, but can be specified here |
77 # auth-url: https://yourkeystoneurl:443/v2.0/ | 79 # auth-url: https://yourkeystoneurl:443/v2.0/ |
78 # override if your workstation is running a different series to which you are
deploying | 80 # override if your workstation is running a different series to which you are
deploying |
79 # default-series: precise | 81 # default-series: precise |
80 # The following are used for userpass authentication (the default) | 82 # The following are used for userpass authentication (the default) |
81 auth-mode: userpass | 83 auth-mode: userpass |
82 # Usually set via the env variable OS_USERNAME, but can be specified here | 84 # Usually set via the env variable OS_USERNAME, but can be specified here |
83 # username: <your username> | 85 # username: <your username> |
84 # Usually set via the env variable OS_PASSWORD, but can be specified here | 86 # Usually set via the env variable OS_PASSWORD, but can be specified here |
85 # password: <secret> | 87 # password: <secret> |
86 # Usually set via the env variable OS_TENANT_NAME, but can be specified here | 88 # Usually set via the env variable OS_TENANT_NAME, but can be specified here |
87 # tenant-name: <your tenant name> | 89 # tenant-name: <your tenant name> |
88 # Usually set via the env variable OS_REGION_NAME, but can be specified here | 90 # Usually set via the env variable OS_REGION_NAME, but can be specified here |
89 # region: <your region> | 91 # region: <your region> |
90 | 92 |
91 ## https://juju.ubuntu.com/docs/config-hpcloud.html | 93 ## https://juju.ubuntu.com/docs/config-hpcloud.html |
92 hpcloud: | 94 hpcloud: |
93 type: openstack | 95 type: openstack |
94 # Specifies whether the use of a floating IP address is required to give the n
odes | 96 # Specifies whether the use of a floating IP address is required to give the n
odes |
95 # a public IP address. Some installations assign public IP addresses by defaul
t without | 97 # a public IP address. Some installations assign public IP addresses by defaul
t without |
96 # requiring a floating IP address. | 98 # requiring a floating IP address. |
97 use-floating-ip: false | 99 use-floating-ip: false |
98 admin-secret: {{rand}} | 100 admin-secret: {{rand}} |
99 # Globally unique swift bucket name | 101 # Globally unique swift bucket name |
100 control-bucket: juju-{{rand}} | 102 control-bucket: juju-{{rand}} |
101 # Not required if env variable OS_AUTH_URL is set | 103 # Not required if env variable OS_AUTH_URL is set |
102 auth-url: https://yourkeystoneurl:35357/v2.0/ | 104 auth-url: https://yourkeystoneurl:35357/v2.0/ |
103 # URL denoting a public container holding the juju tools. | 105 # URL denoting a location holding the juju tools. |
104 public-bucket-url: https://region-a.geo-1.objects.hpcloudsvc.com/v1/6050252975
3910 | 106 tools-url: https://region-a.geo-1.objects.hpcloudsvc.com:443/v1/60502529753910
/juju-dist/tools |
105 # override if your workstation is running a different series to which you are
deploying | 107 # override if your workstation is running a different series to which you are
deploying |
106 # default-series: precise | 108 # default-series: precise |
107 # The following are used for userpass authentication (the default) | 109 # The following are used for userpass authentication (the default) |
108 auth-mode: userpass | 110 auth-mode: userpass |
109 # Usually set via the env variable OS_USERNAME, but can be specified here | 111 # Usually set via the env variable OS_USERNAME, but can be specified here |
110 # username: <your username> | 112 # username: <your username> |
111 # Usually set via the env variable OS_PASSWORD, but can be specified here | 113 # Usually set via the env variable OS_PASSWORD, but can be specified here |
112 # password: <secret> | 114 # password: <secret> |
113 # Usually set via the env variable OS_TENANT_NAME, but can be specified here | 115 # Usually set via the env variable OS_TENANT_NAME, but can be specified here |
114 # tenant-name: <your tenant name> | 116 # tenant-name: <your tenant name> |
(...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
215 } | 217 } |
216 if err != nil { | 218 if err != nil { |
217 return nil, fmt.Errorf("cannot get %q: %v", uri, err) | 219 return nil, fmt.Errorf("cannot get %q: %v", uri, err) |
218 } | 220 } |
219 return | 221 return |
220 } | 222 } |
221 | 223 |
222 type environ struct { | 224 type environ struct { |
223 name string | 225 name string |
224 | 226 |
225 » ecfgMutex sync.Mutex | 227 » ecfgMutex sync.Mutex |
226 » publicStorageMutex sync.Mutex | 228 » imageBaseMutex sync.Mutex |
227 » imageBaseMutex sync.Mutex | 229 » toolsBaseMutex sync.Mutex |
228 » toolsBaseMutex sync.Mutex | 230 » ecfgUnlocked *environConfig |
229 » ecfgUnlocked *environConfig | 231 » client client.AuthenticatingClient |
230 » client client.AuthenticatingClient | 232 » novaUnlocked *nova.Client |
231 » novaUnlocked *nova.Client | 233 » storageUnlocked storage.Storage |
232 » storageUnlocked storage.Storage | |
233 » publicStorageUnlocked storage.StorageReader // optional. | |
234 // An ordered list of sources in which to find the simplestreams index f
iles used to | 234 // An ordered list of sources in which to find the simplestreams index f
iles used to |
235 // look up image ids. | 235 // look up image ids. |
236 imageSources []simplestreams.DataSource | 236 imageSources []simplestreams.DataSource |
237 // An ordered list of paths in which to find the simplestreams index fil
es used to | 237 // An ordered list of paths in which to find the simplestreams index fil
es used to |
238 // look up tools ids. | 238 // look up tools ids. |
239 toolsSources []simplestreams.DataSource | 239 toolsSources []simplestreams.DataSource |
240 } | 240 } |
241 | 241 |
242 var _ environs.Environ = (*environ)(nil) | 242 var _ environs.Environ = (*environ)(nil) |
243 var _ imagemetadata.SupportsCustomSources = (*environ)(nil) | 243 var _ imagemetadata.SupportsCustomSources = (*environ)(nil) |
(...skipping 169 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
413 return e.name | 413 return e.name |
414 } | 414 } |
415 | 415 |
416 func (e *environ) Storage() storage.Storage { | 416 func (e *environ) Storage() storage.Storage { |
417 e.ecfgMutex.Lock() | 417 e.ecfgMutex.Lock() |
418 stor := e.storageUnlocked | 418 stor := e.storageUnlocked |
419 e.ecfgMutex.Unlock() | 419 e.ecfgMutex.Unlock() |
420 return stor | 420 return stor |
421 } | 421 } |
422 | 422 |
423 // publicBucketURL gets the public bucket URL, either from env or keystone catal
og. | |
424 func (e *environ) publicBucketURL() string { | |
425 ecfg := e.ecfg() | |
426 publicBucketURL := ecfg.publicBucketURL() | |
427 if publicBucketURL == "" { | |
428 // No public bucket in env, so authenticate and look in keystone
catalog. | |
429 if !e.client.IsAuthenticated() { | |
430 e.client.Authenticate() | |
431 } | |
432 var err error | |
433 publicBucketURL, err = e.client.MakeServiceURL("juju-tools", nil
) | |
434 if err != nil { | |
435 return "" | |
436 } | |
437 } | |
438 return publicBucketURL | |
439 } | |
440 | |
441 func (e *environ) PublicStorage() storage.StorageReader { | 423 func (e *environ) PublicStorage() storage.StorageReader { |
442 » e.publicStorageMutex.Lock() | 424 » // No public storage required. Tools are fetched from tools-url. |
443 » defer e.publicStorageMutex.Unlock() | 425 » return environs.EmptyStorage |
444 » ecfg := e.ecfg() | |
445 » // If public storage has already been determined, return that instance. | |
446 » publicStorage := e.publicStorageUnlocked | |
447 » if publicStorage == nil && ecfg.publicBucket() == "" { | |
448 » » // If there is no public bucket name, then there can be no publi
c storage. | |
449 » » e.publicStorageUnlocked = environs.EmptyStorage | |
450 » » publicStorage = e.publicStorageUnlocked | |
451 » } | |
452 » if publicStorage != nil { | |
453 » » return publicStorage | |
454 » } | |
455 » // If there is a public bucket URL defined, set up a public storage clie
nt referencing that URL, | |
456 » // otherwise create a new public bucket using the user's credentials on
the authenticated client. | |
457 » publicBucketURL := e.publicBucketURL() | |
458 » if publicBucketURL == "" { | |
459 » » e.publicStorageUnlocked = &openstackstorage{ | |
460 » » » containerName: ecfg.publicBucket(), | |
461 » » » // this is possibly just a hack - if the ACL is swift.Pr
ivate, | |
462 » » » // the machine won't be able to get the tools (401 error
) | |
463 » » » containerACL: swift.PublicRead, | |
464 » » » swift: swift.New(e.client)} | |
465 » } else { | |
466 » » pc := client.NewPublicClient(publicBucketURL, nil) | |
467 » » e.publicStorageUnlocked = &openstackstorage{ | |
468 » » » containerName: ecfg.publicBucket(), | |
469 » » » containerACL: swift.PublicRead, | |
470 » » » swift: swift.New(pc)} | |
471 » } | |
472 » publicStorage = e.publicStorageUnlocked | |
473 » return publicStorage | |
474 } | 426 } |
475 | 427 |
476 func (e *environ) Bootstrap(cons constraints.Value, possibleTools tools.List, ma
chineID string) error { | 428 func (e *environ) Bootstrap(cons constraints.Value, possibleTools tools.List, ma
chineID string) error { |
477 // The client's authentication may have been reset when finding tools if
the agent-version | 429 // The client's authentication may have been reset when finding tools if
the agent-version |
478 // attribute was updated so we need to re-authenticate. This will be a n
o-op if already authenticated. | 430 // attribute was updated so we need to re-authenticate. This will be a n
o-op if already authenticated. |
479 // An authenticated client is needed for the URL() call below. | 431 // An authenticated client is needed for the URL() call below. |
480 err := e.client.Authenticate() | 432 err := e.client.Authenticate() |
481 if err != nil { | 433 if err != nil { |
482 return err | 434 return err |
483 } | 435 } |
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
534 // create new control storage instance, existing instances continue | 486 // create new control storage instance, existing instances continue |
535 // to reference their existing configuration. | 487 // to reference their existing configuration. |
536 // public storage instance creation is deferred until needed since authe
nticated | 488 // public storage instance creation is deferred until needed since authe
nticated |
537 // access to the identity service is required so that any juju-tools end
point can be used. | 489 // access to the identity service is required so that any juju-tools end
point can be used. |
538 e.storageUnlocked = &openstackstorage{ | 490 e.storageUnlocked = &openstackstorage{ |
539 containerName: ecfg.controlBucket(), | 491 containerName: ecfg.controlBucket(), |
540 // this is possibly just a hack - if the ACL is swift.Private, | 492 // this is possibly just a hack - if the ACL is swift.Private, |
541 // the machine won't be able to get the tools (401 error) | 493 // the machine won't be able to get the tools (401 error) |
542 containerACL: swift.PublicRead, | 494 containerACL: swift.PublicRead, |
543 swift: swift.New(e.client)} | 495 swift: swift.New(e.client)} |
544 e.publicStorageUnlocked = nil | |
545 return nil | 496 return nil |
546 } | 497 } |
547 | 498 |
548 // GetImageSources returns a list of sources which are used to search for simple
streams image metadata. | 499 // GetImageSources returns a list of sources which are used to search for simple
streams image metadata. |
549 func (e *environ) GetImageSources() ([]simplestreams.DataSource, error) { | 500 func (e *environ) GetImageSources() ([]simplestreams.DataSource, error) { |
550 e.imageBaseMutex.Lock() | 501 e.imageBaseMutex.Lock() |
551 defer e.imageBaseMutex.Unlock() | 502 defer e.imageBaseMutex.Unlock() |
552 | 503 |
553 if e.imageSources != nil { | 504 if e.imageSources != nil { |
554 return e.imageSources, nil | 505 return e.imageSources, nil |
555 } | 506 } |
556 if !e.client.IsAuthenticated() { | 507 if !e.client.IsAuthenticated() { |
557 err := e.client.Authenticate() | 508 err := e.client.Authenticate() |
558 if err != nil { | 509 if err != nil { |
559 return nil, err | 510 return nil, err |
560 } | 511 } |
561 } | 512 } |
562 // Add the simplestreams source off the control bucket. | 513 // Add the simplestreams source off the control bucket. |
563 e.imageSources = append(e.imageSources, storage.NewStorageSimpleStreamsD
ataSource(e.Storage(), "")) | 514 e.imageSources = append(e.imageSources, storage.NewStorageSimpleStreamsD
ataSource(e.Storage(), "")) |
564 // Add the simplestreams source off the public bucket. | |
565 e.imageSources = append(e.imageSources, storage.NewStorageSimpleStreamsD
ataSource(e.PublicStorage(), "")) | |
566 // Add the simplestreams base URL from keystone if it is defined. | 515 // Add the simplestreams base URL from keystone if it is defined. |
567 productStreamsURL, err := e.client.MakeServiceURL("product-streams", nil
) | 516 productStreamsURL, err := e.client.MakeServiceURL("product-streams", nil
) |
568 if err == nil { | 517 if err == nil { |
569 e.imageSources = append(e.imageSources, simplestreams.NewURLData
Source(productStreamsURL)) | 518 e.imageSources = append(e.imageSources, simplestreams.NewURLData
Source(productStreamsURL)) |
570 } | 519 } |
571 return e.imageSources, nil | 520 return e.imageSources, nil |
572 } | 521 } |
573 | 522 |
574 // GetToolsSources returns a list of sources which are used to search for simple
streams tools metadata. | 523 // GetToolsSources returns a list of sources which are used to search for simple
streams tools metadata. |
575 func (e *environ) GetToolsSources() ([]simplestreams.DataSource, error) { | 524 func (e *environ) GetToolsSources() ([]simplestreams.DataSource, error) { |
(...skipping 542 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1118 }, nil | 1067 }, nil |
1119 } | 1068 } |
1120 | 1069 |
1121 // Region is specified in the HasRegion interface. | 1070 // Region is specified in the HasRegion interface. |
1122 func (e *environ) Region() (simplestreams.CloudSpec, error) { | 1071 func (e *environ) Region() (simplestreams.CloudSpec, error) { |
1123 return simplestreams.CloudSpec{ | 1072 return simplestreams.CloudSpec{ |
1124 Region: e.ecfg().region(), | 1073 Region: e.ecfg().region(), |
1125 Endpoint: e.ecfg().authURL(), | 1074 Endpoint: e.ecfg().authURL(), |
1126 }, nil | 1075 }, nil |
1127 } | 1076 } |
OLD | NEW |