util.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  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 util
  14. import (
  15. "bytes"
  16. "context"
  17. "fmt"
  18. "net/http"
  19. "os"
  20. "path/filepath"
  21. "strings"
  22. "time"
  23. fluxhelm "github.com/fluxcd/helm-controller/api/v2"
  24. fluxsrc "github.com/fluxcd/source-controller/api/v1"
  25. v1 "k8s.io/api/core/v1"
  26. apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
  27. apierrors "k8s.io/apimachinery/pkg/api/errors"
  28. "k8s.io/apimachinery/pkg/api/meta"
  29. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  30. "k8s.io/apimachinery/pkg/runtime"
  31. utilruntime "k8s.io/apimachinery/pkg/util/runtime"
  32. "k8s.io/apimachinery/pkg/util/wait"
  33. "k8s.io/client-go/kubernetes"
  34. clientgoscheme "k8s.io/client-go/kubernetes/scheme"
  35. "k8s.io/client-go/rest"
  36. restclient "k8s.io/client-go/rest"
  37. "k8s.io/client-go/tools/clientcmd"
  38. "k8s.io/client-go/tools/remotecommand"
  39. "k8s.io/client-go/util/homedir"
  40. crclient "sigs.k8s.io/controller-runtime/pkg/client"
  41. esv1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1"
  42. esv1alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v1alpha1"
  43. esv2alpha1 "github.com/external-secrets/external-secrets/apis/externalsecrets/v2alpha1"
  44. genv1alpha1 "github.com/external-secrets/external-secrets/apis/generators/v1alpha1"
  45. awsv2alpha1 "github.com/external-secrets/external-secrets/apis/provider/aws/v2alpha1"
  46. fakev2alpha1 "github.com/external-secrets/external-secrets/apis/provider/fake/v2alpha1"
  47. k8sv2alpha1 "github.com/external-secrets/external-secrets/apis/provider/kubernetes/v2alpha1"
  48. // nolint
  49. . "github.com/onsi/ginkgo/v2"
  50. )
  51. var scheme = runtime.NewScheme()
  52. func init() {
  53. // kubernetes schemes
  54. utilruntime.Must(clientgoscheme.AddToScheme(scheme))
  55. utilruntime.Must(apiextensionsv1.AddToScheme(scheme))
  56. // external-secrets schemes
  57. utilruntime.Must(esv1.AddToScheme(scheme))
  58. utilruntime.Must(esv1alpha1.AddToScheme(scheme))
  59. utilruntime.Must(esv2alpha1.AddToScheme(scheme))
  60. utilruntime.Must(genv1alpha1.AddToScheme(scheme))
  61. // other schemes
  62. utilruntime.Must(fluxhelm.AddToScheme(scheme))
  63. utilruntime.Must(fluxsrc.AddToScheme(scheme))
  64. // v2alpha1 provider schemes
  65. utilruntime.Must(awsv2alpha1.AddToScheme(scheme))
  66. utilruntime.Must(fakev2alpha1.AddToScheme(scheme))
  67. utilruntime.Must(k8sv2alpha1.AddToScheme(scheme))
  68. }
  69. const (
  70. // How often to poll for conditions.
  71. Poll = 2 * time.Second
  72. e2eNamespacePrefix = "e2e-tests-"
  73. )
  74. // CreateKubeNamespace creates a new Kubernetes Namespace for a test.
  75. func CreateKubeNamespace(baseName string, kubeClientSet kubernetes.Interface) (*v1.Namespace, error) {
  76. ns := &v1.Namespace{
  77. ObjectMeta: metav1.ObjectMeta{
  78. GenerateName: fmt.Sprintf("e2e-tests-%v-", baseName),
  79. },
  80. }
  81. return kubeClientSet.CoreV1().Namespaces().Create(GinkgoT().Context(), ns, metav1.CreateOptions{})
  82. }
  83. // DeleteKubeNamespace will delete a namespace resource.
  84. func DeleteKubeNamespace(namespace string, kubeClientSet kubernetes.Interface) error {
  85. return kubeClientSet.CoreV1().Namespaces().Delete(GinkgoT().Context(), namespace, metav1.DeleteOptions{})
  86. }
  87. func IsE2ETestNamespace(namespace string) bool {
  88. return strings.HasPrefix(namespace, e2eNamespacePrefix)
  89. }
  90. func IsMissingAPIResourceError(err error) bool {
  91. return err != nil && meta.IsNoMatchError(err)
  92. }
  93. func ClearKnownNamespaceFinalizers(ctx context.Context, c crclient.Client, namespace string) error {
  94. var secretList v1.SecretList
  95. if err := c.List(ctx, &secretList, crclient.InNamespace(namespace)); err != nil {
  96. return err
  97. }
  98. for i := range secretList.Items {
  99. if len(secretList.Items[i].Finalizers) == 0 {
  100. continue
  101. }
  102. secret := secretList.Items[i].DeepCopy()
  103. secret.Finalizers = nil
  104. if err := c.Update(ctx, secret); err != nil {
  105. return err
  106. }
  107. }
  108. var pushSecretList esv1alpha1.PushSecretList
  109. if err := c.List(ctx, &pushSecretList, crclient.InNamespace(namespace)); err != nil {
  110. return err
  111. }
  112. for i := range pushSecretList.Items {
  113. if len(pushSecretList.Items[i].Finalizers) == 0 {
  114. continue
  115. }
  116. pushSecret := pushSecretList.Items[i].DeepCopy()
  117. pushSecret.Finalizers = nil
  118. if err := c.Update(ctx, pushSecret); err != nil {
  119. return err
  120. }
  121. }
  122. return nil
  123. }
  124. func CleanupTerminatingE2ENamespaces(ctx context.Context, c crclient.Client) error {
  125. var namespaceList v1.NamespaceList
  126. if err := c.List(ctx, &namespaceList); err != nil {
  127. return err
  128. }
  129. for i := range namespaceList.Items {
  130. namespace := namespaceList.Items[i]
  131. if !IsE2ETestNamespace(namespace.Name) || namespace.DeletionTimestamp == nil {
  132. continue
  133. }
  134. if err := ClearKnownNamespaceFinalizers(ctx, c, namespace.Name); err != nil {
  135. return err
  136. }
  137. }
  138. return nil
  139. }
  140. // WaitForKubeNamespaceNotExist will wait for the namespace with the given name
  141. // to not exist for up to 2 minutes.
  142. func WaitForKubeNamespaceNotExist(namespace string, kubeClientSet kubernetes.Interface) error {
  143. return wait.PollUntilContextTimeout(GinkgoT().Context(), Poll, time.Minute*2, true, namespaceNotExist(kubeClientSet, namespace))
  144. }
  145. func namespaceNotExist(c kubernetes.Interface, namespace string) wait.ConditionWithContextFunc {
  146. return func(ctx context.Context) (bool, error) {
  147. _, err := c.CoreV1().Namespaces().Get(ctx, namespace, metav1.GetOptions{})
  148. if apierrors.IsNotFound(err) {
  149. return true, nil
  150. }
  151. if err != nil {
  152. return false, err
  153. }
  154. return false, nil
  155. }
  156. }
  157. // ExecCmd exec command on specific pod and wait the command's output.
  158. func ExecCmd(client kubernetes.Interface, config *restclient.Config, podName, namespace string,
  159. command string) (string, error) {
  160. return execCmd(client, config, podName, "", namespace, command)
  161. }
  162. // ExecCmdWithContainer exec command on specific container in a specific pod and wait the command's output.
  163. func ExecCmdWithContainer(client kubernetes.Interface, config *restclient.Config, podName, containerName, namespace string,
  164. command string) (string, error) {
  165. return execCmd(client, config, podName, containerName, namespace, command)
  166. }
  167. func execCmd(client kubernetes.Interface, config *restclient.Config, podName, containerName, namespace string,
  168. command string) (string, error) {
  169. cmd := []string{
  170. "sh",
  171. "-c",
  172. command,
  173. }
  174. req := client.CoreV1().RESTClient().Post().Resource("pods").Name(podName).
  175. Namespace(namespace).SubResource("exec")
  176. option := &v1.PodExecOptions{
  177. Command: cmd,
  178. Container: containerName,
  179. Stdin: false,
  180. Stdout: true,
  181. Stderr: true,
  182. TTY: false,
  183. }
  184. req.VersionedParams(
  185. option,
  186. clientgoscheme.ParameterCodec,
  187. )
  188. exec, err := remotecommand.NewSPDYExecutor(config, "POST", req.URL())
  189. if err != nil {
  190. return "", err
  191. }
  192. var stdout, stderr bytes.Buffer
  193. err = exec.Stream(remotecommand.StreamOptions{
  194. Stdout: &stdout,
  195. Stderr: &stderr,
  196. Tty: false,
  197. })
  198. if err != nil {
  199. return "", fmt.Errorf("unable to exec stream: %w: \n%s\n%s", err, stdout.String(), stderr.String())
  200. }
  201. return stdout.String() + stderr.String(), nil
  202. }
  203. // WaitForPodsRunning waits for a given amount of time until a group of Pods is running in the given namespace.
  204. func WaitForPodsRunning(kubeClientSet kubernetes.Interface, expectedReplicas int, namespace string, opts metav1.ListOptions) (*v1.PodList, error) {
  205. var pods *v1.PodList
  206. err := wait.PollUntilContextTimeout(GinkgoT().Context(), 1*time.Second, time.Minute*5, true, func(ctx context.Context) (bool, error) {
  207. pl, err := kubeClientSet.CoreV1().Pods(namespace).List(ctx, opts)
  208. if err != nil {
  209. return false, nil
  210. }
  211. r := 0
  212. for i := range pl.Items {
  213. if pl.Items[i].Status.Phase == v1.PodRunning {
  214. r++
  215. }
  216. }
  217. if r == expectedReplicas {
  218. pods = pl
  219. return true, nil
  220. }
  221. return false, nil
  222. })
  223. return pods, err
  224. }
  225. // WaitForPodsReady waits for a given amount of time until a group of Pods is running in the given namespace.
  226. func WaitForPodsReady(kubeClientSet kubernetes.Interface, expectedReplicas int, namespace string, opts metav1.ListOptions) error {
  227. return wait.PollUntilContextTimeout(GinkgoT().Context(), 1*time.Second, time.Minute*5, true, func(ctx context.Context) (bool, error) {
  228. pl, err := kubeClientSet.CoreV1().Pods(namespace).List(ctx, opts)
  229. if err != nil {
  230. return false, nil
  231. }
  232. r := 0
  233. for i := range pl.Items {
  234. if isRunning, _ := podRunningReady(&pl.Items[i]); isRunning {
  235. r++
  236. }
  237. }
  238. if r == expectedReplicas {
  239. return true, nil
  240. }
  241. return false, nil
  242. })
  243. }
  244. // podRunningReady checks whether pod p's phase is running and it has a ready
  245. // condition of status true.
  246. func podRunningReady(p *v1.Pod) (bool, error) {
  247. // Check the phase is running.
  248. if p.Status.Phase != v1.PodRunning {
  249. return false, fmt.Errorf("want pod '%s' on '%s' to be '%v' but was '%v'",
  250. p.ObjectMeta.Name, p.Spec.NodeName, v1.PodRunning, p.Status.Phase)
  251. }
  252. // Check the ready condition is true.
  253. if !isPodReady(p) {
  254. return false, fmt.Errorf("pod '%s' on '%s' didn't have condition {%v %v}; conditions: %v",
  255. p.ObjectMeta.Name, p.Spec.NodeName, v1.PodReady, v1.ConditionTrue, p.Status.Conditions)
  256. }
  257. return true, nil
  258. }
  259. func isPodReady(p *v1.Pod) bool {
  260. for _, condition := range p.Status.Conditions {
  261. if condition.Type != v1.ContainersReady {
  262. continue
  263. }
  264. return condition.Status == v1.ConditionTrue
  265. }
  266. return false
  267. }
  268. // WaitForURL tests the provided url. Once a http 200 is returned the func returns with no error.
  269. // Timeout is 5min.
  270. func WaitForURL(url string) error {
  271. return wait.PollUntilContextTimeout(GinkgoT().Context(), 2*time.Second, time.Minute*5, true, func(ctx context.Context) (bool, error) {
  272. req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, http.NoBody)
  273. if err != nil {
  274. return false, nil
  275. }
  276. res, err := http.DefaultClient.Do(req)
  277. if err != nil {
  278. return false, nil
  279. }
  280. defer func() {
  281. _ = res.Body.Close()
  282. }()
  283. if res.StatusCode == http.StatusOK {
  284. return true, nil
  285. }
  286. return false, err
  287. })
  288. }
  289. // UpdateKubeSA updates a new Kubernetes Service Account for a test.
  290. func UpdateKubeSA(baseName string, kubeClientSet kubernetes.Interface, ns string, annotations map[string]string) (*v1.ServiceAccount, error) {
  291. sa := &v1.ServiceAccount{
  292. ObjectMeta: metav1.ObjectMeta{
  293. Name: baseName,
  294. Annotations: annotations,
  295. },
  296. }
  297. return kubeClientSet.CoreV1().ServiceAccounts(ns).Update(GinkgoT().Context(), sa, metav1.UpdateOptions{})
  298. }
  299. // UpdateKubeSA updates a new Kubernetes Service Account for a test.
  300. func GetKubeSA(baseName string, kubeClientSet kubernetes.Interface, ns string) (*v1.ServiceAccount, error) {
  301. return kubeClientSet.CoreV1().ServiceAccounts(ns).Get(GinkgoT().Context(), baseName, metav1.GetOptions{})
  302. }
  303. func GetKubeSecret(client kubernetes.Interface, namespace, secretName string) (*v1.Secret, error) {
  304. return client.CoreV1().Secrets(namespace).Get(GinkgoT().Context(), secretName, metav1.GetOptions{})
  305. }
  306. // NewConfig loads and returns the kubernetes credentials from the environment.
  307. // KUBECONFIG env var takes precedence, then ~/.kube/config, then in-cluster config.
  308. func NewConfig() (*restclient.Config, *kubernetes.Clientset, crclient.Client) {
  309. var kubeConfig *restclient.Config
  310. var err error
  311. kcPath := os.Getenv("KUBECONFIG")
  312. if kcPath != "" {
  313. kubeConfig, err = clientcmd.BuildConfigFromFlags("", kcPath)
  314. if err != nil {
  315. Fail(err.Error())
  316. }
  317. } else {
  318. // Try ~/.kube/config
  319. homeDir, err := os.UserHomeDir()
  320. if err == nil {
  321. defaultKubeconfig := homeDir + "/.kube/config"
  322. if _, err := os.Stat(defaultKubeconfig); err == nil {
  323. kubeConfig, err = clientcmd.BuildConfigFromFlags("", defaultKubeconfig)
  324. if err != nil {
  325. Fail(err.Error())
  326. }
  327. }
  328. }
  329. // Fall back to in-cluster config if ~/.kube/config doesn't exist
  330. if kubeConfig == nil {
  331. kubeConfig, err = restclient.InClusterConfig()
  332. if err != nil {
  333. Fail(err.Error())
  334. }
  335. }
  336. }
  337. kubeClientSet, err := kubernetes.NewForConfig(kubeConfig)
  338. if err != nil {
  339. Fail(err.Error())
  340. }
  341. CRClient, err := crclient.New(kubeConfig, crclient.Options{Scheme: scheme})
  342. if err != nil {
  343. Fail(err.Error())
  344. }
  345. return kubeConfig, kubeClientSet, CRClient
  346. }
  347. func BuildKubeConfig() (*rest.Config, error) {
  348. // 1. If KUBECONFIG is explicitly set, use it
  349. if kubeconfigEnv := os.Getenv("KUBECONFIG"); kubeconfigEnv != "" {
  350. cfg, err := clientcmd.BuildConfigFromFlags("", kubeconfigEnv)
  351. if err == nil {
  352. return cfg, nil
  353. }
  354. return nil, fmt.Errorf("failed to load KUBECONFIG=%s: %w", kubeconfigEnv, err)
  355. }
  356. // 2. Try default kubeconfig location (~/.kube/config)
  357. if home := homedir.HomeDir(); home != "" {
  358. kubeconfigPath := filepath.Join(home, ".kube", "config")
  359. if _, err := os.Stat(kubeconfigPath); err == nil {
  360. cfg, err := clientcmd.BuildConfigFromFlags("", kubeconfigPath)
  361. if err == nil {
  362. return cfg, nil
  363. }
  364. return nil, fmt.Errorf("failed to load default kubeconfig: %w", err)
  365. }
  366. }
  367. // 3. Fallback to in-cluster config
  368. cfg, err := rest.InClusterConfig()
  369. if err != nil {
  370. return nil, fmt.Errorf("failed to load in-cluster config: %w", err)
  371. }
  372. return cfg, nil
  373. }