vault_test.go 31 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072
  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. "bytes"
  15. "context"
  16. "encoding/json"
  17. "errors"
  18. "fmt"
  19. "io/ioutil"
  20. "net/http"
  21. "testing"
  22. "github.com/crossplane/crossplane-runtime/pkg/test"
  23. "github.com/google/go-cmp/cmp"
  24. vault "github.com/hashicorp/vault/api"
  25. corev1 "k8s.io/api/core/v1"
  26. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  27. kclient "sigs.k8s.io/controller-runtime/pkg/client"
  28. esv1alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1alpha1"
  29. esmeta "github.com/external-secrets/external-secrets/apis/meta/v1"
  30. "github.com/external-secrets/external-secrets/pkg/provider/vault/fake"
  31. )
  32. const (
  33. tokenSecretName = "example-secret-token"
  34. secretDataString = "some-creds"
  35. )
  36. var (
  37. secretStorePath = "secret"
  38. )
  39. func makeValidSecretStoreWithVersion(v esv1alpha1.VaultKVStoreVersion) *esv1alpha1.SecretStore {
  40. return &esv1alpha1.SecretStore{
  41. ObjectMeta: metav1.ObjectMeta{
  42. Name: "vault-store",
  43. Namespace: "default",
  44. },
  45. Spec: esv1alpha1.SecretStoreSpec{
  46. Provider: &esv1alpha1.SecretStoreProvider{
  47. Vault: &esv1alpha1.VaultProvider{
  48. Server: "vault.example.com",
  49. Path: &secretStorePath,
  50. Version: v,
  51. Auth: esv1alpha1.VaultAuth{
  52. Kubernetes: &esv1alpha1.VaultKubernetesAuth{
  53. Path: "kubernetes",
  54. Role: "kubernetes-auth-role",
  55. ServiceAccountRef: &esmeta.ServiceAccountSelector{
  56. Name: "example-sa",
  57. },
  58. },
  59. },
  60. },
  61. },
  62. },
  63. }
  64. }
  65. func makeValidSecretStore() *esv1alpha1.SecretStore {
  66. return makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV2)
  67. }
  68. func makeValidSecretStoreWithCerts() *esv1alpha1.SecretStore {
  69. return &esv1alpha1.SecretStore{
  70. ObjectMeta: metav1.ObjectMeta{
  71. Name: "vault-store",
  72. Namespace: "default",
  73. },
  74. Spec: esv1alpha1.SecretStoreSpec{
  75. Provider: &esv1alpha1.SecretStoreProvider{
  76. Vault: &esv1alpha1.VaultProvider{
  77. Server: "vault.example.com",
  78. Path: &secretStorePath,
  79. Version: esv1alpha1.VaultKVStoreV2,
  80. Auth: esv1alpha1.VaultAuth{
  81. Cert: &esv1alpha1.VaultCertAuth{
  82. ClientCert: esmeta.SecretKeySelector{
  83. Name: "tls-auth-certs",
  84. Key: "tls.crt",
  85. },
  86. SecretRef: esmeta.SecretKeySelector{
  87. Name: "tls-auth-certs",
  88. Key: "tls.key",
  89. },
  90. },
  91. },
  92. },
  93. },
  94. },
  95. }
  96. }
  97. func makeValidSecretStoreWithK8sCerts(isSecret bool) *esv1alpha1.SecretStore {
  98. store := makeSecretStore()
  99. caProvider := &esv1alpha1.CAProvider{
  100. Name: "vault-cert",
  101. Key: "cert",
  102. }
  103. if isSecret {
  104. caProvider.Type = "Secret"
  105. } else {
  106. caProvider.Type = "ConfigMap"
  107. }
  108. store.Spec.Provider.Vault.CAProvider = caProvider
  109. return store
  110. }
  111. func makeInvalidClusterSecretStoreWithK8sCerts() *esv1alpha1.ClusterSecretStore {
  112. return &esv1alpha1.ClusterSecretStore{
  113. TypeMeta: metav1.TypeMeta{
  114. Kind: "ClusterSecretStore",
  115. },
  116. ObjectMeta: metav1.ObjectMeta{
  117. Name: "vault-store",
  118. Namespace: "default",
  119. },
  120. Spec: esv1alpha1.SecretStoreSpec{
  121. Provider: &esv1alpha1.SecretStoreProvider{
  122. Vault: &esv1alpha1.VaultProvider{
  123. Server: "vault.example.com",
  124. Path: &secretStorePath,
  125. Version: "v2",
  126. Auth: esv1alpha1.VaultAuth{
  127. Kubernetes: &esv1alpha1.VaultKubernetesAuth{
  128. Path: "kubernetes",
  129. Role: "kubernetes-auth-role",
  130. ServiceAccountRef: &esmeta.ServiceAccountSelector{
  131. Name: "example-sa",
  132. },
  133. },
  134. },
  135. CAProvider: &esv1alpha1.CAProvider{
  136. Name: "vault-cert",
  137. Key: "cert",
  138. Type: "Secret",
  139. },
  140. },
  141. },
  142. },
  143. }
  144. }
  145. type secretStoreTweakFn func(s *esv1alpha1.SecretStore)
  146. func makeSecretStore(tweaks ...secretStoreTweakFn) *esv1alpha1.SecretStore {
  147. store := makeValidSecretStore()
  148. for _, fn := range tweaks {
  149. fn(store)
  150. }
  151. return store
  152. }
  153. func newVaultResponse(data *vault.Secret) *vault.Response {
  154. jsonData, _ := json.Marshal(data)
  155. return &vault.Response{
  156. Response: &http.Response{
  157. Body: ioutil.NopCloser(bytes.NewReader(jsonData)),
  158. },
  159. }
  160. }
  161. func newVaultResponseWithData(data map[string]interface{}) *vault.Response {
  162. return newVaultResponse(&vault.Secret{
  163. Data: data,
  164. })
  165. }
  166. func newVaultTokenIDResponse(token string) *vault.Response {
  167. return newVaultResponseWithData(map[string]interface{}{
  168. "id": token,
  169. })
  170. }
  171. type args struct {
  172. newClientFunc func(c *vault.Config) (Client, error)
  173. store esv1alpha1.GenericStore
  174. kube kclient.Client
  175. ns string
  176. }
  177. type want struct {
  178. err error
  179. }
  180. type testCase struct {
  181. reason string
  182. args args
  183. want want
  184. }
  185. func clientWithLoginMock(c *vault.Config) (Client, error) {
  186. return &fake.VaultClient{
  187. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  188. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  189. newVaultTokenIDResponse("test-token"), nil, func(got *vault.Request) error { return nil }),
  190. MockSetToken: fake.NewSetTokenFn(),
  191. }, nil
  192. }
  193. func kubeMockWithSecretTokenAndServiceAcc(obj kclient.Object) error {
  194. if o, ok := obj.(*corev1.ServiceAccount); ok {
  195. o.Secrets = []corev1.ObjectReference{
  196. {
  197. Name: tokenSecretName,
  198. },
  199. }
  200. return nil
  201. }
  202. if o, ok := obj.(*corev1.Secret); ok {
  203. o.Data = map[string][]byte{
  204. "token": []byte(secretDataString),
  205. }
  206. return nil
  207. }
  208. return nil
  209. }
  210. func TestNewVault(t *testing.T) {
  211. errBoom := errors.New("boom")
  212. secretClientKey := []byte(`-----BEGIN RSA PRIVATE KEY-----
  213. MIIEpAIBAAKCAQEArfZ4HV1obFVlVNiA24tX/UOakqRnEtWXpIvaOsMaPGvvODgGe4XnyJGO32idPv85sIr7vDH9p+OhactVlJV1fu5SZoZ7pg4jTCLqVDCb3IRD++yik2Sw58YayNe3HiaCTsJQWeMXLzfaqOeyk6bEpBCJo09+3QxUWxijgJ7YZCb+Gi8pf3ZWeSZG+rGNNvXHmTs1Yu1H849SYXu+uJOd/R3ZSTw8CxFe4eTLgbCnPf6tgA8Sg2hc+CAZxunPP2JLZWbiJXxjNRoypso6MAJ1FRkx5sTJiLg6UoLvd95/S/lCVOR2PDlM1hg7ox8VEd4QHky7tLx7gji/5hHQKJQSTwIDAQABAoIBAQCYPICQ8hVX+MNcpLrfZenycR7sBYNOMC0silbH5cUn6yzFfgHuRxi3pOnrCJnTb3cE0BvMbdMVAVdYReD2znSsR9NEdZvvjZ/GGSgH1SIQsI7t//+mDQ/jRLJb4KsXb4vJcLLwdpLrd22bMmhMXjzndrF8gSz8NLX9omozPM8RlLxjzPzYOdlX/Zw8V68qQH2Ic04KbtnCwyAUIgAJxYtn/uYB8lzILBkyzQqwhQKkDDZQ0wbZT0hP6z+HgsdifwQvHG1GZAgCuzzyXrL/4TgDaDhYdMVoBA4+HPmzqm5MkBvjH4oqroxjRofUroVix0OGXZJMI1OJ0z/ubzmwCq5BAoGBANqbwzAydUJs0P+GFL94K/Y6tXULKA2c9N0crbxoxheobRpuJvhpW1ZE/9UGpaYX1Rw3nW4x+Jwvt83YkgHAlR4LgEwDvdJPZobybfqifQDiraUO0t62Crn8mSxOsFCugtRIFniwnX67w3uKxiSdCZYbJGs9JEDTpxRG/PSWq3QlAoGBAMu3zOv1PJAhOky7VcxFxWQPEMY+t2PA/sneD01/qgGuhlTwL4QlpywmBqXcI070dcvcBkP0flnWI7y5cnuE1+55twmsrvfaS8s1+AYje0b35DsaF2vtKuJrXC0AGKP+/eiycd9cbvVW2GWOxE7Ui76Mj95MARK8ZNjt0wJagQhjAoGASm9dD80uhhadN1RFPkjB1054OMk6sx/tdFhug8e9I5MSyzwUguME2aQW5EcmIh7dToVVUo8rUqsgz7NdS8FyRM+vuLJRcQneJDbp4bxwCdwlOh2JCZI8psVutlp4yJATNgrxs9iXV+7BChDflNnvyK+nP+iKrpQiwNHHEdU3vg0CgYEAvEpwD4+loJn1psJn9NxwK6F5IaMKIhtZ4/9pKXpcCh3jb1JouL2MnFOxRVAJGor87aW57Mlol2RDt8W4OM56PqMlOL3xIokUEQka66GT6e5pdu8QwuJ9BrWwhq9WFw4yZQe6FHb836qbbJLegvYVC9QjjZW2UDjtBUwcAkrghH0CgYBUMmMOCwIfMEtMaWxZRGdxRabazLhn7TXhBpVTuv7WouPaXYd7ZGjCTMKAuVa/E4afBlxgemnqBuX90gHpK/dDmn9l+lp8GZey0grJ7G0x5HEMiKziaX5PrgAcKbQ70m9ZNZ1deYhsC05X8rHNexZB6ns7Yms9L7qnlAy51ZH2zw==
  214. -----END RSA PRIVATE KEY-----`)
  215. clientCrt := []byte(`-----BEGIN CERTIFICATE-----
  216. MIICsTCCAZkCFEJJ4daz5sxkFlzq9n1djLEuG7bmMA0GCSqGSIb3DQEBCwUAMBMxETAPBgNVBAMMCHZhdWx0LWNhMB4XDTIxMDcyMDA4MTQxM1oXDTIyMDcyMDA4MTQxM1owFzEVMBMGA1UEAwwMdmF1bHQtY2xpZW50MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEArfZ4HV1obFVlVNiA24tX/UOakqRnEtWXpIvaOsMaPGvvODgGe4XnyJGO32idPv85sIr7vDH9p+OhactVlJV1fu5SZoZ7pg4jTCLqVDCb3IRD++yik2Sw58YayNe3HiaCTsJQWeMXLzfaqOeyk6bEpBCJo09+3QxUWxijgJ7YZCb+Gi8pf3ZWeSZG+rGNNvXHmTs1Yu1H849SYXu+uJOd/R3ZSTw8CxFe4eTLgbCnPf6tgA8Sg2hc+CAZxunPP2JLZWbiJXxjNRoypso6MAJ1FRkx5sTJiLg6UoLvd95/S/lCVOR2PDlM1hg7ox8VEd4QHky7tLx7gji/5hHQKJQSTwIDAQABMA0GCSqGSIb3DQEBCwUAA4IBAQAsDYKtzScIA7bqIOmqF8rr+oLSjRhPt5OfT+KGNdXk8G3VAy1ED2tyCHaRNC7dPLq4EvcxbIXQnXPy1iZMofriGbFPAcQ2fyWUesAD6bYSpI+bYxwz6Ebb93hU5nc/FyXg8yh0kgiGbY3MrACPjxqP2+z5kcOC3u3hx3SZylgW7TeOXDTdqSbNfH1b+1rR/bVNgQQshjhU9d+c4Yv/t0u07uykBhHLWZDSnYiAeOZ8+mWuOSDkcZHE1zznx74fWgtN0zRDtr0L0w9evT9R2CnNSZGxXcEQxAlQ7SL/Jyw82TFCGEw0L4jj7jjvx0N5J8KX/DulUDE9vuVyQEJ88Epe
  217. -----END CERTIFICATE-----
  218. `)
  219. secretData := []byte(secretDataString)
  220. cases := map[string]testCase{
  221. "InvalidVaultStore": {
  222. reason: "Should return error if given an invalid vault store.",
  223. args: args{
  224. store: &esv1alpha1.SecretStore{},
  225. },
  226. want: want{
  227. err: errors.New(errVaultStore),
  228. },
  229. },
  230. "AddVaultStoreCertsError": {
  231. reason: "Should return error if given an invalid CA certificate.",
  232. args: args{
  233. store: makeSecretStore(func(s *esv1alpha1.SecretStore) {
  234. s.Spec.Provider.Vault.CABundle = []byte("badcertdata")
  235. }),
  236. },
  237. want: want{
  238. err: errors.New(errVaultCert),
  239. },
  240. },
  241. "VaultAuthFormatError": {
  242. reason: "Should return error if no valid authentication method is given.",
  243. args: args{
  244. store: makeSecretStore(func(s *esv1alpha1.SecretStore) {
  245. s.Spec.Provider.Vault.Auth = esv1alpha1.VaultAuth{}
  246. }),
  247. },
  248. want: want{
  249. err: errors.New(errAuthFormat),
  250. },
  251. },
  252. "GetKubeServiceAccountError": {
  253. reason: "Should return error if fetching kubernetes secret fails.",
  254. args: args{
  255. store: makeSecretStore(),
  256. kube: &test.MockClient{
  257. MockGet: test.NewMockGetFn(errBoom),
  258. },
  259. },
  260. want: want{
  261. err: fmt.Errorf(errGetKubeSA, "example-sa", errBoom),
  262. },
  263. },
  264. "GetKubeSecretError": {
  265. reason: "Should return error if fetching kubernetes secret fails.",
  266. args: args{
  267. store: makeSecretStore(func(s *esv1alpha1.SecretStore) {
  268. s.Spec.Provider.Vault.Auth.Kubernetes.ServiceAccountRef = nil
  269. s.Spec.Provider.Vault.Auth.Kubernetes.SecretRef = &esmeta.SecretKeySelector{
  270. Name: "vault-secret",
  271. Key: "key",
  272. }
  273. }),
  274. kube: &test.MockClient{
  275. MockGet: test.NewMockGetFn(errBoom),
  276. },
  277. },
  278. want: want{
  279. err: fmt.Errorf(errGetKubeSecret, "vault-secret", errBoom),
  280. },
  281. },
  282. "SuccessfulVaultStore": {
  283. reason: "Should return a Vault provider successfully",
  284. args: args{
  285. store: makeSecretStore(),
  286. kube: &test.MockClient{
  287. MockGet: test.NewMockGetFn(nil, kubeMockWithSecretTokenAndServiceAcc),
  288. },
  289. newClientFunc: func(c *vault.Config) (Client, error) {
  290. return &fake.VaultClient{
  291. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  292. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  293. newVaultTokenIDResponse("test-token"), nil, func(got *vault.Request) error {
  294. kubeRole := makeValidSecretStore().Spec.Provider.Vault.Auth.Kubernetes.Role
  295. want := kubeParameters(kubeRole, string(secretData))
  296. if diff := cmp.Diff(want, got.Obj); diff != "" {
  297. t.Errorf("RawRequestWithContext(...): -want, +got:\n%s", diff)
  298. }
  299. return nil
  300. }),
  301. MockSetToken: fake.NewSetTokenFn(),
  302. MockToken: fake.NewTokenFn(""),
  303. MockClearToken: fake.NewClearTokenFn(),
  304. }, nil
  305. },
  306. },
  307. want: want{
  308. err: nil,
  309. },
  310. },
  311. "SuccessfulVaultStoreWithCertAuth": {
  312. reason: "Should return a Vault provider successfully",
  313. args: args{
  314. store: makeValidSecretStoreWithCerts(),
  315. kube: &test.MockClient{
  316. MockGet: test.NewMockGetFn(nil, func(obj kclient.Object) error {
  317. if o, ok := obj.(*corev1.Secret); ok {
  318. o.Data = map[string][]byte{
  319. "tls.key": secretClientKey,
  320. "tls.crt": clientCrt,
  321. }
  322. return nil
  323. }
  324. return nil
  325. }),
  326. },
  327. newClientFunc: clientWithLoginMock,
  328. },
  329. want: want{
  330. err: nil,
  331. },
  332. },
  333. "SuccessfulVaultStoreWithK8sCertSecret": {
  334. reason: "Should return a Vault prodvider with the cert from k8s",
  335. args: args{
  336. store: makeValidSecretStoreWithK8sCerts(true),
  337. kube: &test.MockClient{
  338. MockGet: test.NewMockGetFn(nil, func(obj kclient.Object) error {
  339. if o, ok := obj.(*corev1.Secret); ok {
  340. o.Data = map[string][]byte{
  341. "cert": clientCrt,
  342. "token": secretData,
  343. }
  344. return nil
  345. }
  346. if o, ok := obj.(*corev1.ServiceAccount); ok {
  347. o.Secrets = []corev1.ObjectReference{
  348. {
  349. Name: tokenSecretName,
  350. },
  351. }
  352. return nil
  353. }
  354. return nil
  355. }),
  356. },
  357. newClientFunc: clientWithLoginMock,
  358. },
  359. want: want{
  360. err: nil,
  361. },
  362. },
  363. "GetCertNamespaceMissingError": {
  364. reason: "Should return an error if namespace is missing and is a ClusterSecretStore",
  365. args: args{
  366. store: makeInvalidClusterSecretStoreWithK8sCerts(),
  367. kube: &test.MockClient{
  368. MockGet: test.NewMockGetFn(nil, kubeMockWithSecretTokenAndServiceAcc),
  369. },
  370. },
  371. want: want{
  372. err: errors.New(errCANamespace),
  373. },
  374. },
  375. "GetCertSecretKeyMissingError": {
  376. reason: "Should return an error if the secret key is missing",
  377. args: args{
  378. store: makeValidSecretStoreWithK8sCerts(true),
  379. kube: &test.MockClient{
  380. MockGet: test.NewMockGetFn(nil, kubeMockWithSecretTokenAndServiceAcc),
  381. },
  382. newClientFunc: clientWithLoginMock,
  383. },
  384. want: want{
  385. err: fmt.Errorf(errVaultCert, errors.New(`cannot find secret data for key: "cert"`)),
  386. },
  387. },
  388. "SuccessfulVaultStoreWithK8sCertConfigMap": {
  389. reason: "Should return a Vault prodvider with the cert from k8s",
  390. args: args{
  391. store: makeValidSecretStoreWithK8sCerts(false),
  392. kube: &test.MockClient{
  393. MockGet: test.NewMockGetFn(nil, func(obj kclient.Object) error {
  394. if o, ok := obj.(*corev1.ConfigMap); ok {
  395. o.Data = map[string]string{
  396. "cert": string(clientCrt),
  397. }
  398. return nil
  399. }
  400. if o, ok := obj.(*corev1.ServiceAccount); ok {
  401. o.Secrets = []corev1.ObjectReference{
  402. {
  403. Name: tokenSecretName,
  404. },
  405. }
  406. return nil
  407. }
  408. if o, ok := obj.(*corev1.Secret); ok {
  409. o.Data = map[string][]byte{
  410. "token": secretData,
  411. }
  412. return nil
  413. }
  414. return nil
  415. }),
  416. },
  417. newClientFunc: clientWithLoginMock,
  418. },
  419. want: want{
  420. err: nil,
  421. },
  422. },
  423. "GetCertConfigMapMissingError": {
  424. reason: "Should return an error if the config map key is missing",
  425. args: args{
  426. store: makeValidSecretStoreWithK8sCerts(false),
  427. kube: &test.MockClient{
  428. MockGet: test.NewMockGetFn(nil, func(obj kclient.Object) error {
  429. if o, ok := obj.(*corev1.ServiceAccount); ok {
  430. o.Secrets = []corev1.ObjectReference{
  431. {
  432. Name: tokenSecretName,
  433. },
  434. }
  435. return nil
  436. }
  437. if o, ok := obj.(*corev1.Secret); ok {
  438. o.Data = map[string][]byte{
  439. "token": secretData,
  440. }
  441. return nil
  442. }
  443. return nil
  444. }),
  445. },
  446. newClientFunc: clientWithLoginMock,
  447. },
  448. want: want{
  449. err: fmt.Errorf(errConfigMapFmt, "cert"),
  450. },
  451. },
  452. "GetCertificateFormatError": {
  453. reason: "Should return error if client certificate is in wrong format.",
  454. args: args{
  455. store: makeValidSecretStoreWithCerts(),
  456. kube: &test.MockClient{
  457. MockGet: test.NewMockGetFn(nil, func(obj kclient.Object) error {
  458. if o, ok := obj.(*corev1.Secret); ok {
  459. o.Data = map[string][]byte{
  460. "tls.key": secretClientKey,
  461. "tls.crt": []byte("cert with mistak"),
  462. }
  463. return nil
  464. }
  465. return nil
  466. }),
  467. },
  468. newClientFunc: clientWithLoginMock,
  469. },
  470. want: want{
  471. err: fmt.Errorf(errClientTLSAuth, "tls: failed to find any PEM data in certificate input"),
  472. },
  473. },
  474. "GetKeyFormatError": {
  475. reason: "Should return error if client key is in wrong format.",
  476. args: args{
  477. store: makeValidSecretStoreWithCerts(),
  478. kube: &test.MockClient{
  479. MockGet: test.NewMockGetFn(nil, func(obj kclient.Object) error {
  480. if o, ok := obj.(*corev1.Secret); ok {
  481. o.Data = map[string][]byte{
  482. "tls.key": []byte("key with mistake"),
  483. "tls.crt": clientCrt,
  484. }
  485. return nil
  486. }
  487. return nil
  488. }),
  489. },
  490. newClientFunc: clientWithLoginMock,
  491. },
  492. want: want{
  493. err: fmt.Errorf(errClientTLSAuth, "tls: failed to find any PEM data in key input"),
  494. },
  495. },
  496. }
  497. for name, tc := range cases {
  498. t.Run(name, func(t *testing.T) {
  499. vaultTest(t, name, tc)
  500. })
  501. }
  502. }
  503. func vaultTest(t *testing.T, name string, tc testCase) {
  504. conn := &connector{
  505. newVaultClient: tc.args.newClientFunc,
  506. }
  507. if tc.args.newClientFunc == nil {
  508. conn.newVaultClient = newVaultClient
  509. }
  510. _, err := conn.NewClient(context.Background(), tc.args.store, tc.args.kube, tc.args.ns)
  511. if diff := cmp.Diff(tc.want.err, err, test.EquateErrors()); diff != "" {
  512. t.Errorf("\n%s\nvault.New(...): -want error, +got error:\n%s", tc.reason, diff)
  513. }
  514. }
  515. func TestGetSecret(t *testing.T) {
  516. errBoom := errors.New("boom")
  517. secret := map[string]interface{}{
  518. "access_key": "access_key",
  519. "access_secret": "access_secret",
  520. }
  521. secretWithNilVal := map[string]interface{}{
  522. "access_key": "access_key",
  523. "access_secret": "access_secret",
  524. "token": nil,
  525. }
  526. secretWithNestedVal := map[string]interface{}{
  527. "access_key": "access_key",
  528. "access_secret": "access_secret",
  529. "nested.bar": "something different",
  530. "nested": map[string]string{
  531. "foo": "oke",
  532. "bar": "also ok?",
  533. },
  534. }
  535. type args struct {
  536. store *esv1alpha1.VaultProvider
  537. kube kclient.Client
  538. vClient Client
  539. ns string
  540. data esv1alpha1.ExternalSecretDataRemoteRef
  541. }
  542. type want struct {
  543. err error
  544. val []byte
  545. }
  546. cases := map[string]struct {
  547. reason string
  548. args args
  549. want want
  550. }{
  551. "ReadSecret": {
  552. reason: "Should return the secret with property",
  553. args: args{
  554. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  555. data: esv1alpha1.ExternalSecretDataRemoteRef{
  556. Property: "access_key",
  557. },
  558. vClient: &fake.VaultClient{
  559. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  560. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  561. newVaultResponseWithData(secret), nil,
  562. ),
  563. },
  564. },
  565. want: want{
  566. err: nil,
  567. val: []byte("access_key"),
  568. },
  569. },
  570. "ReadSecretWithNil": {
  571. reason: "Should return the secret with property if it has a nil val",
  572. args: args{
  573. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  574. data: esv1alpha1.ExternalSecretDataRemoteRef{
  575. Property: "access_key",
  576. },
  577. vClient: &fake.VaultClient{
  578. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  579. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  580. newVaultResponseWithData(secretWithNilVal), nil,
  581. ),
  582. },
  583. },
  584. want: want{
  585. err: nil,
  586. val: []byte("access_key"),
  587. },
  588. },
  589. "ReadSecretWithoutProperty": {
  590. reason: "Should return the json encoded secret without property",
  591. args: args{
  592. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  593. data: esv1alpha1.ExternalSecretDataRemoteRef{},
  594. vClient: &fake.VaultClient{
  595. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  596. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  597. newVaultResponseWithData(secret), nil,
  598. ),
  599. },
  600. },
  601. want: want{
  602. err: nil,
  603. val: []byte(`{"access_key":"access_key","access_secret":"access_secret"}`),
  604. },
  605. },
  606. "ReadSecretWithNestedValue": {
  607. reason: "Should return a nested property",
  608. args: args{
  609. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  610. data: esv1alpha1.ExternalSecretDataRemoteRef{
  611. Property: "nested.foo",
  612. },
  613. vClient: &fake.VaultClient{
  614. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  615. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  616. newVaultResponseWithData(secretWithNestedVal), nil,
  617. ),
  618. },
  619. },
  620. want: want{
  621. err: nil,
  622. val: []byte("oke"),
  623. },
  624. },
  625. "ReadSecretWithNestedValueFromData": {
  626. reason: "Should return a nested property",
  627. args: args{
  628. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  629. data: esv1alpha1.ExternalSecretDataRemoteRef{
  630. //
  631. Property: "nested.bar",
  632. },
  633. vClient: &fake.VaultClient{
  634. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  635. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  636. newVaultResponseWithData(secretWithNestedVal), nil,
  637. ),
  638. },
  639. },
  640. want: want{
  641. err: nil,
  642. val: []byte("something different"),
  643. },
  644. },
  645. "NonexistentProperty": {
  646. reason: "Should return error property does not exist.",
  647. args: args{
  648. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  649. data: esv1alpha1.ExternalSecretDataRemoteRef{
  650. Property: "nop.doesnt.exist",
  651. },
  652. vClient: &fake.VaultClient{
  653. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  654. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  655. newVaultResponseWithData(secretWithNestedVal), nil,
  656. ),
  657. },
  658. },
  659. want: want{
  660. err: fmt.Errorf(errSecretKeyFmt, "nop.doesnt.exist"),
  661. },
  662. },
  663. "ReadSecretError": {
  664. reason: "Should return error if vault client fails to read secret.",
  665. args: args{
  666. store: makeSecretStore().Spec.Provider.Vault,
  667. vClient: &fake.VaultClient{
  668. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  669. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(nil, errBoom),
  670. },
  671. },
  672. want: want{
  673. err: fmt.Errorf(errReadSecret, errBoom),
  674. },
  675. },
  676. }
  677. for name, tc := range cases {
  678. t.Run(name, func(t *testing.T) {
  679. vStore := &client{
  680. kube: tc.args.kube,
  681. client: tc.args.vClient,
  682. store: tc.args.store,
  683. namespace: tc.args.ns,
  684. }
  685. val, err := vStore.GetSecret(context.Background(), tc.args.data)
  686. if diff := cmp.Diff(tc.want.err, err, test.EquateErrors()); diff != "" {
  687. t.Errorf("\n%s\nvault.GetSecret(...): -want error, +got error:\n%s", tc.reason, diff)
  688. }
  689. if diff := cmp.Diff(string(tc.want.val), string(val)); diff != "" {
  690. t.Errorf("\n%s\nvault.GetSecret(...): -want val, +got val:\n%s", tc.reason, diff)
  691. }
  692. })
  693. }
  694. }
  695. func TestGetSecretMap(t *testing.T) {
  696. errBoom := errors.New("boom")
  697. secret := map[string]interface{}{
  698. "access_key": "access_key",
  699. "access_secret": "access_secret",
  700. }
  701. secretWithNilVal := map[string]interface{}{
  702. "access_key": "access_key",
  703. "access_secret": "access_secret",
  704. "token": nil,
  705. }
  706. secretWithNestedVal := map[string]interface{}{
  707. "access_key": "access_key",
  708. "access_secret": "access_secret",
  709. "nested": map[string]interface{}{
  710. "foo": map[string]string{
  711. "oke": "yup",
  712. "mhkeih": "yada yada",
  713. },
  714. },
  715. }
  716. secretWithTypes := map[string]interface{}{
  717. "access_secret": "access_secret",
  718. "f32": float32(2.12),
  719. "f64": float64(2.1234534153423423),
  720. "int": 42,
  721. "bool": true,
  722. "bt": []byte("foobar"),
  723. }
  724. type args struct {
  725. store *esv1alpha1.VaultProvider
  726. kube kclient.Client
  727. vClient Client
  728. ns string
  729. data esv1alpha1.ExternalSecretDataRemoteRef
  730. }
  731. type want struct {
  732. err error
  733. val map[string][]byte
  734. }
  735. cases := map[string]struct {
  736. reason string
  737. args args
  738. want want
  739. }{
  740. "ReadSecretKV1": {
  741. reason: "Should map the secret even if it has a nil value",
  742. args: args{
  743. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  744. vClient: &fake.VaultClient{
  745. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  746. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  747. newVaultResponseWithData(secret), nil,
  748. ),
  749. },
  750. },
  751. want: want{
  752. err: nil,
  753. val: map[string][]byte{
  754. "access_key": []byte("access_key"),
  755. "access_secret": []byte("access_secret"),
  756. },
  757. },
  758. },
  759. "ReadSecretKV2": {
  760. reason: "Should map the secret even if it has a nil value",
  761. args: args{
  762. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV2).Spec.Provider.Vault,
  763. vClient: &fake.VaultClient{
  764. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  765. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  766. newVaultResponseWithData(
  767. map[string]interface{}{
  768. "data": secret,
  769. },
  770. ), nil,
  771. ),
  772. },
  773. },
  774. want: want{
  775. err: nil,
  776. val: map[string][]byte{
  777. "access_key": []byte("access_key"),
  778. "access_secret": []byte("access_secret"),
  779. },
  780. },
  781. },
  782. "ReadSecretWithNilValueKV1": {
  783. reason: "Should map the secret even if it has a nil value",
  784. args: args{
  785. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1).Spec.Provider.Vault,
  786. vClient: &fake.VaultClient{
  787. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  788. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  789. newVaultResponseWithData(secretWithNilVal), nil,
  790. ),
  791. },
  792. },
  793. want: want{
  794. err: nil,
  795. val: map[string][]byte{
  796. "access_key": []byte("access_key"),
  797. "access_secret": []byte("access_secret"),
  798. "token": []byte(nil),
  799. },
  800. },
  801. },
  802. "ReadSecretWithNilValueKV2": {
  803. reason: "Should map the secret even if it has a nil value",
  804. args: args{
  805. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV2).Spec.Provider.Vault,
  806. vClient: &fake.VaultClient{
  807. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  808. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  809. newVaultResponseWithData(
  810. map[string]interface{}{
  811. "data": secretWithNilVal,
  812. },
  813. ), nil,
  814. ),
  815. },
  816. },
  817. want: want{
  818. err: nil,
  819. val: map[string][]byte{
  820. "access_key": []byte("access_key"),
  821. "access_secret": []byte("access_secret"),
  822. "token": []byte(nil),
  823. },
  824. },
  825. },
  826. "ReadSecretWithTypesKV2": {
  827. reason: "Should map the secret even if it has other types",
  828. args: args{
  829. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV2).Spec.Provider.Vault,
  830. vClient: &fake.VaultClient{
  831. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  832. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  833. newVaultResponseWithData(
  834. map[string]interface{}{
  835. "data": secretWithTypes,
  836. },
  837. ), nil,
  838. ),
  839. },
  840. },
  841. want: want{
  842. err: nil,
  843. val: map[string][]byte{
  844. "access_secret": []byte("access_secret"),
  845. "f32": []byte("2.12"),
  846. "f64": []byte("2.1234534153423423"),
  847. "int": []byte("42"),
  848. "bool": []byte("true"),
  849. "bt": []byte("Zm9vYmFy"), // base64
  850. },
  851. },
  852. },
  853. "ReadNestedSecret": {
  854. reason: "Should map the secret for deeply nested property",
  855. args: args{
  856. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV2).Spec.Provider.Vault,
  857. data: esv1alpha1.ExternalSecretDataRemoteRef{
  858. Property: "nested",
  859. },
  860. vClient: &fake.VaultClient{
  861. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  862. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  863. newVaultResponseWithData(
  864. map[string]interface{}{
  865. "data": secretWithNestedVal,
  866. },
  867. ), nil,
  868. ),
  869. },
  870. },
  871. want: want{
  872. err: nil,
  873. val: map[string][]byte{
  874. "foo": []byte(`{"mhkeih":"yada yada","oke":"yup"}`),
  875. },
  876. },
  877. },
  878. "ReadDeeplyNestedSecret": {
  879. reason: "Should map the secret for deeply nested property",
  880. args: args{
  881. store: makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV2).Spec.Provider.Vault,
  882. data: esv1alpha1.ExternalSecretDataRemoteRef{
  883. Property: "nested.foo",
  884. },
  885. vClient: &fake.VaultClient{
  886. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  887. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(
  888. newVaultResponseWithData(
  889. map[string]interface{}{
  890. "data": secretWithNestedVal,
  891. },
  892. ), nil,
  893. ),
  894. },
  895. },
  896. want: want{
  897. err: nil,
  898. val: map[string][]byte{
  899. "oke": []byte("yup"),
  900. "mhkeih": []byte("yada yada"),
  901. },
  902. },
  903. },
  904. "ReadSecretError": {
  905. reason: "Should return error if vault client fails to read secret.",
  906. args: args{
  907. store: makeSecretStore().Spec.Provider.Vault,
  908. vClient: &fake.VaultClient{
  909. MockNewRequest: fake.NewMockNewRequestFn(&vault.Request{}),
  910. MockRawRequestWithContext: fake.NewMockRawRequestWithContextFn(nil, errBoom),
  911. },
  912. },
  913. want: want{
  914. err: fmt.Errorf(errReadSecret, errBoom),
  915. },
  916. },
  917. }
  918. for name, tc := range cases {
  919. t.Run(name, func(t *testing.T) {
  920. vStore := &client{
  921. kube: tc.args.kube,
  922. client: tc.args.vClient,
  923. store: tc.args.store,
  924. namespace: tc.args.ns,
  925. }
  926. val, err := vStore.GetSecretMap(context.Background(), tc.args.data)
  927. if diff := cmp.Diff(tc.want.err, err, test.EquateErrors()); diff != "" {
  928. t.Errorf("\n%s\nvault.GetSecretMap(...): -want error, +got error:\n%s", tc.reason, diff)
  929. }
  930. if diff := cmp.Diff(tc.want.val, val); diff != "" {
  931. t.Errorf("\n%s\nvault.GetSecretMap(...): -want val, +got val:\n%s", tc.reason, diff)
  932. }
  933. })
  934. }
  935. }
  936. func TestGetSecretPath(t *testing.T) {
  937. storeV2 := makeValidSecretStore()
  938. storeV2NoPath := storeV2.DeepCopy()
  939. storeV2NoPath.Spec.Provider.Vault.Path = nil
  940. storeV1 := makeValidSecretStoreWithVersion(esv1alpha1.VaultKVStoreV1)
  941. storeV1NoPath := storeV1.DeepCopy()
  942. storeV1NoPath.Spec.Provider.Vault.Path = nil
  943. type args struct {
  944. store *esv1alpha1.VaultProvider
  945. path string
  946. expected string
  947. }
  948. cases := map[string]struct {
  949. reason string
  950. args args
  951. }{
  952. "PathWithoutFormatV2": {
  953. reason: "Data needs to be found in path",
  954. args: args{
  955. store: storeV2.Spec.Provider.Vault,
  956. path: "secret/test",
  957. expected: "secret/data/test",
  958. },
  959. },
  960. "PathWithDataV2": {
  961. reason: "Data needs to be found only once in path",
  962. args: args{
  963. store: storeV2.Spec.Provider.Vault,
  964. path: "secret/data/test",
  965. expected: "secret/data/test",
  966. },
  967. },
  968. "PathWithoutFormatV2_NoPath": {
  969. reason: "Data needs to be found in path and correct mountpoint is set",
  970. args: args{
  971. store: storeV2NoPath.Spec.Provider.Vault,
  972. path: "secret/test",
  973. expected: "secret/data/test",
  974. },
  975. },
  976. "PathWithoutFormatV1": {
  977. reason: "Data needs to be found in path",
  978. args: args{
  979. store: storeV1.Spec.Provider.Vault,
  980. path: "secret/test",
  981. expected: "secret/test",
  982. },
  983. },
  984. "PathWithoutFormatV1_NoPath": {
  985. reason: "Data needs to be found in path and correct mountpoint is set",
  986. args: args{
  987. store: storeV1NoPath.Spec.Provider.Vault,
  988. path: "secret/test",
  989. expected: "secret/test",
  990. },
  991. },
  992. "WithoutPathButMountpointV2": {
  993. reason: "Mountpoint needs to be set in addition to data",
  994. args: args{
  995. store: storeV2.Spec.Provider.Vault,
  996. path: "test",
  997. expected: "secret/data/test",
  998. },
  999. },
  1000. "WithoutPathButMountpointV1": {
  1001. reason: "Mountpoint needs to be set in addition to data",
  1002. args: args{
  1003. store: storeV1.Spec.Provider.Vault,
  1004. path: "test",
  1005. expected: "secret/test",
  1006. },
  1007. },
  1008. }
  1009. for name, tc := range cases {
  1010. t.Run(name, func(t *testing.T) {
  1011. vStore := &client{
  1012. store: tc.args.store,
  1013. }
  1014. want := vStore.buildPath(tc.args.path)
  1015. if diff := cmp.Diff(want, tc.args.expected); diff != "" {
  1016. t.Errorf("\n%s\nvault.buildPath(...): -want expected, +got error:\n%s", tc.reason, diff)
  1017. }
  1018. })
  1019. }
  1020. }