mirror of
https://github.com/qelectrotech/qelectrotech-source-mirror.git
synced 2026-06-12 06:43:14 +02:00
fb35027624
Extends the headless command-line interface with three read-only tools
aimed at validating projects and element libraries (useful for batch
import / migration pipelines):
qelectrotech --info <project.qet> [output.json]
qelectrotech --export-bom <project.qet> <output.csv>
qelectrotech --check-elements <element.elmt | directory>
- --info dumps a structural summary as JSON straight from QET's loaded
model: per-diagram element / conductor counts, page size, and the
number of unconnected ("free") terminals, plus project totals. Because
it uses the real loader it reports what the editor actually sees.
- --export-bom writes a bill of materials (one row per element) as CSV,
querying the project's own element_nomenclature_view (the same source
as the GUI BOM export). updateDB() is called first so the database is
populated in a headless run.
- --check-elements validates one .elmt file, or every .elmt under a
directory (recursively), against the element schema: XML well-formed,
root <definition type="element">, a usable bounding box, and terminal
count. Reports OK / WARN / FAIL per file and a summary; exit code is
non-zero if any file fails. Verified against the full bundled
collection (8483 elements): 0 false failures, agreeing with QET's own
loader (e.g. a negative-height element it tolerates is a WARN, not a
FAIL).
run() is restructured to handle the differing argument arity (info takes
an optional output, check-elements takes a path rather than a project).
502 lines
14 KiB
C++
502 lines
14 KiB
C++
/*
|
|
Copyright 2006-2025 The QElectroTech Team
|
|
This file is part of QElectroTech.
|
|
|
|
QElectroTech is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
QElectroTech is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with QElectroTech. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "cli_export.h"
|
|
|
|
#include "bordertitleblock.h"
|
|
#include "conductornumexport.h"
|
|
#include "dataBase/projectdatabase.h"
|
|
#include "diagram.h"
|
|
#include "diagramcontext.h"
|
|
#include "qetgraphicsitem/element.h"
|
|
#include "qetgraphicsitem/terminal.h"
|
|
#include "qetproject.h"
|
|
#include "wiringlistexport.h"
|
|
|
|
#include <QDir>
|
|
#include <QDirIterator>
|
|
#include <QDomDocument>
|
|
#include <QFile>
|
|
#include <QFileInfo>
|
|
#include <QJsonArray>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
#include <QPainter>
|
|
#include <QPdfWriter>
|
|
#include <QSqlError>
|
|
#include <QSqlQuery>
|
|
#include <QSvgGenerator>
|
|
#include <QTextStream>
|
|
|
|
namespace {
|
|
|
|
QTextStream out(stdout);
|
|
QTextStream err(stderr);
|
|
|
|
/// All CLI option flags, mapped to a short format name.
|
|
const QHash<QString, QString> &exportFlags()
|
|
{
|
|
static const QHash<QString, QString> flags {
|
|
{"--export-pdf", "pdf"},
|
|
{"--export-png", "png"},
|
|
{"--export-svg", "svg"},
|
|
{"--export-cables", "cables"},
|
|
{"--export-wires", "wires"},
|
|
{"--export-bom", "bom"},
|
|
{"--info", "info"},
|
|
{"--check-elements", "check"},
|
|
};
|
|
return flags;
|
|
}
|
|
|
|
/// Pixel rect of a diagram's border + title block (the printable page area).
|
|
QRect diagramRect(Diagram *diagram)
|
|
{
|
|
QRectF r = diagram->border_and_titleblock.borderAndTitleBlockRect();
|
|
r.adjust(0, 0, 1, 1); // include the 1px border line
|
|
return r.toAlignedRect();
|
|
}
|
|
|
|
/// A filesystem-safe per-diagram file stem: "01_Title".
|
|
QString diagramStem(Diagram *diagram, int index)
|
|
{
|
|
QString title = diagram->title();
|
|
title.replace(QRegularExpression("[^\\w \\-]"), "_");
|
|
title = title.simplified();
|
|
if (title.isEmpty())
|
|
title = "diagram";
|
|
return QStringLiteral("%1_%2")
|
|
.arg(index, 2, 10, QChar('0'))
|
|
.arg(title);
|
|
}
|
|
|
|
/// Render @p diagram into @p painter, fitting @p target to the page rect.
|
|
void renderDiagram(Diagram *diagram, QPainter &painter, const QRectF &target)
|
|
{
|
|
const QRect source = diagramRect(diagram);
|
|
// Export without the editor grid: drawBackground() only paints it when
|
|
// draw_grid_ is set (default true), so toggle it off around the render
|
|
// and restore it afterwards.
|
|
const bool was_drawing_grid = diagram->displayGrid();
|
|
diagram->setDisplayGrid(false);
|
|
diagram->render(&painter, target, source, Qt::KeepAspectRatio);
|
|
diagram->setDisplayGrid(was_drawing_grid);
|
|
}
|
|
|
|
int exportPdf(QETProject &project, const QString &output)
|
|
{
|
|
const QList<Diagram *> diagrams = project.diagrams();
|
|
if (diagrams.isEmpty()) {
|
|
err << "No diagrams to export.\n";
|
|
return 1;
|
|
}
|
|
|
|
QPdfWriter writer(output);
|
|
writer.setCreator("QElectroTech");
|
|
writer.setResolution(96);
|
|
|
|
QPainter painter;
|
|
bool first = true;
|
|
for (Diagram *diagram : diagrams) {
|
|
const QRect r = diagramRect(diagram);
|
|
// Match the page to the diagram (in points: 1px @ 96dpi = 0.75pt).
|
|
const QPageSize page(QSizeF(r.width() * 72.0 / 96.0,
|
|
r.height() * 72.0 / 96.0),
|
|
QPageSize::Point);
|
|
writer.setPageSize(page);
|
|
writer.setPageMargins(QMarginsF(0, 0, 0, 0));
|
|
|
|
if (first) {
|
|
if (!painter.begin(&writer)) {
|
|
err << "Cannot open '" << output << "' for writing.\n";
|
|
return 1;
|
|
}
|
|
first = false;
|
|
} else {
|
|
writer.newPage();
|
|
}
|
|
const QRectF target(0, 0,
|
|
writer.width(), writer.height());
|
|
renderDiagram(diagram, painter, target);
|
|
}
|
|
painter.end();
|
|
out << "Exported " << diagrams.size() << " page(s) -> " << output << "\n";
|
|
return 0;
|
|
}
|
|
|
|
int exportImages(QETProject &project, const QString &format,
|
|
const QString &out_dir)
|
|
{
|
|
const QList<Diagram *> diagrams = project.diagrams();
|
|
if (diagrams.isEmpty()) {
|
|
err << "No diagrams to export.\n";
|
|
return 1;
|
|
}
|
|
QDir().mkpath(out_dir);
|
|
|
|
int index = 0;
|
|
for (Diagram *diagram : diagrams) {
|
|
++index;
|
|
const QRect r = diagramRect(diagram);
|
|
const QString path = QDir(out_dir).filePath(
|
|
diagramStem(diagram, index) + "." + format);
|
|
|
|
if (format == "svg") {
|
|
QSvgGenerator gen;
|
|
gen.setFileName(path);
|
|
gen.setSize(r.size());
|
|
gen.setViewBox(QRect(0, 0, r.width(), r.height()));
|
|
gen.setTitle(diagram->title());
|
|
QPainter painter(&gen);
|
|
renderDiagram(diagram, painter, QRectF(QPointF(0, 0), r.size()));
|
|
painter.end();
|
|
} else { // png
|
|
QImage image(r.size(), QImage::Format_ARGB32);
|
|
image.fill(Qt::white);
|
|
QPainter painter(&image);
|
|
painter.setRenderHint(QPainter::Antialiasing, true);
|
|
renderDiagram(diagram, painter, QRectF(QPointF(0, 0), r.size()));
|
|
painter.end();
|
|
if (!image.save(path)) {
|
|
err << "Failed to write '" << path << "'.\n";
|
|
return 1;
|
|
}
|
|
}
|
|
out << " " << path << "\n";
|
|
}
|
|
out << "Exported " << diagrams.size() << " diagram(s) -> " << out_dir << "\n";
|
|
return 0;
|
|
}
|
|
|
|
int exportCsv(QETProject &project, const QString &format, const QString &output)
|
|
{
|
|
QString csv;
|
|
if (format == "cables") {
|
|
WiringListExport wle(&project, nullptr);
|
|
csv = wle.toCsvString();
|
|
} else { // wires
|
|
ConductorNumExport cne(&project, nullptr);
|
|
csv = cne.wiresNum();
|
|
}
|
|
if (csv.isEmpty()) {
|
|
err << "Nothing to export (empty list).\n";
|
|
return 1;
|
|
}
|
|
|
|
QFile file(output);
|
|
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
|
err << "Cannot open '" << output << "' for writing.\n";
|
|
return 1;
|
|
}
|
|
QTextStream fout(&file);
|
|
fout << csv;
|
|
file.close();
|
|
out << "Exported " << format << " list -> " << output << "\n";
|
|
return 0;
|
|
}
|
|
|
|
/// Quote a field for CSV output (RFC-4180 style, ';' delimiter).
|
|
QString csvField(const QString &value)
|
|
{
|
|
if (value.contains(';') || value.contains('"')
|
|
|| value.contains('\n') || value.contains('\r')) {
|
|
QString v = value;
|
|
v.replace('"', "\"\"");
|
|
return '"' % v % '"';
|
|
}
|
|
return value;
|
|
}
|
|
|
|
/// Bill of materials: one row per element, key component-data fields.
|
|
/// Pulls from QET's own project database (the same source as the GUI BOM
|
|
/// export), so the output matches what the editor produces.
|
|
int exportBom(QETProject &project, const QString &output)
|
|
{
|
|
// The project database is built lazily; force a (re)build before querying.
|
|
project.dataBase()->updateDB();
|
|
|
|
static const QStringList columns {
|
|
"label", "designation", "manufacturer", "manufacturer_reference",
|
|
"quantity", "location", "function", "title", "folio"
|
|
};
|
|
|
|
QSqlQuery query = project.dataBase()->newQuery(
|
|
"SELECT " % columns.join(", ") %
|
|
" FROM element_nomenclature_view ORDER BY label");
|
|
if (!query.exec()) {
|
|
err << "BOM query failed: " << query.lastError().text() << "\n";
|
|
return 1;
|
|
}
|
|
|
|
QString csv = columns.join(";") % "\n";
|
|
int rows = 0;
|
|
while (query.next()) {
|
|
QStringList values;
|
|
for (int i = 0; i < columns.size(); ++i)
|
|
values << csvField(query.value(i).toString());
|
|
csv += values.join(";") % "\n";
|
|
++rows;
|
|
}
|
|
|
|
QFile file(output);
|
|
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
|
err << "Cannot open '" << output << "' for writing.\n";
|
|
return 1;
|
|
}
|
|
QTextStream fout(&file);
|
|
fout << csv;
|
|
file.close();
|
|
out << "Exported " << rows << " component(s) -> " << output << "\n";
|
|
return 0;
|
|
}
|
|
|
|
/// Count terminals on @p element that no conductor connects to.
|
|
int freeTerminals(Element *element)
|
|
{
|
|
int free = 0;
|
|
const QList<Terminal *> terminals = element->terminals();
|
|
for (Terminal *t : terminals)
|
|
if (t->conductorsCount() == 0)
|
|
++free;
|
|
return free;
|
|
}
|
|
|
|
/// Structural ground-truth dump of a project, as JSON, to stdout (or a file).
|
|
/// Uses QET's own loaded model, so it reports what the editor actually sees:
|
|
/// per-page element / conductor counts and unconnected terminals.
|
|
int exportInfo(QETProject &project, const QString &output)
|
|
{
|
|
const QList<Diagram *> diagrams = project.diagrams();
|
|
|
|
int total_elements = 0, total_conductors = 0, total_free = 0;
|
|
QJsonArray pages;
|
|
int index = 0;
|
|
for (Diagram *diagram : diagrams) {
|
|
++index;
|
|
const QList<Element *> elements = diagram->elements();
|
|
const int conductors = diagram->conductors().size();
|
|
int page_free = 0;
|
|
for (Element *e : elements)
|
|
page_free += freeTerminals(e);
|
|
|
|
const QRect r = diagramRect(diagram);
|
|
QJsonObject page;
|
|
page["index"] = index;
|
|
page["title"] = diagram->title();
|
|
page["folio"] = QStringLiteral("%1 of %2")
|
|
.arg(index).arg(diagrams.size());
|
|
page["width_px"] = r.width();
|
|
page["height_px"] = r.height();
|
|
page["elements"] = elements.size();
|
|
page["conductors"] = conductors;
|
|
page["free_terminals"] = page_free;
|
|
pages.append(page);
|
|
|
|
total_elements += elements.size();
|
|
total_conductors += conductors;
|
|
total_free += page_free;
|
|
}
|
|
|
|
QJsonObject root;
|
|
root["project"] = project.title();
|
|
root["diagrams"] = diagrams.size();
|
|
root["elements"] = total_elements;
|
|
root["conductors"] = total_conductors;
|
|
root["free_terminals"] = total_free;
|
|
root["pages"] = pages;
|
|
|
|
const QByteArray json =
|
|
QJsonDocument(root).toJson(QJsonDocument::Indented);
|
|
|
|
if (output.isEmpty()) {
|
|
out << QString::fromUtf8(json);
|
|
} else {
|
|
QFile file(output);
|
|
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
|
err << "Cannot open '" << output << "' for writing.\n";
|
|
return 1;
|
|
}
|
|
file.write(json);
|
|
file.close();
|
|
out << "Wrote project info -> " << output << "\n";
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/// Validate one .elmt file against QET's element schema.
|
|
/// @return 0 = OK, 1 = warning (loads but suspicious), 2 = failure.
|
|
int checkOneElement(const QString &path)
|
|
{
|
|
QFile file(path);
|
|
if (!file.open(QIODevice::ReadOnly)) {
|
|
out << "FAIL " << path << " (cannot open)\n";
|
|
return 2;
|
|
}
|
|
QDomDocument doc;
|
|
QString error;
|
|
int line = 0;
|
|
if (!doc.setContent(&file, &error, &line)) {
|
|
file.close();
|
|
out << "FAIL " << path << " (XML error line "
|
|
<< line << ": " << error << ")\n";
|
|
return 2;
|
|
}
|
|
file.close();
|
|
|
|
const QDomElement root = doc.documentElement();
|
|
if (root.tagName() != "definition" || root.attribute("type") != "element") {
|
|
out << "FAIL " << path << " (root is not <definition type=\"element\">)\n";
|
|
return 2;
|
|
}
|
|
|
|
bool w_ok = false, h_ok = false;
|
|
const double w = root.attribute("width").toDouble(&w_ok);
|
|
const double h = root.attribute("height").toDouble(&h_ok);
|
|
if (!w_ok || !h_ok || w == 0 || h == 0) {
|
|
out << "FAIL " << path << " (missing/zero bounding box "
|
|
<< root.attribute("width") << "x"
|
|
<< root.attribute("height") << ")\n";
|
|
return 2;
|
|
}
|
|
|
|
const int terminals = root.elementsByTagName("terminal").count();
|
|
|
|
// Negative dimensions are malformed but QET still loads them; surface as a
|
|
// warning rather than a failure so this agrees with QET's own loader.
|
|
if (w < 0 || h < 0) {
|
|
out << "WARN " << path << " (negative bounding box "
|
|
<< w << "x" << h << ", " << terminals << " terminals)\n";
|
|
return 1;
|
|
}
|
|
|
|
if (terminals == 0) {
|
|
out << "WARN " << path << " (loads, but 0 terminals)\n";
|
|
return 1;
|
|
}
|
|
|
|
out << "OK " << path << " (" << terminals << " terminals)\n";
|
|
return 0;
|
|
}
|
|
|
|
/// Validate a single .elmt file or every .elmt under a directory.
|
|
int checkElements(const QString &path)
|
|
{
|
|
QStringList files;
|
|
const QFileInfo info(path);
|
|
if (info.isDir()) {
|
|
QDirIterator it(path, {"*.elmt"}, QDir::Files,
|
|
QDirIterator::Subdirectories);
|
|
while (it.hasNext())
|
|
files << it.next();
|
|
files.sort();
|
|
} else if (info.isFile()) {
|
|
files << path;
|
|
} else {
|
|
err << "Not found: " << path << "\n";
|
|
return 2;
|
|
}
|
|
|
|
if (files.isEmpty()) {
|
|
err << "No .elmt files found under: " << path << "\n";
|
|
return 2;
|
|
}
|
|
|
|
int warnings = 0, failures = 0;
|
|
for (const QString &f : files) {
|
|
const int r = checkOneElement(f);
|
|
if (r == 1) ++warnings;
|
|
else if (r == 2) ++failures;
|
|
}
|
|
out << files.size() << " file(s), " << warnings
|
|
<< " warning(s), " << failures << " failure(s)\n";
|
|
return failures > 0 ? 1 : 0;
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
namespace CLIExport {
|
|
|
|
bool isExportRequest(const QStringList &args)
|
|
{
|
|
for (const QString &a : args)
|
|
if (exportFlags().contains(a))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
int run(const QStringList &args)
|
|
{
|
|
QString flag;
|
|
QStringList rest;
|
|
for (int i = 0; i < args.size(); ++i) {
|
|
if (exportFlags().contains(args.at(i))) {
|
|
flag = args.at(i);
|
|
for (int j = i + 1; j < args.size(); ++j)
|
|
rest << args.at(j);
|
|
break;
|
|
}
|
|
}
|
|
const QString format = exportFlags().value(flag);
|
|
|
|
// --check-elements operates on an element file/directory, not a project.
|
|
if (format == "check") {
|
|
if (rest.isEmpty()) {
|
|
err << "Usage: qelectrotech --check-elements "
|
|
"<element.elmt | directory>\n";
|
|
return 2;
|
|
}
|
|
return checkElements(rest.at(0));
|
|
}
|
|
|
|
const QString input = rest.value(0);
|
|
if (input.isEmpty()) {
|
|
err << "Usage: qelectrotech " << flag << " <project.qet> <output>\n";
|
|
return 2;
|
|
}
|
|
if (!QFileInfo::exists(input)) {
|
|
err << "Project not found: " << input << "\n";
|
|
return 2;
|
|
}
|
|
|
|
QETProject project(input);
|
|
if (project.state() != QETProject::Ok) {
|
|
err << "Failed to open project: " << input
|
|
<< " (state " << project.state() << ")\n";
|
|
return 1;
|
|
}
|
|
|
|
// --info writes JSON to stdout, or to an optional output file.
|
|
if (format == "info")
|
|
return exportInfo(project, rest.value(1));
|
|
|
|
const QString output = rest.value(1);
|
|
if (output.isEmpty()) {
|
|
err << "Usage: qelectrotech " << flag
|
|
<< " <project.qet> <output>\n";
|
|
return 2;
|
|
}
|
|
if (format == "pdf")
|
|
return exportPdf(project, output);
|
|
if (format == "cables" || format == "wires")
|
|
return exportCsv(project, format, output);
|
|
if (format == "bom")
|
|
return exportBom(project, output);
|
|
return exportImages(project, format, output);
|
|
}
|
|
|
|
} // namespace CLIExport
|