vault.go 33 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178
  1. /*
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. You may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.
  11. */
  12. package vault
  13. import (
  14. "context"
  15. "crypto/tls"
  16. "crypto/x509"
  17. "encoding/json"
  18. "errors"
  19. "fmt"
  20. "net/http"
  21. "os"
  22. "strconv"
  23. "strings"
  24. "github.com/go-logr/logr"
  25. vault "github.com/hashicorp/vault/api"
  26. approle "github.com/hashicorp/vault/api/auth/approle"
  27. authkubernetes "github.com/hashicorp/vault/api/auth/kubernetes"
  28. authldap "github.com/hashicorp/vault/api/auth/ldap"
  29. "github.com/tidwall/gjson"
  30. authenticationv1 "k8s.io/api/authentication/v1"
  31. corev1 "k8s.io/api/core/v1"
  32. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  33. "k8s.io/apimachinery/pkg/types"
  34. "k8s.io/client-go/kubernetes"
  35. typedcorev1 "k8s.io/client-go/kubernetes/typed/core/v1"
  36. ctrl "sigs.k8s.io/controller-runtime"
  37. kclient "sigs.k8s.io/controller-runtime/pkg/client"
  38. ctrlcfg "sigs.k8s.io/controller-runtime/pkg/client/config"
  39. esv1beta1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1beta1"
  40. esmeta "github.com/external-secrets/external-secrets/apis/meta/v1"
  41. "github.com/external-secrets/external-secrets/pkg/find"
  42. "github.com/external-secrets/external-secrets/pkg/utils"
  43. )
  44. var (
  45. _ esv1beta1.Provider = &connector{}
  46. _ esv1beta1.SecretsClient = &client{}
  47. )
  48. const (
  49. serviceAccTokenPath = "/var/run/secrets/kubernetes.io/serviceaccount/token"
  50. errVaultStore = "received invalid Vault SecretStore resource: %w"
  51. errVaultClient = "cannot setup new vault client: %w"
  52. errVaultCert = "cannot set Vault CA certificate: %w"
  53. errReadSecret = "cannot read secret data from Vault: %w"
  54. errAuthFormat = "cannot initialize Vault client: no valid auth method specified"
  55. errInvalidCredentials = "invalid vault credentials: %w"
  56. errDataField = "failed to find data field"
  57. errJSONUnmarshall = "failed to unmarshall JSON"
  58. errPathInvalid = "provided Path isn't a valid kv v2 path"
  59. errSecretFormat = "secret data not in expected format"
  60. errUnexpectedKey = "unexpected key in data: %s"
  61. errVaultToken = "cannot parse Vault authentication token: %w"
  62. errVaultRequest = "error from Vault request: %w"
  63. errServiceAccount = "cannot read Kubernetes service account token from file system: %w"
  64. errJwtNoTokenSource = "neither `secretRef` nor `kubernetesServiceAccountToken` was supplied as token source for jwt authentication"
  65. errUnsupportedKvVersion = "cannot perform find operations with kv version v1"
  66. errNotFound = "secret not found"
  67. errGetKubeSA = "cannot get Kubernetes service account %q: %w"
  68. errGetKubeSASecrets = "cannot find secrets bound to service account: %q"
  69. errGetKubeSANoToken = "cannot find token in secrets bound to service account: %q"
  70. errGetKubeSATokenRequest = "cannot request Kubernetes service account token for service account %q: %w"
  71. errGetKubeSecret = "cannot get Kubernetes secret %q: %w"
  72. errSecretKeyFmt = "cannot find secret data for key: %q"
  73. errConfigMapFmt = "cannot find config map data for key: %q"
  74. errClientTLSAuth = "error from Client TLS Auth: %q"
  75. errVaultRevokeToken = "error while revoking token: %w"
  76. errUnknownCAProvider = "unknown caProvider type given"
  77. errCANamespace = "cannot read secret for CAProvider due to missing namespace on kind ClusterSecretStore"
  78. errInvalidStore = "invalid store"
  79. errInvalidStoreSpec = "invalid store spec"
  80. errInvalidStoreProv = "invalid store provider"
  81. errInvalidVaultProv = "invalid vault provider"
  82. errInvalidAppRoleSec = "invalid Auth.AppRole.SecretRef: %w"
  83. errInvalidClientCert = "invalid Auth.Cert.ClientCert: %w"
  84. errInvalidCertSec = "invalid Auth.Cert.SecretRef: %w"
  85. errInvalidJwtSec = "invalid Auth.Jwt.SecretRef: %w"
  86. errInvalidJwtK8sSA = "invalid Auth.Jwt.KubernetesServiceAccountToken.ServiceAccountRef: %w"
  87. errInvalidKubeSA = "invalid Auth.Kubernetes.ServiceAccountRef: %w"
  88. errInvalidKubeSec = "invalid Auth.Kubernetes.SecretRef: %w"
  89. errInvalidLdapSec = "invalid Auth.Ldap.SecretRef: %w"
  90. errInvalidTokenRef = "invalid Auth.TokenSecretRef: %w"
  91. )
  92. // https://github.com/external-secrets/external-secrets/issues/644
  93. var _ esv1beta1.SecretsClient = &client{}
  94. var _ esv1beta1.Provider = &connector{}
  95. type Auth interface {
  96. Login(ctx context.Context, authMethod vault.AuthMethod) (*vault.Secret, error)
  97. }
  98. type Token interface {
  99. RevokeSelfWithContext(ctx context.Context, token string) error
  100. LookupSelfWithContext(ctx context.Context) (*vault.Secret, error)
  101. }
  102. type Logical interface {
  103. ReadWithDataWithContext(ctx context.Context, path string, data map[string][]string) (*vault.Secret, error)
  104. ListWithContext(ctx context.Context, path string) (*vault.Secret, error)
  105. WriteWithContext(ctx context.Context, path string, data map[string]interface{}) (*vault.Secret, error)
  106. }
  107. type Client interface {
  108. SetToken(v string)
  109. Token() string
  110. ClearToken()
  111. Auth() Auth
  112. Logical() Logical
  113. AuthToken() Token
  114. SetNamespace(namespace string)
  115. AddHeader(key, value string)
  116. }
  117. type VClient struct {
  118. setToken func(v string)
  119. token func() string
  120. clearToken func()
  121. auth Auth
  122. logical Logical
  123. authToken Token
  124. setNamespace func(namespace string)
  125. addHeader func(key, value string)
  126. }
  127. func (v VClient) AddHeader(key, value string) {
  128. v.addHeader(key, value)
  129. }
  130. func (v VClient) SetNamespace(namespace string) {
  131. v.setNamespace(namespace)
  132. }
  133. func (v VClient) ClearToken() {
  134. v.clearToken()
  135. }
  136. func (v VClient) Token() string {
  137. return v.token()
  138. }
  139. func (v VClient) SetToken(token string) {
  140. v.setToken(token)
  141. }
  142. func (v VClient) Auth() Auth {
  143. return v.auth
  144. }
  145. func (v VClient) AuthToken() Token {
  146. return v.authToken
  147. }
  148. func (v VClient) Logical() Logical {
  149. return v.logical
  150. }
  151. type client struct {
  152. kube kclient.Client
  153. store *esv1beta1.VaultProvider
  154. log logr.Logger
  155. corev1 typedcorev1.CoreV1Interface
  156. client Client
  157. auth Auth
  158. logical Logical
  159. token Token
  160. namespace string
  161. storeKind string
  162. }
  163. func init() {
  164. esv1beta1.Register(&connector{
  165. newVaultClient: newVaultClient,
  166. }, &esv1beta1.SecretStoreProvider{
  167. Vault: &esv1beta1.VaultProvider{},
  168. })
  169. }
  170. func newVaultClient(c *vault.Config) (Client, error) {
  171. cl, err := vault.NewClient(c)
  172. if err != nil {
  173. return nil, err
  174. }
  175. auth := cl.Auth()
  176. logical := cl.Logical()
  177. token := cl.Auth().Token()
  178. out := VClient{
  179. setToken: cl.SetToken,
  180. token: cl.Token,
  181. clearToken: cl.ClearToken,
  182. auth: auth,
  183. authToken: token,
  184. logical: logical,
  185. setNamespace: cl.SetNamespace,
  186. addHeader: cl.AddHeader,
  187. }
  188. return out, nil
  189. }
  190. type connector struct {
  191. newVaultClient func(c *vault.Config) (Client, error)
  192. }
  193. // Capabilities return the provider supported capabilities (ReadOnly, WriteOnly, ReadWrite).
  194. func (c *connector) Capabilities() esv1beta1.SecretStoreCapabilities {
  195. return esv1beta1.SecretStoreReadOnly
  196. }
  197. func (c *connector) NewClient(ctx context.Context, store esv1beta1.GenericStore, kube kclient.Client, namespace string) (esv1beta1.SecretsClient, error) {
  198. // controller-runtime/client does not support TokenRequest or other subresource APIs
  199. // so we need to construct our own client and use it to fetch tokens
  200. // (for Kubernetes service account token auth)
  201. restCfg, err := ctrlcfg.GetConfig()
  202. if err != nil {
  203. return nil, err
  204. }
  205. clientset, err := kubernetes.NewForConfig(restCfg)
  206. if err != nil {
  207. return nil, err
  208. }
  209. return c.newClient(ctx, store, kube, clientset.CoreV1(), namespace)
  210. }
  211. func (c *connector) newClient(ctx context.Context, store esv1beta1.GenericStore, kube kclient.Client, corev1 typedcorev1.CoreV1Interface, namespace string) (esv1beta1.SecretsClient, error) {
  212. storeSpec := store.GetSpec()
  213. if storeSpec == nil || storeSpec.Provider == nil || storeSpec.Provider.Vault == nil {
  214. return nil, errors.New(errVaultStore)
  215. }
  216. vaultSpec := storeSpec.Provider.Vault
  217. vStore := &client{
  218. kube: kube,
  219. corev1: corev1,
  220. store: vaultSpec,
  221. log: ctrl.Log.WithName("provider").WithName("vault"),
  222. namespace: namespace,
  223. storeKind: store.GetObjectKind().GroupVersionKind().Kind,
  224. }
  225. cfg, err := vStore.newConfig()
  226. if err != nil {
  227. return nil, err
  228. }
  229. client, err := c.newVaultClient(cfg)
  230. if err != nil {
  231. return nil, fmt.Errorf(errVaultClient, err)
  232. }
  233. if vaultSpec.Namespace != nil {
  234. client.SetNamespace(*vaultSpec.Namespace)
  235. }
  236. if vaultSpec.ReadYourWrites && vaultSpec.ForwardInconsistent {
  237. client.AddHeader("X-Vault-Inconsistent", "forward-active-node")
  238. }
  239. vStore.client = client
  240. vStore.auth = client.Auth()
  241. vStore.logical = client.Logical()
  242. vStore.token = client.AuthToken()
  243. // allow SecretStore controller validation to pass
  244. // when using referent namespace.
  245. if vStore.storeKind == esv1beta1.ClusterSecretStoreKind && vStore.namespace == "" && isReferentSpec(vaultSpec) {
  246. return vStore, nil
  247. }
  248. if err := vStore.setAuth(ctx, cfg); err != nil {
  249. return nil, err
  250. }
  251. return vStore, nil
  252. }
  253. func (c *connector) ValidateStore(store esv1beta1.GenericStore) error {
  254. if store == nil {
  255. return fmt.Errorf(errInvalidStore)
  256. }
  257. spc := store.GetSpec()
  258. if spc == nil {
  259. return fmt.Errorf(errInvalidStoreSpec)
  260. }
  261. if spc.Provider == nil {
  262. return fmt.Errorf(errInvalidStoreProv)
  263. }
  264. p := spc.Provider.Vault
  265. if p == nil {
  266. return fmt.Errorf(errInvalidVaultProv)
  267. }
  268. if p.Auth.AppRole != nil {
  269. if err := utils.ValidateReferentSecretSelector(store, p.Auth.AppRole.SecretRef); err != nil {
  270. return fmt.Errorf(errInvalidAppRoleSec, err)
  271. }
  272. }
  273. if p.Auth.Cert != nil {
  274. if err := utils.ValidateReferentSecretSelector(store, p.Auth.Cert.ClientCert); err != nil {
  275. return fmt.Errorf(errInvalidClientCert, err)
  276. }
  277. if err := utils.ValidateReferentSecretSelector(store, p.Auth.Cert.SecretRef); err != nil {
  278. return fmt.Errorf(errInvalidCertSec, err)
  279. }
  280. }
  281. if p.Auth.Jwt != nil {
  282. if p.Auth.Jwt.SecretRef != nil {
  283. if err := utils.ValidateReferentSecretSelector(store, *p.Auth.Jwt.SecretRef); err != nil {
  284. return fmt.Errorf(errInvalidJwtSec, err)
  285. }
  286. } else if p.Auth.Jwt.KubernetesServiceAccountToken != nil {
  287. if err := utils.ValidateReferentServiceAccountSelector(store, p.Auth.Jwt.KubernetesServiceAccountToken.ServiceAccountRef); err != nil {
  288. return fmt.Errorf(errInvalidJwtK8sSA, err)
  289. }
  290. } else {
  291. return fmt.Errorf(errJwtNoTokenSource)
  292. }
  293. }
  294. if p.Auth.Kubernetes != nil {
  295. if p.Auth.Kubernetes.ServiceAccountRef != nil {
  296. if err := utils.ValidateReferentServiceAccountSelector(store, *p.Auth.Kubernetes.ServiceAccountRef); err != nil {
  297. return fmt.Errorf(errInvalidKubeSA, err)
  298. }
  299. }
  300. if p.Auth.Kubernetes.SecretRef != nil {
  301. if err := utils.ValidateReferentSecretSelector(store, *p.Auth.Kubernetes.SecretRef); err != nil {
  302. return fmt.Errorf(errInvalidKubeSec, err)
  303. }
  304. }
  305. }
  306. if p.Auth.Ldap != nil {
  307. if err := utils.ValidateReferentSecretSelector(store, p.Auth.Ldap.SecretRef); err != nil {
  308. return fmt.Errorf(errInvalidLdapSec, err)
  309. }
  310. }
  311. if p.Auth.TokenSecretRef != nil {
  312. if err := utils.ValidateReferentSecretSelector(store, *p.Auth.TokenSecretRef); err != nil {
  313. return fmt.Errorf(errInvalidTokenRef, err)
  314. }
  315. }
  316. return nil
  317. }
  318. // Not Implemented SetSecret.
  319. func (v *client) SetSecret(ctx context.Context, value []byte, remoteRef esv1beta1.PushRemoteRef) error {
  320. return fmt.Errorf("not implemented")
  321. }
  322. // GetAllSecrets gets multiple secrets from the provider and loads into a kubernetes secret.
  323. // First load all secrets from secretStore path configuration
  324. // Then, gets secrets from a matching name or matching custom_metadata.
  325. func (v *client) GetAllSecrets(ctx context.Context, ref esv1beta1.ExternalSecretFind) (map[string][]byte, error) {
  326. if v.store.Version == esv1beta1.VaultKVStoreV1 {
  327. return nil, errors.New(errUnsupportedKvVersion)
  328. }
  329. searchPath := ""
  330. if ref.Path != nil {
  331. searchPath = *ref.Path + "/"
  332. }
  333. potentialSecrets, err := v.listSecrets(ctx, searchPath)
  334. if err != nil {
  335. return nil, err
  336. }
  337. if ref.Name != nil {
  338. return v.findSecretsFromName(ctx, potentialSecrets, *ref.Name)
  339. }
  340. return v.findSecretsFromTags(ctx, potentialSecrets, ref.Tags)
  341. }
  342. func (v *client) findSecretsFromTags(ctx context.Context, candidates []string, tags map[string]string) (map[string][]byte, error) {
  343. secrets := make(map[string][]byte)
  344. for _, name := range candidates {
  345. match := true
  346. metadata, err := v.readSecretMetadata(ctx, name)
  347. if err != nil {
  348. return nil, err
  349. }
  350. for tk, tv := range tags {
  351. p, ok := metadata[tk]
  352. if !ok || p != tv {
  353. match = false
  354. break
  355. }
  356. }
  357. if match {
  358. secret, err := v.GetSecret(ctx, esv1beta1.ExternalSecretDataRemoteRef{Key: name})
  359. if err != nil {
  360. return nil, err
  361. }
  362. secrets[name] = secret
  363. }
  364. }
  365. return secrets, nil
  366. }
  367. func (v *client) findSecretsFromName(ctx context.Context, candidates []string, ref esv1beta1.FindName) (map[string][]byte, error) {
  368. secrets := make(map[string][]byte)
  369. matcher, err := find.New(ref)
  370. if err != nil {
  371. return nil, err
  372. }
  373. for _, name := range candidates {
  374. ok := matcher.MatchName(name)
  375. if ok {
  376. secret, err := v.GetSecret(ctx, esv1beta1.ExternalSecretDataRemoteRef{Key: name})
  377. if err != nil {
  378. return nil, err
  379. }
  380. secrets[name] = secret
  381. }
  382. }
  383. return secrets, nil
  384. }
  385. func (v *client) listSecrets(ctx context.Context, path string) ([]string, error) {
  386. secrets := make([]string, 0)
  387. url, err := v.buildMetadataPath(path)
  388. if err != nil {
  389. return nil, err
  390. }
  391. secret, err := v.logical.ListWithContext(ctx, url)
  392. if err != nil {
  393. return nil, fmt.Errorf(errReadSecret, err)
  394. }
  395. t, ok := secret.Data["keys"]
  396. if !ok {
  397. return nil, nil
  398. }
  399. paths := t.([]interface{})
  400. for _, p := range paths {
  401. strPath := p.(string)
  402. fullPath := path + strPath // because path always ends with a /
  403. if path == "" {
  404. fullPath = strPath
  405. }
  406. // Recurrently find secrets
  407. if !strings.HasSuffix(p.(string), "/") {
  408. secrets = append(secrets, fullPath)
  409. } else {
  410. partial, err := v.listSecrets(ctx, fullPath)
  411. if err != nil {
  412. return nil, err
  413. }
  414. secrets = append(secrets, partial...)
  415. }
  416. }
  417. return secrets, nil
  418. }
  419. func (v *client) readSecretMetadata(ctx context.Context, path string) (map[string]string, error) {
  420. metadata := make(map[string]string)
  421. url, err := v.buildMetadataPath(path)
  422. if err != nil {
  423. return nil, err
  424. }
  425. secret, err := v.logical.ReadWithDataWithContext(ctx, url, nil)
  426. if err != nil {
  427. return nil, fmt.Errorf(errReadSecret, err)
  428. }
  429. if secret == nil {
  430. return nil, errors.New(errNotFound)
  431. }
  432. t, ok := secret.Data["custom_metadata"]
  433. if !ok {
  434. return nil, nil
  435. }
  436. d, ok := t.(map[string]interface{})
  437. if !ok {
  438. return metadata, nil
  439. }
  440. for k, v := range d {
  441. metadata[k] = v.(string)
  442. }
  443. return metadata, nil
  444. }
  445. // GetSecret supports two types:
  446. // 1. get the full secret as json-encoded value
  447. // by leaving the ref.Property empty.
  448. // 2. get a key from the secret.
  449. // Nested values are supported by specifying a gjson expression
  450. func (v *client) GetSecret(ctx context.Context, ref esv1beta1.ExternalSecretDataRemoteRef) ([]byte, error) {
  451. data, err := v.readSecret(ctx, ref.Key, ref.Version)
  452. if err != nil {
  453. return nil, err
  454. }
  455. jsonStr, err := json.Marshal(data)
  456. if err != nil {
  457. return nil, err
  458. }
  459. // (1): return raw json if no property is defined
  460. if ref.Property == "" {
  461. return jsonStr, nil
  462. }
  463. // For backwards compatibility we want the
  464. // actual keys to take precedence over gjson syntax
  465. // (2): extract key from secret with property
  466. if _, ok := data[ref.Property]; ok {
  467. return getTypedKey(data, ref.Property)
  468. }
  469. // (3): extract key from secret using gjson
  470. val := gjson.Get(string(jsonStr), ref.Property)
  471. if !val.Exists() {
  472. return nil, fmt.Errorf(errSecretKeyFmt, ref.Property)
  473. }
  474. return []byte(val.String()), nil
  475. }
  476. // GetSecretMap supports two modes of operation:
  477. // 1. get the full secret from the vault data payload (by leaving .property empty).
  478. // 2. extract key/value pairs from a (nested) object.
  479. func (v *client) GetSecretMap(ctx context.Context, ref esv1beta1.ExternalSecretDataRemoteRef) (map[string][]byte, error) {
  480. data, err := v.GetSecret(ctx, ref)
  481. if err != nil {
  482. return nil, err
  483. }
  484. var secretData map[string]interface{}
  485. err = json.Unmarshal(data, &secretData)
  486. if err != nil {
  487. return nil, err
  488. }
  489. byteMap := make(map[string][]byte, len(secretData))
  490. for k := range secretData {
  491. byteMap[k], err = getTypedKey(secretData, k)
  492. if err != nil {
  493. return nil, err
  494. }
  495. }
  496. return byteMap, nil
  497. }
  498. func getTypedKey(data map[string]interface{}, key string) ([]byte, error) {
  499. v, ok := data[key]
  500. if !ok {
  501. return nil, fmt.Errorf(errUnexpectedKey, key)
  502. }
  503. switch t := v.(type) {
  504. case string:
  505. return []byte(t), nil
  506. case map[string]interface{}:
  507. return json.Marshal(t)
  508. case []byte:
  509. return t, nil
  510. // also covers int and float32 due to json.Marshal
  511. case float64:
  512. return []byte(strconv.FormatFloat(t, 'f', -1, 64)), nil
  513. case bool:
  514. return []byte(strconv.FormatBool(t)), nil
  515. case nil:
  516. return []byte(nil), nil
  517. default:
  518. return nil, errors.New(errSecretFormat)
  519. }
  520. }
  521. func (v *client) Close(ctx context.Context) error {
  522. // Revoke the token if we have one set and it wasn't sourced from a TokenSecretRef
  523. if v.client.Token() != "" && v.store.Auth.TokenSecretRef == nil {
  524. revoke, err := checkToken(ctx, v)
  525. if err != nil {
  526. return fmt.Errorf(errVaultRevokeToken, err)
  527. }
  528. if revoke {
  529. err = v.token.RevokeSelfWithContext(ctx, v.client.Token())
  530. if err != nil {
  531. return fmt.Errorf(errVaultRevokeToken, err)
  532. }
  533. v.client.ClearToken()
  534. }
  535. }
  536. return nil
  537. }
  538. func isReferentSpec(prov *esv1beta1.VaultProvider) bool {
  539. if prov.Auth.TokenSecretRef != nil && prov.Auth.TokenSecretRef.Namespace == nil {
  540. return true
  541. }
  542. if prov.Auth.AppRole != nil && prov.Auth.AppRole.SecretRef.Namespace == nil {
  543. return true
  544. }
  545. if prov.Auth.Kubernetes != nil && prov.Auth.Kubernetes.SecretRef != nil && prov.Auth.Kubernetes.SecretRef.Namespace == nil {
  546. return true
  547. }
  548. if prov.Auth.Kubernetes != nil && prov.Auth.Kubernetes.ServiceAccountRef != nil && prov.Auth.Kubernetes.ServiceAccountRef.Namespace == nil {
  549. return true
  550. }
  551. if prov.Auth.Ldap != nil && prov.Auth.Ldap.SecretRef.Namespace == nil {
  552. return true
  553. }
  554. if prov.Auth.Jwt != nil && prov.Auth.Jwt.SecretRef != nil && prov.Auth.Jwt.SecretRef.Namespace == nil {
  555. return true
  556. }
  557. if prov.Auth.Jwt != nil && prov.Auth.Jwt.KubernetesServiceAccountToken != nil && prov.Auth.Jwt.KubernetesServiceAccountToken.ServiceAccountRef.Namespace == nil {
  558. return true
  559. }
  560. if prov.Auth.Cert != nil && prov.Auth.Cert.SecretRef.Namespace == nil {
  561. return true
  562. }
  563. return false
  564. }
  565. func (v *client) Validate() (esv1beta1.ValidationResult, error) {
  566. // when using referent namespace we can not validate the token
  567. // because the namespace is not known yet when Validate() is called
  568. // from the SecretStore controller.
  569. if v.storeKind == esv1beta1.ClusterSecretStoreKind && isReferentSpec(v.store) {
  570. return esv1beta1.ValidationResultUnknown, nil
  571. }
  572. _, err := checkToken(context.Background(), v)
  573. if err != nil {
  574. return esv1beta1.ValidationResultError, fmt.Errorf(errInvalidCredentials, err)
  575. }
  576. return esv1beta1.ValidationResultReady, nil
  577. }
  578. func (v *client) buildMetadataPath(path string) (string, error) {
  579. var url string
  580. if v.store.Path == nil && !strings.Contains(path, "data") {
  581. return "", fmt.Errorf(errPathInvalid)
  582. }
  583. if v.store.Path == nil {
  584. path = strings.Replace(path, "data", "metadata", 1)
  585. url = path
  586. } else {
  587. url = fmt.Sprintf("%s/metadata/%s", *v.store.Path, path)
  588. }
  589. return url, nil
  590. }
  591. func (v *client) buildPath(path string) string {
  592. optionalMount := v.store.Path
  593. origPath := strings.Split(path, "/")
  594. newPath := make([]string, 0)
  595. cursor := 0
  596. if optionalMount != nil && origPath[0] != *optionalMount {
  597. // Default case before path was optional
  598. // Ensure that the requested path includes the SecretStores paths as prefix
  599. newPath = append(newPath, *optionalMount)
  600. } else {
  601. newPath = append(newPath, origPath[cursor])
  602. cursor++
  603. }
  604. if v.store.Version == esv1beta1.VaultKVStoreV2 {
  605. // Add the required `data` part of the URL for the v2 API
  606. if len(origPath) < 2 || origPath[1] != "data" {
  607. newPath = append(newPath, "data")
  608. }
  609. }
  610. newPath = append(newPath, origPath[cursor:]...)
  611. returnPath := strings.Join(newPath, "/")
  612. return returnPath
  613. }
  614. func (v *client) readSecret(ctx context.Context, path, version string) (map[string]interface{}, error) {
  615. dataPath := v.buildPath(path)
  616. // path formated according to vault docs for v1 and v2 API
  617. // v1: https://www.vaultproject.io/api-docs/secret/kv/kv-v1#read-secret
  618. // v2: https://www.vaultproject.io/api/secret/kv/kv-v2#read-secret-version
  619. var params map[string][]string
  620. if version != "" {
  621. params = make(map[string][]string)
  622. params["version"] = []string{version}
  623. }
  624. vaultSecret, err := v.logical.ReadWithDataWithContext(ctx, dataPath, params)
  625. if err != nil {
  626. return nil, fmt.Errorf(errReadSecret, err)
  627. }
  628. if vaultSecret == nil {
  629. return nil, errors.New(errNotFound)
  630. }
  631. secretData := vaultSecret.Data
  632. if v.store.Version == esv1beta1.VaultKVStoreV2 {
  633. // Vault KV2 has data embedded within sub-field
  634. // reference - https://www.vaultproject.io/api/secret/kv/kv-v2#read-secret-version
  635. dataInt, ok := vaultSecret.Data["data"]
  636. if !ok {
  637. return nil, errors.New(errDataField)
  638. }
  639. secretData, ok = dataInt.(map[string]interface{})
  640. if !ok {
  641. return nil, errors.New(errJSONUnmarshall)
  642. }
  643. }
  644. return secretData, nil
  645. }
  646. func (v *client) newConfig() (*vault.Config, error) {
  647. cfg := vault.DefaultConfig()
  648. cfg.Address = v.store.Server
  649. // In a controller-runtime context, we rely on the reconciliation process for retrying
  650. cfg.MaxRetries = 0
  651. if len(v.store.CABundle) == 0 && v.store.CAProvider == nil {
  652. return cfg, nil
  653. }
  654. caCertPool := x509.NewCertPool()
  655. if len(v.store.CABundle) > 0 {
  656. ok := caCertPool.AppendCertsFromPEM(v.store.CABundle)
  657. if !ok {
  658. return nil, errors.New(errVaultCert)
  659. }
  660. }
  661. if v.store.CAProvider != nil && v.storeKind == esv1beta1.ClusterSecretStoreKind && v.store.CAProvider.Namespace == nil {
  662. return nil, errors.New(errCANamespace)
  663. }
  664. if v.store.CAProvider != nil {
  665. var cert []byte
  666. var err error
  667. switch v.store.CAProvider.Type {
  668. case esv1beta1.CAProviderTypeSecret:
  669. cert, err = getCertFromSecret(v)
  670. case esv1beta1.CAProviderTypeConfigMap:
  671. cert, err = getCertFromConfigMap(v)
  672. default:
  673. return nil, errors.New(errUnknownCAProvider)
  674. }
  675. if err != nil {
  676. return nil, err
  677. }
  678. ok := caCertPool.AppendCertsFromPEM(cert)
  679. if !ok {
  680. return nil, errors.New(errVaultCert)
  681. }
  682. }
  683. if transport, ok := cfg.HttpClient.Transport.(*http.Transport); ok {
  684. transport.TLSClientConfig.RootCAs = caCertPool
  685. }
  686. // If either read-after-write consistency feature is enabled, enable ReadYourWrites
  687. cfg.ReadYourWrites = v.store.ReadYourWrites || v.store.ForwardInconsistent
  688. return cfg, nil
  689. }
  690. func getCertFromSecret(v *client) ([]byte, error) {
  691. secretRef := esmeta.SecretKeySelector{
  692. Name: v.store.CAProvider.Name,
  693. Key: v.store.CAProvider.Key,
  694. }
  695. if v.store.CAProvider.Namespace != nil {
  696. secretRef.Namespace = v.store.CAProvider.Namespace
  697. }
  698. ctx := context.Background()
  699. res, err := v.secretKeyRef(ctx, &secretRef)
  700. if err != nil {
  701. return nil, fmt.Errorf(errVaultCert, err)
  702. }
  703. return []byte(res), nil
  704. }
  705. func getCertFromConfigMap(v *client) ([]byte, error) {
  706. objKey := types.NamespacedName{
  707. Name: v.store.CAProvider.Name,
  708. }
  709. if v.store.CAProvider.Namespace != nil {
  710. objKey.Namespace = *v.store.CAProvider.Namespace
  711. }
  712. configMapRef := &corev1.ConfigMap{}
  713. ctx := context.Background()
  714. err := v.kube.Get(ctx, objKey, configMapRef)
  715. if err != nil {
  716. return nil, fmt.Errorf(errVaultCert, err)
  717. }
  718. val, ok := configMapRef.Data[v.store.CAProvider.Key]
  719. if !ok {
  720. return nil, fmt.Errorf(errConfigMapFmt, v.store.CAProvider.Key)
  721. }
  722. return []byte(val), nil
  723. }
  724. func (v *client) setAuth(ctx context.Context, cfg *vault.Config) error {
  725. tokenExists, err := setSecretKeyToken(ctx, v)
  726. if tokenExists {
  727. return err
  728. }
  729. tokenExists, err = setAppRoleToken(ctx, v)
  730. if tokenExists {
  731. return err
  732. }
  733. tokenExists, err = setKubernetesAuthToken(ctx, v)
  734. if tokenExists {
  735. return err
  736. }
  737. tokenExists, err = setLdapAuthToken(ctx, v)
  738. if tokenExists {
  739. return err
  740. }
  741. tokenExists, err = setJwtAuthToken(ctx, v)
  742. if tokenExists {
  743. return err
  744. }
  745. tokenExists, err = setCertAuthToken(ctx, v, cfg)
  746. if tokenExists {
  747. return err
  748. }
  749. return errors.New(errAuthFormat)
  750. }
  751. func setAppRoleToken(ctx context.Context, v *client) (bool, error) {
  752. tokenRef := v.store.Auth.TokenSecretRef
  753. if tokenRef != nil {
  754. token, err := v.secretKeyRef(ctx, tokenRef)
  755. if err != nil {
  756. return true, err
  757. }
  758. v.client.SetToken(token)
  759. return true, nil
  760. }
  761. return false, nil
  762. }
  763. func setSecretKeyToken(ctx context.Context, v *client) (bool, error) {
  764. appRole := v.store.Auth.AppRole
  765. if appRole != nil {
  766. err := v.requestTokenWithAppRoleRef(ctx, appRole)
  767. if err != nil {
  768. return true, err
  769. }
  770. return true, nil
  771. }
  772. return false, nil
  773. }
  774. func setKubernetesAuthToken(ctx context.Context, v *client) (bool, error) {
  775. kubernetesAuth := v.store.Auth.Kubernetes
  776. if kubernetesAuth != nil {
  777. err := v.requestTokenWithKubernetesAuth(ctx, kubernetesAuth)
  778. if err != nil {
  779. return true, err
  780. }
  781. return true, nil
  782. }
  783. return false, nil
  784. }
  785. func setLdapAuthToken(ctx context.Context, v *client) (bool, error) {
  786. ldapAuth := v.store.Auth.Ldap
  787. if ldapAuth != nil {
  788. err := v.requestTokenWithLdapAuth(ctx, ldapAuth)
  789. if err != nil {
  790. return true, err
  791. }
  792. return true, nil
  793. }
  794. return false, nil
  795. }
  796. func setJwtAuthToken(ctx context.Context, v *client) (bool, error) {
  797. jwtAuth := v.store.Auth.Jwt
  798. if jwtAuth != nil {
  799. err := v.requestTokenWithJwtAuth(ctx, jwtAuth)
  800. if err != nil {
  801. return true, err
  802. }
  803. return true, nil
  804. }
  805. return false, nil
  806. }
  807. func setCertAuthToken(ctx context.Context, v *client, cfg *vault.Config) (bool, error) {
  808. certAuth := v.store.Auth.Cert
  809. if certAuth != nil {
  810. err := v.requestTokenWithCertAuth(ctx, certAuth, cfg)
  811. if err != nil {
  812. return true, err
  813. }
  814. return true, nil
  815. }
  816. return false, nil
  817. }
  818. func (v *client) secretKeyRefForServiceAccount(ctx context.Context, serviceAccountRef *esmeta.ServiceAccountSelector) (string, error) {
  819. serviceAccount := &corev1.ServiceAccount{}
  820. ref := types.NamespacedName{
  821. Namespace: v.namespace,
  822. Name: serviceAccountRef.Name,
  823. }
  824. if (v.storeKind == esv1beta1.ClusterSecretStoreKind) &&
  825. (serviceAccountRef.Namespace != nil) {
  826. ref.Namespace = *serviceAccountRef.Namespace
  827. }
  828. err := v.kube.Get(ctx, ref, serviceAccount)
  829. if err != nil {
  830. return "", fmt.Errorf(errGetKubeSA, ref.Name, err)
  831. }
  832. if len(serviceAccount.Secrets) == 0 {
  833. return "", fmt.Errorf(errGetKubeSASecrets, ref.Name)
  834. }
  835. for _, tokenRef := range serviceAccount.Secrets {
  836. retval, err := v.secretKeyRef(ctx, &esmeta.SecretKeySelector{
  837. Name: tokenRef.Name,
  838. Namespace: &ref.Namespace,
  839. Key: "token",
  840. })
  841. if err != nil {
  842. continue
  843. }
  844. return retval, nil
  845. }
  846. return "", fmt.Errorf(errGetKubeSANoToken, ref.Name)
  847. }
  848. func (v *client) secretKeyRef(ctx context.Context, secretRef *esmeta.SecretKeySelector) (string, error) {
  849. secret := &corev1.Secret{}
  850. ref := types.NamespacedName{
  851. Namespace: v.namespace,
  852. Name: secretRef.Name,
  853. }
  854. if (v.storeKind == esv1beta1.ClusterSecretStoreKind) &&
  855. (secretRef.Namespace != nil) {
  856. ref.Namespace = *secretRef.Namespace
  857. }
  858. err := v.kube.Get(ctx, ref, secret)
  859. if err != nil {
  860. return "", fmt.Errorf(errGetKubeSecret, ref.Name, err)
  861. }
  862. keyBytes, ok := secret.Data[secretRef.Key]
  863. if !ok {
  864. return "", fmt.Errorf(errSecretKeyFmt, secretRef.Key)
  865. }
  866. value := string(keyBytes)
  867. valueStr := strings.TrimSpace(value)
  868. return valueStr, nil
  869. }
  870. func (v *client) serviceAccountToken(ctx context.Context, serviceAccountRef esmeta.ServiceAccountSelector, audiences []string, expirationSeconds int64) (string, error) {
  871. tokenRequest := &authenticationv1.TokenRequest{
  872. ObjectMeta: metav1.ObjectMeta{
  873. Namespace: v.namespace,
  874. },
  875. Spec: authenticationv1.TokenRequestSpec{
  876. Audiences: audiences,
  877. ExpirationSeconds: &expirationSeconds,
  878. },
  879. }
  880. if (v.storeKind == esv1beta1.ClusterSecretStoreKind) &&
  881. (serviceAccountRef.Namespace != nil) {
  882. tokenRequest.Namespace = *serviceAccountRef.Namespace
  883. }
  884. tokenResponse, err := v.corev1.ServiceAccounts(tokenRequest.Namespace).CreateToken(ctx, serviceAccountRef.Name, tokenRequest, metav1.CreateOptions{})
  885. if err != nil {
  886. return "", fmt.Errorf(errGetKubeSATokenRequest, serviceAccountRef.Name, err)
  887. }
  888. return tokenResponse.Status.Token, nil
  889. }
  890. // checkToken does a lookup and checks if the provided token exists.
  891. func checkToken(ctx context.Context, vStore *client) (bool, error) {
  892. // https://www.vaultproject.io/api-docs/auth/token#lookup-a-token-self
  893. resp, err := vStore.token.LookupSelfWithContext(ctx)
  894. if err != nil {
  895. return false, err
  896. }
  897. t, ok := resp.Data["type"]
  898. if !ok {
  899. return false, fmt.Errorf("could not assert token type")
  900. }
  901. tokenType := t.(string)
  902. if tokenType == "batch" {
  903. return false, nil
  904. }
  905. return true, nil
  906. }
  907. func (v *client) requestTokenWithAppRoleRef(ctx context.Context, appRole *esv1beta1.VaultAppRole) error {
  908. roleID := strings.TrimSpace(appRole.RoleID)
  909. secretID, err := v.secretKeyRef(ctx, &appRole.SecretRef)
  910. if err != nil {
  911. return err
  912. }
  913. secret := approle.SecretID{FromString: secretID}
  914. appRoleClient, err := approle.NewAppRoleAuth(roleID, &secret, approle.WithMountPath(appRole.Path))
  915. if err != nil {
  916. return err
  917. }
  918. _, err = v.auth.Login(ctx, appRoleClient)
  919. if err != nil {
  920. return err
  921. }
  922. return nil
  923. }
  924. func (v *client) requestTokenWithKubernetesAuth(ctx context.Context, kubernetesAuth *esv1beta1.VaultKubernetesAuth) error {
  925. jwtString, err := getJwtString(ctx, v, kubernetesAuth)
  926. if err != nil {
  927. return err
  928. }
  929. k, err := authkubernetes.NewKubernetesAuth(kubernetesAuth.Role, authkubernetes.WithServiceAccountToken(jwtString), authkubernetes.WithMountPath(kubernetesAuth.Path))
  930. if err != nil {
  931. return err
  932. }
  933. _, err = v.auth.Login(ctx, k)
  934. if err != nil {
  935. return err
  936. }
  937. return nil
  938. }
  939. func getJwtString(ctx context.Context, v *client, kubernetesAuth *esv1beta1.VaultKubernetesAuth) (string, error) {
  940. if kubernetesAuth.ServiceAccountRef != nil {
  941. jwt, err := v.secretKeyRefForServiceAccount(ctx, kubernetesAuth.ServiceAccountRef)
  942. if err != nil {
  943. return "", err
  944. }
  945. return jwt, nil
  946. } else if kubernetesAuth.SecretRef != nil {
  947. tokenRef := kubernetesAuth.SecretRef
  948. if tokenRef.Key == "" {
  949. tokenRef = kubernetesAuth.SecretRef.DeepCopy()
  950. tokenRef.Key = "token"
  951. }
  952. jwt, err := v.secretKeyRef(ctx, tokenRef)
  953. if err != nil {
  954. return "", err
  955. }
  956. return jwt, nil
  957. } else {
  958. // Kubernetes authentication is specified, but without a referenced
  959. // Kubernetes secret. We check if the file path for in-cluster service account
  960. // exists and attempt to use the token for Vault Kubernetes auth.
  961. if _, err := os.Stat(serviceAccTokenPath); err != nil {
  962. return "", fmt.Errorf(errServiceAccount, err)
  963. }
  964. jwtByte, err := os.ReadFile(serviceAccTokenPath)
  965. if err != nil {
  966. return "", fmt.Errorf(errServiceAccount, err)
  967. }
  968. return string(jwtByte), nil
  969. }
  970. }
  971. func (v *client) requestTokenWithLdapAuth(ctx context.Context, ldapAuth *esv1beta1.VaultLdapAuth) error {
  972. username := strings.TrimSpace(ldapAuth.Username)
  973. password, err := v.secretKeyRef(ctx, &ldapAuth.SecretRef)
  974. if err != nil {
  975. return err
  976. }
  977. pass := authldap.Password{FromString: password}
  978. l, err := authldap.NewLDAPAuth(username, &pass, authldap.WithMountPath(ldapAuth.Path))
  979. if err != nil {
  980. return err
  981. }
  982. _, err = v.auth.Login(ctx, l)
  983. if err != nil {
  984. return err
  985. }
  986. return nil
  987. }
  988. func (v *client) requestTokenWithJwtAuth(ctx context.Context, jwtAuth *esv1beta1.VaultJwtAuth) error {
  989. role := strings.TrimSpace(jwtAuth.Role)
  990. var jwt string
  991. var err error
  992. if jwtAuth.SecretRef != nil {
  993. jwt, err = v.secretKeyRef(ctx, jwtAuth.SecretRef)
  994. } else if k8sServiceAccountToken := jwtAuth.KubernetesServiceAccountToken; k8sServiceAccountToken != nil {
  995. audiences := k8sServiceAccountToken.Audiences
  996. if audiences == nil {
  997. audiences = &[]string{"vault"}
  998. }
  999. expirationSeconds := k8sServiceAccountToken.ExpirationSeconds
  1000. if expirationSeconds == nil {
  1001. tmp := int64(600)
  1002. expirationSeconds = &tmp
  1003. }
  1004. jwt, err = v.serviceAccountToken(ctx, k8sServiceAccountToken.ServiceAccountRef, *audiences, *expirationSeconds)
  1005. } else {
  1006. err = fmt.Errorf(errJwtNoTokenSource)
  1007. }
  1008. if err != nil {
  1009. return err
  1010. }
  1011. parameters := map[string]interface{}{
  1012. "role": role,
  1013. "jwt": jwt,
  1014. }
  1015. url := strings.Join([]string{"auth", jwtAuth.Path, "login"}, "/")
  1016. vaultResult, err := v.logical.WriteWithContext(ctx, url, parameters)
  1017. if err != nil {
  1018. return err
  1019. }
  1020. token, err := vaultResult.TokenID()
  1021. if err != nil {
  1022. return fmt.Errorf(errVaultToken, err)
  1023. }
  1024. v.client.SetToken(token)
  1025. return nil
  1026. }
  1027. func (v *client) requestTokenWithCertAuth(ctx context.Context, certAuth *esv1beta1.VaultCertAuth, cfg *vault.Config) error {
  1028. clientKey, err := v.secretKeyRef(ctx, &certAuth.SecretRef)
  1029. if err != nil {
  1030. return err
  1031. }
  1032. clientCert, err := v.secretKeyRef(ctx, &certAuth.ClientCert)
  1033. if err != nil {
  1034. return err
  1035. }
  1036. cert, err := tls.X509KeyPair([]byte(clientCert), []byte(clientKey))
  1037. if err != nil {
  1038. return fmt.Errorf(errClientTLSAuth, err)
  1039. }
  1040. if transport, ok := cfg.HttpClient.Transport.(*http.Transport); ok {
  1041. transport.TLSClientConfig.Certificates = []tls.Certificate{cert}
  1042. }
  1043. url := strings.Join([]string{"auth", "cert", "login"}, "/")
  1044. vaultResult, err := v.logical.WriteWithContext(ctx, url, nil)
  1045. if err != nil {
  1046. return fmt.Errorf(errVaultRequest, err)
  1047. }
  1048. token, err := vaultResult.TokenID()
  1049. if err != nil {
  1050. return fmt.Errorf(errVaultToken, err)
  1051. }
  1052. v.client.SetToken(token)
  1053. return nil
  1054. }