2017-10-31 19:44:45 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "bookmaker.h"
|
|
|
|
|
|
|
|
#include "SkOSFile.h"
|
|
|
|
#include "SkOSPath.h"
|
|
|
|
|
2018-08-28 18:53:37 +00:00
|
|
|
const string kCatalogFileName("catalog.htm");
|
|
|
|
|
2018-04-16 12:37:38 +00:00
|
|
|
bool Catalog::appendFile(string path) {
|
2017-10-31 19:44:45 +00:00
|
|
|
FILE* file = fopen(path.c_str(), "r");
|
|
|
|
if (!file) {
|
|
|
|
SkDebugf("could not append %s\n", path.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
fseek(file, 0L, SEEK_END);
|
|
|
|
int sz = (int) ftell(file);
|
|
|
|
rewind(file);
|
|
|
|
char* buffer = new char[sz];
|
|
|
|
memset(buffer, ' ', sz);
|
2017-11-01 14:52:55 +00:00
|
|
|
SkAssertResult(sz == (int)fread(buffer, 1, sz, file));
|
2017-10-31 19:44:45 +00:00
|
|
|
fclose(file);
|
|
|
|
this->writeBlock(sz, buffer);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-28 18:53:37 +00:00
|
|
|
bool Catalog::openCatalog(const char* inDir) {
|
2017-10-31 19:44:45 +00:00
|
|
|
fDocsDir = inDir;
|
|
|
|
if ('/' != fDocsDir.back()) {
|
|
|
|
fDocsDir += '/';
|
|
|
|
}
|
2018-08-28 18:53:37 +00:00
|
|
|
fOut = fopen(kCatalogFileName.c_str(), "wb");
|
|
|
|
fFullName = kCatalogFileName;
|
2017-10-31 19:44:45 +00:00
|
|
|
if (!fOut) {
|
2018-08-28 18:53:37 +00:00
|
|
|
SkDebugf("could not open output file %s\n", kCatalogFileName.c_str());
|
2017-10-31 19:44:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
fContinuation = false;
|
|
|
|
if (!appendFile(fDocsDir + "catalogHeader.txt")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
this->lf(1);
|
2018-08-28 18:53:37 +00:00
|
|
|
|
2017-10-31 19:44:45 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-28 18:53:37 +00:00
|
|
|
bool Catalog::openStatus(const char* statusFile) {
|
2017-12-11 21:03:17 +00:00
|
|
|
StatusIter iter(statusFile, ".bmh", StatusFilter::kInProgress);
|
|
|
|
// FIXME: iterate through only chosen files by setting fDocsDir to iter
|
|
|
|
// read one file to find directory
|
2018-10-08 18:57:48 +00:00
|
|
|
if (!iter.next(nullptr, nullptr)) {
|
2017-12-11 21:03:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
2018-08-28 18:53:37 +00:00
|
|
|
return openCatalog(iter.baseDir().c_str());
|
2017-12-11 21:03:17 +00:00
|
|
|
}
|
|
|
|
|
2018-08-28 18:53:37 +00:00
|
|
|
bool Catalog::closeCatalog(const char* outDir) {
|
2017-10-31 19:44:45 +00:00
|
|
|
if (fOut) {
|
|
|
|
this->lf(1);
|
|
|
|
this->writeString("}");
|
|
|
|
this->lf(1);
|
|
|
|
if (!appendFile(fDocsDir + "catalogTrailer.txt")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
this->lf(1);
|
|
|
|
this->writePending();
|
|
|
|
fclose(fOut);
|
2018-08-28 18:53:37 +00:00
|
|
|
string outie = outDir;
|
|
|
|
if ('/' != outie.back()) {
|
|
|
|
outie += '/';
|
|
|
|
}
|
|
|
|
string fullName = outie + kCatalogFileName;
|
|
|
|
if (ParserCommon::WrittenFileDiffers(fullName, kCatalogFileName)) {
|
|
|
|
ParserCommon::CopyToFile(fullName, kCatalogFileName);
|
|
|
|
SkDebugf("wrote %s\n", fullName.c_str());
|
2018-09-04 20:15:11 +00:00
|
|
|
} else {
|
|
|
|
remove(kCatalogFileName.c_str());
|
2018-08-28 18:53:37 +00:00
|
|
|
}
|
2017-10-31 19:44:45 +00:00
|
|
|
fOut = nullptr;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Catalog::parseFromFile(const char* path) {
|
2018-07-18 19:10:08 +00:00
|
|
|
if (!INHERITED::parseFromFile(path)) {
|
2017-10-31 19:44:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
fIndent = 4;
|
|
|
|
this->writeString("var text = {");
|
|
|
|
this->lf(1);
|
|
|
|
fTextOut = true;
|
|
|
|
if (!parseFiddles()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
this->lf(1);
|
|
|
|
this->writeString("}");
|
|
|
|
this->lf(2);
|
|
|
|
this->writeString("var pngs = {");
|
|
|
|
fTextOut = false;
|
|
|
|
fPngOut = true;
|
2018-07-18 19:10:08 +00:00
|
|
|
JsonStatus* status = &fStack.back();
|
|
|
|
status->fIter = status->fObject.begin();
|
2017-10-31 19:44:45 +00:00
|
|
|
fContinuation = false;
|
|
|
|
return parseFiddles();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Catalog::pngOut(Definition* example) {
|
|
|
|
string result;
|
2018-03-05 18:26:16 +00:00
|
|
|
if (!fBmhParser->exampleToScript(example, BmhParser::ExampleOptions::kPng, &result)) {
|
2017-10-31 19:44:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (result.length() > 0) {
|
|
|
|
if (fContinuation) {
|
|
|
|
this->writeString(",");
|
|
|
|
this->lf(1);
|
|
|
|
} else {
|
|
|
|
fContinuation = true;
|
|
|
|
}
|
|
|
|
this->writeBlock(result.size(), result.c_str());
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-12-12 16:38:09 +00:00
|
|
|
bool Catalog::textOut(Definition* def, const char* stdOutStart,
|
2017-10-31 19:44:45 +00:00
|
|
|
const char* stdOutEnd) {
|
|
|
|
string result;
|
2018-03-05 18:26:16 +00:00
|
|
|
if (!fBmhParser->exampleToScript(def, BmhParser::ExampleOptions::kText, &result)) {
|
2017-10-31 19:44:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (result.length() > 0) {
|
|
|
|
if (fContinuation) {
|
|
|
|
this->writeString(",");
|
|
|
|
this->lf(1);
|
|
|
|
} else {
|
|
|
|
fContinuation = true;
|
|
|
|
}
|
|
|
|
fIndent = 8;
|
|
|
|
this->writeBlock(result.size(), result.c_str());
|
|
|
|
this->lf(1);
|
|
|
|
this->writeString("\"stdout\": \"");
|
|
|
|
size_t pos = 0;
|
|
|
|
size_t len = stdOutEnd - stdOutStart;
|
|
|
|
string example;
|
2017-12-12 16:38:09 +00:00
|
|
|
const Definition* stdOut = def->hasChild(MarkType::kStdOut);
|
|
|
|
const Definition* outVolatile = stdOut ? stdOut->hasChild(MarkType::kVolatile) : nullptr;
|
|
|
|
if (outVolatile) {
|
|
|
|
stdOutStart = outVolatile->fContentStart;
|
|
|
|
while ('\n' == stdOutStart[0]) {
|
|
|
|
++stdOutStart;
|
|
|
|
}
|
|
|
|
len = stdOut->fContentEnd - stdOutStart;
|
|
|
|
}
|
2017-10-31 19:44:45 +00:00
|
|
|
while ((size_t) pos < len) {
|
|
|
|
example += '"' == stdOutStart[pos] ? "\\\"" :
|
|
|
|
'\\' == stdOutStart[pos] ? "\\\\" :
|
2017-12-12 16:38:09 +00:00
|
|
|
'\n' == stdOutStart[pos] ? "\\\\n" :
|
2017-10-31 19:44:45 +00:00
|
|
|
string(&stdOutStart[pos], 1);
|
2017-12-12 16:38:09 +00:00
|
|
|
if (outVolatile && '\n' == stdOutStart[pos]) {
|
|
|
|
++pos;
|
|
|
|
while ((size_t) pos < len && ' ' == stdOutStart[pos]) {
|
|
|
|
++pos;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2017-10-31 19:44:45 +00:00
|
|
|
++pos;
|
|
|
|
}
|
2017-12-12 16:38:09 +00:00
|
|
|
if (outVolatile) {
|
|
|
|
example += "\\\\n";
|
|
|
|
}
|
2017-10-31 19:44:45 +00:00
|
|
|
this->writeBlock(example.length(), example.c_str());
|
|
|
|
this->writeString("\"");
|
|
|
|
this->lf(1);
|
|
|
|
fIndent = 4;
|
|
|
|
this->writeString("}");
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|