--- parser3/src/classes/table.C 2016/09/06 22:38:44 1.320 +++ parser3/src/classes/table.C 2016/09/08 17:01:49 1.323 @@ -22,7 +22,7 @@ #define USE_STRINGSTREAM #endif -volatile const char * IDENT_TABLE_C="$Id: table.C,v 1.320 2016/09/06 22:38:44 moko Exp $"; +volatile const char * IDENT_TABLE_C="$Id: table.C,v 1.323 2016/09/08 17:01:49 moko Exp $"; // class @@ -111,7 +111,7 @@ struct TableControlChars { if(Value* vseparator=options.get(PA_COLUMN_SEPARATOR_NAME)) { sseparator=&vseparator->as_string(); if(sseparator->length()!=1) - throw Exception(PARSER_RUNTIME, sseparator, "separator must be one character long"); + throw Exception(PARSER_RUNTIME, sseparator, "separator must be one byte character"); separator=sseparator->first_char(); separators[0]=separator; result++; @@ -122,7 +122,7 @@ struct TableControlChars { encloser=0; } else { if(sencloser->length()!=1) - throw Exception(PARSER_RUNTIME, sencloser, "encloser must be one character long"); + throw Exception(PARSER_RUNTIME, sencloser, "encloser must be one byte character"); encloser=sencloser->first_char(); } result++; @@ -132,6 +132,117 @@ struct TableControlChars { }; +struct lsplit_sresult { + String* piece; + char delim; + + lsplit_sresult() : piece(0), delim(0){} + + operator bool() { return piece!=0; } + + void append(String *str){ + if(piece) + *piece << *str; + else + piece = str; + } +}; + +class StringSplitHelper : public String { +public: + char* base; + + StringSplitHelper(String astring) : String(astring), base(cstrm()) {} + + bool check_lang(const char *pos){ + return langs.check_lang(L_AS_IS, pos-base, 1); + } + + String *extract(char *pos){ + String *result=new String; + // first: their langs + result->langs.append(result->body, langs, pos-base, strlen(pos)); + // next: letters themselves + result->body=Body(pos); + return result; + } +}; + +inline lsplit_sresult lsplit(char* *string_ref, const char* delims, StringSplitHelper& helper) { + lsplit_sresult result; + if(char *pos=*string_ref) { + while(char* v=strpbrk(pos, delims)) { + if(helper.check_lang(v)){ + result.delim=*v; + *v=0; + result.piece=helper.extract(*string_ref); + *string_ref=v+1; + return result; + } + pos=v+1; + } + result.piece=helper.extract(*string_ref); + *string_ref=0; + } + return result; +} + +static lsplit_sresult lsplit(char** string_ref, const char* delims, char encloser, StringSplitHelper& helper) { + lsplit_sresult result; + + if(char *pos=*string_ref) { + if(encloser && *pos==encloser && helper.check_lang(pos)) { + *string_ref=++pos; + + // we are enclosed, searching for second encloser + while(1) { + if(char* v=strchr(pos, encloser)){ + if(helper.check_lang(v)){ + *(v++)=0; + result.append(helper.extract(*string_ref)); + if(*v==encloser && helper.check_lang(v)){ // double-encloser stands for encloser + *string_ref=v+1; + } else { + *string_ref=pos=v; + break; + } + } + pos=v+1; + }{ + result.append(helper.extract(*string_ref)); + *string_ref=0; + return result; + } + } + + // we are no longer enclosed, searching for delimiter + while(char* v=strpbrk(pos, delims)) { + if(helper.check_lang(v)){ + result.delim=*v; + if(v>*string_ref){ + *v=0; + result.append(helper.extract(*string_ref)); + } + *string_ref=v+1; + return result; + } + pos=v+1; + } + result.append(helper.extract(*string_ref)); + *string_ref=0; + } else + return lsplit(string_ref, delims, helper); + } + return result; +} + +static void skip_clean_empty_lines(char** data_ref, StringSplitHelper& helper) { + if(*data_ref) { + while(**data_ref == '\n' && helper.check_lang(*data_ref)) + (*data_ref)++; + } +} + static void _create(Request& r, MethodParams& params) { // clone/copy part? if(Table *source=params[0].get_table()) { @@ -159,61 +270,50 @@ static void _create(Request& r, MethodPa TableControlChars control_chars; size_t options_param_index=data_param_index+1; - if( - options_param_indexis_empty()) - *columns += new String(); - else - head[0]->split(*columns, col_pos_after, *control_chars.sseparator, String::L_AS_IS); + columns=new ArrayString; + while( lsplit_sresult sr=lsplit(&data, control_chars.separators, control_chars.encloser, sdata) ) { + *columns+=sr.piece; + if(sr.delim=='\n') + break; } } - + Table& table=*new Table(columns); - // parse cells + int columns_count=columns ? columns->count(): 0; - ArrayString rows; - data.split(rows, raw_pos_after, "\n", String::L_AS_IS); - Array_iterator i(rows); - while(i.has_next()) { - Table::element_type row(new ArrayString); - const String& string=*i.next(); - // remove comment lines - if(string.is_empty()) - continue; - - size_t col_pos_after=0; - string.split(*row, col_pos_after, *control_chars.sseparator, String::L_AS_IS); - table+=row; + // parse cells + Table::element_type row(new ArrayString(columns_count)); + skip_clean_empty_lines(&data, sdata); + while( lsplit_sresult sr=lsplit(&data, control_chars.separators, control_chars.encloser, sdata) ) { + if(sr.piece->is_empty() && !sr.delim && !row->count()) // append last empty column [if without \n] + break; + *row+=sr.piece; + if(sr.delim=='\n') { + table+=row; + row=new ArrayString(columns_count); + skip_clean_empty_lines(&data, sdata); + } } - + // last line [if without \n] + if(row->count()) + table+=row; + // replace any previous table value GET_SELF(r, VTable).set_table(table); } @@ -222,44 +322,36 @@ struct lsplit_result { char* piece; char delim; + lsplit_result(char *apiece=0) : piece(apiece), delim(0){} operator bool() { return piece!=0; } }; -inline lsplit_result lsplit(char* string, const char* delims) { - lsplit_result result; - if(string) { - if(char* v=strpbrk(string, delims)) { +inline lsplit_result lsplit(char* *string_ref, const char* delims) { + lsplit_result result(*string_ref); + if(result.piece) { + if(char* v=strpbrk(result.piece, delims)) { result.delim=*v; *v=0; - result.piece=v+1; + *string_ref=v+1; return result; } + *string_ref=0; } - result.piece=0; - result.delim=0; - return result; -} - -inline lsplit_result lsplit(char* *string_ref, const char* delims) { - lsplit_result result; - result.piece=*string_ref; - lsplit_result next=lsplit(*string_ref, delims); - result.delim=next.delim; - *string_ref=next.piece; return result; } static lsplit_result lsplit(char** string_ref, const char* delims, char encloser) { - lsplit_result result; + lsplit_result result(*string_ref); - if(char* string=*string_ref) { - if(encloser && *string==encloser) { - string++; + if(result.piece) { + if(encloser && *result.piece==encloser) { + result.piece++; + char c; char *read; char *write; - write=read=string; - char c; + write=read=result.piece; + // we are enclosed, searching for second encloser while(c=*read++) { if(c==encloser) { @@ -270,6 +362,7 @@ static lsplit_result lsplit(char** strin } *write++=c; } + // we are no longer enclosed, searching for delimiter, skipping extra enclosers while(c=*read++) { if(c==delims[0] || c==delims[1]) { @@ -278,42 +371,33 @@ static lsplit_result lsplit(char** strin } else if(c!=encloser) *write++=c; } + *write=0; // terminate - *string_ref=c? read: 0; - result.piece=string; + *string_ref=c ? read : 0; return result; } else return lsplit(string_ref, delims); } - result.piece=0; return result; } static void skip_empty_and_comment_lines( char** data_ref ) { - if(char *data=*data_ref) { - while( char c=*data ) { - if( c== '\n' || c == '#' ) { - /*nowhere=*/getrow(&data); // remove empty&comment lines - if(!(*data_ref=data)) - break; - continue; - } - break; + while(*data_ref) { + if(**data_ref == '\n'){ + (*data_ref)++; + } else { + if(**data_ref == '#' ) + /*nowhere=*/getrow(data_ref); + else + break; } } } static void skip_empty_lines( char** data_ref ) { - if(char *data=*data_ref) { - while( char c=*data ) { - if( c== '\n' ) { - /*nowhere=*/getrow(&data); // remove empty lines - if(!(*data_ref=data)) - break; - continue; - } - break; - } + if(*data_ref) { + while(**data_ref == '\n') + (*data_ref)++; } } @@ -337,20 +421,16 @@ static void _load(Request& r, MethodPara } // loading text - char *data=file_load_text(r, - r.absolute(params.as_string(filename_param_index, FILE_NAME_MUST_BE_STRING)), - true, - options - ); + char *data=file_load_text(r, r.absolute(params.as_string(filename_param_index, FILE_NAME_MUST_BE_STRING)), true, options); Skip_lines_action skip_lines_action = (control_chars.separator=='#' || control_chars.encloser=='#') ? skip_empty_lines : skip_empty_and_comment_lines; // parse columns Table::columns_type columns; if(nameless) { - columns=Table::columns_type(0); // nameless + columns=0; // nameless } else { - columns=Table::columns_type(new ArrayString); + columns=new ArrayString; skip_lines_action(&data); while( lsplit_result sr=lsplit(&data, control_chars.separators, control_chars.encloser) ) { @@ -361,7 +441,7 @@ static void _load(Request& r, MethodPara } Table& table=*new Table(columns); - int columns_count=columns? columns->count(): 0; + int columns_count=columns ? columns->count(): 0; // parse cells Table::element_type row(new ArrayString(columns_count));