project.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. package models
  2. import (
  3. "fmt"
  4. "gorm.io/gorm"
  5. "github.com/launchdarkly/go-sdk-common/v3/ldcontext"
  6. "github.com/porter-dev/porter/api/types"
  7. "github.com/porter-dev/porter/internal/features"
  8. ints "github.com/porter-dev/porter/internal/models/integrations"
  9. )
  10. // FeatureFlagLabel strongly types project feature flags
  11. type FeatureFlagLabel string
  12. const (
  13. // APITokensEnabled allows users to create Bearer tokens for use with the Porter API
  14. // #nosec G101 - Not actually an api token
  15. APITokensEnabled FeatureFlagLabel = "api_tokens_enabled"
  16. // AzureEnabled enables Azure Provisioning
  17. AzureEnabled FeatureFlagLabel = "azure_enabled"
  18. // CapiProvisionerEnabled enables the CAPI Provisioning flow
  19. CapiProvisionerEnabled FeatureFlagLabel = "capi_provisioner_enabled"
  20. // DBEnabled enables the "Databases" tab
  21. DBEnabled FeatureFlagLabel = "db_enabled"
  22. // EFSEnabled enables the "EFS" checkbox in App Settings
  23. EFSEnabled FeatureFlagLabel = "efs_enabled"
  24. // EnableReprovision enables the provisioning button after initial creation of the cluster
  25. EnableReprovision FeatureFlagLabel = "enable_reprovision"
  26. // FullAddOns shows all addons, not just curated
  27. FullAddOns FeatureFlagLabel = "full_add_ons"
  28. // GPUEnabled enables the "GPU for users"
  29. GPUEnabled FeatureFlagLabel = "gpu_enabled"
  30. // HelmValuesEnabled shows the helm values tab for porter apps (when simplified_view_enabled=true)
  31. HelmValuesEnabled FeatureFlagLabel = "helm_values_enabled"
  32. // ManagedInfraEnabled uses terraform provisioning instead of capi
  33. ManagedInfraEnabled FeatureFlagLabel = "managed_infra_enabled"
  34. // MultiCluster allows multiple clusters in simplified view (simplified_view_enabled=true)
  35. MultiCluster FeatureFlagLabel = "multi_cluster"
  36. // PreviewEnvsEnabled allows legacy user the ability to see preview environments in sidebar (simplified_view_enabled=false)
  37. PreviewEnvsEnabled FeatureFlagLabel = "preview_envs_enabled"
  38. // RDSDatabasesEnabled allows for users to provision RDS instances within their cluster vpc
  39. RDSDatabasesEnabled FeatureFlagLabel = "rds_databases_enabled"
  40. // QuotaIncrease enables whether we allow for auto increase of quota_increase
  41. QuotaIncrease FeatureFlagLabel = "quota_increase"
  42. // SimplifiedViewEnabled shows the new UI dashboard or not
  43. SimplifiedViewEnabled FeatureFlagLabel = "simplified_view_enabled"
  44. // StacksEnabled uses stack view for legacy (simplified_view_enabled=false)
  45. StacksEnabled FeatureFlagLabel = "stacks_enabled"
  46. // ValidateApplyV2 controls whether apps deploys use a porter app revision contract vs helm
  47. ValidateApplyV2 FeatureFlagLabel = "validate_apply_v2"
  48. // BetaFeaturesEnabled controls whether a project uses beta features
  49. BetaFeaturesEnabled FeatureFlagLabel = "beta_features_enabled"
  50. )
  51. // ProjectFeatureFlags keeps track of all project-related feature flags
  52. var ProjectFeatureFlags = map[FeatureFlagLabel]bool{
  53. APITokensEnabled: false,
  54. AzureEnabled: false,
  55. CapiProvisionerEnabled: true,
  56. DBEnabled: false,
  57. EFSEnabled: false,
  58. EnableReprovision: false,
  59. FullAddOns: false,
  60. GPUEnabled: false,
  61. HelmValuesEnabled: false,
  62. ManagedInfraEnabled: false,
  63. MultiCluster: false,
  64. PreviewEnvsEnabled: false,
  65. RDSDatabasesEnabled: false,
  66. QuotaIncrease: false,
  67. SimplifiedViewEnabled: true,
  68. StacksEnabled: false,
  69. ValidateApplyV2: true,
  70. BetaFeaturesEnabled: false,
  71. }
  72. type ProjectPlan string
  73. const (
  74. ProjectPlanBasic ProjectPlan = "basic"
  75. ProjectPlanTeam ProjectPlan = "team"
  76. ProjectPlanGrowth ProjectPlan = "growth"
  77. ProjectPlanEnterprise ProjectPlan = "enterprise"
  78. )
  79. // Project type that extends gorm.Model
  80. type Project struct {
  81. gorm.Model `gorm:"embedded"`
  82. Name string `json:"name"`
  83. Roles []Role `json:"roles"`
  84. ProjectUsageID uint
  85. ProjectUsageCacheID uint
  86. // linked repos
  87. GitRepos []GitRepo `json:"git_repos,omitempty"`
  88. // linked registries
  89. Registries []Registry `json:"registries,omitempty"`
  90. // linked clusters
  91. Clusters []Cluster `json:"clusters"`
  92. ClusterCandidates []ClusterCandidate `json:"cluster_candidates"`
  93. // linked databases
  94. Databases []Database `json:"databases"`
  95. // linked helm repos
  96. HelmRepos []HelmRepo `json:"helm_repos"`
  97. // invitations to the project
  98. Invites []Invite `json:"invites"`
  99. // provisioned aws infra
  100. Infras []Infra `json:"infras"`
  101. // auth mechanisms
  102. KubeIntegrations []ints.KubeIntegration `json:"kube_integrations"`
  103. BasicIntegrations []ints.BasicIntegration `json:"basic_integrations"`
  104. OIDCIntegrations []ints.OIDCIntegration `json:"oidc_integrations"`
  105. OAuthIntegrations []ints.OAuthIntegration `json:"oauth_integrations"`
  106. AWSIntegrations []ints.AWSIntegration `json:"aws_integrations"`
  107. GCPIntegrations []ints.GCPIntegration `json:"gcp_integrations"`
  108. AzureIntegrations []ints.AzureIntegration `json:"azure_integrations"`
  109. GitlabIntegrations []ints.GitlabIntegration `json:"gitlab_integrations"`
  110. // Deprecated: use p.GetFeatureFlag(PreviewEnvsEnabled, *features.Client) instead
  111. PreviewEnvsEnabled bool
  112. // Deprecated: use p.GetFeatureFlag(RDSDatabasesEnabled, *features.Client) instead
  113. RDSDatabasesEnabled bool
  114. // Deprecated: use p.GetFeatureFlag(ManagedInfraEnabled, *features.Client) instead
  115. ManagedInfraEnabled bool
  116. // Deprecated: use p.GetFeatureFlag(StacksEnabled, *features.Client) instead
  117. StacksEnabled bool
  118. // Deprecated: use p.GetFeatureFlag(APITokensEnabled, *features.Client) instead
  119. APITokensEnabled bool
  120. // Deprecated: use p.GetFeatureFlag(CapiProvisionerEnabled, *features.Client) instead
  121. CapiProvisionerEnabled bool
  122. // Deprecated: use p.GetFeatureFlag(SimplifiedViewEnabled, *features.Client) instead
  123. SimplifiedViewEnabled bool
  124. // Deprecated: use p.GetFeatureFlag(AzureEnabled, *features.Client) instead
  125. AzureEnabled bool
  126. // Deprecated: use p.GetFeatureFlag(HelmValuesEnabled, *features.Client) instead
  127. HelmValuesEnabled bool
  128. // Deprecated: use p.GetFeatureFlag(MultiCluster, *features.Client) instead
  129. MultiCluster bool `gorm:"default:false"`
  130. // Deprecated: use p.GetFeatureFlag(FullAddOns, *features.Client) instead
  131. FullAddOns bool `gorm:"default:false"`
  132. // Deprecated: use p.GetFeatureFlag(ValidateApplyV2, *features.Client) instead
  133. ValidateApplyV2 bool `gorm:"default:false"`
  134. // Deprecated: use p.GetFeatureFlag(EnableReprovision, *features.Client) instead
  135. EnableReprovision bool `gorm:"default:false"`
  136. }
  137. // GetFeatureFlag calls launchdarkly for the specified flag
  138. // and returns the configured value
  139. func (p *Project) GetFeatureFlag(flagName FeatureFlagLabel, launchDarklyClient *features.Client) bool {
  140. if launchDarklyClient.UseDatabase() {
  141. // case switch things
  142. switch flagName {
  143. case "api_tokens_enabled":
  144. return p.APITokensEnabled
  145. case "azure_enabled":
  146. return p.AzureEnabled
  147. case "capi_provisioner_enabled":
  148. return p.CapiProvisionerEnabled
  149. case "db_enabled":
  150. return false
  151. case "enable_reprovision":
  152. return p.EnableReprovision
  153. case "full_add_ons":
  154. return p.FullAddOns
  155. case "gpu_enabled":
  156. return false
  157. case "helm_values_enabled":
  158. return p.HelmValuesEnabled
  159. case "managed_infra_enabled":
  160. return p.ManagedInfraEnabled
  161. case "multi_cluster":
  162. return p.MultiCluster
  163. case "preview_envs_enabled":
  164. return p.PreviewEnvsEnabled
  165. case "quota_increase":
  166. return false
  167. case "rds_databases_enabled":
  168. return p.RDSDatabasesEnabled
  169. case "simplified_view_enabled":
  170. return p.SimplifiedViewEnabled
  171. case "stacks_enabled":
  172. return p.StacksEnabled
  173. case "validate_apply_v2":
  174. return p.ValidateApplyV2
  175. case "efs_enabled":
  176. return false
  177. }
  178. }
  179. projectID := p.ID
  180. projectName := p.Name
  181. ldContext := getProjectContext(projectID, projectName)
  182. defaultValue := ProjectFeatureFlags[flagName]
  183. value, _ := launchDarklyClient.BoolVariation(string(flagName), ldContext, defaultValue)
  184. return value
  185. }
  186. // ToProjectType generates an external types.Project to be shared over REST
  187. func (p *Project) ToProjectType(launchDarklyClient *features.Client) types.Project {
  188. roles := make([]*types.Role, 0)
  189. for _, role := range p.Roles {
  190. roles = append(roles, role.ToRoleType())
  191. }
  192. projectID := p.ID
  193. projectName := p.Name
  194. return types.Project{
  195. ID: projectID,
  196. Name: projectName,
  197. Roles: roles,
  198. PreviewEnvsEnabled: p.GetFeatureFlag(PreviewEnvsEnabled, launchDarklyClient),
  199. RDSDatabasesEnabled: p.GetFeatureFlag(RDSDatabasesEnabled, launchDarklyClient),
  200. ManagedInfraEnabled: p.GetFeatureFlag(ManagedInfraEnabled, launchDarklyClient),
  201. StacksEnabled: p.GetFeatureFlag(StacksEnabled, launchDarklyClient),
  202. APITokensEnabled: p.GetFeatureFlag(APITokensEnabled, launchDarklyClient),
  203. CapiProvisionerEnabled: p.GetFeatureFlag(CapiProvisionerEnabled, launchDarklyClient),
  204. DBEnabled: p.GetFeatureFlag(DBEnabled, launchDarklyClient),
  205. SimplifiedViewEnabled: p.GetFeatureFlag(SimplifiedViewEnabled, launchDarklyClient),
  206. AzureEnabled: p.GetFeatureFlag(AzureEnabled, launchDarklyClient),
  207. GPUEnabled: p.GetFeatureFlag(GPUEnabled, launchDarklyClient),
  208. HelmValuesEnabled: p.GetFeatureFlag(HelmValuesEnabled, launchDarklyClient),
  209. MultiCluster: p.GetFeatureFlag(MultiCluster, launchDarklyClient),
  210. EnableReprovision: p.GetFeatureFlag(EnableReprovision, launchDarklyClient),
  211. ValidateApplyV2: p.GetFeatureFlag(ValidateApplyV2, launchDarklyClient),
  212. FullAddOns: p.GetFeatureFlag(FullAddOns, launchDarklyClient),
  213. QuotaIncrease: p.GetFeatureFlag(QuotaIncrease, launchDarklyClient),
  214. EFSEnabled: p.GetFeatureFlag(EFSEnabled, launchDarklyClient),
  215. BetaFeaturesEnabled: p.GetFeatureFlag(BetaFeaturesEnabled, launchDarklyClient),
  216. }
  217. }
  218. // ToProjectListType returns a "minified" version of a Project
  219. // suitable for api responses to GET /projects
  220. // TODO: update this in the future to use default values for all
  221. // the feature flags instead of trying to retrieve them from the database
  222. func (p *Project) ToProjectListType() *types.ProjectList {
  223. var roles []types.Role
  224. for _, role := range p.Roles {
  225. roles = append(roles, *role.ToRoleType())
  226. }
  227. return &types.ProjectList{
  228. ID: p.ID,
  229. Name: p.Name,
  230. // note: all of these fields should be considered deprecated
  231. // in an api response
  232. Roles: roles,
  233. PreviewEnvsEnabled: p.PreviewEnvsEnabled,
  234. RDSDatabasesEnabled: p.RDSDatabasesEnabled,
  235. ManagedInfraEnabled: p.ManagedInfraEnabled,
  236. StacksEnabled: p.StacksEnabled,
  237. APITokensEnabled: p.APITokensEnabled,
  238. DBEnabled: false,
  239. CapiProvisionerEnabled: p.CapiProvisionerEnabled,
  240. SimplifiedViewEnabled: p.SimplifiedViewEnabled,
  241. AzureEnabled: p.AzureEnabled,
  242. HelmValuesEnabled: p.HelmValuesEnabled,
  243. MultiCluster: p.MultiCluster,
  244. EnableReprovision: p.EnableReprovision,
  245. ValidateApplyV2: p.ValidateApplyV2,
  246. FullAddOns: p.FullAddOns,
  247. }
  248. }
  249. func getProjectContext(projectID uint, projectName string) ldcontext.Context {
  250. projectIdentifier := fmt.Sprintf("project-%d", projectID)
  251. launchDarklyName := fmt.Sprintf("%s: %s", projectIdentifier, projectName)
  252. return ldcontext.NewBuilder(projectIdentifier).
  253. Kind("project").
  254. Name(launchDarklyName).
  255. SetInt("project_id", int(projectID)).
  256. Build()
  257. }