package commands
import (
"fmt"
"io/ioutil"
"strings"
"testing"
"time"
"github.com/stretchr/testify/require"
)
func TestComment(t *testing.T) {
const golden = "testdata/comment/message-only"
env, bug := newTestEnvAndBug(t)
require.NoError(t, runComment(env.env, []string{bug}))
requireCommentsEqual(t, golden, env)
}
const gitDateFormat = "Mon Jan 2 15:04:05 2006 -0700"
type parsedComment struct {
author string
id string
date time.Time
message string
}
type parseFunc func(*parsedComment, string)
type commentParser struct {
t *testing.T
fn parseFunc
comments []parsedComment
}
func parseComments(t *testing.T, env *testEnv) []parsedComment {
t.Helper()
parser := &commentParser{
t: t,
comments: []parsedComment{},
}
comment := &parsedComment{}
parser.fn = parser.parseAuthor
for _, line := range strings.Split(env.out.String(), "\n") {
parser.fn(comment, line)
}
parser.comments = append(parser.comments, *comment)
return parser.comments
}
func (p *commentParser) parseAuthor(comment *parsedComment, line string) {
p.t.Helper()
tkns := strings.Split(line, ": ")
require.Len(p.t, tkns, 2)
require.Equal(p.t, "Author", tkns[0])
comment.author = tkns[1]
p.fn = p.parseID
}
func (p *commentParser) parseID(comment *parsedComment, line string) {
p.t.Helper()
tkns := strings.Split(line, ": ")
require.Len(p.t, tkns, 2)
require.Equal(p.t, "Id", tkns[0])
comment.id = tkns[1]
p.fn = p.parseDate
}
func (p *commentParser) parseDate(comment *parsedComment, line string) {
p.t.Helper()
tkns := strings.Split(line, ": ")
require.Len(p.t, tkns, 2)
require.Equal(p.t, "Date", tkns[0])
date, err := time.Parse(gitDateFormat, tkns[1])
require.NoError(p.t, err)
comment.date = date
p.fn = p.parseMessage
}
func (p *commentParser) parseMessage(comment *parsedComment, line string) {
p.t.Helper()
if strings.HasPrefix(line, "Author: ") {
p.comments = append(p.comments, *comment)
comment = &parsedComment{}
p.parseAuthor(comment, line)
return
}
require.True(p.t, line == "" || strings.HasPrefix(line, " "))
comment.message = strings.Join([]string{comment.message, line}, "\n")
}
func normalizeParsedComments(t *testing.T, comments []parsedComment) []parsedComment {
t.Helper()
prefix := 0x1234567
date, err := time.Parse(gitDateFormat, "Fri Aug 19 07:00:00 2022 +1900")
require.NoError(t, err)
out := []parsedComment{}
for i, comment := range comments {
comment.id = fmt.Sprintf("%7x", prefix+i)
comment.date = date.Add(time.Duration(i) * time.Minute)
out = append(out, comment)
}
return out
}
func requireCommentsEqual(t *testing.T, golden string, env *testEnv) {
t.Helper()
const goldenFilePatter = "%s-%d-golden.txt"
comments := parseComments(t, env)
comments = normalizeParsedComments(t, comments)
if *update {
t.Log("Got here")
for i, comment := range comments {
fileName := fmt.Sprintf(goldenFilePatter, golden, i)
require.NoError(t, ioutil.WriteFile(fileName, []byte(comment.message), 0644))
}
}
prefix := 0x1234567
date, err := time.Parse(gitDateFormat, "Fri Aug 19 07:00:00 2022 +1900")
require.NoError(t, err)
for i, comment := range comments {
require.Equal(t, "John Doe", comment.author)
require.Equal(t, fmt.Sprintf("%7x", prefix+i), comment.id)
require.Equal(t, date.Add(time.Duration(i)*time.Minute), comment.date)
fileName := fmt.Sprintf(goldenFilePatter, golden, i)
exp, err := ioutil.ReadFile(fileName)
require.NoError(t, err)
require.Equal(t, string(exp), strings.ReplaceAll(comment.message, "\r", ""))
}
}