FLOATEXPSIGN,
STRING,
STRING_ESCAPE,
+ STRING_BASE64,
ACCEPT,
ZERO,
DECIMAL,
FLOAT,
FLOATEXP,
STRING_END,
+ STRING_BASE64_PAD,
IDENTIFIER
};
Token::SPECIAL,
Token::SPECIAL,
Token::SPECIAL,
+ Token::SPECIAL,
Token::INTEGER,
Token::INTEGER,
Token::INTEGER,
Token::FLOAT,
Token::FLOAT,
Token::STRING,
+ Token::STRING,
Token::IDENTIFIER
};
state = FLOAT;
else if(c=='"')
state = STRING;
+ else if(c=='=')
+ state = STRING_BASE64;
else if(c=='{' || c=='}' || c==';')
return Token(Token::SPECIAL, string(1, c));
else if(isdigit(c))
state = STRING;
break;
+ case STRING_BASE64:
+ if(c=='=')
+ state = STRING_BASE64_PAD;
+ else if(!isalnum(c) && c!='+' && c!='/')
+ throw parse_error(buf);
+ break;
+
+ case STRING_BASE64_PAD:
+ if(c!='=')
+ throw parse_error(buf);
+ break;
+
case IDENTIFIER:
if(!isalpha(c) && !isdigit(c) && c!='_' && c!='-' && c!='/')
throw parse_error(buf);
return Token(Token::IDENTIFIER, buf.substr(1));
else if(state==STRING_END)
return Token(Token::STRING, c_unescape(buf.substr(1, buf.size()-2)));
+ else if(state==STRING_BASE64 || state==STRING_BASE64_PAD)
+ return Token(Token::STRING, base64_decode(buf));
else
return Token(token_type[state], buf);
}
return (c>='0' && c<='7');
}
+string TextParser::base64_decode(const string &data)
+{
+ string bin;
+ unsigned accum = 0;
+ unsigned a_bits = 0;
+ for(string::const_iterator i=data.begin(); i!=data.end(); ++i)
+ {
+ unsigned d;
+ if(*i>='A' && *i<='Z')
+ d = *i-'A';
+ else if(*i>='a' && *i<='z')
+ d = 26+(*i-'a');
+ else if(*i>='0' && *i<='9')
+ d = 52+(*i-'0');
+ else if(*i=='+')
+ d = 62;
+ else if(*i=='/')
+ d = 63;
+ else if(*i=='=')
+ continue;
+ else
+ throw invalid_argument("TextParser::base64_decode");
+
+ accum = (accum<<6)|d;
+ a_bits += 6;
+
+ if(a_bits>=8)
+ {
+ bin += (accum>>(a_bits-8))&0xFF;
+ a_bits -= 8;
+ }
+ }
+
+ return bin;
+}
+
} // namespace DataFile
} // namespace Msp