blob: d8c4a1913d5b038d8451df0f419c80a41eeefd6a [file] [log] [blame]
Jamie Hannaford2aaf1a62014-10-16 12:55:50 +02001package rackspace
2
3import (
4 "fmt"
5
6 "github.com/rackspace/gophercloud"
7 os "github.com/rackspace/gophercloud/openstack"
8 "github.com/rackspace/gophercloud/openstack/utils"
9 tokens2 "github.com/rackspace/gophercloud/rackspace/identity/v2/tokens"
10)
11
12const (
13 // RackspaceUSIdentity is an identity endpoint located in the United States.
14 RackspaceUSIdentity = "https://identity.api.rackspacecloud.com/v2.0/"
15
16 // RackspaceUKIdentity is an identity endpoint located in the UK.
17 RackspaceUKIdentity = "https://lon.identity.api.rackspacecloud.com/v2.0/"
18)
19
20const (
21 v20 = "v2.0"
22)
23
24// NewClient creates a client that's prepared to communicate with the Rackspace API, but is not
25// yet authenticated. Most users will probably prefer using the AuthenticatedClient function
26// instead.
27//
28// Provide the base URL of the identity endpoint you wish to authenticate against as "endpoint".
29// Often, this will be either RackspaceUSIdentity or RackspaceUKIdentity.
30func NewClient(endpoint string) (*gophercloud.ProviderClient, error) {
31 if endpoint == "" {
32 return os.NewClient(RackspaceUSIdentity)
33 }
34 return os.NewClient(endpoint)
35}
36
37// AuthenticatedClient logs in to Rackspace with the provided credentials and constructs a
38// ProviderClient that's ready to operate.
39//
40// If the provided AuthOptions does not specify an explicit IdentityEndpoint, it will default to
41// the canonical, production Rackspace US identity endpoint.
42func AuthenticatedClient(options gophercloud.AuthOptions) (*gophercloud.ProviderClient, error) {
43 client, err := NewClient(options.IdentityEndpoint)
44 if err != nil {
45 return nil, err
46 }
47
48 err = Authenticate(client, options)
49 if err != nil {
50 return nil, err
51 }
52 return client, nil
53}
54
55// Authenticate or re-authenticate against the most recent identity service supported at the
56// provided endpoint.
57func Authenticate(client *gophercloud.ProviderClient, options gophercloud.AuthOptions) error {
58 versions := []*utils.Version{
59 &utils.Version{ID: v20, Priority: 20, Suffix: "/v2.0/"},
60 }
61
62 chosen, endpoint, err := utils.ChooseVersion(client.IdentityBase, client.IdentityEndpoint, versions)
63 if err != nil {
64 return err
65 }
66
67 switch chosen.ID {
68 case v20:
69 return v2auth(client, endpoint, options)
70 default:
71 // The switch statement must be out of date from the versions list.
72 return fmt.Errorf("Unrecognized identity version: %s", chosen.ID)
73 }
74}
75
76// AuthenticateV2 explicitly authenticates with v2 of the identity service.
77func AuthenticateV2(client *gophercloud.ProviderClient, options gophercloud.AuthOptions) error {
78 return v2auth(client, "", options)
79}
80
81func v2auth(client *gophercloud.ProviderClient, endpoint string, options gophercloud.AuthOptions) error {
82 v2Client := NewIdentityV2(client)
83 if endpoint != "" {
84 v2Client.Endpoint = endpoint
85 }
86
87 result := tokens2.Create(v2Client, tokens2.WrapOptions(options))
88
89 token, err := result.ExtractToken()
90 if err != nil {
91 return err
92 }
93
94 catalog, err := result.ExtractServiceCatalog()
95 if err != nil {
96 return err
97 }
98
99 client.TokenID = token.ID
100 client.EndpointLocator = func(opts gophercloud.EndpointOpts) (string, error) {
101 return os.V2EndpointURL(catalog, opts)
102 }
103
104 return nil
105}
106
107// NewIdentityV2 creates a ServiceClient that may be used to access the v2 identity service.
108func NewIdentityV2(client *gophercloud.ProviderClient) *gophercloud.ServiceClient {
109 v2Endpoint := client.IdentityBase + "v2.0/"
110
111 return &gophercloud.ServiceClient{
112 Provider: client,
113 Endpoint: v2Endpoint,
114 }
115}