set output stream in write(), not in builder

This commit is contained in:
Christopher Dunn 2015-02-09 18:44:53 -06:00
parent b56381a636
commit c41609b9f9
2 changed files with 51 additions and 68 deletions

View File

@ -31,15 +31,15 @@ Usage:
using namespace Json; using namespace Json;
void writeToStdout(StreamWriter::Factory const& factory, Value const& value) { void writeToStdout(StreamWriter::Factory const& factory, Value const& value) {
std::unique_ptr<StreamWriter> const writer( std::unique_ptr<StreamWriter> const writer(
factory.newStreamWriter(&std::cout)); factory.newStreamWriter());
writer->write(value); writer->write(value, &std::cout);
std::cout << std::endl; // add lf and flush std::cout << std::endl; // add lf and flush
} }
\endcode \endcode
*/ */
class JSON_API StreamWriter { class JSON_API StreamWriter {
protected: protected:
std::ostream& sout_; // not owned; will not delete std::ostream* sout_; // not owned; will not delete
public: public:
/// Scoped enums are not available until C++11. /// Scoped enums are not available until C++11.
struct CommentStyle { struct CommentStyle {
@ -51,13 +51,15 @@ public:
}; };
}; };
/// Keep a reference, but do not take ownership of `sout`. StreamWriter();
StreamWriter(std::ostream* sout);
virtual ~StreamWriter(); virtual ~StreamWriter();
/// Write Value into document as configured in sub-class. /** Write Value into document as configured in sub-class.
/// \return zero on success Do not take ownership of sout, but maintain a reference during function.
/// \throw std::exception possibly, depending on configuration \pre sout != NULL
virtual int write(Value const& root) = 0; \return zero on success
\throw std::exception possibly, depending on configuration
*/
virtual int write(Value const& root, std::ostream* sout) = 0;
/** \brief A simple abstract factory. /** \brief A simple abstract factory.
*/ */
@ -65,10 +67,9 @@ public:
public: public:
virtual ~Factory(); virtual ~Factory();
/** \brief Allocate a CharReader via operator new(). /** \brief Allocate a CharReader via operator new().
* Do not take ownership of sout, but maintain a reference.
* \throw std::exception if something goes wrong (e.g. invalid settings) * \throw std::exception if something goes wrong (e.g. invalid settings)
*/ */
virtual StreamWriter* newStreamWriter(std::ostream* sout) const = 0; virtual StreamWriter* newStreamWriter() const = 0;
}; // Factory }; // Factory
}; // StreamWriter }; // StreamWriter
@ -88,8 +89,8 @@ Usage:
builder.settings_["commentStyle"] = "None"; builder.settings_["commentStyle"] = "None";
builder.settings_["indentation"] = " "; // or whatever you like builder.settings_["indentation"] = " "; // or whatever you like
std::unique_ptr<Json::StreamWriter> writer( std::unique_ptr<Json::StreamWriter> writer(
builder.newStreamWriter(&std::cout)); builder.newStreamWriter());
writer->write(value); writer->write(value, &std::cout);
std::cout << std::endl; // add lf and flush std::cout << std::endl; // add lf and flush
\endcode \endcode
*/ */
@ -112,10 +113,10 @@ public:
StreamWriterBuilder(); StreamWriterBuilder();
virtual ~StreamWriterBuilder(); virtual ~StreamWriterBuilder();
/** Do not take ownership of sout, but maintain a reference. /**
* \throw std::exception if something goes wrong (e.g. invalid settings) * \throw std::exception if something goes wrong (e.g. invalid settings)
*/ */
virtual StreamWriter* newStreamWriter(std::ostream* sout) const; virtual StreamWriter* newStreamWriter() const;
/** \return true if 'settings' are legal and consistent; /** \return true if 'settings' are legal and consistent;
* otherwise, indicate bad settings via 'invalid'. * otherwise, indicate bad settings via 'invalid'.
@ -137,8 +138,8 @@ public:
* \code * \code
* OldCompressingStreamWriterBuilder b; * OldCompressingStreamWriterBuilder b;
* b.dropNullPlaceHolders_ = true; // etc. * b.dropNullPlaceHolders_ = true; // etc.
* StreamWriter* w = b.newStreamWriter(&std::cout); * StreamWriter* w = b.newStreamWriter();
* w->write(value); * w->write(value, &std::cout);
* delete w; * delete w;
* \endcode * \endcode
* *
@ -174,7 +175,7 @@ public:
, omitEndingLineFeed_(false) , omitEndingLineFeed_(false)
, enableYAMLCompatibility_(false) , enableYAMLCompatibility_(false)
{} {}
virtual StreamWriter* newStreamWriter(std::ostream*) const; virtual StreamWriter* newStreamWriter() const;
}; };
/** \brief Abstract class for writers. /** \brief Abstract class for writers.

View File

@ -679,13 +679,12 @@ bool StyledStreamWriter::hasCommentForValue(const Value& value) {
struct BuiltStyledStreamWriter : public StreamWriter struct BuiltStyledStreamWriter : public StreamWriter
{ {
BuiltStyledStreamWriter( BuiltStyledStreamWriter(
std::ostream* sout,
std::string const& indentation, std::string const& indentation,
StreamWriter::CommentStyle::Enum cs, StreamWriter::CommentStyle::Enum cs,
std::string const& colonSymbol, std::string const& colonSymbol,
std::string const& nullSymbol, std::string const& nullSymbol,
std::string const& endingLineFeedSymbol); std::string const& endingLineFeedSymbol);
virtual int write(Value const& root); virtual int write(Value const& root, std::ostream* sout);
private: private:
void writeValue(Value const& value); void writeValue(Value const& value);
void writeArrayValue(Value const& value); void writeArrayValue(Value const& value);
@ -713,14 +712,12 @@ private:
bool indented_ : 1; bool indented_ : 1;
}; };
BuiltStyledStreamWriter::BuiltStyledStreamWriter( BuiltStyledStreamWriter::BuiltStyledStreamWriter(
std::ostream* sout,
std::string const& indentation, std::string const& indentation,
StreamWriter::CommentStyle::Enum cs, StreamWriter::CommentStyle::Enum cs,
std::string const& colonSymbol, std::string const& colonSymbol,
std::string const& nullSymbol, std::string const& nullSymbol,
std::string const& endingLineFeedSymbol) std::string const& endingLineFeedSymbol)
: StreamWriter(sout) : rightMargin_(74)
, rightMargin_(74)
, indentation_(indentation) , indentation_(indentation)
, cs_(cs) , cs_(cs)
, colonSymbol_(colonSymbol) , colonSymbol_(colonSymbol)
@ -730,8 +727,9 @@ BuiltStyledStreamWriter::BuiltStyledStreamWriter(
, indented_(false) , indented_(false)
{ {
} }
int BuiltStyledStreamWriter::write(Value const& root) int BuiltStyledStreamWriter::write(Value const& root, std::ostream* sout)
{ {
sout_ = sout;
addChildValues_ = false; addChildValues_ = false;
indented_ = true; indented_ = true;
indentString_ = ""; indentString_ = "";
@ -740,7 +738,8 @@ int BuiltStyledStreamWriter::write(Value const& root)
indented_ = true; indented_ = true;
writeValue(root); writeValue(root);
writeCommentAfterValueOnSameLine(root); writeCommentAfterValueOnSameLine(root);
sout_ << endingLineFeedSymbol_; *sout_ << endingLineFeedSymbol_;
sout_ = NULL;
return 0; return 0;
} }
void BuiltStyledStreamWriter::writeValue(Value const& value) { void BuiltStyledStreamWriter::writeValue(Value const& value) {
@ -779,13 +778,13 @@ void BuiltStyledStreamWriter::writeValue(Value const& value) {
Value const& childValue = value[name]; Value const& childValue = value[name];
writeCommentBeforeValue(childValue); writeCommentBeforeValue(childValue);
writeWithIndent(valueToQuotedString(name.c_str())); writeWithIndent(valueToQuotedString(name.c_str()));
sout_ << colonSymbol_; *sout_ << colonSymbol_;
writeValue(childValue); writeValue(childValue);
if (++it == members.end()) { if (++it == members.end()) {
writeCommentAfterValueOnSameLine(childValue); writeCommentAfterValueOnSameLine(childValue);
break; break;
} }
sout_ << ","; *sout_ << ",";
writeCommentAfterValueOnSameLine(childValue); writeCommentAfterValueOnSameLine(childValue);
} }
unindent(); unindent();
@ -821,7 +820,7 @@ void BuiltStyledStreamWriter::writeArrayValue(Value const& value) {
writeCommentAfterValueOnSameLine(childValue); writeCommentAfterValueOnSameLine(childValue);
break; break;
} }
sout_ << ","; *sout_ << ",";
writeCommentAfterValueOnSameLine(childValue); writeCommentAfterValueOnSameLine(childValue);
} }
unindent(); unindent();
@ -829,15 +828,15 @@ void BuiltStyledStreamWriter::writeArrayValue(Value const& value) {
} else // output on a single line } else // output on a single line
{ {
assert(childValues_.size() == size); assert(childValues_.size() == size);
sout_ << "["; *sout_ << "[";
if (!indentation_.empty()) sout_ << " "; if (!indentation_.empty()) *sout_ << " ";
for (unsigned index = 0; index < size; ++index) { for (unsigned index = 0; index < size; ++index) {
if (index > 0) if (index > 0)
sout_ << ", "; *sout_ << ", ";
sout_ << childValues_[index]; *sout_ << childValues_[index];
} }
if (!indentation_.empty()) sout_ << " "; if (!indentation_.empty()) *sout_ << " ";
sout_ << "]"; *sout_ << "]";
} }
} }
} }
@ -874,7 +873,7 @@ void BuiltStyledStreamWriter::pushValue(std::string const& value) {
if (addChildValues_) if (addChildValues_)
childValues_.push_back(value); childValues_.push_back(value);
else else
sout_ << value; *sout_ << value;
} }
void BuiltStyledStreamWriter::writeIndent() { void BuiltStyledStreamWriter::writeIndent() {
@ -885,13 +884,13 @@ void BuiltStyledStreamWriter::writeIndent() {
if (!indentation_.empty()) { if (!indentation_.empty()) {
// In this case, drop newlines too. // In this case, drop newlines too.
sout_ << '\n' << indentString_; *sout_ << '\n' << indentString_;
} }
} }
void BuiltStyledStreamWriter::writeWithIndent(std::string const& value) { void BuiltStyledStreamWriter::writeWithIndent(std::string const& value) {
if (!indented_) writeIndent(); if (!indented_) writeIndent();
sout_ << value; *sout_ << value;
indented_ = false; indented_ = false;
} }
@ -911,11 +910,11 @@ void BuiltStyledStreamWriter::writeCommentBeforeValue(Value const& root) {
const std::string& comment = root.getComment(commentBefore); const std::string& comment = root.getComment(commentBefore);
std::string::const_iterator iter = comment.begin(); std::string::const_iterator iter = comment.begin();
while (iter != comment.end()) { while (iter != comment.end()) {
sout_ << *iter; *sout_ << *iter;
if (*iter == '\n' && if (*iter == '\n' &&
(iter != comment.end() && *(iter + 1) == '/')) (iter != comment.end() && *(iter + 1) == '/'))
// writeIndent(); // would write extra newline // writeIndent(); // would write extra newline
sout_ << indentString_; *sout_ << indentString_;
++iter; ++iter;
} }
indented_ = false; indented_ = false;
@ -924,11 +923,11 @@ void BuiltStyledStreamWriter::writeCommentBeforeValue(Value const& root) {
void BuiltStyledStreamWriter::writeCommentAfterValueOnSameLine(Value const& root) { void BuiltStyledStreamWriter::writeCommentAfterValueOnSameLine(Value const& root) {
if (cs_ == CommentStyle::None) return; if (cs_ == CommentStyle::None) return;
if (root.hasComment(commentAfterOnSameLine)) if (root.hasComment(commentAfterOnSameLine))
sout_ << " " + root.getComment(commentAfterOnSameLine); *sout_ << " " + root.getComment(commentAfterOnSameLine);
if (root.hasComment(commentAfter)) { if (root.hasComment(commentAfter)) {
writeIndent(); writeIndent();
sout_ << root.getComment(commentAfter); *sout_ << root.getComment(commentAfter);
} }
} }
@ -942,8 +941,8 @@ bool BuiltStyledStreamWriter::hasCommentForValue(const Value& value) {
/////////////// ///////////////
// StreamWriter // StreamWriter
StreamWriter::StreamWriter(std::ostream* sout) StreamWriter::StreamWriter()
: sout_(*sout) : sout_(NULL)
{ {
} }
StreamWriter::~StreamWriter() StreamWriter::~StreamWriter()
@ -957,7 +956,7 @@ StreamWriterBuilder::StreamWriterBuilder()
} }
StreamWriterBuilder::~StreamWriterBuilder() StreamWriterBuilder::~StreamWriterBuilder()
{} {}
StreamWriter* StreamWriterBuilder::newStreamWriter(std::ostream* stream) const StreamWriter* StreamWriterBuilder::newStreamWriter() const
{ {
if (!validate(NULL)) throw std::runtime_error("invalid settings"); if (!validate(NULL)) throw std::runtime_error("invalid settings");
// TODO: Maybe serialize the invalid settings into the exception. // TODO: Maybe serialize the invalid settings into the exception.
@ -978,7 +977,7 @@ StreamWriter* StreamWriterBuilder::newStreamWriter(std::ostream* stream) const
} }
std::string nullSymbol = "null"; std::string nullSymbol = "null";
std::string endingLineFeedSymbol = ""; std::string endingLineFeedSymbol = "";
return new BuiltStyledStreamWriter(stream, return new BuiltStyledStreamWriter(
indentation, cs, indentation, cs,
colonSymbol, nullSymbol, endingLineFeedSymbol); colonSymbol, nullSymbol, endingLineFeedSymbol);
} }
@ -1015,24 +1014,7 @@ void StreamWriterBuilder::setDefaults(Json::Value* settings)
//! [StreamWriterBuilderDefaults] //! [StreamWriterBuilderDefaults]
} }
/* StreamWriter* OldCompressingStreamWriterBuilder::newStreamWriter() const
// This might become public someday.
class StreamWriterBuilderFactory {
public:
virtual ~StreamWriterBuilderFactory();
virtual StreamWriterBuilder* newStreamWriterBuilder() const;
};
StreamWriterBuilderFactory::~StreamWriterBuilderFactory()
{
}
StreamWriterBuilder* StreamWriterBuilderFactory::newStreamWriterBuilder() const
{
return new StreamWriterBuilder;
}
*/
StreamWriter* OldCompressingStreamWriterBuilder::newStreamWriter(
std::ostream* stream) const
{ {
std::string colonSymbol = " : "; std::string colonSymbol = " : ";
if (enableYAMLCompatibility_) { if (enableYAMLCompatibility_) {
@ -1048,22 +1030,22 @@ StreamWriter* OldCompressingStreamWriterBuilder::newStreamWriter(
if (omitEndingLineFeed_) { if (omitEndingLineFeed_) {
endingLineFeedSymbol = ""; endingLineFeedSymbol = "";
} }
return new BuiltStyledStreamWriter(stream, return new BuiltStyledStreamWriter(
"", StreamWriter::CommentStyle::None, "", StreamWriter::CommentStyle::None,
colonSymbol, nullSymbol, endingLineFeedSymbol); colonSymbol, nullSymbol, endingLineFeedSymbol);
} }
std::string writeString(StreamWriter::Factory const& builder, Value const& root) { std::string writeString(StreamWriter::Factory const& builder, Value const& root) {
std::ostringstream sout; std::ostringstream sout;
StreamWriterPtr const writer(builder.newStreamWriter(&sout)); StreamWriterPtr const writer(builder.newStreamWriter());
writer->write(root); writer->write(root, &sout);
return sout.str(); return sout.str();
} }
std::ostream& operator<<(std::ostream& sout, Value const& root) { std::ostream& operator<<(std::ostream& sout, Value const& root) {
StreamWriterBuilder builder; StreamWriterBuilder builder;
StreamWriterPtr const writer(builder.newStreamWriter(&sout)); StreamWriterPtr const writer(builder.newStreamWriter());
writer->write(root); writer->write(root, &sout);
return sout; return sout;
} }