auth_test.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  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. "errors"
  16. "testing"
  17. "github.com/google/go-cmp/cmp"
  18. "github.com/google/go-cmp/cmp/cmpopts"
  19. vault "github.com/hashicorp/vault/api"
  20. corev1 "k8s.io/api/core/v1"
  21. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  22. "k8s.io/utils/ptr"
  23. clientfake "sigs.k8s.io/controller-runtime/pkg/client/fake"
  24. esv1beta1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1beta1"
  25. esmeta "github.com/external-secrets/external-secrets/apis/meta/v1"
  26. "github.com/external-secrets/external-secrets/pkg/provider/vault/fake"
  27. )
  28. // Test Vault Namespace logic.
  29. func TestSetAuthNamespace(t *testing.T) {
  30. store := makeValidSecretStore()
  31. kube := clientfake.NewClientBuilder().WithObjects(&corev1.Secret{
  32. ObjectMeta: metav1.ObjectMeta{
  33. Name: "vault-secret",
  34. Namespace: "default",
  35. },
  36. Data: map[string][]byte{
  37. "key": []byte("token"),
  38. },
  39. }).Build()
  40. store.Spec.Provider.Vault.Auth.Kubernetes.ServiceAccountRef = nil
  41. store.Spec.Provider.Vault.Auth.Kubernetes.SecretRef = &esmeta.SecretKeySelector{
  42. Name: "vault-secret",
  43. Namespace: ptr.To("default"),
  44. Key: "key",
  45. }
  46. adminNS := "admin"
  47. teamNS := "admin/team-a"
  48. type result struct {
  49. Before string
  50. During string
  51. After string
  52. }
  53. type args struct {
  54. store *esv1beta1.SecretStore
  55. expected result
  56. }
  57. cases := map[string]struct {
  58. reason string
  59. args args
  60. }{
  61. "StoreNoNamespace": {
  62. reason: "no namespace should ever be set",
  63. args: args{
  64. store: store,
  65. expected: result{Before: "", During: "", After: ""},
  66. },
  67. },
  68. "StoreWithNamespace": {
  69. reason: "use the team namespace throughout",
  70. args: args{
  71. store: func(store *esv1beta1.SecretStore) *esv1beta1.SecretStore {
  72. s := store.DeepCopy()
  73. s.Spec.Provider.Vault.Namespace = ptr.To(teamNS)
  74. return s
  75. }(store),
  76. expected: result{Before: teamNS, During: teamNS, After: teamNS},
  77. },
  78. },
  79. "StoreWithAuthNamespace": {
  80. reason: "switch to the auth namespace during login then revert",
  81. args: args{
  82. store: func(store *esv1beta1.SecretStore) *esv1beta1.SecretStore {
  83. s := store.DeepCopy()
  84. s.Spec.Provider.Vault.Auth.Namespace = ptr.To(adminNS)
  85. return s
  86. }(store),
  87. expected: result{Before: "", During: adminNS, After: ""},
  88. },
  89. },
  90. "StoreWithSameNamespace": {
  91. reason: "the admin namespace throughout",
  92. args: args{
  93. store: func(store *esv1beta1.SecretStore) *esv1beta1.SecretStore {
  94. s := store.DeepCopy()
  95. s.Spec.Provider.Vault.Namespace = ptr.To(adminNS)
  96. s.Spec.Provider.Vault.Auth.Namespace = ptr.To(adminNS)
  97. return s
  98. }(store),
  99. expected: result{Before: adminNS, During: adminNS, After: adminNS},
  100. },
  101. },
  102. "StoreWithDistinctNamespace": {
  103. reason: "switch from team namespace, to admin, then back",
  104. args: args{
  105. store: func(store *esv1beta1.SecretStore) *esv1beta1.SecretStore {
  106. s := store.DeepCopy()
  107. s.Spec.Provider.Vault.Namespace = ptr.To(teamNS)
  108. s.Spec.Provider.Vault.Auth.Namespace = ptr.To(adminNS)
  109. return s
  110. }(store),
  111. expected: result{Before: teamNS, During: adminNS, After: teamNS},
  112. },
  113. },
  114. }
  115. for name, tc := range cases {
  116. t.Run(name, func(t *testing.T) {
  117. prov := &Provider{
  118. NewVaultClient: fake.ClientWithLoginMock,
  119. }
  120. c, cfg, err := prov.prepareConfig(context.Background(), kube, nil, tc.args.store.Spec.Provider.Vault, nil, "default", store.GetObjectKind().GroupVersionKind().Kind)
  121. if err != nil {
  122. t.Errorf(err.Error())
  123. }
  124. client, err := getVaultClient(prov, tc.args.store, cfg)
  125. if err != nil {
  126. t.Errorf("vault.useAuthNamespace: failed to create client: %s", err.Error())
  127. }
  128. _, err = prov.initClient(context.Background(), c, client, cfg, tc.args.store.Spec.Provider.Vault)
  129. if err != nil {
  130. t.Errorf("vault.useAuthNamespace: failed to init client: %s", err.Error())
  131. }
  132. c.client = client
  133. // before auth
  134. actual := result{
  135. Before: c.client.Namespace(),
  136. }
  137. // during authentication (getting a token)
  138. resetNS := c.useAuthNamespace(context.Background())
  139. actual.During = c.client.Namespace()
  140. resetNS()
  141. // after getting the token
  142. actual.After = c.client.Namespace()
  143. if diff := cmp.Diff(tc.args.expected, actual, cmpopts.EquateComparable()); diff != "" {
  144. t.Errorf("\n%s\nvault.useAuthNamepsace(...): -want namespace, +got namespace:\n%s", tc.reason, diff)
  145. }
  146. })
  147. }
  148. }
  149. func TestCheckTokenErrors(t *testing.T) {
  150. cases := map[string]struct {
  151. message string
  152. secret *vault.Secret
  153. err error
  154. }{
  155. "SuccessWithNoData": {
  156. message: "should not cache if token lookup returned no data",
  157. secret: &vault.Secret{},
  158. err: nil,
  159. },
  160. "Error": {
  161. message: "should not cache if token lookup errored",
  162. secret: nil,
  163. err: errors.New(""),
  164. },
  165. // This happens when a token is expired and the Vault server returns:
  166. // {"errors":["permission denied"]}
  167. "NoDataNorError": {
  168. message: "should not cache if token lookup returned no data nor error",
  169. secret: nil,
  170. err: nil,
  171. },
  172. }
  173. for name, tc := range cases {
  174. t.Run(name, func(t *testing.T) {
  175. token := fake.Token{
  176. LookupSelfWithContextFn: func(ctx context.Context) (*vault.Secret, error) {
  177. return tc.secret, tc.err
  178. },
  179. }
  180. cached, _ := checkToken(context.Background(), token)
  181. if cached {
  182. t.Errorf("%v", tc.message)
  183. }
  184. })
  185. }
  186. }