utils_test.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. /*
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. You may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.
  11. */
  12. package utils
  13. import (
  14. "reflect"
  15. "testing"
  16. "time"
  17. vault "github.com/oracle/oci-go-sdk/v56/vault"
  18. v1 "k8s.io/api/core/v1"
  19. esv1beta1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1beta1"
  20. )
  21. const (
  22. base64DecodedValue string = "foo%_?bar"
  23. base64EncodedValue string = "Zm9vJV8/YmFy"
  24. base64URLEncodedValue string = "Zm9vJV8_YmFy"
  25. )
  26. func TestObjectHash(t *testing.T) {
  27. tests := []struct {
  28. name string
  29. input interface{}
  30. want string
  31. }{
  32. {
  33. name: "A nil should be still working",
  34. input: nil,
  35. want: "60046f14c917c18a9a0f923e191ba0dc",
  36. },
  37. {
  38. name: "We accept a simple scalar value, i.e. string",
  39. input: "hello there",
  40. want: "161bc25962da8fed6d2f59922fb642aa",
  41. },
  42. {
  43. name: "A complex object like a secret is not an issue",
  44. input: v1.Secret{Data: map[string][]byte{
  45. "xx": []byte("yyy"),
  46. }},
  47. want: "7b6fb27a17a8e4a4be8fda6cb499f3d5",
  48. },
  49. {
  50. name: "map also works",
  51. input: map[string][]byte{
  52. "foo": []byte("value1"),
  53. "bar": []byte("value2"),
  54. },
  55. want: "caa0155759a6a9b3b6ada5a6883ee2bb",
  56. },
  57. }
  58. for _, tt := range tests {
  59. t.Run(tt.name, func(t *testing.T) {
  60. if got := ObjectHash(tt.input); got != tt.want {
  61. t.Errorf("ObjectHash() = %v, want %v", got, tt.want)
  62. }
  63. })
  64. }
  65. }
  66. func TestIsNil(t *testing.T) {
  67. tbl := []struct {
  68. name string
  69. val interface{}
  70. exp bool
  71. }{
  72. {
  73. name: "simple nil val",
  74. val: nil,
  75. exp: true,
  76. },
  77. {
  78. name: "nil slice",
  79. val: (*[]struct{})(nil),
  80. exp: true,
  81. },
  82. {
  83. name: "struct pointer",
  84. val: &testing.T{},
  85. exp: false,
  86. },
  87. {
  88. name: "struct",
  89. val: testing.T{},
  90. exp: false,
  91. },
  92. {
  93. name: "slice of struct",
  94. val: []struct{}{{}},
  95. exp: false,
  96. },
  97. {
  98. name: "slice of ptr",
  99. val: []*testing.T{nil},
  100. exp: false,
  101. },
  102. {
  103. name: "slice",
  104. val: []struct{}(nil),
  105. exp: false,
  106. },
  107. {
  108. name: "int default value",
  109. val: 0,
  110. exp: false,
  111. },
  112. {
  113. name: "empty str",
  114. val: "",
  115. exp: false,
  116. },
  117. {
  118. name: "oracle vault",
  119. val: vault.VaultsClient{},
  120. exp: false,
  121. },
  122. {
  123. name: "func",
  124. val: func() {
  125. // noop for testing and to make linter happy
  126. },
  127. exp: false,
  128. },
  129. {
  130. name: "channel",
  131. val: make(chan struct{}),
  132. exp: false,
  133. },
  134. {
  135. name: "map",
  136. val: map[string]string{},
  137. exp: false,
  138. },
  139. }
  140. for _, row := range tbl {
  141. t.Run(row.name, func(t *testing.T) {
  142. res := IsNil(row.val)
  143. if res != row.exp {
  144. t.Errorf("IsNil(%#v)=%t, expected %t", row.val, res, row.exp)
  145. }
  146. })
  147. }
  148. }
  149. func TestConvertKeys(t *testing.T) {
  150. type args struct {
  151. strategy esv1beta1.ExternalSecretConversionStrategy
  152. in map[string][]byte
  153. }
  154. tests := []struct {
  155. name string
  156. args args
  157. want map[string][]byte
  158. wantErr bool
  159. }{
  160. {
  161. name: "convert with special chars",
  162. args: args{
  163. strategy: esv1beta1.ExternalSecretConversionDefault,
  164. in: map[string][]byte{
  165. "foo$bar%baz*bing": []byte(`noop`),
  166. },
  167. },
  168. want: map[string][]byte{
  169. "foo_bar_baz_bing": []byte(`noop`),
  170. },
  171. },
  172. {
  173. name: "error on collision",
  174. args: args{
  175. strategy: esv1beta1.ExternalSecretConversionDefault,
  176. in: map[string][]byte{
  177. "foo$bar%baz*bing": []byte(`noop`),
  178. "foo_bar_baz$bing": []byte(`noop`),
  179. },
  180. },
  181. wantErr: true,
  182. },
  183. {
  184. name: "convert path",
  185. args: args{
  186. strategy: esv1beta1.ExternalSecretConversionDefault,
  187. in: map[string][]byte{
  188. "/foo/bar/baz/bing": []byte(`noop`),
  189. "foo/bar/baz/bing/": []byte(`noop`),
  190. },
  191. },
  192. want: map[string][]byte{
  193. "_foo_bar_baz_bing": []byte(`noop`),
  194. "foo_bar_baz_bing_": []byte(`noop`),
  195. },
  196. },
  197. {
  198. name: "convert unicode",
  199. args: args{
  200. strategy: esv1beta1.ExternalSecretConversionUnicode,
  201. in: map[string][]byte{
  202. "😀foo😁bar😂baz😈bing": []byte(`noop`),
  203. },
  204. },
  205. want: map[string][]byte{
  206. "_U1f600_foo_U1f601_bar_U1f602_baz_U1f608_bing": []byte(`noop`),
  207. },
  208. },
  209. }
  210. for _, tt := range tests {
  211. t.Run(tt.name, func(t *testing.T) {
  212. got, err := ConvertKeys(tt.args.strategy, tt.args.in)
  213. if (err != nil) != tt.wantErr {
  214. t.Errorf("ConvertKeys() error = %v, wantErr %v", err, tt.wantErr)
  215. return
  216. }
  217. if !reflect.DeepEqual(got, tt.want) {
  218. t.Errorf("ConvertKeys() = %v, want %v", got, tt.want)
  219. }
  220. })
  221. }
  222. }
  223. func TestDecode(t *testing.T) {
  224. type args struct {
  225. strategy esv1beta1.ExternalSecretDecodingStrategy
  226. in map[string][]byte
  227. }
  228. tests := []struct {
  229. name string
  230. args args
  231. want map[string][]byte
  232. wantErr bool
  233. }{
  234. {
  235. name: "base64 decoded",
  236. args: args{
  237. strategy: esv1beta1.ExternalSecretDecodeBase64,
  238. in: map[string][]byte{
  239. "foo": []byte("YmFy"),
  240. },
  241. },
  242. want: map[string][]byte{
  243. "foo": []byte("bar"),
  244. },
  245. },
  246. {
  247. name: "invalid base64",
  248. args: args{
  249. strategy: esv1beta1.ExternalSecretDecodeBase64,
  250. in: map[string][]byte{
  251. "foo": []byte("foo"),
  252. },
  253. },
  254. wantErr: true,
  255. },
  256. {
  257. name: "base64url decoded",
  258. args: args{
  259. strategy: esv1beta1.ExternalSecretDecodeBase64URL,
  260. in: map[string][]byte{
  261. "foo": []byte(base64URLEncodedValue),
  262. },
  263. },
  264. want: map[string][]byte{
  265. "foo": []byte(base64DecodedValue),
  266. },
  267. },
  268. {
  269. name: "invalid base64url",
  270. args: args{
  271. strategy: esv1beta1.ExternalSecretDecodeBase64URL,
  272. in: map[string][]byte{
  273. "foo": []byte("foo"),
  274. },
  275. },
  276. wantErr: true,
  277. },
  278. {
  279. name: "none",
  280. args: args{
  281. strategy: esv1beta1.ExternalSecretDecodeNone,
  282. in: map[string][]byte{
  283. "foo": []byte(base64URLEncodedValue),
  284. },
  285. },
  286. want: map[string][]byte{
  287. "foo": []byte(base64URLEncodedValue),
  288. },
  289. },
  290. {
  291. name: "auto",
  292. args: args{
  293. strategy: esv1beta1.ExternalSecretDecodeAuto,
  294. in: map[string][]byte{
  295. "b64": []byte(base64EncodedValue),
  296. "invalidb64": []byte("foo"),
  297. "b64url": []byte(base64URLEncodedValue),
  298. },
  299. },
  300. want: map[string][]byte{
  301. "b64": []byte(base64DecodedValue),
  302. "invalidb64": []byte("foo"),
  303. "b64url": []byte(base64DecodedValue),
  304. },
  305. },
  306. }
  307. for _, tt := range tests {
  308. t.Run(tt.name, func(t *testing.T) {
  309. got, err := DecodeMap(tt.args.strategy, tt.args.in)
  310. if (err != nil) != tt.wantErr {
  311. t.Errorf("DecodeMap() error = %v, wantErr %v", err, tt.wantErr)
  312. return
  313. }
  314. if !reflect.DeepEqual(got, tt.want) {
  315. t.Errorf("DecodeMap() = %v, want %v", got, tt.want)
  316. }
  317. })
  318. }
  319. }
  320. func TestValidate(t *testing.T) {
  321. err := NetworkValidate("http://google.com", 10*time.Second)
  322. if err != nil {
  323. t.Errorf("Connection problem: %v", err)
  324. }
  325. }
  326. func TestRewriteRegexp(t *testing.T) {
  327. type args struct {
  328. operations []esv1beta1.ExternalSecretRewrite
  329. in map[string][]byte
  330. }
  331. tests := []struct {
  332. name string
  333. args args
  334. want map[string][]byte
  335. wantErr bool
  336. }{
  337. {
  338. name: "replace of a single key",
  339. args: args{
  340. operations: []esv1beta1.ExternalSecretRewrite{
  341. {
  342. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  343. Source: "-",
  344. Target: "_",
  345. },
  346. },
  347. },
  348. in: map[string][]byte{
  349. "foo-bar": []byte("bar"),
  350. },
  351. },
  352. want: map[string][]byte{
  353. "foo_bar": []byte("bar"),
  354. },
  355. },
  356. {
  357. name: "no operation",
  358. args: args{
  359. operations: []esv1beta1.ExternalSecretRewrite{
  360. {
  361. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  362. Source: "hello",
  363. Target: "world",
  364. },
  365. },
  366. },
  367. in: map[string][]byte{
  368. "foo": []byte("bar"),
  369. },
  370. },
  371. want: map[string][]byte{
  372. "foo": []byte("bar"),
  373. },
  374. },
  375. {
  376. name: "removing prefix from keys",
  377. args: args{
  378. operations: []esv1beta1.ExternalSecretRewrite{
  379. {
  380. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  381. Source: "^my/initial/path/",
  382. Target: "",
  383. },
  384. },
  385. },
  386. in: map[string][]byte{
  387. "my/initial/path/foo": []byte("bar"),
  388. },
  389. },
  390. want: map[string][]byte{
  391. "foo": []byte("bar"),
  392. },
  393. },
  394. {
  395. name: "using un-named capture groups",
  396. args: args{
  397. operations: []esv1beta1.ExternalSecretRewrite{
  398. {
  399. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  400. Source: "f(.*)o",
  401. Target: "a_new_path_$1",
  402. },
  403. },
  404. },
  405. in: map[string][]byte{
  406. "foo": []byte("bar"),
  407. "foodaloo": []byte("barr"),
  408. },
  409. },
  410. want: map[string][]byte{
  411. "a_new_path_o": []byte("bar"),
  412. "a_new_path_oodalo": []byte("barr"),
  413. },
  414. },
  415. {
  416. name: "using named and numbered capture groups",
  417. args: args{
  418. operations: []esv1beta1.ExternalSecretRewrite{
  419. {
  420. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  421. Source: "f(?P<content>.*)o",
  422. Target: "a_new_path_${content}_${1}",
  423. },
  424. },
  425. },
  426. in: map[string][]byte{
  427. "foo": []byte("bar"),
  428. "floo": []byte("barr"),
  429. },
  430. },
  431. want: map[string][]byte{
  432. "a_new_path_o_o": []byte("bar"),
  433. "a_new_path_lo_lo": []byte("barr"),
  434. },
  435. },
  436. {
  437. name: "using sequenced rewrite operations",
  438. args: args{
  439. operations: []esv1beta1.ExternalSecretRewrite{
  440. {
  441. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  442. Source: "my/(.*?)/bar/(.*)",
  443. Target: "$1-$2",
  444. },
  445. },
  446. {
  447. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  448. Source: "-",
  449. Target: "_",
  450. },
  451. },
  452. {
  453. Regexp: &esv1beta1.ExternalSecretRewriteRegexp{
  454. Source: "ass",
  455. Target: "***",
  456. },
  457. },
  458. },
  459. in: map[string][]byte{
  460. "my/app/bar/key": []byte("bar"),
  461. "my/app/bar/password": []byte("barr"),
  462. },
  463. },
  464. want: map[string][]byte{
  465. "app_key": []byte("bar"),
  466. "app_p***word": []byte("barr"),
  467. },
  468. },
  469. }
  470. for _, tt := range tests {
  471. t.Run(tt.name, func(t *testing.T) {
  472. got, err := RewriteMap(tt.args.operations, tt.args.in)
  473. if (err != nil) != tt.wantErr {
  474. t.Errorf("RewriteMap() error = %v, wantErr %v", err, tt.wantErr)
  475. return
  476. }
  477. if !reflect.DeepEqual(got, tt.want) {
  478. t.Errorf("RewriteMap() = %v, want %v", got, tt.want)
  479. }
  480. })
  481. }
  482. }