jsonx.cpp 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. // JSON to JSONx conversion exmaple, using SAX API.
  2. // JSONx is an IBM standard format to represent JSON as XML.
  3. // https://www-01.ibm.com/support/knowledgecenter/SS9H2Y_7.1.0/com.ibm.dp.doc/json_jsonx.html
  4. // This example parses JSON text from stdin with validation,
  5. // and convert to JSONx format to stdout.
  6. // Need compile with -D__STDC_FORMAT_MACROS for defining PRId64 and PRIu64 macros.
  7. #include "rapidjson/reader.h"
  8. #include "rapidjson/stringbuffer.h"
  9. #include "rapidjson/filereadstream.h"
  10. #include "rapidjson/filewritestream.h"
  11. #include "rapidjson/error/en.h"
  12. #include <cstdio>
  13. using namespace rapidjson;
  14. // For simplicity, this example only read/write in UTF-8 encoding
  15. template <typename OutputStream>
  16. class JsonxWriter {
  17. public:
  18. JsonxWriter(OutputStream& os) : os_(os), name_(), level_(0), hasName_(false) {
  19. }
  20. bool Null() {
  21. return WriteStartElement("null", true);
  22. }
  23. bool Bool(bool b) {
  24. return
  25. WriteStartElement("boolean") &&
  26. WriteString(b ? "true" : "false") &&
  27. WriteEndElement("boolean");
  28. }
  29. bool Int(int i) {
  30. char buffer[12];
  31. return WriteNumberElement(buffer, sprintf(buffer, "%d", i));
  32. }
  33. bool Uint(unsigned i) {
  34. char buffer[11];
  35. return WriteNumberElement(buffer, sprintf(buffer, "%u", i));
  36. }
  37. bool Int64(int64_t i) {
  38. char buffer[21];
  39. return WriteNumberElement(buffer, sprintf(buffer, "%" PRId64, i));
  40. }
  41. bool Uint64(uint64_t i) {
  42. char buffer[21];
  43. return WriteNumberElement(buffer, sprintf(buffer, "%" PRIu64, i));
  44. }
  45. bool Double(double d) {
  46. char buffer[30];
  47. return WriteNumberElement(buffer, sprintf(buffer, "%.17g", d));
  48. }
  49. bool String(const char* str, SizeType length, bool) {
  50. return
  51. WriteStartElement("string") &&
  52. WriteEscapedText(str, length) &&
  53. WriteEndElement("string");
  54. }
  55. bool StartObject() {
  56. return WriteStartElement("object");
  57. }
  58. bool Key(const char* str, SizeType length, bool) {
  59. // backup key to name_
  60. name_.Clear();
  61. for (SizeType i = 0; i < length; i++)
  62. name_.Put(str[i]);
  63. hasName_ = true;
  64. return true;
  65. }
  66. bool EndObject(SizeType) {
  67. return WriteEndElement("object");
  68. }
  69. bool StartArray() {
  70. return WriteStartElement("array");
  71. }
  72. bool EndArray(SizeType) {
  73. return WriteEndElement("array");
  74. }
  75. private:
  76. bool WriteString(const char* s) {
  77. while (*s)
  78. os_.Put(*s++);
  79. return true;
  80. }
  81. bool WriteEscapedAttributeValue(const char* s, size_t length) {
  82. for (size_t i = 0; i < length; i++) {
  83. switch (s[i]) {
  84. case '&': WriteString("&amp;"); break;
  85. case '<': WriteString("&lt;"); break;
  86. case '"': WriteString("&quot;"); break;
  87. default: os_.Put(s[i]); break;
  88. }
  89. }
  90. return true;
  91. }
  92. bool WriteEscapedText(const char* s, size_t length) {
  93. for (size_t i = 0; i < length; i++) {
  94. switch (s[i]) {
  95. case '&': WriteString("&amp;"); break;
  96. case '<': WriteString("&lt;"); break;
  97. default: os_.Put(s[i]); break;
  98. }
  99. }
  100. return true;
  101. }
  102. bool WriteStartElement(const char* type, bool emptyElement = false) {
  103. if (level_ == 0)
  104. if (!WriteString("<?xml version=\"1.0\" encoding=\"UTF-8\"?>"))
  105. return false;
  106. if (!WriteString("<json:") || !WriteString(type))
  107. return false;
  108. // For root element, need to add declarations
  109. if (level_ == 0) {
  110. if (!WriteString(
  111. " xsi:schemaLocation=\"http://www.datapower.com/schemas/json jsonx.xsd\""
  112. " xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""
  113. " xmlns:json=\"http://www.ibm.com/xmlns/prod/2009/jsonx\""))
  114. return false;
  115. }
  116. if (hasName_) {
  117. hasName_ = false;
  118. if (!WriteString(" name=\"") ||
  119. !WriteEscapedAttributeValue(name_.GetString(), name_.GetSize()) ||
  120. !WriteString("\""))
  121. return false;
  122. }
  123. if (emptyElement)
  124. return WriteString("/>");
  125. else {
  126. level_++;
  127. return WriteString(">");
  128. }
  129. }
  130. bool WriteEndElement(const char* type) {
  131. if (!WriteString("</json:") ||
  132. !WriteString(type) ||
  133. !WriteString(">"))
  134. return false;
  135. // For the last end tag, flush the output stream.
  136. if (--level_ == 0)
  137. os_.Flush();
  138. return true;
  139. }
  140. bool WriteNumberElement(const char* buffer, int length) {
  141. if (!WriteStartElement("number"))
  142. return false;
  143. for (int j = 0; j < length; j++)
  144. os_.Put(buffer[j]);
  145. return WriteEndElement("number");
  146. }
  147. OutputStream& os_;
  148. StringBuffer name_;
  149. unsigned level_;
  150. bool hasName_;
  151. };
  152. int main(int, char*[]) {
  153. // Prepare JSON reader and input stream.
  154. Reader reader;
  155. char readBuffer[65536];
  156. FileReadStream is(stdin, readBuffer, sizeof(readBuffer));
  157. // Prepare JSON writer and output stream.
  158. char writeBuffer[65536];
  159. FileWriteStream os(stdout, writeBuffer, sizeof(writeBuffer));
  160. JsonxWriter<FileWriteStream> writer(os);
  161. // JSON reader parse from the input stream and let writer generate the output.
  162. if (!reader.Parse(is, writer)) {
  163. fprintf(stderr, "\nError(%u): %s\n", static_cast<unsigned>(reader.GetErrorOffset()), GetParseError_En(reader.GetParseErrorCode()));
  164. return 1;
  165. }
  166. return 0;
  167. }