2012-03-27 23:13:14 +00:00
|
|
|
// Copyright 2011 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package csv
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"io"
|
|
|
|
"strings"
|
|
|
|
"unicode"
|
|
|
|
"unicode/utf8"
|
|
|
|
)
|
|
|
|
|
|
|
|
// A Writer writes records to a CSV encoded file.
|
|
|
|
//
|
|
|
|
// As returned by NewWriter, a Writer writes records terminated by a
|
2017-10-07 00:16:47 +00:00
|
|
|
// newline and uses ',' as the field delimiter. The exported fields can be
|
2012-03-27 23:13:14 +00:00
|
|
|
// changed to customize the details before the first call to Write or WriteAll.
|
|
|
|
//
|
|
|
|
// Comma is the field delimiter.
|
|
|
|
//
|
|
|
|
// If UseCRLF is true, the Writer ends each record with \r\n instead of \n.
|
|
|
|
type Writer struct {
|
2014-09-21 17:33:12 +00:00
|
|
|
Comma rune // Field delimiter (set to ',' by NewWriter)
|
2012-03-27 23:13:14 +00:00
|
|
|
UseCRLF bool // True to use \r\n as the line terminator
|
|
|
|
w *bufio.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewWriter returns a new Writer that writes to w.
|
|
|
|
func NewWriter(w io.Writer) *Writer {
|
|
|
|
return &Writer{
|
|
|
|
Comma: ',',
|
|
|
|
w: bufio.NewWriter(w),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Writer writes a single CSV record to w along with any necessary quoting.
|
|
|
|
// A record is a slice of strings with each string being one field.
|
2017-10-07 00:16:47 +00:00
|
|
|
func (w *Writer) Write(record []string) error {
|
2012-03-27 23:13:14 +00:00
|
|
|
for n, field := range record {
|
|
|
|
if n > 0 {
|
2017-10-07 00:16:47 +00:00
|
|
|
if _, err := w.w.WriteRune(w.Comma); err != nil {
|
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we don't have to have a quoted field then just
|
|
|
|
// write out the field and continue to the next field.
|
|
|
|
if !w.fieldNeedsQuotes(field) {
|
2017-10-07 00:16:47 +00:00
|
|
|
if _, err := w.w.WriteString(field); err != nil {
|
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
2017-10-07 00:16:47 +00:00
|
|
|
if err := w.w.WriteByte('"'); err != nil {
|
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, r1 := range field {
|
2017-10-07 00:16:47 +00:00
|
|
|
var err error
|
2012-03-27 23:13:14 +00:00
|
|
|
switch r1 {
|
|
|
|
case '"':
|
|
|
|
_, err = w.w.WriteString(`""`)
|
|
|
|
case '\r':
|
|
|
|
if !w.UseCRLF {
|
|
|
|
err = w.w.WriteByte('\r')
|
|
|
|
}
|
|
|
|
case '\n':
|
|
|
|
if w.UseCRLF {
|
|
|
|
_, err = w.w.WriteString("\r\n")
|
|
|
|
} else {
|
|
|
|
err = w.w.WriteByte('\n')
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
_, err = w.w.WriteRune(r1)
|
|
|
|
}
|
|
|
|
if err != nil {
|
2017-10-07 00:16:47 +00:00
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-07 00:16:47 +00:00
|
|
|
if err := w.w.WriteByte('"'); err != nil {
|
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
}
|
2017-10-07 00:16:47 +00:00
|
|
|
var err error
|
2012-03-27 23:13:14 +00:00
|
|
|
if w.UseCRLF {
|
|
|
|
_, err = w.w.WriteString("\r\n")
|
|
|
|
} else {
|
|
|
|
err = w.w.WriteByte('\n')
|
|
|
|
}
|
2017-10-07 00:16:47 +00:00
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Flush writes any buffered data to the underlying io.Writer.
|
2014-09-21 17:33:12 +00:00
|
|
|
// To check if an error occurred during the Flush, call Error.
|
2012-03-27 23:13:14 +00:00
|
|
|
func (w *Writer) Flush() {
|
|
|
|
w.w.Flush()
|
|
|
|
}
|
|
|
|
|
2014-09-21 17:33:12 +00:00
|
|
|
// Error reports any error that has occurred during a previous Write or Flush.
|
|
|
|
func (w *Writer) Error() error {
|
|
|
|
_, err := w.w.Write(nil)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2012-03-27 23:13:14 +00:00
|
|
|
// WriteAll writes multiple CSV records to w using Write and then calls Flush.
|
2017-10-07 00:16:47 +00:00
|
|
|
func (w *Writer) WriteAll(records [][]string) error {
|
2012-03-27 23:13:14 +00:00
|
|
|
for _, record := range records {
|
2017-10-07 00:16:47 +00:00
|
|
|
err := w.Write(record)
|
2012-03-27 23:13:14 +00:00
|
|
|
if err != nil {
|
2014-09-21 17:33:12 +00:00
|
|
|
return err
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
}
|
2014-09-21 17:33:12 +00:00
|
|
|
return w.w.Flush()
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
|
2017-04-10 11:32:00 +00:00
|
|
|
// fieldNeedsQuotes reports whether our field must be enclosed in quotes.
|
2015-08-28 15:33:40 +00:00
|
|
|
// Fields with a Comma, fields with a quote or newline, and
|
2012-03-27 23:13:14 +00:00
|
|
|
// fields which start with a space must be enclosed in quotes.
|
2015-08-28 15:33:40 +00:00
|
|
|
// We used to quote empty strings, but we do not anymore (as of Go 1.4).
|
|
|
|
// The two representations should be equivalent, but Postgres distinguishes
|
|
|
|
// quoted vs non-quoted empty string during database imports, and it has
|
|
|
|
// an option to force the quoted behavior for non-quoted CSV but it has
|
|
|
|
// no option to force the non-quoted behavior for quoted CSV, making
|
|
|
|
// CSV with quoted empty strings strictly less useful.
|
|
|
|
// Not quoting the empty string also makes this package match the behavior
|
|
|
|
// of Microsoft Excel and Google Drive.
|
2017-04-10 11:32:00 +00:00
|
|
|
// For Postgres, quote the data terminating string `\.`.
|
2012-03-27 23:13:14 +00:00
|
|
|
func (w *Writer) fieldNeedsQuotes(field string) bool {
|
2015-08-28 15:33:40 +00:00
|
|
|
if field == "" {
|
|
|
|
return false
|
|
|
|
}
|
2017-10-07 00:16:47 +00:00
|
|
|
if field == `\.` || strings.ContainsRune(field, w.Comma) || strings.ContainsAny(field, "\"\r\n") {
|
2012-03-27 23:13:14 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
r1, _ := utf8.DecodeRuneInString(field)
|
|
|
|
return unicode.IsSpace(r1)
|
|
|
|
}
|