/vendor/github.com/hashicorp/hcl/hcl/fmtcmd/fmtcmd_test.go

https://bitbucket.org/gollariel/dockertool · Go · 440 lines · 383 code · 52 blank · 5 comment · 82 complexity · 4acfe32c2d4874f423a1e0d33263686f MD5 · raw file

  1. // +build !windows
  2. // TODO(jen20): These need fixing on Windows but fmt is not used right now
  3. // and red CI is making it harder to process other bugs, so ignore until
  4. // we get around to fixing them.
  5. package fmtcmd
  6. import (
  7. "bytes"
  8. "fmt"
  9. "io/ioutil"
  10. "os"
  11. "path/filepath"
  12. "reflect"
  13. "regexp"
  14. "sort"
  15. "syscall"
  16. "testing"
  17. "github.com/hashicorp/hcl/testhelper"
  18. )
  19. var fixtureExtensions = []string{"hcl"}
  20. func init() {
  21. sort.Sort(ByFilename(fixtures))
  22. }
  23. func TestIsValidFile(t *testing.T) {
  24. const fixtureDir = "./test-fixtures"
  25. cases := []struct {
  26. Path string
  27. Expected bool
  28. }{
  29. {"good.hcl", true},
  30. {".hidden.ignore", false},
  31. {"file.ignore", false},
  32. {"dir.ignore", false},
  33. }
  34. for _, tc := range cases {
  35. file, err := os.Stat(filepath.Join(fixtureDir, tc.Path))
  36. if err != nil {
  37. t.Errorf("unexpected error: %s", err)
  38. }
  39. if res := isValidFile(file, fixtureExtensions); res != tc.Expected {
  40. t.Errorf("want: %b, got: %b", tc.Expected, res)
  41. }
  42. }
  43. }
  44. func TestRunMultiplePaths(t *testing.T) {
  45. path1, err := renderFixtures("")
  46. if err != nil {
  47. t.Errorf("unexpected error: %s", err)
  48. }
  49. defer os.RemoveAll(path1)
  50. path2, err := renderFixtures("")
  51. if err != nil {
  52. t.Errorf("unexpected error: %s", err)
  53. }
  54. defer os.RemoveAll(path2)
  55. var expectedOut bytes.Buffer
  56. for _, path := range []string{path1, path2} {
  57. for _, fixture := range fixtures {
  58. if !bytes.Equal(fixture.golden, fixture.input) {
  59. expectedOut.WriteString(filepath.Join(path, fixture.filename) + "\n")
  60. }
  61. }
  62. }
  63. _, stdout := mockIO()
  64. err = Run(
  65. []string{path1, path2},
  66. fixtureExtensions,
  67. nil, stdout,
  68. Options{
  69. List: true,
  70. },
  71. )
  72. if err != nil {
  73. t.Errorf("unexpected error: %s", err)
  74. }
  75. if stdout.String() != expectedOut.String() {
  76. t.Errorf("stdout want:\n%s\ngot:\n%s", expectedOut, stdout)
  77. }
  78. }
  79. func TestRunSubDirectories(t *testing.T) {
  80. pathParent, err := ioutil.TempDir("", "")
  81. if err != nil {
  82. t.Errorf("unexpected error: %s", err)
  83. }
  84. defer os.RemoveAll(pathParent)
  85. path1, err := renderFixtures(pathParent)
  86. if err != nil {
  87. t.Errorf("unexpected error: %s", err)
  88. }
  89. path2, err := renderFixtures(pathParent)
  90. if err != nil {
  91. t.Errorf("unexpected error: %s", err)
  92. }
  93. paths := []string{path1, path2}
  94. sort.Strings(paths)
  95. var expectedOut bytes.Buffer
  96. for _, path := range paths {
  97. for _, fixture := range fixtures {
  98. if !bytes.Equal(fixture.golden, fixture.input) {
  99. expectedOut.WriteString(filepath.Join(path, fixture.filename) + "\n")
  100. }
  101. }
  102. }
  103. _, stdout := mockIO()
  104. err = Run(
  105. []string{pathParent},
  106. fixtureExtensions,
  107. nil, stdout,
  108. Options{
  109. List: true,
  110. },
  111. )
  112. if err != nil {
  113. t.Errorf("unexpected error: %s", err)
  114. }
  115. if stdout.String() != expectedOut.String() {
  116. t.Errorf("stdout want:\n%s\ngot:\n%s", expectedOut, stdout)
  117. }
  118. }
  119. func TestRunStdin(t *testing.T) {
  120. var expectedOut bytes.Buffer
  121. for i, fixture := range fixtures {
  122. if i != 0 {
  123. expectedOut.WriteString("\n")
  124. }
  125. expectedOut.Write(fixture.golden)
  126. }
  127. stdin, stdout := mockIO()
  128. for _, fixture := range fixtures {
  129. stdin.Write(fixture.input)
  130. }
  131. err := Run(
  132. []string{},
  133. fixtureExtensions,
  134. stdin, stdout,
  135. Options{},
  136. )
  137. if err != nil {
  138. t.Errorf("unexpected error: %s", err)
  139. }
  140. if !bytes.Equal(stdout.Bytes(), expectedOut.Bytes()) {
  141. t.Errorf("stdout want:\n%s\ngot:\n%s", expectedOut, stdout)
  142. }
  143. }
  144. func TestRunStdinAndWrite(t *testing.T) {
  145. var expectedOut = []byte{}
  146. stdin, stdout := mockIO()
  147. stdin.WriteString("")
  148. err := Run(
  149. []string{}, []string{},
  150. stdin, stdout,
  151. Options{
  152. Write: true,
  153. },
  154. )
  155. if err != ErrWriteStdin {
  156. t.Errorf("error want:\n%s\ngot:\n%s", ErrWriteStdin, err)
  157. }
  158. if !bytes.Equal(stdout.Bytes(), expectedOut) {
  159. t.Errorf("stdout want:\n%s\ngot:\n%s", expectedOut, stdout)
  160. }
  161. }
  162. func TestRunFileError(t *testing.T) {
  163. path, err := ioutil.TempDir("", "")
  164. if err != nil {
  165. t.Errorf("unexpected error: %s", err)
  166. }
  167. defer os.RemoveAll(path)
  168. filename := filepath.Join(path, "unreadable.hcl")
  169. var expectedError = &os.PathError{
  170. Op: "open",
  171. Path: filename,
  172. Err: syscall.EACCES,
  173. }
  174. err = ioutil.WriteFile(filename, []byte{}, 0000)
  175. if err != nil {
  176. t.Errorf("unexpected error: %s", err)
  177. }
  178. _, stdout := mockIO()
  179. err = Run(
  180. []string{path},
  181. fixtureExtensions,
  182. nil, stdout,
  183. Options{},
  184. )
  185. if !reflect.DeepEqual(err, expectedError) {
  186. t.Errorf("error want: %#v, got: %#v", expectedError, err)
  187. }
  188. }
  189. func TestRunNoOptions(t *testing.T) {
  190. path, err := renderFixtures("")
  191. if err != nil {
  192. t.Errorf("unexpected error: %s", err)
  193. }
  194. defer os.RemoveAll(path)
  195. var expectedOut bytes.Buffer
  196. for _, fixture := range fixtures {
  197. expectedOut.Write(fixture.golden)
  198. }
  199. _, stdout := mockIO()
  200. err = Run(
  201. []string{path},
  202. fixtureExtensions,
  203. nil, stdout,
  204. Options{},
  205. )
  206. if err != nil {
  207. t.Errorf("unexpected error: %s", err)
  208. }
  209. if stdout.String() != expectedOut.String() {
  210. t.Errorf("stdout want:\n%s\ngot:\n%s", expectedOut, stdout)
  211. }
  212. }
  213. func TestRunList(t *testing.T) {
  214. path, err := renderFixtures("")
  215. if err != nil {
  216. t.Errorf("unexpected error: %s", err)
  217. }
  218. defer os.RemoveAll(path)
  219. var expectedOut bytes.Buffer
  220. for _, fixture := range fixtures {
  221. if !bytes.Equal(fixture.golden, fixture.input) {
  222. expectedOut.WriteString(fmt.Sprintln(filepath.Join(path, fixture.filename)))
  223. }
  224. }
  225. _, stdout := mockIO()
  226. err = Run(
  227. []string{path},
  228. fixtureExtensions,
  229. nil, stdout,
  230. Options{
  231. List: true,
  232. },
  233. )
  234. if err != nil {
  235. t.Errorf("unexpected error: %s", err)
  236. }
  237. if stdout.String() != expectedOut.String() {
  238. t.Errorf("stdout want:\n%s\ngot:\n%s", expectedOut, stdout)
  239. }
  240. }
  241. func TestRunWrite(t *testing.T) {
  242. path, err := renderFixtures("")
  243. if err != nil {
  244. t.Errorf("unexpected error: %s", err)
  245. }
  246. defer os.RemoveAll(path)
  247. _, stdout := mockIO()
  248. err = Run(
  249. []string{path},
  250. fixtureExtensions,
  251. nil, stdout,
  252. Options{
  253. Write: true,
  254. },
  255. )
  256. if err != nil {
  257. t.Errorf("unexpected error: %s", err)
  258. }
  259. for _, fixture := range fixtures {
  260. res, err := ioutil.ReadFile(filepath.Join(path, fixture.filename))
  261. if err != nil {
  262. t.Errorf("unexpected error: %s", err)
  263. }
  264. if !bytes.Equal(res, fixture.golden) {
  265. t.Errorf("file %q contents want:\n%s\ngot:\n%s", fixture.filename, fixture.golden, res)
  266. }
  267. }
  268. }
  269. func TestRunDiff(t *testing.T) {
  270. path, err := renderFixtures("")
  271. if err != nil {
  272. t.Errorf("unexpected error: %s", err)
  273. }
  274. defer os.RemoveAll(path)
  275. var expectedOut bytes.Buffer
  276. for _, fixture := range fixtures {
  277. if len(fixture.diff) > 0 {
  278. expectedOut.WriteString(
  279. regexp.QuoteMeta(
  280. fmt.Sprintf("diff a/%s/%s b/%s/%s\n", path, fixture.filename, path, fixture.filename),
  281. ),
  282. )
  283. // Need to use regex to ignore datetimes in diff.
  284. expectedOut.WriteString(`--- .+?\n`)
  285. expectedOut.WriteString(`\+\+\+ .+?\n`)
  286. expectedOut.WriteString(regexp.QuoteMeta(string(fixture.diff)))
  287. }
  288. }
  289. expectedOutString := testhelper.Unix2dos(expectedOut.String())
  290. _, stdout := mockIO()
  291. err = Run(
  292. []string{path},
  293. fixtureExtensions,
  294. nil, stdout,
  295. Options{
  296. Diff: true,
  297. },
  298. )
  299. if err != nil {
  300. t.Errorf("unexpected error: %s", err)
  301. }
  302. if !regexp.MustCompile(expectedOutString).Match(stdout.Bytes()) {
  303. t.Errorf("stdout want match:\n%s\ngot:\n%q", expectedOutString, stdout)
  304. }
  305. }
  306. func mockIO() (stdin, stdout *bytes.Buffer) {
  307. return new(bytes.Buffer), new(bytes.Buffer)
  308. }
  309. type fixture struct {
  310. filename string
  311. input, golden, diff []byte
  312. }
  313. type ByFilename []fixture
  314. func (s ByFilename) Len() int { return len(s) }
  315. func (s ByFilename) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
  316. func (s ByFilename) Less(i, j int) bool { return len(s[i].filename) > len(s[j].filename) }
  317. var fixtures = []fixture{
  318. {
  319. "noop.hcl",
  320. []byte(`resource "aws_security_group" "firewall" {
  321. count = 5
  322. }
  323. `),
  324. []byte(`resource "aws_security_group" "firewall" {
  325. count = 5
  326. }
  327. `),
  328. []byte(``),
  329. }, {
  330. "align_equals.hcl",
  331. []byte(`variable "foo" {
  332. default = "bar"
  333. description = "bar"
  334. }
  335. `),
  336. []byte(`variable "foo" {
  337. default = "bar"
  338. description = "bar"
  339. }
  340. `),
  341. []byte(`@@ -1,4 +1,4 @@
  342. variable "foo" {
  343. - default = "bar"
  344. + default = "bar"
  345. description = "bar"
  346. }
  347. `),
  348. }, {
  349. "indentation.hcl",
  350. []byte(`provider "aws" {
  351. access_key = "foo"
  352. secret_key = "bar"
  353. }
  354. `),
  355. []byte(`provider "aws" {
  356. access_key = "foo"
  357. secret_key = "bar"
  358. }
  359. `),
  360. []byte(`@@ -1,4 +1,4 @@
  361. provider "aws" {
  362. - access_key = "foo"
  363. - secret_key = "bar"
  364. + access_key = "foo"
  365. + secret_key = "bar"
  366. }
  367. `),
  368. },
  369. }
  370. // parent can be an empty string, in which case the system's default
  371. // temporary directory will be used.
  372. func renderFixtures(parent string) (path string, err error) {
  373. path, err = ioutil.TempDir(parent, "")
  374. if err != nil {
  375. return "", err
  376. }
  377. for _, fixture := range fixtures {
  378. err = ioutil.WriteFile(filepath.Join(path, fixture.filename), []byte(fixture.input), 0644)
  379. if err != nil {
  380. os.RemoveAll(path)
  381. return "", err
  382. }
  383. }
  384. return path, nil
  385. }