package csv import "core:io" import "core:strings" import "core:unicode/utf8" // Writer is a data structure used for writing records using a CSV-encoding. Writer :: struct { // Field delimiter (set to ',' with writer_init) comma: rune, // if set to true, \r\n will be used as the line terminator use_crlf: bool, w: io.Writer, } // writer_init initializes a Writer that writes to w writer_init :: proc(writer: ^Writer, w: io.Writer) { writer.comma = ',' writer.w = w } // write writes a single CSV records to w with any of the necessarily quoting. // A record is a slice of strings, where each string is a single field. // // If the underlying io.Writer requires flushing, make sure to call io.flush write :: proc(w: ^Writer, record: []string) -> io.Error { CHAR_SET :: "\n\r\"" field_needs_quoting :: proc(w: ^Writer, field: string) -> bool { switch { case field == "": // No need to quote empty strings return false case field == `\.`: // Postgres is weird return true case w.comma < utf8.RUNE_SELF: // ASCII optimization for i in 0..= 0 { return true } if strings.contains_any(field, CHAR_SET) { return true } } // Leading spaces need quoting r, _ := utf8.decode_rune_in_string(field) return strings.is_space(r) } if !is_valid_delim(w.comma) { return .No_Progress // TODO(bill): Is this a good error? } for _, field_idx in record { // NOTE(bill): declared like this so that the field can be modified later if necessary field := record[field_idx] if field_idx > 0 { io.write_rune(w.w, w.comma) or_return } if !field_needs_quoting(w, field) { io.write_string(w.w, field) or_return continue } io.write_byte(w.w, '"') or_return for len(field) > 0 { i := strings.index_any(field, CHAR_SET) if i < 0 { i = len(field) } io.write_string(w.w, field[:i]) or_return field = field[i:] if len(field) > 0 { switch field[0] { case '\r': if !w.use_crlf { io.write_byte(w.w, '\r') or_return } case '\n': if w.use_crlf { io.write_string(w.w, "\r\n") or_return } else { io.write_byte(w.w, '\n') or_return } case '"': io.write_string(w.w, `""`) or_return } field = field[1:] } } io.write_byte(w.w, '"') or_return } if w.use_crlf { _, err := io.write_string(w.w, "\r\n") return err } return io.write_byte(w.w, '\n') } // write_all writes multiple CSV records to w using write, and then flushes (if necessary). write_all :: proc(w: ^Writer, records: [][]string) -> io.Error { for record in records { write(w, record) or_return } return writer_flush(w) } // writer_flush flushes the underlying io.Writer. // If the underlying io.Writer does not support flush, nil is returned. writer_flush :: proc(w: ^Writer) -> io.Error { return io.flush(auto_cast w.w) }