template_test.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. package report
  2. import (
  3. "os"
  4. "path/filepath"
  5. "testing"
  6. "github.com/stretchr/testify/assert"
  7. "github.com/stretchr/testify/require"
  8. )
  9. func TestWriteTemplate(t *testing.T) {
  10. tests := []struct {
  11. findings []Finding
  12. testReportName string
  13. expected string
  14. wantEmpty bool
  15. }{
  16. {
  17. testReportName: "markdown",
  18. expected: filepath.Join(expectPath, "report", "template_markdown.md"),
  19. findings: []Finding{
  20. {
  21. RuleID: "test-rule",
  22. Description: "A test rule",
  23. Match: "line containing secret",
  24. Secret: "a secret",
  25. StartLine: 1,
  26. EndLine: 2,
  27. StartColumn: 1,
  28. EndColumn: 2,
  29. Message: "opps",
  30. File: "auth.py",
  31. Commit: "0000000000000000",
  32. Author: "John Doe",
  33. Email: "johndoe@gmail.com",
  34. Date: "10-19-2003",
  35. Tags: []string{"tag1", "tag2", "tag3"},
  36. },
  37. },
  38. },
  39. {
  40. testReportName: "jsonextra",
  41. expected: filepath.Join(expectPath, "report", "template_jsonextra.json"),
  42. findings: []Finding{
  43. {
  44. RuleID: "test-rule",
  45. Description: "A test rule",
  46. Line: "whole line containing secret",
  47. Match: "line containing secret",
  48. Secret: "a secret",
  49. StartLine: 1,
  50. EndLine: 2,
  51. StartColumn: 1,
  52. EndColumn: 2,
  53. Message: "opps",
  54. File: "auth.py",
  55. Commit: "0000000000000000",
  56. Author: "John Doe",
  57. Email: "johndoe@gmail.com",
  58. Date: "10-19-2003",
  59. Tags: []string{"tag1", "tag2", "tag3"},
  60. },
  61. },
  62. },
  63. }
  64. for _, test := range tests {
  65. t.Run(test.testReportName, func(t *testing.T) {
  66. reporter, err := NewTemplateReporter(templatePath + test.testReportName + ".tmpl")
  67. require.NoError(t, err)
  68. tmpfile, err := os.Create(filepath.Join(t.TempDir(), test.testReportName+filepath.Ext(test.expected)))
  69. require.NoError(t, err)
  70. defer tmpfile.Close()
  71. err = reporter.Write(tmpfile, test.findings)
  72. require.NoError(t, err)
  73. assert.FileExists(t, tmpfile.Name())
  74. got, err := os.ReadFile(tmpfile.Name())
  75. require.NoError(t, err)
  76. if test.wantEmpty {
  77. assert.Empty(t, got)
  78. return
  79. }
  80. want, err := os.ReadFile(test.expected)
  81. require.NoError(t, err)
  82. wantStr := lineEndingReplacer.Replace(string(want))
  83. gotStr := lineEndingReplacer.Replace(string(got))
  84. assert.Equal(t, wantStr, gotStr)
  85. })
  86. }
  87. }
  88. func TestTemplateDangerousFunctions(t *testing.T) {
  89. tests := []struct {
  90. name string
  91. template string
  92. wantErr string
  93. }{
  94. {
  95. name: "env is blocked",
  96. template: `{{ env "SECRET" }}`,
  97. wantErr: `function "env" not defined`,
  98. },
  99. {
  100. name: "expandenv is blocked",
  101. template: `{{ expandenv "$SECRET" }}`,
  102. wantErr: `function "expandenv" not defined`,
  103. },
  104. {
  105. name: "getHostByName is blocked",
  106. template: `{{ getHostByName "localhost" }}`,
  107. wantErr: `function "getHostByName" not defined`,
  108. },
  109. {
  110. name: "now is allowed (benign)",
  111. template: `{{ now | date "2006-01-02" }}`,
  112. wantErr: "", // should not error on parse
  113. },
  114. }
  115. for _, tt := range tests {
  116. t.Run(tt.name, func(t *testing.T) {
  117. tmpfile, err := os.CreateTemp(t.TempDir(), "test*.tmpl")
  118. require.NoError(t, err)
  119. defer os.Remove(tmpfile.Name())
  120. _, err = tmpfile.WriteString(tt.template)
  121. require.NoError(t, err)
  122. tmpfile.Close()
  123. _, err = NewTemplateReporter(tmpfile.Name())
  124. if tt.wantErr != "" {
  125. assert.Error(t, err)
  126. assert.Contains(t, err.Error(), tt.wantErr)
  127. } else {
  128. assert.NoError(t, err)
  129. }
  130. })
  131. }
  132. }