/* * MinIO Cloud Storage, (C) 2019 MinIO, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package csv import ( "encoding/xml" "errors" "fmt" "io" "strings" "unicode/utf8" ) const ( none = "none" use = "use" ignore = "ignore" defaultRecordDelimiter = "\n" defaultFieldDelimiter = "," defaultQuoteCharacter = `"` defaultQuoteEscapeCharacter = `"` defaultCommentCharacter = "#" always = "always" asneeded = "asneeded" ) // ReaderArgs - represents elements inside in request XML. type ReaderArgs struct { FileHeaderInfo string `xml:"FileHeaderInfo"` RecordDelimiter string `xml:"RecordDelimiter"` FieldDelimiter string `xml:"FieldDelimiter"` QuoteCharacter string `xml:"QuoteCharacter"` QuoteEscapeCharacter string `xml:"QuoteEscapeCharacter"` CommentCharacter string `xml:"Comments"` AllowQuotedRecordDelimiter bool `xml:"AllowQuotedRecordDelimiter"` unmarshaled bool } // IsEmpty - returns whether reader args is empty or not. func (args *ReaderArgs) IsEmpty() bool { return !args.unmarshaled } // UnmarshalXML - decodes XML data. func (args *ReaderArgs) UnmarshalXML(d *xml.Decoder, start xml.StartElement) (err error) { args.FileHeaderInfo = none args.RecordDelimiter = defaultRecordDelimiter args.FieldDelimiter = defaultFieldDelimiter args.QuoteCharacter = defaultQuoteCharacter args.QuoteEscapeCharacter = defaultQuoteEscapeCharacter args.CommentCharacter = defaultCommentCharacter args.AllowQuotedRecordDelimiter = false for { // Read tokens from the XML document in a stream. t, err := d.Token() if err != nil { if err == io.EOF { break } return err } switch se := t.(type) { case xml.StartElement: tagName := se.Name.Local switch tagName { case "AllowQuotedRecordDelimiter": var b bool if err = d.DecodeElement(&b, &se); err != nil { return err } args.AllowQuotedRecordDelimiter = b default: var s string if err = d.DecodeElement(&s, &se); err != nil { return err } switch tagName { case "FileHeaderInfo": args.FileHeaderInfo = strings.ToLower(s) case "RecordDelimiter": args.RecordDelimiter = s case "FieldDelimiter": args.FieldDelimiter = s case "QuoteCharacter": if utf8.RuneCountInString(s) > 1 { return fmt.Errorf("unsupported QuoteCharacter '%v'", s) } args.QuoteCharacter = s // Not supported yet case "QuoteEscapeCharacter": case "Comments": args.CommentCharacter = s default: return errors.New("unrecognized option") } } } } args.QuoteEscapeCharacter = args.QuoteCharacter args.unmarshaled = true return nil } // WriterArgs - represents elements inside in request XML. type WriterArgs struct { QuoteFields string `xml:"QuoteFields"` RecordDelimiter string `xml:"RecordDelimiter"` FieldDelimiter string `xml:"FieldDelimiter"` QuoteCharacter string `xml:"QuoteCharacter"` QuoteEscapeCharacter string `xml:"QuoteEscapeCharacter"` unmarshaled bool } // IsEmpty - returns whether writer args is empty or not. func (args *WriterArgs) IsEmpty() bool { return !args.unmarshaled } // UnmarshalXML - decodes XML data. func (args *WriterArgs) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error { args.QuoteFields = asneeded args.RecordDelimiter = defaultRecordDelimiter args.FieldDelimiter = defaultFieldDelimiter args.QuoteCharacter = defaultQuoteCharacter args.QuoteEscapeCharacter = defaultQuoteCharacter for { // Read tokens from the XML document in a stream. t, err := d.Token() if err != nil { if err == io.EOF { break } return err } switch se := t.(type) { case xml.StartElement: var s string if err = d.DecodeElement(&s, &se); err != nil { return err } switch se.Name.Local { case "QuoteFields": args.QuoteFields = strings.ToLower(s) case "RecordDelimiter": args.RecordDelimiter = s case "FieldDelimiter": args.FieldDelimiter = s case "QuoteCharacter": switch utf8.RuneCountInString(s) { case 0: args.QuoteCharacter = "\x00" case 1: args.QuoteCharacter = s default: return fmt.Errorf("unsupported QuoteCharacter '%v'", s) } // Not supported yet case "QuoteEscapeCharacter": default: return errors.New("unrecognized option") } } } args.QuoteEscapeCharacter = args.QuoteCharacter args.unmarshaled = true return nil }