Skip to content

Commit

Permalink
detect csv delimiter in csv rendering
Browse files Browse the repository at this point in the history
  • Loading branch information
noerw committed Aug 14, 2019
1 parent 14230ca commit 28e5897
Show file tree
Hide file tree
Showing 2 changed files with 61 additions and 5 deletions.
56 changes: 54 additions & 2 deletions modules/markup/csv/csv.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ import (
"encoding/csv"
"html"
"io"
"math"
"strings"

"code.gitea.io/gitea/modules/markup"
)
Expand All @@ -28,12 +30,13 @@ func (Parser) Name() string {

// Extensions implements markup.Parser
func (Parser) Extensions() []string {
return []string{".csv"}
return []string{".csv", ".tsv"}
}

// Render implements markup.Parser
func (Parser) Render(rawBytes []byte, urlPrefix string, metas map[string]string, isWiki bool) []byte {
func (p Parser) Render(rawBytes []byte, urlPrefix string, metas map[string]string, isWiki bool) []byte {
rd := csv.NewReader(bytes.NewReader(rawBytes))
rd.Comma = p.bestDelimiter(rawBytes)
var tmpBlock bytes.Buffer
tmpBlock.WriteString(`<table class="table">`)
for {
Expand All @@ -56,3 +59,52 @@ func (Parser) Render(rawBytes []byte, urlPrefix string, metas map[string]string,

return tmpBlock.Bytes()
}

func (p Parser) bestDelimiter(data []byte) rune {
// Scores the input data against delimiters, and returns the best matching.
// Reads at most 10k bytes & 10 lines.
maxLines := 10
maxBytes := int(math.Min(float64(len(data)), 1e4))
text := string(data[:maxBytes])
lines := strings.SplitN(text, "\n", maxLines+1)[:maxLines]

delimiters := []rune{',', ';', '\t', '|'}
bestDelim := delimiters[0]
bestScore := 0.0

for _, delim := range delimiters {
score := p.scoreDelimiter(lines, delim)
if score > bestScore {
bestScore = score
bestDelim = delim
}
}

return bestDelim
}

func (Parser) scoreDelimiter(lines []string, delim rune) (score float64) {
// Scores a delimiter against input csv data with a count and regularity metric.

countTotal := 0.0
countLineMax := 0.0
linesNotEqual := 0.0

for _, line := range lines {
if len(line) == 0 {
continue
}

countLine := float64(strings.Count(line, string(delim)))
countTotal += countLine

if countLine != countLineMax {
if countLineMax != 0 {
linesNotEqual += 1
}
countLineMax = math.Max(countLine, countLineMax)
}
}

return countTotal * (1 - linesNotEqual/float64(len(lines)))
}
10 changes: 7 additions & 3 deletions modules/markup/csv/csv_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,13 @@ import (
func TestRenderCSV(t *testing.T) {
var parser Parser
var kases = map[string]string{
"a": "<table class=\"table\"><tr><td>a</td><tr></table>",
"1,2": "<table class=\"table\"><tr><td>1</td><td>2</td><tr></table>",
"<br/>": "<table class=\"table\"><tr><td>&lt;br/&gt;</td><tr></table>",
"a": "<table class=\"table\"><tr><td>a</td><tr></table>",
"1,2": "<table class=\"table\"><tr><td>1</td><td>2</td><tr></table>",
"1;2": "<table class=\"table\"><tr><td>1</td><td>2</td><tr></table>",
"1\t2": "<table class=\"table\"><tr><td>1</td><td>2</td><tr></table>",
"1|2": "<table class=\"table\"><tr><td>1</td><td>2</td><tr></table>",
"1,2,3;4,5,6;7,8,9\na;b;c": "<table class=\"table\"><tr><td>1,2,3</td><td>4,5,6</td><td>7,8,9</td><tr><tr><td>a</td><td>b</td<td>c</td><tr></table>",
"<br/>": "<table class=\"table\"><tr><td>&lt;br/&gt;</td><tr></table>",
}

for k, v := range kases {
Expand Down

0 comments on commit 28e5897

Please sign in to comment.