/* This file is part of qjson * * Copyright (C) 2009 Till Adam * Copyright (C) 2009 Flavio Castelli * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include "serializer.h" #include #include #include #include #ifdef Q_OS_SOLARIS # ifndef isinf # include # define isinf(x) (!finite((x)) && (x)==(x)) # endif #endif #ifdef _MSC_VER // using MSVC compiler #include #endif using namespace QJson; class Serializer::SerializerPrivate { public: SerializerPrivate() : specialNumbersAllowed(false), indentMode(QJson::IndentNone), doublePrecision(6) { errorMessage.clear(); } QString errorMessage; bool specialNumbersAllowed; IndentMode indentMode; int doublePrecision; QByteArray buildIndent(int spaces); QByteArray serialize( const QVariant &v, bool *ok, int indentLevel = 0); QString sanitizeString( QString str ); QByteArray join( const QList& list, const QByteArray& sep ); }; QByteArray Serializer::SerializerPrivate::join( const QList& list, const QByteArray& sep ) { QByteArray res; Q_FOREACH( const QByteArray& i, list ) { if ( !res.isEmpty() ) res += sep; res += i; } return res; } QByteArray Serializer::SerializerPrivate::serialize( const QVariant &v, bool *ok, int indentLevel) { QByteArray str; if ( ! v.isValid() ) { // invalid or null? str = "null"; } else if (( v.type() == QVariant::List ) || ( v.type() == QVariant::StringList )){ // an array or a stringlist? const QVariantList list = v.toList(); QList values; Q_FOREACH( const QVariant& var, list ) { QByteArray serializedValue; serializedValue = serialize( var, ok, indentLevel+1); if ( !*ok ) { break; } switch(indentMode) { case QJson::IndentFull : case QJson::IndentMedium : case QJson::IndentMinimum : values << serializedValue; break; case QJson::IndentCompact : case QJson::IndentNone : default: values << serializedValue.trimmed(); break; } } if (indentMode == QJson::IndentMedium || indentMode == QJson::IndentFull ) { QByteArray indent = buildIndent(indentLevel); str = indent + "[\n" + join( values, ",\n" ) + "\n" + indent + "]"; } else if (indentMode == QJson::IndentMinimum) { QByteArray indent = buildIndent(indentLevel); str = indent + "[\n" + join( values, ",\n" ) + "\n" + indent + "]"; } else if (indentMode == QJson::IndentCompact) { str = "[" + join( values, "," ) + "]"; } else { str = "[ " + join( values, ", " ) + " ]"; } } else if ( v.type() == QVariant::Map ) { // variant is a map? const QVariantMap vmap = v.toMap(); QMapIterator it( vmap ); if (indentMode == QJson::IndentMinimum) { QByteArray indent = buildIndent(indentLevel); str = indent + "{ "; } else if (indentMode == QJson::IndentMedium || indentMode == QJson::IndentFull) { QByteArray indent = buildIndent(indentLevel); QByteArray nextindent = buildIndent(indentLevel + 1); str = indent + "{\n" + nextindent; } else if (indentMode == QJson::IndentCompact) { str = "{"; } else { str = "{ "; } QList pairs; while ( it.hasNext() ) { it.next(); indentLevel++; QByteArray serializedValue = serialize( it.value(), ok, indentLevel); indentLevel--; if ( !*ok ) { break; } QByteArray key = sanitizeString( it.key() ).toUtf8(); QByteArray value = serializedValue.trimmed(); if (indentMode == QJson::IndentCompact) { pairs << key + ":" + value; } else { pairs << key + " : " + value; } } if (indentMode == QJson::IndentFull) { QByteArray indent = buildIndent(indentLevel + 1); str += join( pairs, ",\n" + indent); } else if (indentMode == QJson::IndentCompact) { str += join( pairs, "," ); } else { str += join( pairs, ", " ); } if (indentMode == QJson::IndentMedium || indentMode == QJson::IndentFull) { QByteArray indent = buildIndent(indentLevel); str += "\n" + indent + "}"; } else if (indentMode == QJson::IndentCompact) { str += "}"; } else { str += " }"; } } else if ( v.type() == QVariant::Hash ) { // variant is a hash? const QVariantHash vhash = v.toHash(); QHashIterator it( vhash ); if (indentMode == QJson::IndentMinimum) { QByteArray indent = buildIndent(indentLevel); str = indent + "{ "; } else if (indentMode == QJson::IndentMedium || indentMode == QJson::IndentFull) { QByteArray indent = buildIndent(indentLevel); QByteArray nextindent = buildIndent(indentLevel + 1); str = indent + "{\n" + nextindent; } else if (indentMode == QJson::IndentCompact) { str = "{"; } else { str = "{ "; } QList pairs; while ( it.hasNext() ) { it.next(); QByteArray serializedValue = serialize( it.value(), ok, indentLevel + 1); if ( !*ok ) { break; } QByteArray key = sanitizeString( it.key() ).toUtf8(); QByteArray value = serializedValue.trimmed(); if (indentMode == QJson::IndentCompact) { pairs << key + ":" + value; } else { pairs << key + " : " + value; } } if (indentMode == QJson::IndentFull) { QByteArray indent = buildIndent(indentLevel + 1); str += join( pairs, ",\n" + indent); } else if (indentMode == QJson::IndentCompact) { str += join( pairs, "," ); } else { str += join( pairs, ", " ); } if (indentMode == QJson::IndentMedium || indentMode == QJson::IndentFull) { QByteArray indent = buildIndent(indentLevel); str += "\n" + indent + "}"; } else if (indentMode == QJson::IndentCompact) { str += "}"; } else { str += " }"; } } else { // Add indent, we may need to remove it later for some layouts switch(indentMode) { case QJson::IndentFull : case QJson::IndentMedium : case QJson::IndentMinimum : str += buildIndent(indentLevel); break; case QJson::IndentCompact : case QJson::IndentNone : default: break; } if (( v.type() == QVariant::String ) || ( v.type() == QVariant::ByteArray )) { // a string or a byte array? str = sanitizeString( v.toString() ).toUtf8(); } else if (( v.type() == QVariant::Double) || ((QMetaType::Type)v.type() == QMetaType::Float)) { // a double or a float? const double value = v.toDouble(); #if defined _WIN32 && !defined(Q_OS_SYMBIAN) const bool special = _isnan(value) || !_finite(value); #elif defined(Q_OS_SYMBIAN) || defined(Q_OS_ANDROID) || defined(Q_OS_BLACKBERRY) || defined(Q_OS_SOLARIS) const bool special = isnan(value) || isinf(value); #else const bool special = std::isnan(value) || std::isinf(value); #endif if (special) { if (specialNumbersAllowed) { #if defined _WIN32 && !defined(Q_OS_SYMBIAN) if (_isnan(value)) { #elif defined(Q_OS_SYMBIAN) || defined(Q_OS_ANDROID) || defined(Q_OS_BLACKBERRY) || defined(Q_OS_SOLARIS) if (isnan(value)) { #else if (std::isnan(value)) { #endif str += "NaN"; } else { if (value<0) { str += '-'; } str += "Infinity"; } } else { errorMessage += QLatin1String("Attempt to write NaN or infinity, which is not supported by json\n"); *ok = false; } } else { str = QByteArray::number( value , 'g', doublePrecision); if( ! str.contains( "." ) && ! str.contains( "e" ) ) { str += ".0"; } } } else if ( v.type() == QVariant::Bool ) { // boolean value? str += ( v.toBool() ? "true" : "false" ); } else if ( v.type() == QVariant::ULongLong ) { // large unsigned number? str += QByteArray::number( v.value() ); } else if ( v.type() == QVariant::UInt ) { // unsigned int number? str += QByteArray::number( v.value() ); } else if ( v.canConvert() ) { // any signed number? str += QByteArray::number( v.value() ); } else if ( v.canConvert() ) { // unsigned short number? str += QByteArray::number( v.value() ); } else if ( v.canConvert() ){ // can value be converted to string? // this will catch QDate, QDateTime, QUrl, ... str += sanitizeString( v.toString() ).toUtf8(); //TODO: catch other values like QImage, QRect, ... } else { *ok = false; errorMessage += QLatin1String("Cannot serialize "); errorMessage += v.toString(); errorMessage += QLatin1String(" because type "); errorMessage += QLatin1String(v.typeName()); errorMessage += QLatin1String(" is not supported by QJson\n"); } } if ( *ok ) { return str; } else return QByteArray(); } QByteArray Serializer::SerializerPrivate::buildIndent(int spaces) { QByteArray indent; if (spaces < 0) { spaces = 0; } for (int i = 0; i < spaces; i++ ) { indent += " "; } return indent; } QString Serializer::SerializerPrivate::sanitizeString( QString str ) { str.replace( QLatin1String( "\\" ), QLatin1String( "\\\\" ) ); // escape unicode chars QString result; const ushort* unicode = str.utf16(); unsigned int i = 0; while ( unicode[ i ] ) { if ( unicode[ i ] < 128 ) { result.append( QChar( unicode[ i ] ) ); } else { QString hexCode = QString::number( unicode[ i ], 16 ).rightJustified( 4, QLatin1Char('0') ); result.append( QLatin1String ("\\u") ).append( hexCode ); } ++i; } str = result; str.replace( QLatin1String( "\"" ), QLatin1String( "\\\"" ) ); str.replace( QLatin1String( "\b" ), QLatin1String( "\\b" ) ); str.replace( QLatin1String( "\f" ), QLatin1String( "\\f" ) ); str.replace( QLatin1String( "\n" ), QLatin1String( "\\n" ) ); str.replace( QLatin1String( "\r" ), QLatin1String( "\\r" ) ); str.replace( QLatin1String( "\t" ), QLatin1String( "\\t" ) ); return QString( QLatin1String( "\"%1\"" ) ).arg( str ); } Serializer::Serializer() : d( new SerializerPrivate ) { } Serializer::~Serializer() { delete d; } void Serializer::serialize( const QVariant& v, QIODevice* io, bool* ok) { Q_ASSERT( io ); *ok = true; if (!io->isOpen()) { if (!io->open(QIODevice::WriteOnly)) { d->errorMessage = QLatin1String("Error opening device"); *ok = false; return; } } if (!io->isWritable()) { d->errorMessage = QLatin1String("Device is not readable"); io->close(); *ok = false; return; } const QByteArray str = serialize( v, ok); if (*ok && (io->write(str) != str.count())) { *ok = false; d->errorMessage = QLatin1String("Something went wrong while writing to IO device"); } } QByteArray Serializer::serialize( const QVariant &v) { bool ok; return serialize(v, &ok); } QByteArray Serializer::serialize( const QVariant &v, bool *ok) { bool _ok = true; d->errorMessage.clear(); if (ok) { *ok = true; } else { ok = &_ok; } return d->serialize(v, ok); } void QJson::Serializer::allowSpecialNumbers(bool allow) { d->specialNumbersAllowed = allow; } bool QJson::Serializer::specialNumbersAllowed() const { return d->specialNumbersAllowed; } void QJson::Serializer::setIndentMode(IndentMode mode) { d->indentMode = mode; } void QJson::Serializer::setDoublePrecision(int precision) { d->doublePrecision = precision; } IndentMode QJson::Serializer::indentMode() const { return d->indentMode; } QString QJson::Serializer::errorMessage() const { return d->errorMessage; }