utils_test.go 34 KB

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