Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
101 lines
2.8 KiB
C++
101 lines
2.8 KiB
C++
// This file is part of Desktop App Toolkit,
|
|
// a set of libraries for developing nice desktop applications.
|
|
//
|
|
// For license and copyright information please follow this link:
|
|
// https://github.com/desktop-app/legal/blob/master/LEGAL
|
|
//
|
|
#include "base/parse_helper.h"
|
|
|
|
namespace base {
|
|
namespace parse {
|
|
|
|
// inspired by https://github.com/sindresorhus/strip-json-comments
|
|
QByteArray stripComments(const QByteArray &content) {
|
|
enum class InsideComment {
|
|
None,
|
|
SingleLine,
|
|
MultiLine,
|
|
};
|
|
auto insideComment = InsideComment::None;
|
|
auto insideString = false;
|
|
|
|
QByteArray result;
|
|
auto begin = content.cbegin(), end = content.cend(), offset = begin;
|
|
auto feedContent = [&result, &offset, end](const char *ch) {
|
|
if (ch > offset) {
|
|
if (result.isEmpty()) result.reserve(end - offset - 2);
|
|
result.append(offset, ch - offset);
|
|
offset = ch;
|
|
}
|
|
};
|
|
auto feedComment = [&result, &offset, end](const char *ch) {
|
|
if (ch > offset) {
|
|
if (result.isEmpty()) result.reserve(end - offset - 2);
|
|
result.append(' ');
|
|
offset = ch;
|
|
}
|
|
};
|
|
for (auto ch = offset; ch != end;) {
|
|
auto currentChar = *ch;
|
|
auto nextChar = (ch + 1 == end) ? 0 : *(ch + 1);
|
|
|
|
if (insideComment == InsideComment::None && currentChar == '"') {
|
|
auto escaped = ((ch > begin) && *(ch - 1) == '\\') && ((ch - 1 < begin) || *(ch - 2) != '\\');
|
|
if (!escaped) {
|
|
insideString = !insideString;
|
|
}
|
|
}
|
|
if (insideString) {
|
|
++ch;
|
|
continue;
|
|
}
|
|
|
|
if (insideComment == InsideComment::None && currentChar == '/' && nextChar == '/') {
|
|
feedContent(ch);
|
|
insideComment = InsideComment::SingleLine;
|
|
ch += 2;
|
|
} else if (insideComment == InsideComment::SingleLine && currentChar == '\r' && nextChar == '\n') {
|
|
feedComment(ch);
|
|
ch += 2;
|
|
insideComment = InsideComment::None;
|
|
} else if (insideComment == InsideComment::SingleLine && currentChar == '\n') {
|
|
feedComment(ch);
|
|
++ch;
|
|
insideComment = InsideComment::None;
|
|
} else if (insideComment == InsideComment::None && currentChar == '/' && nextChar == '*') {
|
|
feedContent(ch);
|
|
ch += 2;
|
|
insideComment = InsideComment::MultiLine;
|
|
} else if (insideComment == InsideComment::MultiLine && currentChar == '*' && nextChar == '/') {
|
|
ch += 2;
|
|
feedComment(ch);
|
|
insideComment = InsideComment::None;
|
|
} else if (insideComment == InsideComment::MultiLine && currentChar == '\r' && nextChar == '\n') {
|
|
feedComment(ch);
|
|
ch += 2;
|
|
feedContent(ch);
|
|
} else if (insideComment == InsideComment::MultiLine && currentChar == '\n') {
|
|
feedComment(ch);
|
|
++ch;
|
|
feedContent(ch);
|
|
} else {
|
|
++ch;
|
|
}
|
|
}
|
|
|
|
if (insideComment == InsideComment::MultiLine) {
|
|
// unexpected end of content
|
|
}
|
|
if (insideComment == InsideComment::None && end > offset) {
|
|
if (result.isEmpty()) {
|
|
return content;
|
|
} else {
|
|
result.append(offset, end - offset);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
} // namespace parse
|
|
} // namespace base
|