utils_test.go 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549
  1. /*
  2. Copyright © The ESO Authors
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. https://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package esutils
  14. import (
  15. "context"
  16. "encoding/json"
  17. "errors"
  18. "reflect"
  19. "testing"
  20. "time"
  21. "github.com/aws/aws-sdk-go-v2/aws"
  22. "github.com/oracle/oci-go-sdk/v65/vault"
  23. "github.com/stretchr/testify/assert"
  24. v1 "k8s.io/api/core/v1"
  25. apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
  26. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  27. clientfake "sigs.k8s.io/controller-runtime/pkg/client/fake"
  28. esv1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1"
  29. esv1alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1alpha1"
  30. esmetav1 "github.com/external-secrets/external-secrets/apis/meta/v1"
  31. )
  32. const (
  33. base64DecodedValue string = "foo%_?bar"
  34. base64EncodedValue string = "Zm9vJV8/YmFy"
  35. base64URLEncodedValue string = "Zm9vJV8_YmFy"
  36. keyWithEmojis string = "😀foo😁bar😂baz😈bing"
  37. keyWithInvalidChars string = "some-array[0].entity"
  38. keyWithEncodedInvalidChars string = "some-array_U005b_0_U005d_.entity"
  39. )
  40. func TestObjectHash(t *testing.T) {
  41. tests := []struct {
  42. name string
  43. input any
  44. want string
  45. }{
  46. {
  47. name: "A nil should be still working",
  48. input: nil,
  49. want: "c461202a18e99215f121936fb2452e03843828e448a00a53f285a6fc",
  50. },
  51. {
  52. name: "We accept a simple scalar value, i.e. string",
  53. input: "hello there",
  54. want: "f78681ec611ebaeea0689bff6c7812a83ff98a7faba986d9af76c999",
  55. },
  56. {
  57. name: "A complex object like a secret is not an issue",
  58. input: v1.Secret{Data: map[string][]byte{
  59. "xx": []byte("yyy"),
  60. }},
  61. want: "9c717e13e4281db3cdad3f56c6e7faab1d7029c4b4fbbf12fbec9b1e",
  62. },
  63. {
  64. name: "map also works",
  65. input: map[string][]byte{
  66. "foo": []byte("value1"),
  67. "bar": []byte("value2"),
  68. },
  69. want: "1bed8bcbcb4547ffe19a19cd47d9078e84aa6598266d86b99f992d64",
  70. },
  71. }
  72. for _, tt := range tests {
  73. t.Run(tt.name, func(t *testing.T) {
  74. if got := ObjectHash(tt.input); got != tt.want {
  75. t.Errorf("ObjectHash() = %v, want %v", got, tt.want)
  76. }
  77. })
  78. }
  79. }
  80. func TestIsNil(t *testing.T) {
  81. tbl := []struct {
  82. name string
  83. val any
  84. exp bool
  85. }{
  86. {
  87. name: "simple nil val",
  88. val: nil,
  89. exp: true,
  90. },
  91. {
  92. name: "nil slice",
  93. val: (*[]struct{})(nil),
  94. exp: true,
  95. },
  96. {
  97. name: "struct pointer",
  98. val: &testing.T{},
  99. exp: false,
  100. },
  101. {
  102. name: "struct",
  103. val: testing.T{},
  104. exp: false,
  105. },
  106. {
  107. name: "slice of struct",
  108. val: []struct{}{{}},
  109. exp: false,
  110. },
  111. {
  112. name: "slice of ptr",
  113. val: []*testing.T{nil},
  114. exp: false,
  115. },
  116. {
  117. name: "slice",
  118. val: []struct{}(nil),
  119. exp: false,
  120. },
  121. {
  122. name: "int default value",
  123. val: 0,
  124. exp: false,
  125. },
  126. {
  127. name: "empty str",
  128. val: "",
  129. exp: false,
  130. },
  131. {
  132. name: "oracle vault",
  133. val: vault.VaultsClient{},
  134. exp: false,
  135. },
  136. {
  137. name: "func",
  138. val: func() {
  139. // noop for testing and to make linter happy
  140. },
  141. exp: false,
  142. },
  143. {
  144. name: "channel",
  145. val: make(chan struct{}),
  146. exp: false,
  147. },
  148. {
  149. name: "map",
  150. val: map[string]string{},
  151. exp: false,
  152. },
  153. }
  154. for _, row := range tbl {
  155. t.Run(row.name, func(t *testing.T) {
  156. res := IsNil(row.val)
  157. if res != row.exp {
  158. t.Errorf("IsNil(%#v)=%t, expected %t", row.val, res, row.exp)
  159. }
  160. })
  161. }
  162. }
  163. func TestConvertKeys(t *testing.T) {
  164. type args struct {
  165. strategy esv1.ExternalSecretConversionStrategy
  166. in map[string][]byte
  167. }
  168. tests := []struct {
  169. name string
  170. args args
  171. want map[string][]byte
  172. wantErr bool
  173. }{
  174. {
  175. name: "convert with special chars",
  176. args: args{
  177. strategy: esv1.ExternalSecretConversionDefault,
  178. in: map[string][]byte{
  179. "foo$bar%baz*bing": []byte(`noop`),
  180. },
  181. },
  182. want: map[string][]byte{
  183. "foo_bar_baz_bing": []byte(`noop`),
  184. },
  185. },
  186. {
  187. name: "error on collision",
  188. args: args{
  189. strategy: esv1.ExternalSecretConversionDefault,
  190. in: map[string][]byte{
  191. "foo$bar%baz*bing": []byte(`noop`),
  192. "foo_bar_baz$bing": []byte(`noop`),
  193. },
  194. },
  195. wantErr: true,
  196. },
  197. {
  198. name: "convert path",
  199. args: args{
  200. strategy: esv1.ExternalSecretConversionDefault,
  201. in: map[string][]byte{
  202. "/foo/bar/baz/bing": []byte(`noop`),
  203. "foo/bar/baz/bing/": []byte(`noop`),
  204. },
  205. },
  206. want: map[string][]byte{
  207. "_foo_bar_baz_bing": []byte(`noop`),
  208. "foo_bar_baz_bing_": []byte(`noop`),
  209. },
  210. },
  211. {
  212. name: "convert unicode",
  213. args: args{
  214. strategy: esv1.ExternalSecretConversionUnicode,
  215. in: map[string][]byte{
  216. keyWithEmojis: []byte(`noop`),
  217. },
  218. },
  219. want: map[string][]byte{
  220. "_U1f600_foo_U1f601_bar_U1f602_baz_U1f608_bing": []byte(`noop`),
  221. },
  222. },
  223. }
  224. for _, tt := range tests {
  225. t.Run(tt.name, func(t *testing.T) {
  226. got, err := ConvertKeys(tt.args.strategy, tt.args.in)
  227. if (err != nil) != tt.wantErr {
  228. t.Errorf("ConvertKeys() error = %v, wantErr %v", err, tt.wantErr)
  229. return
  230. }
  231. if !reflect.DeepEqual(got, tt.want) {
  232. t.Errorf("ConvertKeys() = %v, want %v", got, tt.want)
  233. }
  234. })
  235. }
  236. }
  237. func TestReverseKeys(t *testing.T) {
  238. type args struct {
  239. encodingStrategy esv1.ExternalSecretConversionStrategy
  240. decodingStrategy esv1alpha1.PushSecretConversionStrategy
  241. in map[string][]byte
  242. }
  243. tests := []struct {
  244. name string
  245. args args
  246. want map[string][]byte
  247. wantErr bool
  248. }{
  249. {
  250. name: "encoding and decoding strategy are selecting Unicode conversion and reverse unicode, so the in and want should match, this test covers Unicode characters beyond the Basic Multilingual Plane (BMP)",
  251. args: args{
  252. encodingStrategy: esv1.ExternalSecretConversionUnicode,
  253. decodingStrategy: esv1alpha1.PushSecretConversionReverseUnicode,
  254. in: map[string][]byte{
  255. keyWithEmojis: []byte(`noop`),
  256. },
  257. },
  258. want: map[string][]byte{
  259. keyWithEmojis: []byte(`noop`),
  260. },
  261. },
  262. {
  263. name: "encoding and decoding strategy are selecting Unicode conversion and reverse unicode, so the in and want should match, this test covers Unicode characters in the Basic Multilingual Plane (BMP)",
  264. args: args{
  265. encodingStrategy: esv1.ExternalSecretConversionUnicode,
  266. decodingStrategy: esv1alpha1.PushSecretConversionReverseUnicode,
  267. in: map[string][]byte{
  268. keyWithInvalidChars: []byte(`noop`),
  269. },
  270. },
  271. want: map[string][]byte{
  272. keyWithInvalidChars: []byte(`noop`),
  273. },
  274. },
  275. {
  276. name: "the encoding strategy is selecting Unicode conversion, but the decoding strategy is none, so we want an encoded representation of the content",
  277. args: args{
  278. encodingStrategy: esv1.ExternalSecretConversionUnicode,
  279. decodingStrategy: esv1alpha1.PushSecretConversionNone,
  280. in: map[string][]byte{
  281. keyWithInvalidChars: []byte(`noop`),
  282. },
  283. },
  284. want: map[string][]byte{
  285. keyWithEncodedInvalidChars: []byte(`noop`),
  286. },
  287. },
  288. }
  289. for _, tt := range tests {
  290. t.Run(tt.name, func(t *testing.T) {
  291. got, err := ConvertKeys(tt.args.encodingStrategy, tt.args.in)
  292. if (err != nil) != tt.wantErr {
  293. t.Errorf("ConvertKeys() error = %v, wantErr %v", err, tt.wantErr)
  294. return
  295. }
  296. got, err = ReverseKeys(tt.args.decodingStrategy, got)
  297. if (err != nil) != tt.wantErr {
  298. t.Errorf("ReverseKeys() error = %v, wantErr %v", err, tt.wantErr)
  299. return
  300. }
  301. if !reflect.DeepEqual(got, tt.want) {
  302. t.Errorf("ReverseKeys() = %v, want %v", got, tt.want)
  303. }
  304. })
  305. }
  306. }
  307. func TestDecode(t *testing.T) {
  308. type args struct {
  309. strategy esv1.ExternalSecretDecodingStrategy
  310. in map[string][]byte
  311. }
  312. tests := []struct {
  313. name string
  314. args args
  315. want map[string][]byte
  316. wantErr bool
  317. }{
  318. {
  319. name: "base64 decoded",
  320. args: args{
  321. strategy: esv1.ExternalSecretDecodeBase64,
  322. in: map[string][]byte{
  323. "foo": []byte("YmFy"),
  324. },
  325. },
  326. want: map[string][]byte{
  327. "foo": []byte("bar"),
  328. },
  329. },
  330. {
  331. name: "invalid base64",
  332. args: args{
  333. strategy: esv1.ExternalSecretDecodeBase64,
  334. in: map[string][]byte{
  335. "foo": []byte("foo"),
  336. },
  337. },
  338. wantErr: true,
  339. },
  340. {
  341. name: "base64url decoded",
  342. args: args{
  343. strategy: esv1.ExternalSecretDecodeBase64URL,
  344. in: map[string][]byte{
  345. "foo": []byte(base64URLEncodedValue),
  346. },
  347. },
  348. want: map[string][]byte{
  349. "foo": []byte(base64DecodedValue),
  350. },
  351. },
  352. {
  353. name: "invalid base64url",
  354. args: args{
  355. strategy: esv1.ExternalSecretDecodeBase64URL,
  356. in: map[string][]byte{
  357. "foo": []byte("foo"),
  358. },
  359. },
  360. wantErr: true,
  361. },
  362. {
  363. name: "none",
  364. args: args{
  365. strategy: esv1.ExternalSecretDecodeNone,
  366. in: map[string][]byte{
  367. "foo": []byte(base64URLEncodedValue),
  368. },
  369. },
  370. want: map[string][]byte{
  371. "foo": []byte(base64URLEncodedValue),
  372. },
  373. },
  374. {
  375. name: "auto",
  376. args: args{
  377. strategy: esv1.ExternalSecretDecodeAuto,
  378. in: map[string][]byte{
  379. "b64": []byte(base64EncodedValue),
  380. "invalidb64": []byte("foo"),
  381. "b64url": []byte(base64URLEncodedValue),
  382. },
  383. },
  384. want: map[string][]byte{
  385. "b64": []byte(base64DecodedValue),
  386. "invalidb64": []byte("foo"),
  387. "b64url": []byte(base64DecodedValue),
  388. },
  389. },
  390. }
  391. for _, tt := range tests {
  392. t.Run(tt.name, func(t *testing.T) {
  393. got, err := DecodeMap(tt.args.strategy, tt.args.in)
  394. if (err != nil) != tt.wantErr {
  395. t.Errorf("DecodeMap() error = %v, wantErr %v", err, tt.wantErr)
  396. return
  397. }
  398. if !reflect.DeepEqual(got, tt.want) {
  399. t.Errorf("DecodeMap() = %v, want %v", got, tt.want)
  400. }
  401. })
  402. }
  403. }
  404. func TestValidate(t *testing.T) {
  405. err := NetworkValidate("http://google.com", 10*time.Second)
  406. if err != nil {
  407. t.Errorf("Connection problem: %v", err)
  408. }
  409. }
  410. func TestRewrite(t *testing.T) {
  411. type args struct {
  412. operations []esv1.ExternalSecretRewrite
  413. in map[string][]byte
  414. }
  415. tests := []struct {
  416. name string
  417. args args
  418. want map[string][]byte
  419. wantErr bool
  420. }{
  421. {
  422. name: "using double merge",
  423. args: args{
  424. operations: []esv1.ExternalSecretRewrite{
  425. {
  426. Merge: &esv1.ExternalSecretRewriteMerge{
  427. Strategy: esv1.ExternalSecretRewriteMergeStrategyJSON,
  428. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyIgnore,
  429. Into: "merged",
  430. Priority: []string{"a"},
  431. },
  432. },
  433. {
  434. Merge: &esv1.ExternalSecretRewriteMerge{
  435. Strategy: esv1.ExternalSecretRewriteMergeStrategyExtract,
  436. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyIgnore,
  437. Priority: []string{"b"},
  438. },
  439. },
  440. },
  441. in: map[string][]byte{
  442. "a": []byte(`{"host": "dba", "pass": "yola", "port": 123}`),
  443. "b": []byte(`{"host": "dbb", "pass": "yolb"}`),
  444. },
  445. },
  446. want: map[string][]byte{
  447. "host": []byte("dbb"),
  448. "pass": []byte("yolb"),
  449. "port": []byte("123"),
  450. },
  451. },
  452. {
  453. name: "using regexp and merge",
  454. args: args{
  455. operations: []esv1.ExternalSecretRewrite{
  456. {
  457. Regexp: &esv1.ExternalSecretRewriteRegexp{
  458. Source: "db/(.*)",
  459. Target: "$1",
  460. },
  461. },
  462. {
  463. Merge: &esv1.ExternalSecretRewriteMerge{
  464. Strategy: esv1.ExternalSecretRewriteMergeStrategyJSON,
  465. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyIgnore,
  466. Into: "merged",
  467. Priority: []string{"a"},
  468. },
  469. },
  470. },
  471. in: map[string][]byte{
  472. "db/a": []byte(`{"host": "dba.example.com"}`),
  473. "db/b": []byte(`{"host": "dbb.example.com", "pass": "yolo"}`),
  474. },
  475. },
  476. want: map[string][]byte{
  477. "a": []byte(`{"host": "dba.example.com"}`),
  478. "b": []byte(`{"host": "dbb.example.com", "pass": "yolo"}`),
  479. "merged": []byte(`{"host":"dba.example.com","pass":"yolo"}`),
  480. },
  481. },
  482. {
  483. name: "replace of a single key",
  484. args: args{
  485. operations: []esv1.ExternalSecretRewrite{
  486. {
  487. Regexp: &esv1.ExternalSecretRewriteRegexp{
  488. Source: "-",
  489. Target: "_",
  490. },
  491. },
  492. },
  493. in: map[string][]byte{
  494. "foo-bar": []byte("bar"),
  495. },
  496. },
  497. want: map[string][]byte{
  498. "foo_bar": []byte("bar"),
  499. },
  500. },
  501. {
  502. name: "no operation",
  503. args: args{
  504. operations: []esv1.ExternalSecretRewrite{
  505. {
  506. Regexp: &esv1.ExternalSecretRewriteRegexp{
  507. Source: "hello",
  508. Target: "world",
  509. },
  510. },
  511. },
  512. in: map[string][]byte{
  513. "foo": []byte("bar"),
  514. },
  515. },
  516. want: map[string][]byte{
  517. "foo": []byte("bar"),
  518. },
  519. },
  520. {
  521. name: "removing prefix from keys",
  522. args: args{
  523. operations: []esv1.ExternalSecretRewrite{
  524. {
  525. Regexp: &esv1.ExternalSecretRewriteRegexp{
  526. Source: "^my/initial/path/",
  527. Target: "",
  528. },
  529. },
  530. },
  531. in: map[string][]byte{
  532. "my/initial/path/foo": []byte("bar"),
  533. },
  534. },
  535. want: map[string][]byte{
  536. "foo": []byte("bar"),
  537. },
  538. },
  539. {
  540. name: "using un-named capture groups",
  541. args: args{
  542. operations: []esv1.ExternalSecretRewrite{
  543. {
  544. Regexp: &esv1.ExternalSecretRewriteRegexp{
  545. Source: "f(.*)o",
  546. Target: "a_new_path_$1",
  547. },
  548. },
  549. },
  550. in: map[string][]byte{
  551. "foo": []byte("bar"),
  552. "foodaloo": []byte("barr"),
  553. },
  554. },
  555. want: map[string][]byte{
  556. "a_new_path_o": []byte("bar"),
  557. "a_new_path_oodalo": []byte("barr"),
  558. },
  559. },
  560. {
  561. name: "using named and numbered capture groups",
  562. args: args{
  563. operations: []esv1.ExternalSecretRewrite{
  564. {
  565. Regexp: &esv1.ExternalSecretRewriteRegexp{
  566. Source: "f(?P<content>.*)o",
  567. Target: "a_new_path_${content}_${1}",
  568. },
  569. },
  570. },
  571. in: map[string][]byte{
  572. "foo": []byte("bar"),
  573. "floo": []byte("barr"),
  574. },
  575. },
  576. want: map[string][]byte{
  577. "a_new_path_o_o": []byte("bar"),
  578. "a_new_path_lo_lo": []byte("barr"),
  579. },
  580. },
  581. {
  582. name: "using sequenced rewrite operations",
  583. args: args{
  584. operations: []esv1.ExternalSecretRewrite{
  585. {
  586. Regexp: &esv1.ExternalSecretRewriteRegexp{
  587. Source: "my/(.*?)/bar/(.*)",
  588. Target: "$1-$2",
  589. },
  590. },
  591. {
  592. Regexp: &esv1.ExternalSecretRewriteRegexp{
  593. Source: "-",
  594. Target: "_",
  595. },
  596. },
  597. {
  598. Regexp: &esv1.ExternalSecretRewriteRegexp{
  599. Source: "ass",
  600. Target: "***",
  601. },
  602. },
  603. },
  604. in: map[string][]byte{
  605. "my/app/bar/key": []byte("bar"),
  606. "my/app/bar/password": []byte("barr"),
  607. },
  608. },
  609. want: map[string][]byte{
  610. "app_key": []byte("bar"),
  611. "app_p***word": []byte("barr"),
  612. },
  613. },
  614. {
  615. name: "using transform rewrite operation to create env var format keys",
  616. args: args{
  617. operations: []esv1.ExternalSecretRewrite{
  618. {
  619. Regexp: &esv1.ExternalSecretRewriteRegexp{
  620. Source: "my/(.*?)/bar/(.*)",
  621. Target: "$1-$2",
  622. },
  623. },
  624. {
  625. Transform: &esv1.ExternalSecretRewriteTransform{
  626. Template: `{{ .value | upper | replace "-" "_" }}`,
  627. },
  628. },
  629. },
  630. in: map[string][]byte{
  631. "my/app/bar/key": []byte("bar"),
  632. },
  633. },
  634. want: map[string][]byte{
  635. "APP_KEY": []byte("bar"),
  636. },
  637. },
  638. }
  639. for _, tt := range tests {
  640. t.Run(tt.name, func(t *testing.T) {
  641. got, err := RewriteMap(tt.args.operations, tt.args.in)
  642. if (err != nil) != tt.wantErr {
  643. t.Errorf("RewriteMap() error = %v, wantErr %v", err, tt.wantErr)
  644. return
  645. }
  646. if !reflect.DeepEqual(got, tt.want) {
  647. t.Errorf("RewriteMap() = %v, want %v", got, tt.want)
  648. }
  649. })
  650. }
  651. }
  652. func TestRewriteMerge(t *testing.T) {
  653. type args struct {
  654. operation esv1.ExternalSecretRewriteMerge
  655. in map[string][]byte
  656. }
  657. tests := []struct {
  658. name string
  659. args args
  660. want map[string][]byte
  661. wantErr bool
  662. }{
  663. {
  664. name: "using empty merge",
  665. args: args{
  666. operation: esv1.ExternalSecretRewriteMerge{},
  667. in: map[string][]byte{
  668. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  669. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  670. },
  671. },
  672. want: map[string][]byte{
  673. "username": []byte("foz"),
  674. "password": []byte("baz"),
  675. "host": []byte("redis.example.com"),
  676. "port": []byte("6379"),
  677. },
  678. wantErr: false,
  679. },
  680. {
  681. name: "using priority",
  682. args: args{
  683. operation: esv1.ExternalSecretRewriteMerge{
  684. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyIgnore,
  685. Priority: []string{"mongo-credentials", "redis-credentials"},
  686. },
  687. in: map[string][]byte{
  688. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  689. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  690. "other-credentials": []byte(`{"key": "value", "host": "other.example.com"}`),
  691. },
  692. },
  693. want: map[string][]byte{
  694. "username": []byte("foz"),
  695. "password": []byte("baz"),
  696. "host": []byte("redis.example.com"),
  697. "port": []byte("6379"),
  698. "key": []byte("value"),
  699. },
  700. wantErr: false,
  701. },
  702. {
  703. name: "using priority with keys not in input (default strict)",
  704. args: args{
  705. operation: esv1.ExternalSecretRewriteMerge{
  706. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyIgnore,
  707. Priority: []string{"non-existent-key", "another-missing-key", "mongo-credentials"},
  708. },
  709. in: map[string][]byte{
  710. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  711. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  712. },
  713. },
  714. want: nil,
  715. wantErr: true,
  716. },
  717. {
  718. name: "using priority with keys not in input and ignore policy",
  719. args: args{
  720. operation: esv1.ExternalSecretRewriteMerge{
  721. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyIgnore,
  722. Priority: []string{"non-existent-key", "mongo-credentials"},
  723. PriorityPolicy: esv1.ExternalSecretRewriteMergePriorityPolicyIgnoreNotFound,
  724. },
  725. in: map[string][]byte{
  726. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  727. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  728. },
  729. },
  730. want: map[string][]byte{
  731. "username": []byte("foz"),
  732. "password": []byte("baz"),
  733. "host": []byte("redis.example.com"),
  734. "port": []byte("6379"),
  735. },
  736. wantErr: false,
  737. },
  738. {
  739. name: "using conflict policy error",
  740. args: args{
  741. operation: esv1.ExternalSecretRewriteMerge{
  742. ConflictPolicy: esv1.ExternalSecretRewriteMergeConflictPolicyError,
  743. },
  744. in: map[string][]byte{
  745. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  746. "redis-credentials": []byte(`{"username": "redis", "port": "6379"}`),
  747. },
  748. },
  749. want: nil,
  750. wantErr: true,
  751. },
  752. {
  753. name: "using JSON strategy",
  754. args: args{
  755. operation: esv1.ExternalSecretRewriteMerge{
  756. Strategy: esv1.ExternalSecretRewriteMergeStrategyJSON,
  757. Into: "credentials",
  758. },
  759. in: map[string][]byte{
  760. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  761. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  762. },
  763. },
  764. want: map[string][]byte{
  765. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  766. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  767. "credentials": func() []byte {
  768. expected := map[string]any{
  769. "username": "foz",
  770. "password": "baz",
  771. "host": "redis.example.com",
  772. "port": "6379",
  773. }
  774. b, _ := json.Marshal(expected)
  775. return b
  776. }(),
  777. },
  778. wantErr: false,
  779. },
  780. {
  781. name: "using JSON strategy without into",
  782. args: args{
  783. operation: esv1.ExternalSecretRewriteMerge{
  784. Strategy: esv1.ExternalSecretRewriteMergeStrategyJSON,
  785. },
  786. in: map[string][]byte{
  787. "mongo-credentials": []byte(`{"username": "foz", "password": "baz"}`),
  788. "redis-credentials": []byte(`{"host": "redis.example.com", "port": "6379"}`),
  789. },
  790. },
  791. want: nil,
  792. wantErr: true,
  793. },
  794. {
  795. name: "with invalid JSON",
  796. args: args{
  797. operation: esv1.ExternalSecretRewriteMerge{},
  798. in: map[string][]byte{
  799. "invalid-json": []byte(`{"username": "foz", "password": "baz"`),
  800. },
  801. },
  802. want: nil,
  803. wantErr: true,
  804. },
  805. }
  806. for _, tt := range tests {
  807. t.Run(tt.name, func(t *testing.T) {
  808. got, err := RewriteMerge(tt.args.operation, tt.args.in)
  809. if (err != nil) != tt.wantErr {
  810. t.Errorf("RewriteMerge() error = %v, wantErr %v", err, tt.wantErr)
  811. return
  812. }
  813. if !reflect.DeepEqual(got, tt.want) {
  814. t.Errorf("RewriteMerge() = %v, want %v", got, tt.want)
  815. }
  816. })
  817. }
  818. }
  819. func TestReverse(t *testing.T) {
  820. type args struct {
  821. strategy esv1alpha1.PushSecretConversionStrategy
  822. in string
  823. }
  824. tests := []struct {
  825. name string
  826. args args
  827. want string
  828. }{
  829. {
  830. name: "do not change the key when using the None strategy",
  831. args: args{
  832. strategy: esv1alpha1.PushSecretConversionNone,
  833. in: keyWithEncodedInvalidChars,
  834. },
  835. want: keyWithEncodedInvalidChars,
  836. },
  837. {
  838. name: "reverse an unicode encoded key",
  839. args: args{
  840. strategy: esv1alpha1.PushSecretConversionReverseUnicode,
  841. in: keyWithEncodedInvalidChars,
  842. },
  843. want: keyWithInvalidChars,
  844. },
  845. {
  846. name: "do not attempt to decode an invalid unicode representation",
  847. args: args{
  848. strategy: esv1alpha1.PushSecretConversionReverseUnicode,
  849. in: "_U0xxx_x_U005b_",
  850. },
  851. want: "_U0xxx_x[",
  852. },
  853. }
  854. for _, tt := range tests {
  855. t.Run(tt.name, func(t *testing.T) {
  856. if got := reverse(tt.args.strategy, tt.args.in); got != tt.want {
  857. t.Errorf("reverse() = %v, want %v", got, tt.want)
  858. }
  859. })
  860. }
  861. }
  862. func TestFetchValueFromMetadata(t *testing.T) {
  863. type args struct {
  864. key string
  865. data *apiextensionsv1.JSON
  866. def any
  867. }
  868. type testCase struct {
  869. name string
  870. args args
  871. wantT any
  872. wantErr bool
  873. }
  874. tests := []testCase{
  875. {
  876. name: "plain dig for an existing key",
  877. args: args{
  878. key: "key",
  879. data: &apiextensionsv1.JSON{
  880. Raw: []byte(
  881. `{"key": "value"}`,
  882. ),
  883. },
  884. def: "def",
  885. },
  886. wantT: "value",
  887. wantErr: false,
  888. },
  889. {
  890. name: "return default if key not found",
  891. args: args{
  892. key: "key2",
  893. data: &apiextensionsv1.JSON{
  894. Raw: []byte(
  895. `{"key": "value"}`,
  896. ),
  897. },
  898. def: "def",
  899. },
  900. wantT: "def",
  901. wantErr: false,
  902. },
  903. {
  904. name: "use a different type",
  905. args: args{
  906. key: "key",
  907. data: &apiextensionsv1.JSON{
  908. Raw: []byte(
  909. `{"key": 123}`,
  910. ),
  911. },
  912. def: 1234,
  913. },
  914. wantT: float64(123), // unmarshal is always float64
  915. wantErr: false,
  916. },
  917. {
  918. name: "digging deeper",
  919. args: args{
  920. key: "key2",
  921. data: &apiextensionsv1.JSON{
  922. Raw: []byte(
  923. `{"key": {"key2": "value"}}`,
  924. ),
  925. },
  926. def: "",
  927. },
  928. wantT: "value",
  929. wantErr: false,
  930. },
  931. {
  932. name: "digging for a slice",
  933. args: args{
  934. key: "topics",
  935. data: &apiextensionsv1.JSON{
  936. Raw: []byte(
  937. `{"topics": ["topic1", "topic2"]}`,
  938. ),
  939. },
  940. def: []string{},
  941. },
  942. wantT: []any{"topic1", "topic2"}, // we don't have deep type matching so it's not an []string{} but []any.
  943. wantErr: false,
  944. },
  945. }
  946. for _, tt := range tests {
  947. t.Run(tt.name, func(t *testing.T) {
  948. gotT, err := FetchValueFromMetadata(tt.args.key, tt.args.data, tt.args.def)
  949. if (err != nil) != tt.wantErr {
  950. t.Errorf("FetchValueFromMetadata() error = %v, wantErr %v", err, tt.wantErr)
  951. return
  952. }
  953. assert.Equal(t, tt.wantT, gotT)
  954. })
  955. }
  956. }
  957. func TestGetByteValue(t *testing.T) {
  958. type args struct {
  959. data any
  960. }
  961. type testCase struct {
  962. name string
  963. args args
  964. want []byte
  965. wantErr bool
  966. }
  967. tests := []testCase{
  968. {
  969. name: "string",
  970. args: args{
  971. data: "value",
  972. },
  973. want: []byte("value"),
  974. wantErr: false,
  975. },
  976. {
  977. name: "map of any",
  978. args: args{
  979. data: map[string]any{
  980. "key": "value",
  981. },
  982. },
  983. want: []byte(`{"key":"value"}`),
  984. wantErr: false,
  985. },
  986. {
  987. name: "slice of string",
  988. args: args{
  989. data: []string{"value1", "value2"},
  990. },
  991. want: []byte("value1\nvalue2"),
  992. wantErr: false,
  993. },
  994. {
  995. name: "json.RawMessage",
  996. args: args{
  997. data: json.RawMessage(`{"key":"value"}`),
  998. },
  999. want: []byte(`{"key":"value"}`),
  1000. wantErr: false,
  1001. },
  1002. {
  1003. name: "float64",
  1004. args: args{
  1005. data: 123.45,
  1006. },
  1007. want: []byte("123.45"),
  1008. wantErr: false,
  1009. },
  1010. {
  1011. name: "json.Number",
  1012. args: args{
  1013. data: json.Number("123.45"),
  1014. },
  1015. want: []byte("123.45"),
  1016. wantErr: false,
  1017. },
  1018. {
  1019. name: "slice of any",
  1020. args: args{
  1021. data: []any{"value1", "value2"},
  1022. },
  1023. want: []byte(`["value1","value2"]`),
  1024. wantErr: false,
  1025. },
  1026. {
  1027. name: "boolean",
  1028. args: args{
  1029. data: true,
  1030. },
  1031. want: []byte("true"),
  1032. wantErr: false,
  1033. },
  1034. {
  1035. name: "nil",
  1036. args: args{
  1037. data: nil,
  1038. },
  1039. want: []byte(nil),
  1040. wantErr: false,
  1041. },
  1042. }
  1043. for _, tt := range tests {
  1044. t.Run(tt.name, func(t *testing.T) {
  1045. got, err := GetByteValue(tt.args.data)
  1046. if (err != nil) != tt.wantErr {
  1047. t.Errorf("GetByteValue() error = %v, wantErr %v", err, tt.wantErr)
  1048. return
  1049. }
  1050. if !reflect.DeepEqual(got, tt.want) {
  1051. t.Errorf("GetByteValue() = %v, want %v", got, tt.want)
  1052. }
  1053. })
  1054. }
  1055. }
  1056. func TestCompareStringAndByteSlices(t *testing.T) {
  1057. type args struct {
  1058. stringValue *string
  1059. byteValueSlice []byte
  1060. }
  1061. type testCase struct {
  1062. name string
  1063. args args
  1064. want bool
  1065. wantErr bool
  1066. }
  1067. tests := []testCase{
  1068. {
  1069. name: "same contents",
  1070. args: args{
  1071. stringValue: aws.String("value"),
  1072. byteValueSlice: []byte("value"),
  1073. },
  1074. want: true,
  1075. wantErr: true,
  1076. }, {
  1077. name: "different contents",
  1078. args: args{
  1079. stringValue: aws.String("value89"),
  1080. byteValueSlice: []byte("value"),
  1081. },
  1082. want: true,
  1083. wantErr: false,
  1084. }, {
  1085. name: "same contents with random",
  1086. args: args{
  1087. stringValue: aws.String("value89!3#@212"),
  1088. byteValueSlice: []byte("value89!3#@212"),
  1089. },
  1090. want: true,
  1091. wantErr: true,
  1092. }, {
  1093. name: "check Nil",
  1094. args: args{
  1095. stringValue: nil,
  1096. byteValueSlice: []byte("value89!3#@212"),
  1097. },
  1098. want: false,
  1099. wantErr: false,
  1100. },
  1101. }
  1102. for _, tt := range tests {
  1103. t.Run(tt.name, func(t *testing.T) {
  1104. got := CompareStringAndByteSlices(tt.args.stringValue, tt.args.byteValueSlice)
  1105. if got != tt.wantErr {
  1106. t.Errorf("CompareStringAndByteSlices() got = %v, want = %v", got, tt.wantErr)
  1107. return
  1108. }
  1109. })
  1110. }
  1111. }
  1112. func TestValidateSecretSelector(t *testing.T) {
  1113. tests := []struct {
  1114. desc string
  1115. store esv1.GenericStore
  1116. ref esmetav1.SecretKeySelector
  1117. expected error
  1118. }{
  1119. {
  1120. desc: "cluster secret store with namespace reference",
  1121. store: &esv1.ClusterSecretStore{
  1122. TypeMeta: metav1.TypeMeta{
  1123. Kind: esv1.ClusterSecretStoreKind,
  1124. },
  1125. },
  1126. ref: esmetav1.SecretKeySelector{
  1127. Namespace: new("test"),
  1128. },
  1129. expected: nil,
  1130. },
  1131. {
  1132. desc: "secret store without namespace reference",
  1133. store: &esv1.SecretStore{
  1134. TypeMeta: metav1.TypeMeta{
  1135. Kind: esv1.SecretStoreKind,
  1136. },
  1137. },
  1138. ref: esmetav1.SecretKeySelector{},
  1139. expected: nil,
  1140. },
  1141. {
  1142. desc: "secret store with the same namespace reference",
  1143. store: &esv1.SecretStore{
  1144. TypeMeta: metav1.TypeMeta{
  1145. Kind: esv1.SecretStoreKind,
  1146. },
  1147. ObjectMeta: metav1.ObjectMeta{
  1148. Namespace: "test",
  1149. },
  1150. },
  1151. ref: esmetav1.SecretKeySelector{
  1152. Namespace: new("test"),
  1153. },
  1154. expected: nil,
  1155. },
  1156. {
  1157. desc: "cluster secret store without namespace reference",
  1158. store: &esv1.ClusterSecretStore{
  1159. TypeMeta: metav1.TypeMeta{
  1160. Kind: esv1.ClusterSecretStoreKind,
  1161. },
  1162. },
  1163. ref: esmetav1.SecretKeySelector{},
  1164. expected: errRequireNamespace,
  1165. },
  1166. {
  1167. desc: "secret store with the different namespace reference",
  1168. store: &esv1.SecretStore{
  1169. TypeMeta: metav1.TypeMeta{
  1170. Kind: esv1.SecretStoreKind,
  1171. },
  1172. ObjectMeta: metav1.ObjectMeta{
  1173. Namespace: "test",
  1174. },
  1175. },
  1176. ref: esmetav1.SecretKeySelector{
  1177. Namespace: new("different"),
  1178. },
  1179. expected: errNamespaceNotAllowed,
  1180. },
  1181. }
  1182. for _, tt := range tests {
  1183. t.Run(tt.desc, func(t *testing.T) {
  1184. got := ValidateSecretSelector(tt.store, tt.ref)
  1185. if !errors.Is(got, tt.expected) {
  1186. t.Errorf("ValidateSecretSelector() got = %v, want = %v", got, tt.expected)
  1187. return
  1188. }
  1189. })
  1190. }
  1191. }
  1192. func TestValidateReferentSecretSelector(t *testing.T) {
  1193. tests := []struct {
  1194. desc string
  1195. store esv1.GenericStore
  1196. ref esmetav1.SecretKeySelector
  1197. expected error
  1198. }{
  1199. {
  1200. desc: "cluster secret store with namespace reference",
  1201. store: &esv1.ClusterSecretStore{
  1202. TypeMeta: metav1.TypeMeta{
  1203. Kind: esv1.ClusterSecretStoreKind,
  1204. },
  1205. },
  1206. ref: esmetav1.SecretKeySelector{
  1207. Namespace: new("test"),
  1208. },
  1209. expected: nil,
  1210. },
  1211. {
  1212. desc: "secret store without namespace reference",
  1213. store: &esv1.SecretStore{
  1214. TypeMeta: metav1.TypeMeta{
  1215. Kind: esv1.SecretStoreKind,
  1216. },
  1217. },
  1218. ref: esmetav1.SecretKeySelector{},
  1219. expected: nil,
  1220. },
  1221. {
  1222. desc: "secret store with the same namespace reference",
  1223. store: &esv1.SecretStore{
  1224. TypeMeta: metav1.TypeMeta{
  1225. Kind: esv1.SecretStoreKind,
  1226. },
  1227. ObjectMeta: metav1.ObjectMeta{
  1228. Namespace: "test",
  1229. },
  1230. },
  1231. ref: esmetav1.SecretKeySelector{
  1232. Namespace: new("test"),
  1233. },
  1234. expected: nil,
  1235. },
  1236. {
  1237. desc: "secret store with the different namespace reference",
  1238. store: &esv1.SecretStore{
  1239. TypeMeta: metav1.TypeMeta{
  1240. Kind: esv1.SecretStoreKind,
  1241. },
  1242. ObjectMeta: metav1.ObjectMeta{
  1243. Namespace: "test",
  1244. },
  1245. },
  1246. ref: esmetav1.SecretKeySelector{
  1247. Namespace: new("different"),
  1248. },
  1249. expected: errNamespaceNotAllowed,
  1250. },
  1251. }
  1252. for _, tt := range tests {
  1253. t.Run(tt.desc, func(t *testing.T) {
  1254. got := ValidateReferentSecretSelector(tt.store, tt.ref)
  1255. if !errors.Is(got, tt.expected) {
  1256. t.Errorf("ValidateReferentSecretSelector() got = %v, want = %v", got, tt.expected)
  1257. return
  1258. }
  1259. })
  1260. }
  1261. }
  1262. func TestValidateServiceAccountSelector(t *testing.T) {
  1263. tests := []struct {
  1264. desc string
  1265. store esv1.GenericStore
  1266. ref esmetav1.ServiceAccountSelector
  1267. expected error
  1268. }{
  1269. {
  1270. desc: "cluster secret store with namespace reference",
  1271. store: &esv1.ClusterSecretStore{
  1272. TypeMeta: metav1.TypeMeta{
  1273. Kind: esv1.ClusterSecretStoreKind,
  1274. },
  1275. },
  1276. ref: esmetav1.ServiceAccountSelector{
  1277. Namespace: new("test"),
  1278. },
  1279. expected: nil,
  1280. },
  1281. {
  1282. desc: "secret store without namespace reference",
  1283. store: &esv1.SecretStore{
  1284. TypeMeta: metav1.TypeMeta{
  1285. Kind: esv1.SecretStoreKind,
  1286. },
  1287. },
  1288. ref: esmetav1.ServiceAccountSelector{},
  1289. expected: nil,
  1290. },
  1291. {
  1292. desc: "secret store with the same namespace reference",
  1293. store: &esv1.SecretStore{
  1294. TypeMeta: metav1.TypeMeta{
  1295. Kind: esv1.SecretStoreKind,
  1296. },
  1297. ObjectMeta: metav1.ObjectMeta{
  1298. Namespace: "test",
  1299. },
  1300. },
  1301. ref: esmetav1.ServiceAccountSelector{
  1302. Namespace: new("test"),
  1303. },
  1304. expected: nil,
  1305. },
  1306. {
  1307. desc: "cluster secret store without namespace reference",
  1308. store: &esv1.ClusterSecretStore{
  1309. TypeMeta: metav1.TypeMeta{
  1310. Kind: esv1.ClusterSecretStoreKind,
  1311. },
  1312. },
  1313. ref: esmetav1.ServiceAccountSelector{},
  1314. expected: errRequireNamespace,
  1315. },
  1316. {
  1317. desc: "secret store with the different namespace reference",
  1318. store: &esv1.SecretStore{
  1319. TypeMeta: metav1.TypeMeta{
  1320. Kind: esv1.SecretStoreKind,
  1321. },
  1322. ObjectMeta: metav1.ObjectMeta{
  1323. Namespace: "test",
  1324. },
  1325. },
  1326. ref: esmetav1.ServiceAccountSelector{
  1327. Namespace: new("different"),
  1328. },
  1329. expected: errNamespaceNotAllowed,
  1330. },
  1331. }
  1332. for _, tt := range tests {
  1333. t.Run(tt.desc, func(t *testing.T) {
  1334. got := ValidateServiceAccountSelector(tt.store, tt.ref)
  1335. if !errors.Is(got, tt.expected) {
  1336. t.Errorf("ValidateServiceAccountSelector() got = %v, want = %v", got, tt.expected)
  1337. return
  1338. }
  1339. })
  1340. }
  1341. }
  1342. func TestValidateReferentServiceAccountSelector(t *testing.T) {
  1343. tests := []struct {
  1344. desc string
  1345. store esv1.GenericStore
  1346. ref esmetav1.ServiceAccountSelector
  1347. expected error
  1348. }{
  1349. {
  1350. desc: "cluster secret store with namespace reference",
  1351. store: &esv1.ClusterSecretStore{
  1352. TypeMeta: metav1.TypeMeta{
  1353. Kind: esv1.ClusterSecretStoreKind,
  1354. },
  1355. },
  1356. ref: esmetav1.ServiceAccountSelector{
  1357. Namespace: new("test"),
  1358. },
  1359. expected: nil,
  1360. },
  1361. {
  1362. desc: "secret store without namespace reference",
  1363. store: &esv1.SecretStore{
  1364. TypeMeta: metav1.TypeMeta{
  1365. Kind: esv1.SecretStoreKind,
  1366. },
  1367. },
  1368. ref: esmetav1.ServiceAccountSelector{},
  1369. expected: nil,
  1370. },
  1371. {
  1372. desc: "secret store with the same namespace reference",
  1373. store: &esv1.SecretStore{
  1374. TypeMeta: metav1.TypeMeta{
  1375. Kind: esv1.SecretStoreKind,
  1376. },
  1377. ObjectMeta: metav1.ObjectMeta{
  1378. Namespace: "test",
  1379. },
  1380. },
  1381. ref: esmetav1.ServiceAccountSelector{
  1382. Namespace: new("test"),
  1383. },
  1384. expected: nil,
  1385. },
  1386. {
  1387. desc: "secret store with the different namespace reference",
  1388. store: &esv1.SecretStore{
  1389. TypeMeta: metav1.TypeMeta{
  1390. Kind: esv1.SecretStoreKind,
  1391. },
  1392. ObjectMeta: metav1.ObjectMeta{
  1393. Namespace: "test",
  1394. },
  1395. },
  1396. ref: esmetav1.ServiceAccountSelector{
  1397. Namespace: new("different"),
  1398. },
  1399. expected: errNamespaceNotAllowed,
  1400. },
  1401. }
  1402. for _, tt := range tests {
  1403. t.Run(tt.desc, func(t *testing.T) {
  1404. got := ValidateReferentServiceAccountSelector(tt.store, tt.ref)
  1405. if !errors.Is(got, tt.expected) {
  1406. t.Errorf("ValidateReferentServiceAccountSelector() got = %v, want = %v", got, tt.expected)
  1407. return
  1408. }
  1409. })
  1410. }
  1411. }
  1412. func TestFetchCACertFromSourceRejectsCrossNamespaceCAProviderConfigMapForSecretStore(t *testing.T) {
  1413. fakeClient := clientfake.NewClientBuilder().WithObjects(
  1414. &v1.ConfigMap{
  1415. ObjectMeta: metav1.ObjectMeta{
  1416. Name: "ca-cm",
  1417. Namespace: "default",
  1418. },
  1419. Data: map[string]string{
  1420. "ca.crt": "local-cert",
  1421. },
  1422. },
  1423. &v1.ConfigMap{
  1424. ObjectMeta: metav1.ObjectMeta{
  1425. Name: "ca-cm",
  1426. Namespace: "other",
  1427. },
  1428. Data: map[string]string{
  1429. "ca.crt": "other-cert",
  1430. },
  1431. },
  1432. ).Build()
  1433. cert, err := FetchCACertFromSource(context.Background(), CreateCertOpts{
  1434. CAProvider: &esv1.CAProvider{
  1435. Type: esv1.CAProviderTypeConfigMap,
  1436. Name: "ca-cm",
  1437. Key: "ca.crt",
  1438. Namespace: Ptr("other"),
  1439. },
  1440. StoreKind: esv1.SecretStoreKind,
  1441. Namespace: "default",
  1442. Client: fakeClient,
  1443. })
  1444. assert.ErrorIs(t, err, errNamespaceNotAllowed)
  1445. assert.Nil(t, cert)
  1446. }
  1447. func TestFetchCACertFromSourceRejectsCrossNamespaceCAProviderSecretForSecretStore(t *testing.T) {
  1448. fakeClient := clientfake.NewClientBuilder().WithObjects(
  1449. &v1.Secret{
  1450. ObjectMeta: metav1.ObjectMeta{
  1451. Name: "ca-secret",
  1452. Namespace: "default",
  1453. },
  1454. Data: map[string][]byte{
  1455. "tls.crt": []byte("local-cert"),
  1456. },
  1457. },
  1458. &v1.Secret{
  1459. ObjectMeta: metav1.ObjectMeta{
  1460. Name: "ca-secret",
  1461. Namespace: "other",
  1462. },
  1463. Data: map[string][]byte{
  1464. "tls.crt": []byte("other-cert"),
  1465. },
  1466. },
  1467. ).Build()
  1468. cert, err := FetchCACertFromSource(context.Background(), CreateCertOpts{
  1469. CAProvider: &esv1.CAProvider{
  1470. Type: esv1.CAProviderTypeSecret,
  1471. Name: "ca-secret",
  1472. Key: "tls.crt",
  1473. Namespace: Ptr("other"),
  1474. },
  1475. StoreKind: esv1.SecretStoreKind,
  1476. Namespace: "default",
  1477. Client: fakeClient,
  1478. })
  1479. assert.ErrorIs(t, err, errNamespaceNotAllowed)
  1480. assert.Nil(t, cert)
  1481. }
  1482. const mockJWTToken = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiZXhwIjoxNzAwMDAwMDAwfQ.signature"
  1483. func TestParseJWTClaims(t *testing.T) {
  1484. // Mock JWT token with known payload
  1485. mockToken := mockJWTToken
  1486. claims, err := ParseJWTClaims(mockToken)
  1487. if err != nil {
  1488. t.Fatalf("Failed to get claims: %v", err)
  1489. }
  1490. if claims["sub"] != "1234567890" {
  1491. t.Errorf("Expected sub claim to be '1234567890', got %v", claims["sub"])
  1492. }
  1493. if claims["name"] != "John Doe" {
  1494. t.Errorf("Expected name claim to be 'John Doe', got %v", claims["name"])
  1495. }
  1496. }
  1497. func TestExtractJWTExpiration(t *testing.T) {
  1498. // Mock JWT token with known exp claim
  1499. mockToken := mockJWTToken
  1500. exp, err := ExtractJWTExpiration(mockToken)
  1501. if err != nil {
  1502. t.Fatalf("Failed to get token expiration: %v", err)
  1503. }
  1504. if exp != "1700000000" {
  1505. t.Errorf("Expected expiration to be '1700000000', got %s", exp)
  1506. }
  1507. }