| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 1 | package tokens | 
|  | 2 |  | 
|  | 3 | import ( | 
| Ash Wilson | 2491b4c | 2015-02-12 16:13:39 -0500 | [diff] [blame] | 4 | "net/http" | 
|  | 5 |  | 
| Jon Perritt | 27249f4 | 2016-02-18 10:35:59 -0600 | [diff] [blame] | 6 | "github.com/gophercloud/gophercloud" | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 7 | ) | 
|  | 8 |  | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 9 | // Scope allows a created token to be limited to a specific domain or project. | 
|  | 10 | type Scope struct { | 
|  | 11 | ProjectID   string | 
|  | 12 | ProjectName string | 
|  | 13 | DomainID    string | 
|  | 14 | DomainName  string | 
|  | 15 | } | 
|  | 16 |  | 
| Ash Wilson | 6425a41 | 2014-08-29 12:30:35 -0400 | [diff] [blame] | 17 | func subjectTokenHeaders(c *gophercloud.ServiceClient, subjectToken string) map[string]string { | 
| Krzysztof Kwapisiewicz | baaaf3e | 2016-02-03 15:18:16 +0100 | [diff] [blame] | 18 | return map[string]string{ | 
|  | 19 | "X-Subject-Token": subjectToken, | 
|  | 20 | } | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 21 | } | 
|  | 22 |  | 
| Ash Wilson | e555086 | 2014-08-28 15:37:09 -0400 | [diff] [blame] | 23 | // Create authenticates and either generates a new token, or changes the Scope of an existing token. | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 24 | func Create(c *gophercloud.ServiceClient, options gophercloud.AuthOptions, scope *Scope) CreateResult { | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 25 | type domainReq struct { | 
|  | 26 | ID   *string `json:"id,omitempty"` | 
| Ash Wilson | 417d922 | 2014-08-29 07:58:35 -0400 | [diff] [blame] | 27 | Name *string `json:"name,omitempty"` | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 28 | } | 
|  | 29 |  | 
|  | 30 | type projectReq struct { | 
|  | 31 | Domain *domainReq `json:"domain,omitempty"` | 
|  | 32 | Name   *string    `json:"name,omitempty"` | 
|  | 33 | ID     *string    `json:"id,omitempty"` | 
|  | 34 | } | 
|  | 35 |  | 
|  | 36 | type userReq struct { | 
|  | 37 | ID       *string    `json:"id,omitempty"` | 
|  | 38 | Name     *string    `json:"name,omitempty"` | 
|  | 39 | Password string     `json:"password"` | 
| Ash Wilson | cde6812 | 2014-08-28 16:15:43 -0400 | [diff] [blame] | 40 | Domain   *domainReq `json:"domain,omitempty"` | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 41 | } | 
|  | 42 |  | 
|  | 43 | type passwordReq struct { | 
|  | 44 | User userReq `json:"user"` | 
|  | 45 | } | 
|  | 46 |  | 
|  | 47 | type tokenReq struct { | 
|  | 48 | ID string `json:"id"` | 
|  | 49 | } | 
|  | 50 |  | 
|  | 51 | type identityReq struct { | 
|  | 52 | Methods  []string     `json:"methods"` | 
| Ash Wilson | cde6812 | 2014-08-28 16:15:43 -0400 | [diff] [blame] | 53 | Password *passwordReq `json:"password,omitempty"` | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 54 | Token    *tokenReq    `json:"token,omitempty"` | 
|  | 55 | } | 
|  | 56 |  | 
|  | 57 | type scopeReq struct { | 
|  | 58 | Domain  *domainReq  `json:"domain,omitempty"` | 
|  | 59 | Project *projectReq `json:"project,omitempty"` | 
|  | 60 | } | 
|  | 61 |  | 
|  | 62 | type authReq struct { | 
|  | 63 | Identity identityReq `json:"identity"` | 
| Ash Wilson | cde6812 | 2014-08-28 16:15:43 -0400 | [diff] [blame] | 64 | Scope    *scopeReq   `json:"scope,omitempty"` | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 65 | } | 
|  | 66 |  | 
|  | 67 | type request struct { | 
|  | 68 | Auth authReq `json:"auth"` | 
|  | 69 | } | 
|  | 70 |  | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 71 | // Populate the request structure based on the provided arguments. Create and return an error | 
|  | 72 | // if insufficient or incompatible information is present. | 
|  | 73 | var req request | 
|  | 74 |  | 
|  | 75 | // Test first for unrecognized arguments. | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 76 | if options.APIKey != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 77 | return createErr(ErrAPIKeyProvided) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 78 | } | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 79 | if options.TenantID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 80 | return createErr(ErrTenantIDProvided) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 81 | } | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 82 | if options.TenantName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 83 | return createErr(ErrTenantNameProvided) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 84 | } | 
|  | 85 |  | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 86 | if options.Password == "" { | 
| Ash Wilson | d7f73e9 | 2014-10-22 09:11:49 -0400 | [diff] [blame] | 87 | if c.TokenID != "" { | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 88 | // Because we aren't using password authentication, it's an error to also provide any of the user-based authentication | 
|  | 89 | // parameters. | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 90 | if options.Username != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 91 | return createErr(ErrUsernameWithToken) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 92 | } | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 93 | if options.UserID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 94 | return createErr(ErrUserIDWithToken) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 95 | } | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 96 | if options.DomainID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 97 | return createErr(ErrDomainIDWithToken) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 98 | } | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 99 | if options.DomainName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 100 | return createErr(ErrDomainNameWithToken) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 101 | } | 
|  | 102 |  | 
|  | 103 | // Configure the request for Token authentication. | 
|  | 104 | req.Auth.Identity.Methods = []string{"token"} | 
|  | 105 | req.Auth.Identity.Token = &tokenReq{ | 
| Ash Wilson | d7f73e9 | 2014-10-22 09:11:49 -0400 | [diff] [blame] | 106 | ID: c.TokenID, | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 107 | } | 
|  | 108 | } else { | 
|  | 109 | // If no password or token ID are available, authentication can't continue. | 
| Ash Wilson | 55f2433 | 2014-10-02 09:37:05 -0400 | [diff] [blame] | 110 | return createErr(ErrMissingPassword) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 111 | } | 
|  | 112 | } else { | 
|  | 113 | // Password authentication. | 
|  | 114 | req.Auth.Identity.Methods = []string{"password"} | 
|  | 115 |  | 
|  | 116 | // At least one of Username and UserID must be specified. | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 117 | if options.Username == "" && options.UserID == "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 118 | return createErr(ErrUsernameOrUserID) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 119 | } | 
|  | 120 |  | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 121 | if options.Username != "" { | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 122 | // If Username is provided, UserID may not be provided. | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 123 | if options.UserID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 124 | return createErr(ErrUsernameOrUserID) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 125 | } | 
|  | 126 |  | 
|  | 127 | // Either DomainID or DomainName must also be specified. | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 128 | if options.DomainID == "" && options.DomainName == "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 129 | return createErr(ErrDomainIDOrDomainName) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 130 | } | 
|  | 131 |  | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 132 | if options.DomainID != "" { | 
|  | 133 | if options.DomainName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 134 | return createErr(ErrDomainIDOrDomainName) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 135 | } | 
|  | 136 |  | 
|  | 137 | // Configure the request for Username and Password authentication with a DomainID. | 
|  | 138 | req.Auth.Identity.Password = &passwordReq{ | 
|  | 139 | User: userReq{ | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 140 | Name:     &options.Username, | 
|  | 141 | Password: options.Password, | 
|  | 142 | Domain:   &domainReq{ID: &options.DomainID}, | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 143 | }, | 
|  | 144 | } | 
|  | 145 | } | 
|  | 146 |  | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 147 | if options.DomainName != "" { | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 148 | // Configure the request for Username and Password authentication with a DomainName. | 
|  | 149 | req.Auth.Identity.Password = &passwordReq{ | 
|  | 150 | User: userReq{ | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 151 | Name:     &options.Username, | 
|  | 152 | Password: options.Password, | 
|  | 153 | Domain:   &domainReq{Name: &options.DomainName}, | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 154 | }, | 
|  | 155 | } | 
|  | 156 | } | 
|  | 157 | } | 
|  | 158 |  | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 159 | if options.UserID != "" { | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 160 | // If UserID is specified, neither DomainID nor DomainName may be. | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 161 | if options.DomainID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 162 | return createErr(ErrDomainIDWithUserID) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 163 | } | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 164 | if options.DomainName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 165 | return createErr(ErrDomainNameWithUserID) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 166 | } | 
|  | 167 |  | 
|  | 168 | // Configure the request for UserID and Password authentication. | 
|  | 169 | req.Auth.Identity.Password = &passwordReq{ | 
| Ash Wilson | a87ee06 | 2014-09-03 11:26:06 -0400 | [diff] [blame] | 170 | User: userReq{ID: &options.UserID, Password: options.Password}, | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 171 | } | 
|  | 172 | } | 
|  | 173 | } | 
|  | 174 |  | 
|  | 175 | // Add a "scope" element if a Scope has been provided. | 
|  | 176 | if scope != nil { | 
|  | 177 | if scope.ProjectName != "" { | 
|  | 178 | // ProjectName provided: either DomainID or DomainName must also be supplied. | 
|  | 179 | // ProjectID may not be supplied. | 
|  | 180 | if scope.DomainID == "" && scope.DomainName == "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 181 | return createErr(ErrScopeDomainIDOrDomainName) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 182 | } | 
|  | 183 | if scope.ProjectID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 184 | return createErr(ErrScopeProjectIDOrProjectName) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 185 | } | 
|  | 186 |  | 
|  | 187 | if scope.DomainID != "" { | 
|  | 188 | // ProjectName + DomainID | 
|  | 189 | req.Auth.Scope = &scopeReq{ | 
|  | 190 | Project: &projectReq{ | 
|  | 191 | Name:   &scope.ProjectName, | 
|  | 192 | Domain: &domainReq{ID: &scope.DomainID}, | 
|  | 193 | }, | 
|  | 194 | } | 
|  | 195 | } | 
|  | 196 |  | 
|  | 197 | if scope.DomainName != "" { | 
|  | 198 | // ProjectName + DomainName | 
|  | 199 | req.Auth.Scope = &scopeReq{ | 
|  | 200 | Project: &projectReq{ | 
|  | 201 | Name:   &scope.ProjectName, | 
|  | 202 | Domain: &domainReq{Name: &scope.DomainName}, | 
|  | 203 | }, | 
|  | 204 | } | 
|  | 205 | } | 
|  | 206 | } else if scope.ProjectID != "" { | 
|  | 207 | // ProjectID provided. ProjectName, DomainID, and DomainName may not be provided. | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 208 | if scope.DomainID != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 209 | return createErr(ErrScopeProjectIDAlone) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 210 | } | 
|  | 211 | if scope.DomainName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 212 | return createErr(ErrScopeProjectIDAlone) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 213 | } | 
|  | 214 |  | 
|  | 215 | // ProjectID | 
|  | 216 | req.Auth.Scope = &scopeReq{ | 
|  | 217 | Project: &projectReq{ID: &scope.ProjectID}, | 
|  | 218 | } | 
|  | 219 | } else if scope.DomainID != "" { | 
|  | 220 | // DomainID provided. ProjectID, ProjectName, and DomainName may not be provided. | 
|  | 221 | if scope.DomainName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 222 | return createErr(ErrScopeDomainIDOrDomainName) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 223 | } | 
|  | 224 |  | 
|  | 225 | // DomainID | 
|  | 226 | req.Auth.Scope = &scopeReq{ | 
|  | 227 | Domain: &domainReq{ID: &scope.DomainID}, | 
|  | 228 | } | 
|  | 229 | } else if scope.DomainName != "" { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 230 | return createErr(ErrScopeDomainName) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 231 | } else { | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 232 | return createErr(ErrScopeEmpty) | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 233 | } | 
|  | 234 | } | 
|  | 235 |  | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 236 | var result CreateResult | 
| Ash Wilson | 2491b4c | 2015-02-12 16:13:39 -0500 | [diff] [blame] | 237 | var response *http.Response | 
| Jamie Hannaford | 562a7d5 | 2015-03-24 16:20:16 +0100 | [diff] [blame] | 238 | response, result.Err = c.Post(tokenURL(c), req, &result.Body, nil) | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 239 | if result.Err != nil { | 
|  | 240 | return result | 
| Ash Wilson | 4a52e2a | 2014-08-29 09:28:00 -0400 | [diff] [blame] | 241 | } | 
| Ash Wilson | 2491b4c | 2015-02-12 16:13:39 -0500 | [diff] [blame] | 242 | result.Header = response.Header | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 243 | return result | 
| Ash Wilson | 85d8265 | 2014-08-28 13:57:46 -0400 | [diff] [blame] | 244 | } | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 245 |  | 
| Ash Wilson | 5266e49 | 2014-09-09 15:44:30 -0400 | [diff] [blame] | 246 | // Get validates and retrieves information about another token. | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 247 | func Get(c *gophercloud.ServiceClient, token string) GetResult { | 
|  | 248 | var result GetResult | 
| Ash Wilson | 2491b4c | 2015-02-12 16:13:39 -0500 | [diff] [blame] | 249 | var response *http.Response | 
| Jamie Hannaford | 562a7d5 | 2015-03-24 16:20:16 +0100 | [diff] [blame] | 250 | response, result.Err = c.Get(tokenURL(c), &result.Body, &gophercloud.RequestOpts{ | 
|  | 251 | MoreHeaders: subjectTokenHeaders(c, token), | 
|  | 252 | OkCodes:     []int{200, 203}, | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 253 | }) | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 254 | if result.Err != nil { | 
|  | 255 | return result | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 256 | } | 
| Ash Wilson | 2491b4c | 2015-02-12 16:13:39 -0500 | [diff] [blame] | 257 | result.Header = response.Header | 
| Ash Wilson | f8d546a | 2014-09-30 17:43:25 -0400 | [diff] [blame] | 258 | return result | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 259 | } | 
|  | 260 |  | 
|  | 261 | // Validate determines if a specified token is valid or not. | 
| Ash Wilson | 6425a41 | 2014-08-29 12:30:35 -0400 | [diff] [blame] | 262 | func Validate(c *gophercloud.ServiceClient, token string) (bool, error) { | 
| Ash Wilson | 4bf41a3 | 2015-02-12 15:52:44 -0500 | [diff] [blame] | 263 | response, err := c.Request("HEAD", tokenURL(c), gophercloud.RequestOpts{ | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 264 | MoreHeaders: subjectTokenHeaders(c, token), | 
|  | 265 | OkCodes:     []int{204, 404}, | 
|  | 266 | }) | 
|  | 267 | if err != nil { | 
|  | 268 | return false, err | 
|  | 269 | } | 
|  | 270 |  | 
|  | 271 | return response.StatusCode == 204, nil | 
|  | 272 | } | 
|  | 273 |  | 
|  | 274 | // Revoke immediately makes specified token invalid. | 
| Jamie Hannaford | f38dd2e | 2014-10-27 11:36:54 +0100 | [diff] [blame] | 275 | func Revoke(c *gophercloud.ServiceClient, token string) RevokeResult { | 
|  | 276 | var res RevokeResult | 
| Jamie Hannaford | 562a7d5 | 2015-03-24 16:20:16 +0100 | [diff] [blame] | 277 | _, res.Err = c.Delete(tokenURL(c), &gophercloud.RequestOpts{ | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 278 | MoreHeaders: subjectTokenHeaders(c, token), | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 279 | }) | 
| Jamie Hannaford | f38dd2e | 2014-10-27 11:36:54 +0100 | [diff] [blame] | 280 | return res | 
| Ash Wilson | 46d913f | 2014-08-29 11:00:11 -0400 | [diff] [blame] | 281 | } |