list.go 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. package environment_groups
  2. import (
  3. "encoding/base64"
  4. "net/http"
  5. "strings"
  6. "time"
  7. "connectrpc.com/connect"
  8. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  9. "github.com/porter-dev/porter/api/server/authz"
  10. "github.com/porter-dev/porter/api/server/handlers"
  11. "github.com/porter-dev/porter/api/server/shared"
  12. "github.com/porter-dev/porter/api/server/shared/apierrors"
  13. "github.com/porter-dev/porter/api/server/shared/config"
  14. "github.com/porter-dev/porter/api/types"
  15. environmentgroups "github.com/porter-dev/porter/internal/kubernetes/environment_groups"
  16. "github.com/porter-dev/porter/internal/models"
  17. "github.com/porter-dev/porter/internal/telemetry"
  18. )
  19. type ListEnvironmentGroupsHandler struct {
  20. handlers.PorterHandlerReadWriter
  21. authz.KubernetesAgentGetter
  22. }
  23. func NewListEnvironmentGroupsHandler(
  24. config *config.Config,
  25. decoderValidator shared.RequestDecoderValidator,
  26. writer shared.ResultWriter,
  27. ) *ListEnvironmentGroupsHandler {
  28. return &ListEnvironmentGroupsHandler{
  29. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  30. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  31. }
  32. }
  33. // ListEnvironmentGroupsRequest is the request object for the /environment-groups endpoint
  34. type ListEnvironmentGroupsRequest struct {
  35. // Type of the env group to filter by. If empty, all env groups will be returned.
  36. Type string `json:"type"`
  37. }
  38. type ListEnvironmentGroupsResponse struct {
  39. EnvironmentGroups []EnvironmentGroupListItem `json:"environment_groups,omitempty"`
  40. }
  41. // EnvironmentGroupFile represents a file in an environment group
  42. type EnvironmentGroupFile struct {
  43. Name string `json:"name"`
  44. Contents string `json:"contents"`
  45. }
  46. // EnvironmentGroupListItem represents an environment group in the list response
  47. type EnvironmentGroupListItem struct {
  48. Name string `json:"name"`
  49. Type string `json:"type"`
  50. LatestVersion int `json:"latest_version"`
  51. Variables map[string]string `json:"variables,omitempty"`
  52. SecretVariables map[string]string `json:"secret_variables,omitempty"`
  53. CreatedAtUTC time.Time `json:"created_at"`
  54. LinkedApplications []string `json:"linked_applications,omitempty"`
  55. Files []EnvironmentGroupFile `json:"files,omitempty"`
  56. }
  57. func (c *ListEnvironmentGroupsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  58. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-env-groups")
  59. defer span.End()
  60. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  61. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  62. request := &ListEnvironmentGroupsRequest{}
  63. if ok := c.DecodeAndValidate(w, r, request); !ok {
  64. err := telemetry.Error(ctx, span, nil, "unable to decode or validate request body")
  65. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  66. return
  67. }
  68. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "env-group-type", Value: request.Type})
  69. if project.GetFeatureFlag(models.ValidateApplyV2, c.Config().LaunchDarklyClient) {
  70. listEnvGroupsReq := connect.NewRequest(&porterv1.ListEnvGroupsRequest{
  71. ProjectId: int64(project.ID),
  72. ClusterId: int64(cluster.ID),
  73. IncludeSecrets: false,
  74. })
  75. listEnvGroupResp, err := c.Config().ClusterControlPlaneClient.ListEnvGroups(ctx, listEnvGroupsReq)
  76. if err != nil {
  77. err = telemetry.Error(ctx, span, err, "unable to get linked applications")
  78. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  79. return
  80. }
  81. if listEnvGroupResp == nil || listEnvGroupResp.Msg == nil {
  82. err = telemetry.Error(ctx, span, err, "ccp resp is nil")
  83. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  84. return
  85. }
  86. var envGroups []EnvironmentGroupListItem
  87. for _, envGroup := range listEnvGroupResp.Msg.EnvGroups {
  88. var files []EnvironmentGroupFile
  89. for _, file := range envGroup.Files {
  90. decoded, err := base64.StdEncoding.DecodeString(file.B64Contents)
  91. if err != nil {
  92. err = telemetry.Error(ctx, span, err, "unable to decode base64 contents")
  93. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  94. return
  95. }
  96. files = append(files, EnvironmentGroupFile{
  97. Name: file.Name,
  98. Contents: string(decoded),
  99. })
  100. }
  101. envGroups = append(envGroups, EnvironmentGroupListItem{
  102. Name: envGroup.Name,
  103. Type: translateProtoTypeToEnvGroupType[envGroup.Type],
  104. LatestVersion: int(envGroup.Version),
  105. Variables: envGroup.Variables,
  106. SecretVariables: envGroup.SecretVariables,
  107. Files: files,
  108. CreatedAtUTC: envGroup.CreatedAt.AsTime(),
  109. LinkedApplications: envGroup.LinkedApplications,
  110. })
  111. }
  112. // return early for cleaner change
  113. c.WriteResult(w, r, ListEnvironmentGroupsResponse{EnvironmentGroups: envGroups})
  114. return
  115. }
  116. agent, err := c.GetAgent(r, cluster, "")
  117. if err != nil {
  118. err = telemetry.Error(ctx, span, err, "unable to connect to cluster")
  119. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusServiceUnavailable))
  120. return
  121. }
  122. allEnvGroupVersions, err := environmentgroups.ListEnvironmentGroups(ctx, agent, environmentgroups.WithNamespace(environmentgroups.Namespace_EnvironmentGroups), environmentgroups.WithoutDefaultAppEnvironmentGroups(), environmentgroups.WithoutDefaultAddonEnvironmentGroups())
  123. if err != nil {
  124. err = telemetry.Error(ctx, span, err, "unable to list all environment groups")
  125. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  126. return
  127. }
  128. if request.Type != "" {
  129. var filteredEnvGroupVersions []environmentgroups.EnvironmentGroup
  130. for _, envGroup := range allEnvGroupVersions {
  131. if envGroup.Type == request.Type {
  132. filteredEnvGroupVersions = append(filteredEnvGroupVersions, envGroup)
  133. }
  134. }
  135. allEnvGroupVersions = filteredEnvGroupVersions
  136. }
  137. envGroupSet := make(map[string]struct{})
  138. for _, envGroup := range allEnvGroupVersions {
  139. if envGroup.Name == "" {
  140. continue
  141. }
  142. if _, ok := envGroupSet[envGroup.Name]; !ok {
  143. envGroupSet[envGroup.Name] = struct{}{}
  144. }
  145. }
  146. var envGroups []EnvironmentGroupListItem
  147. for envGroupName := range envGroupSet {
  148. latestVersion, err := environmentgroups.LatestBaseEnvironmentGroup(ctx, agent, envGroupName)
  149. if err != nil {
  150. err = telemetry.Error(ctx, span, err, "unable to get latest environment groups")
  151. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  152. return
  153. }
  154. var linkedApplications []string
  155. applications, err := environmentgroups.LinkedApplications(ctx, agent, latestVersion.Name, true)
  156. if err != nil {
  157. err = telemetry.Error(ctx, span, err, "unable to get linked applications")
  158. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  159. return
  160. }
  161. applicationSetForEnvGroup := make(map[string]struct{})
  162. for _, app := range applications {
  163. if app.Namespace == "" {
  164. continue
  165. }
  166. if _, ok := applicationSetForEnvGroup[app.Namespace]; !ok {
  167. applicationSetForEnvGroup[app.Namespace] = struct{}{}
  168. }
  169. }
  170. for appNamespace := range applicationSetForEnvGroup {
  171. porterAppName := strings.TrimPrefix(appNamespace, "porter-stack-")
  172. linkedApplications = append(linkedApplications, porterAppName)
  173. }
  174. secrets := make(map[string]string)
  175. for k, v := range latestVersion.SecretVariables {
  176. secrets[k] = string(v)
  177. }
  178. envGroups = append(envGroups, EnvironmentGroupListItem{
  179. Name: latestVersion.Name,
  180. Type: latestVersion.Type,
  181. LatestVersion: latestVersion.Version,
  182. Variables: latestVersion.Variables,
  183. SecretVariables: secrets,
  184. CreatedAtUTC: latestVersion.CreatedAtUTC,
  185. LinkedApplications: linkedApplications,
  186. })
  187. }
  188. c.WriteResult(w, r, ListEnvironmentGroupsResponse{EnvironmentGroups: envGroups})
  189. }
  190. var translateProtoTypeToEnvGroupType = map[porterv1.EnumEnvGroupProviderType]string{
  191. porterv1.EnumEnvGroupProviderType_ENUM_ENV_GROUP_PROVIDER_TYPE_DATASTORE: "datastore",
  192. porterv1.EnumEnvGroupProviderType_ENUM_ENV_GROUP_PROVIDER_TYPE_DOPPLER: "doppler",
  193. porterv1.EnumEnvGroupProviderType_ENUM_ENV_GROUP_PROVIDER_TYPE_INFISICAL: "infisical",
  194. porterv1.EnumEnvGroupProviderType_ENUM_ENV_GROUP_PROVIDER_TYPE_PORTER: "porter",
  195. }