terraform.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. package tools
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "os/exec"
  8. "github.com/hashicorp/go-version"
  9. "github.com/hashicorp/hc-install/product"
  10. "github.com/hashicorp/hc-install/releases"
  11. "github.com/hashicorp/terraform-exec/tfexec"
  12. tfjson "github.com/hashicorp/terraform-json"
  13. )
  14. func Terraform(workingDir string) *terraform {
  15. return &terraform{
  16. workingDir: workingDir,
  17. version: "1.14.5",
  18. logger: tfLogParser{},
  19. vars: []string{},
  20. }
  21. }
  22. type (
  23. terraform struct {
  24. tf *tfexec.Terraform
  25. state *tfjson.State
  26. workingDir string
  27. version string
  28. logger io.Writer
  29. vars []string
  30. }
  31. tfLogParser struct{}
  32. )
  33. func (_ tfLogParser) Write(p []byte) (n int, err error) {
  34. var logEntry tfjson.LogMessage
  35. err = json.Unmarshal(p, &logEntry)
  36. if err != nil {
  37. return 0, err
  38. }
  39. var msg []byte
  40. if logEntry.Level() == "" {
  41. msg = []byte(fmt.Sprintf("%s\n", logEntry.Message()))
  42. } else {
  43. msg = []byte(fmt.Sprintf("[%s] %s\n", logEntry.Level(), logEntry.Message()))
  44. }
  45. log.Write(msg)
  46. return len(msg), nil
  47. }
  48. func (t *terraform) VarsFile(path string) *terraform {
  49. t.vars = append(t.vars, path)
  50. return t
  51. }
  52. func (t *terraform) Version(version string) *terraform {
  53. t.version = version
  54. return t
  55. }
  56. func (t *terraform) Init() error {
  57. var err error
  58. var execPath = ""
  59. var install = true
  60. execPath, err = exec.LookPath("terraform")
  61. if err == nil {
  62. t.tf, err = tfexec.NewTerraform(t.workingDir, execPath)
  63. if err == nil {
  64. v, _, err := t.tf.Version(context.Background(), true)
  65. if err == nil {
  66. if v.String() == t.version {
  67. install = false
  68. }
  69. }
  70. }
  71. }
  72. if install {
  73. log.Write([]byte(fmt.Sprintf("Installing Terraform version: %s", t.version)))
  74. installer := &releases.ExactVersion{
  75. Product: product.Terraform,
  76. Version: version.Must(version.NewVersion(t.version)),
  77. }
  78. execPath, err = installer.Install(context.Background())
  79. if err != nil {
  80. return err
  81. }
  82. }
  83. t.tf, err = tfexec.NewTerraform(t.workingDir, execPath)
  84. if err != nil {
  85. return err
  86. }
  87. err = t.tf.InitJSON(context.Background(),
  88. t.logger,
  89. tfexec.Upgrade(true),
  90. )
  91. if err != nil {
  92. return err
  93. }
  94. t.state, err = t.tf.Show(context.Background())
  95. if err != nil {
  96. return err
  97. }
  98. return nil
  99. }
  100. func (t *terraform) Plan() (*tfjson.Plan, error) {
  101. if t.tf == nil {
  102. err := t.Init()
  103. if err != nil {
  104. return nil, err
  105. }
  106. }
  107. opts := []tfexec.PlanOption{
  108. tfexec.Out("terraform.plan"),
  109. }
  110. for _, f := range t.vars {
  111. tfexec.VarFile(f)
  112. }
  113. _, err := t.tf.PlanJSON(context.Background(),
  114. t.logger,
  115. opts...,
  116. )
  117. if err != nil {
  118. return nil, err
  119. }
  120. plan, err := t.tf.ShowPlanFile(context.Background(), t.workingDir+"/terraform.plan")
  121. if err != nil {
  122. return nil, err
  123. }
  124. return plan, nil
  125. }
  126. func (t *terraform) Apply() error {
  127. if t.tf == nil {
  128. err := t.Init()
  129. if err != nil {
  130. return err
  131. }
  132. }
  133. err := t.tf.ApplyJSON(context.Background(), t.logger)
  134. if err != nil {
  135. return err
  136. }
  137. t.state, err = t.tf.Show(context.Background())
  138. if err != nil {
  139. return err
  140. }
  141. return nil
  142. }
  143. func (t *terraform) Show() error {
  144. if t.tf == nil {
  145. err := t.Init()
  146. if err != nil {
  147. return err
  148. }
  149. }
  150. var err error
  151. t.state, err = t.tf.Show(context.Background())
  152. if err != nil {
  153. return err
  154. }
  155. return nil
  156. }
  157. func (t *terraform) Outputs() map[string]any {
  158. outputs := map[string]any{}
  159. if t.state != nil {
  160. if t.state.Values != nil {
  161. if t.state.Values.Outputs != nil {
  162. for k, v := range t.state.Values.Outputs {
  163. outputs[k] = v.Value
  164. }
  165. }
  166. }
  167. }
  168. return outputs
  169. }
  170. func (t *terraform) State() *tfjson.StateModule {
  171. if t.state != nil {
  172. if t.state.Values != nil {
  173. return t.state.Values.RootModule
  174. }
  175. }
  176. return nil
  177. }
  178. func (t *terraform) Destroy() error {
  179. if t.tf == nil {
  180. err := t.Init()
  181. if err != nil {
  182. return err
  183. }
  184. }
  185. err := t.tf.DestroyJSON(context.Background(), t.logger)
  186. if err != nil {
  187. return err
  188. }
  189. t.state, err = t.tf.Show(context.Background())
  190. if err != nil {
  191. return err
  192. }
  193. return nil
  194. }