get_kubeconfig.go 2.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. package cluster
  2. import (
  3. "context"
  4. "encoding/base64"
  5. "errors"
  6. "fmt"
  7. "net/http"
  8. "github.com/bufbuild/connect-go"
  9. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  10. "github.com/porter-dev/porter/api/server/authz"
  11. "github.com/porter-dev/porter/api/server/handlers"
  12. "github.com/porter-dev/porter/api/server/shared"
  13. "github.com/porter-dev/porter/api/server/shared/apierrors"
  14. "github.com/porter-dev/porter/api/server/shared/config"
  15. "github.com/porter-dev/porter/api/types"
  16. "github.com/porter-dev/porter/internal/models"
  17. "k8s.io/client-go/tools/clientcmd"
  18. )
  19. type GetTemporaryKubeconfigHandler struct {
  20. handlers.PorterHandlerWriter
  21. authz.KubernetesAgentGetter
  22. }
  23. func NewGetTemporaryKubeconfigHandler(
  24. config *config.Config,
  25. writer shared.ResultWriter,
  26. ) *GetTemporaryKubeconfigHandler {
  27. return &GetTemporaryKubeconfigHandler{
  28. PorterHandlerWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
  29. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  30. }
  31. }
  32. func (c *GetTemporaryKubeconfigHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  33. if c.Config().ServerConf.DisableTemporaryKubeconfig {
  34. c.HandleAPIError(w, r, apierrors.NewErrNotFound(
  35. errors.New("temporary kubeconfig generation is disabled on this instance"),
  36. ))
  37. return
  38. }
  39. ctx := r.Context()
  40. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  41. outOfClusterConfig := c.GetOutOfClusterConfig(cluster)
  42. if cluster.ProvisionedBy == "CAPI" {
  43. kubeconfigResp, err := c.Config().ClusterControlPlaneClient.KubeConfigForCluster(context.Background(), connect.NewRequest(
  44. &porterv1.KubeConfigForClusterRequest{
  45. ProjectId: int64(cluster.ProjectID),
  46. ClusterId: int64(cluster.ID),
  47. },
  48. ))
  49. if err != nil {
  50. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error getting temporary capi config: %w", err)))
  51. return
  52. }
  53. if kubeconfigResp.Msg == nil {
  54. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error reading temporary capi config: %w", err)))
  55. return
  56. }
  57. b64, err := base64.StdEncoding.DecodeString(kubeconfigResp.Msg.KubeConfig)
  58. if err != nil {
  59. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("unable to decode base64 kubeconfig: %w", err)))
  60. return
  61. }
  62. res := &types.GetTemporaryKubeconfigResponse{
  63. Kubeconfig: b64,
  64. }
  65. c.WriteResult(w, r, res)
  66. return
  67. }
  68. kubeconfig, err := outOfClusterConfig.CreateRawConfigFromCluster()
  69. if err != nil {
  70. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  71. return
  72. }
  73. kubeconfigBytes, err := clientcmd.Write(*kubeconfig)
  74. if err != nil {
  75. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  76. return
  77. }
  78. res := &types.GetTemporaryKubeconfigResponse{
  79. Kubeconfig: kubeconfigBytes,
  80. }
  81. c.WriteResult(w, r, res)
  82. }