1 /*
2 * Copyright 2017 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8 #include "bmhParser.h"
9 #include "fiddleParser.h"
10
11 #include "SkOSFile.h"
12 #include "SkOSPath.h"
13
14 const string kCatalogFileName("catalog.htm");
15
appendFile(string path)16 bool Catalog::appendFile(string path) {
17 FILE* file = fopen(path.c_str(), "r");
18 if (!file) {
19 SkDebugf("could not append %s\n", path.c_str());
20 return false;
21 }
22 fseek(file, 0L, SEEK_END);
23 int sz = (int) ftell(file);
24 rewind(file);
25 char* buffer = new char[sz];
26 memset(buffer, ' ', sz);
27 SkAssertResult(sz == (int)fread(buffer, 1, sz, file));
28 fclose(file);
29 this->writeBlock(sz, buffer);
30 return true;
31 }
32
openCatalog(const char * inDir)33 bool Catalog::openCatalog(const char* inDir) {
34 fDocsDir = inDir;
35 if ('/' != fDocsDir.back()) {
36 fDocsDir += '/';
37 }
38 fOut = fopen(kCatalogFileName.c_str(), "wb");
39 fFullName = kCatalogFileName;
40 if (!fOut) {
41 SkDebugf("could not open output file %s\n", kCatalogFileName.c_str());
42 return false;
43 }
44 fContinuation = false;
45 if (!appendFile(fDocsDir + "catalogHeader.txt")) {
46 return false;
47 }
48 this->lf(1);
49
50 return true;
51 }
52
openStatus(const char * statusFile)53 bool Catalog::openStatus(const char* statusFile) {
54 StatusIter iter(statusFile, ".bmh", StatusFilter::kInProgress);
55 // FIXME: iterate through only chosen files by setting fDocsDir to iter
56 // read one file to find directory
57 if (!iter.next(nullptr, nullptr)) {
58 return false;
59 }
60 return openCatalog(iter.baseDir().c_str());
61 }
62
closeCatalog(const char * outDir)63 bool Catalog::closeCatalog(const char* outDir) {
64 if (fOut) {
65 this->lf(1);
66 this->writeString("}");
67 this->lf(1);
68 if (!appendFile(fDocsDir + "catalogTrailer.txt")) {
69 return false;
70 }
71 this->lf(1);
72 this->writePending();
73 fclose(fOut);
74 string outie = outDir;
75 if ('/' != outie.back()) {
76 outie += '/';
77 }
78 string fullName = outie + kCatalogFileName;
79 if (ParserCommon::WrittenFileDiffers(fullName, kCatalogFileName)) {
80 ParserCommon::CopyToFile(fullName, kCatalogFileName);
81 SkDebugf("wrote %s\n", fullName.c_str());
82 } else {
83 remove(kCatalogFileName.c_str());
84 }
85 fOut = nullptr;
86 }
87 return true;
88 }
89
parseFromFile(const char * path)90 bool Catalog::parseFromFile(const char* path) {
91 if (!INHERITED::parseFromFile(path)) {
92 return false;
93 }
94 fIndent = 4;
95 this->writeString("var text = {");
96 this->lf(1);
97 fTextOut = true;
98 if (!parseFiddles()) {
99 return false;
100 }
101 this->lf(1);
102 this->writeString("}");
103 this->lf(2);
104 this->writeString("var pngs = {");
105 fTextOut = false;
106 fPngOut = true;
107 JsonStatus* status = &fStack.back();
108 status->reset();
109 fContinuation = false;
110 return parseFiddles();
111 }
112
pngOut(Definition * example)113 bool Catalog::pngOut(Definition* example) {
114 string result;
115 if (!fBmhParser->exampleToScript(example, BmhParser::ExampleOptions::kPng, &result)) {
116 return false;
117 }
118 if (result.length() > 0) {
119 if (fContinuation) {
120 this->writeString(",");
121 this->lf(1);
122 } else {
123 fContinuation = true;
124 }
125 this->writeBlock(result.size(), result.c_str());
126 }
127 return true;
128 }
129
textOut(Definition * def,const char * stdOutStart,const char * stdOutEnd)130 bool Catalog::textOut(Definition* def, const char* stdOutStart,
131 const char* stdOutEnd) {
132 string result;
133 if (!fBmhParser->exampleToScript(def, BmhParser::ExampleOptions::kText, &result)) {
134 return false;
135 }
136 if (result.length() > 0) {
137 if (fContinuation) {
138 this->writeString(",");
139 this->lf(1);
140 } else {
141 fContinuation = true;
142 }
143 fIndent = 8;
144 this->writeBlock(result.size(), result.c_str());
145 this->lf(1);
146 this->writeString("\"stdout\": \"");
147 size_t pos = 0;
148 size_t len = stdOutEnd - stdOutStart;
149 string example;
150 const Definition* stdOut = def->hasChild(MarkType::kStdOut);
151 const Definition* outVolatile = stdOut ? stdOut->hasChild(MarkType::kVolatile) : nullptr;
152 if (outVolatile) {
153 stdOutStart = outVolatile->fContentStart;
154 while ('\n' == stdOutStart[0]) {
155 ++stdOutStart;
156 }
157 len = stdOut->fContentEnd - stdOutStart;
158 }
159 while ((size_t) pos < len) {
160 example += '"' == stdOutStart[pos] ? "\\\"" :
161 '\\' == stdOutStart[pos] ? "\\\\" :
162 '\n' == stdOutStart[pos] ? "\\\\n" :
163 string(&stdOutStart[pos], 1);
164 if (outVolatile && '\n' == stdOutStart[pos]) {
165 ++pos;
166 while ((size_t) pos < len && ' ' == stdOutStart[pos]) {
167 ++pos;
168 }
169 continue;
170 }
171 ++pos;
172 }
173 if (outVolatile) {
174 example += "\\\\n";
175 }
176 this->writeBlock(example.length(), example.c_str());
177 this->writeString("\"");
178 this->lf(1);
179 fIndent = 4;
180 this->writeString("}");
181 }
182 return true;
183 }
184