Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
113 lines
2.6 KiB
113 lines
2.6 KiB
// Copyright 2018 The Gitea Authors. All rights reserved. |
|
// Use of this source code is governed by a MIT-style |
|
// license that can be found in the LICENSE file. |
|
|
|
package markup |
|
|
|
import ( |
|
"bytes" |
|
"encoding/csv" |
|
"html" |
|
"io" |
|
"regexp" |
|
"strings" |
|
|
|
"code.gitea.io/gitea/modules/markup" |
|
"code.gitea.io/gitea/modules/util" |
|
) |
|
|
|
var quoteRegexp = regexp.MustCompile(`["'][\s\S]+?["']`) |
|
|
|
func init() { |
|
markup.RegisterParser(Parser{}) |
|
|
|
} |
|
|
|
// Parser implements markup.Parser for orgmode |
|
type Parser struct { |
|
} |
|
|
|
// Name implements markup.Parser |
|
func (Parser) Name() string { |
|
return "csv" |
|
} |
|
|
|
// Extensions implements markup.Parser |
|
func (Parser) Extensions() []string { |
|
return []string{".csv", ".tsv"} |
|
} |
|
|
|
// Render implements markup.Parser |
|
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 { |
|
fields, err := rd.Read() |
|
if err == io.EOF { |
|
break |
|
} |
|
if err != nil { |
|
continue |
|
} |
|
tmpBlock.WriteString("<tr>") |
|
for _, field := range fields { |
|
tmpBlock.WriteString("<td>") |
|
tmpBlock.WriteString(html.EscapeString(field)) |
|
tmpBlock.WriteString("</td>") |
|
} |
|
tmpBlock.WriteString("</tr>") |
|
} |
|
tmpBlock.WriteString("</table>") |
|
|
|
return tmpBlock.Bytes() |
|
} |
|
|
|
// bestDelimiter scores the input CSV data against delimiters, and returns the best match. |
|
// Reads at most 10k bytes & 10 lines. |
|
func (p Parser) bestDelimiter(data []byte) rune { |
|
maxLines := 10 |
|
maxBytes := util.Min(len(data), 1e4) |
|
text := string(data[:maxBytes]) |
|
text = quoteRegexp.ReplaceAllLiteralString(text, "") |
|
lines := strings.SplitN(text, "\n", maxLines+1) |
|
lines = lines[:util.Min(maxLines, len(lines))] |
|
|
|
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 |
|
} |
|
|
|
// scoreDelimiter uses a count & regularity metric to evaluate a delimiter against lines of CSV |
|
func (Parser) scoreDelimiter(lines []string, delim rune) (score float64) { |
|
countTotal := 0 |
|
countLineMax := 0 |
|
linesNotEqual := 0 |
|
|
|
for _, line := range lines { |
|
if len(line) == 0 { |
|
continue |
|
} |
|
|
|
countLine := strings.Count(line, string(delim)) |
|
countTotal += countLine |
|
if countLine != countLineMax { |
|
if countLineMax != 0 { |
|
linesNotEqual++ |
|
} |
|
countLineMax = util.Max(countLine, countLineMax) |
|
} |
|
} |
|
|
|
return float64(countTotal) * (1 - float64(linesNotEqual)/float64(len(lines))) |
|
}
|
|
|