apply.go 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. package cmd
  2. import (
  3. "context"
  4. "fmt"
  5. "io/ioutil"
  6. "os"
  7. "strconv"
  8. "github.com/cli/cli/git"
  9. "github.com/mitchellh/mapstructure"
  10. api "github.com/porter-dev/porter/api/client"
  11. "github.com/porter-dev/porter/api/types"
  12. "github.com/porter-dev/switchboard/pkg/drivers"
  13. "github.com/porter-dev/switchboard/pkg/models"
  14. "github.com/porter-dev/switchboard/pkg/parser"
  15. switchboardTypes "github.com/porter-dev/switchboard/pkg/types"
  16. "github.com/porter-dev/switchboard/pkg/worker"
  17. "github.com/rs/zerolog"
  18. "github.com/spf13/cobra"
  19. )
  20. // applyCmd represents the "porter apply" base command when called
  21. // with a porter.yaml file as an argument
  22. var applyCmd = &cobra.Command{
  23. Use: "apply",
  24. Short: "Applies the provided porter.yaml to a project",
  25. Run: func(cmd *cobra.Command, args []string) {
  26. err := checkLoginAndRun(args, apply)
  27. if err != nil {
  28. os.Exit(1)
  29. }
  30. },
  31. }
  32. var porterYAML string
  33. func init() {
  34. rootCmd.AddCommand(applyCmd)
  35. applyCmd.Flags().StringVarP(&porterYAML, "file", "f", "", "path to porter.yaml")
  36. applyCmd.MarkFlagRequired("file")
  37. }
  38. func apply(_ *types.GetAuthenticatedUserResponse, client *api.Client, args []string) error {
  39. fileBytes, err := ioutil.ReadFile(porterYAML)
  40. if err != nil {
  41. return err
  42. }
  43. resGroup, err := parser.ParseRawBytes(fileBytes)
  44. if err != nil {
  45. return err
  46. }
  47. basePath, err := os.Getwd()
  48. if err != nil {
  49. return err
  50. }
  51. worker := worker.NewWorker()
  52. worker.RegisterDriver("", NewPorterDriver) // FIXME: workaround for when driver is not present
  53. worker.RegisterDriver("porter.deploy", NewPorterDriver)
  54. return worker.Apply(resGroup, &switchboardTypes.ApplyOpts{
  55. BasePath: basePath,
  56. })
  57. }
  58. type Source struct {
  59. Name string
  60. Repo string
  61. Version string
  62. }
  63. type Target struct {
  64. Project uint
  65. Cluster uint
  66. Namespace string
  67. }
  68. type Config struct {
  69. Build struct {
  70. Method string
  71. Context string
  72. Dockerfile string
  73. }
  74. Values map[string]interface{}
  75. }
  76. type Driver struct {
  77. source *Source
  78. target *Target
  79. config *Config
  80. output map[string]interface{}
  81. lookupTable *map[string]drivers.Driver
  82. logger *zerolog.Logger
  83. }
  84. func NewPorterDriver(resource *models.Resource, opts *drivers.SharedDriverOpts) (drivers.Driver, error) {
  85. driver := &Driver{
  86. lookupTable: opts.DriverLookupTable,
  87. logger: opts.Logger,
  88. }
  89. source, err := getSource(resource.Source)
  90. if err != nil {
  91. return nil, err
  92. }
  93. driver.source = source
  94. target, err := getTarget(resource.Target)
  95. if err != nil {
  96. return nil, err
  97. }
  98. driver.target = target
  99. config, err := getConfig(resource.Config)
  100. if err != nil {
  101. return nil, err
  102. }
  103. driver.config = config
  104. return driver, nil
  105. }
  106. func (d *Driver) ShouldApply(resource *models.Resource) bool {
  107. return true
  108. }
  109. func (d *Driver) Apply(resource *models.Resource) (*models.Resource, error) {
  110. client := GetAPIClient(config)
  111. // TODO: use source.repo, source.version, config.values
  112. config.SetProject(d.target.Project)
  113. config.SetCluster(d.target.Cluster)
  114. namespace = d.target.Namespace
  115. existingNamespaces, err := client.GetK8sNamespaces(context.Background(), config.Project, config.Cluster)
  116. if err != nil {
  117. return nil, err
  118. }
  119. namespaceFound := false
  120. for _, ns := range existingNamespaces.Items {
  121. if namespace == ns.Name {
  122. namespaceFound = true
  123. break
  124. }
  125. }
  126. if !namespaceFound {
  127. _, err := client.CreateNewK8sNamespace(
  128. context.Background(), config.Project, config.Cluster, namespace)
  129. if err != nil {
  130. return nil, err
  131. }
  132. }
  133. method = d.config.Build.Method
  134. if method == "" {
  135. return nil, fmt.Errorf("method should either be \"docker\" or \"pack\"")
  136. } else if method == "docker" {
  137. dockerfile = d.config.Build.Dockerfile
  138. }
  139. localPath = d.config.Build.Context
  140. source = "local"
  141. valuesObj = d.config.Values
  142. _, err = client.GetRelease(context.Background(), config.Project, config.Cluster, d.target.Namespace, resource.Name)
  143. if err == nil {
  144. // app exists
  145. if resource.Name == "" {
  146. return nil, fmt.Errorf("empty app name")
  147. }
  148. app = resource.Name
  149. tag = os.Getenv("PORTER_TAG")
  150. if tag == "" {
  151. commit, err := git.LastCommit()
  152. if err != nil {
  153. return nil, err
  154. }
  155. tag = commit.Sha
  156. }
  157. if tag == "" {
  158. return nil, fmt.Errorf("could not find commit SHA to tag the image")
  159. }
  160. err = updateFull(nil, client, []string{})
  161. if err != nil {
  162. return nil, err
  163. }
  164. } else {
  165. // create new app
  166. name = resource.Name
  167. regList, err := client.ListRegistries(context.Background(), config.Project)
  168. if err != nil {
  169. return nil, err
  170. }
  171. if len(*regList) > 0 {
  172. registryURL = (*regList)[0].URL
  173. }
  174. err = createFull(nil, client, []string{d.source.Name})
  175. if err != nil {
  176. return nil, err
  177. }
  178. }
  179. return resource, nil
  180. }
  181. func (d *Driver) Output() (map[string]interface{}, error) {
  182. return d.output, nil
  183. }
  184. func getSource(genericSource map[string]interface{}) (*Source, error) {
  185. source := &Source{}
  186. // first read from env vars
  187. source.Name = os.Getenv("PORTER_SOURCE_NAME")
  188. source.Repo = os.Getenv("PORTER_SOURCE_REPO")
  189. source.Version = os.Getenv("PORTER_SOURCE_VERSION")
  190. // next, check for values in the YAML file
  191. if source.Name == "" {
  192. if name, ok := genericSource["name"]; ok {
  193. source.Name = name.(string)
  194. }
  195. }
  196. if source.Repo == "" {
  197. if repo, ok := genericSource["repo"]; ok {
  198. source.Repo = repo.(string)
  199. }
  200. }
  201. if source.Version == "" {
  202. if version, ok := genericSource["version"]; ok {
  203. source.Version = version.(string)
  204. }
  205. }
  206. // lastly, just put in the defaults
  207. if source.Name == "" || source.Repo == "" || source.Version == "" {
  208. // default to these values when any one of the source values are empty
  209. // this makes sense because it might save us from version mismatches and other mishaps
  210. source.Name = "web"
  211. source.Repo = "https://charts.getporter.dev"
  212. source.Version = "v0.13.0"
  213. }
  214. return source, nil
  215. }
  216. func getTarget(genericTarget map[string]interface{}) (*Target, error) {
  217. target := &Target{}
  218. // first read from env vars
  219. if projectEnv := os.Getenv("PORTER_PROJECT"); projectEnv != "" {
  220. project, err := strconv.Atoi(projectEnv)
  221. if err != nil {
  222. return nil, err
  223. }
  224. target.Project = uint(project)
  225. }
  226. if clusterEnv := os.Getenv("PORTER_CLUSTER"); clusterEnv != "" {
  227. cluster, err := strconv.Atoi(clusterEnv)
  228. if err != nil {
  229. return nil, err
  230. }
  231. target.Cluster = uint(cluster)
  232. }
  233. target.Namespace = os.Getenv("PORTER_NAMESPACE")
  234. // next, check for values in the YAML file
  235. if target.Project == 0 {
  236. if project, ok := genericTarget["project"]; ok {
  237. target.Project = project.(uint)
  238. }
  239. }
  240. if target.Cluster == 0 {
  241. if cluster, ok := genericTarget["cluster"]; ok {
  242. target.Cluster = cluster.(uint)
  243. }
  244. }
  245. if target.Namespace == "" {
  246. if namespace, ok := genericTarget["namespace"]; ok {
  247. target.Namespace = namespace.(string)
  248. }
  249. }
  250. // lastly, just put in the defaults
  251. if target.Project == 0 || target.Cluster == 0 || target.Namespace == "" {
  252. // default to these values when any one of the target values are empty
  253. target.Project = config.Project
  254. target.Cluster = config.Cluster
  255. target.Namespace = "default"
  256. }
  257. return target, nil
  258. }
  259. func getConfig(genericConfig map[string]interface{}) (*Config, error) {
  260. config := &Config{}
  261. err := mapstructure.Decode(genericConfig, config)
  262. if err != nil {
  263. return nil, err
  264. }
  265. return config, nil
  266. }