65 if (shr_ancestor == ancestor)
82 if (shr_descendant == descendant)
84 return shr_descendant;
100 auto shr_this = descendant->findAncestor(
this);
101 if(!shr_this.isNull())
return shr_this;
106 auto shr_this = ancestor->findDescendant(
this);
107 if(!shr_this.isNull())
return shr_this;
115 qRegisterMetaType<QAPropertyMap>();
116 qRegisterMetaType<QAPropagationRules>();
117 qRegisterMetaType<QAShrAlgorithm>();
123 for(
const QString& PropName: parameters.keys())
126 for(
int k = 0; k < metaObject()->propertyCount(); ++k)
128 if(metaObject()->property(k).name() ==
QA_PAR + PropName ||
129 metaObject()->property(k).name() ==
QA_IN + PropName)
133 if (!setProperty(metaObject()->property(k).name(), parameters.value(PropName)))
135 qWarning() <<
"Cannot set parameter/input" << PropName;
140 if(!set) qWarning() <<
"Trying to set" << PropName <<
"but it is not among object's properties";
147 setAutoDelete(
false);
156 if(!shr_this.isNull())
159 foreach(
auto ancestor,
getAncestors().keys()) ancestor->descendants[shr_this] =
true;
163 descendant->ancestors[shr_this] =
true;
164 descendant->getInput(shr_this);
165 if(!descendant->getKeepInput())
170 for(
int k = 0; k < metaObject()->propertyCount(); ++k)
172 if(QString(metaObject()->property(k).name()).startsWith(
QA_IN))
174 setProperty(metaObject()->property(k).name(), QVariant());
178 if(!descendant->isStarted())
180 if (getParallelExecution()) descendant->parallelExecution();
181 else descendant->serialExecution();
192 for(
int k = 0; k < parent->metaObject()->propertyCount(); ++k)
194 QString parentPropName = parent->metaObject()->property(k).name();
197 QString parentPropBaseName;
198 if(parentPropName.startsWith(
QA_OUT))
201 parentPropBaseName = parentPropName.mid(strlen(
QA_OUT));
203 else if(parentPropName.startsWith(
QA_PAR))
206 parentPropBaseName = parentPropName.mid(strlen(
QA_PAR));
210 QString childPropBaseName;
211 if(!getPropagationRules().contains(parentPropBaseName))
213 childPropBaseName = parentPropBaseName;
218 auto values = getPropagationRules().values(parentPropBaseName);
219 if(values.size() > 1)
222 childPropBaseName = values.filter(parent->objectName()).first();
226 childPropBaseName = values.first();
230 if(parentPropName.startsWith(
QA_PAR) && !getPropagationRules().contains(parentPropBaseName))
235 for(
int i = 0; i < child->metaObject()->propertyCount(); ++i)
237 QString childPropName = child->metaObject()->property(i).name();
240 if(childPropName ==
QA_IN+childPropBaseName || childPropName ==
QA_PAR+childPropBaseName)
242 QVariant parentProp = parent->property(parentPropName.toStdString().c_str());
243 if(parentProp.isValid())
245 if(!child->setProperty(childPropName.toStdString().c_str(), parentProp))
247 qWarning() <<
"getInput():" << childPropName <<
"failed to set for" << child->printName();
253 qWarning() <<
"getInput():" << parentPropName <<
"failed to read for" << parent->printName();
278 if(!ancestor->isStarted()) ancestor->parallelExecution();
292 if(!ancestor->isStarted()) ancestor->serialExecution();
296 setParallelExecution(
false);
305 Q_EMIT
raise(message);
314 dotFile.setFileName(QDir::home().absoluteFilePath(
"QAlgorithmTree.gv"));
315 outFileName = QDir::home().absoluteFilePath(
"QAlgorithmTree.svg");
319 dotFile.setFileName(path);
320 QStringList svgpath = path.split(
".");
321 svgpath.removeLast();
322 outFileName = svgpath.join(
".")+
".svg";
324 if (!dotFile.open(QFile::WriteOnly)) Q_EMIT
raise(
"Cannot write graph to given file");
327 QTextStream dot(&dotFile);
329 nospace.setNumberOptions(QLocale::NumberOption::OmitGroupSeparator);
330 dot <<
"digraph g{\n";
333 foreach(
auto alg, flatMap.keys())
335 QString
id = nospace.toString(quint64(alg.data()));
336 QString idspace = QLocale().toString(quint64(alg.data()));
337 QString name = alg->metaObject()->className();
338 QString nickname = alg->objectName();
339 QString dotstring =
"var"+
id.replace(
" ",
"")+
"[label=\""+name+
"\\nID "+idspace;
340 if(!nickname.isEmpty()) dotstring +=
"\\nNick: "+nickname;
341 dotstring +=
"\"];\n";
345 foreach(
auto parent, flatMap.keys())
347 QString parentName =
"var" + nospace.toString(quint64(parent.data()));
348 foreach(
auto child, flatMap[parent])
350 QString childName =
"var" + nospace.toString(quint64(child.data()));
351 dot << parentName <<
" -> " << childName <<
"\n";
357 QStringList cmdArgs =
358 {dotFile.fileName(),
"-Tsvg",
"-o", outFileName}
360 int r = QProcess::execute(
"/usr/local/bin/circo", cmdArgs);
363 qWarning(
"%s",
"The dot process crashed");
367 qWarning(
"%s",
"Cannot start the dot process");
371 r = QProcess::execute(
"/bin/rm",
376 qWarning(
"%s",
"The rm process crashed");
380 qWarning(
"%s",
"Cannot start the rm process");
388 return std::pair<QString, QVariant>({
"PropagationRules", QVariant::fromValue(
QAPropagationRules(pairs))});
394 msg += QString(metaObject()->className()) +
" ";
395 msg += QLocale().toString(qlonglong(
this)) +
" ";
396 if (!objectName().isEmpty()) msg += objectName();
403 auto keysList = tree.keys();
405 auto shr_this_it = std::find_if(keysList.begin(), keysList.end(), [
this](
auto& ptr)
406 {
return ptr ==
this;}
408 if (shr_this_it == keysList.end())
412 if(shr_this.isNull()) qWarning() <<
"This instance has no properly set connection, flattenTree will not work";
416 tree[shr_this] = QSet<QAShrAlgorithm>();
418 for (
auto descendant:
getDescendants().keys()) tree[shr_this] << descendant;
422 if (!tree.contains(relative)) tree = relative->flattenTree(tree);
426 else qWarning() <<
"flattenTree: possible loop";
432 ancestor->descendants[descendant] = descendant->isFinished();
433 descendant->ancestors[ancestor] = ancestor->isFinished();
440 ancestor->descendants.remove(descendant);
441 descendant->ancestors.remove(ancestor);
448 return ancestor->getDescendants().contains(descendant) && descendant->getAncestors().contains(ancestor);
455 return (p2->getDescendants().count() == 1) && (p1->getAncestors().count() == 1);
459 return (p1->getDescendants().count() == 1) && (p2->getAncestors().count() == 1);
472 ancestor >> descendant;
478 QDebugStateSaver saver(debug);
479 const QMetaObject* obj = c.metaObject();
482 debug << endl <<
"------------------------------" << c.
printName() <<
"subclass of QAlgorithm" << endl;
485 debug <<
"Algorithm with input:" << endl;
486 for(
int k = 0; k < obj->propertyCount(); k++)
488 QMetaProperty prop = obj->property(k);
490 QString propName = prop.name();
491 if(propName.startsWith(
QA_IN))
493 propName.remove(
QA_IN);
494 debug << propName.rightJustified(30,
' ',
true) <<
"\t" << prop.read(&c) << endl;
499 debug <<
"Algorithm with parameters:" << endl;
500 for(
int k = 0; k < obj->propertyCount(); k++)
502 QMetaProperty prop = obj->property(k);
504 QString propName = prop.name();
505 if(propName.startsWith(
QA_PAR))
508 debug << propName.rightJustified(30,
' ',
true) <<
"\t" << prop.read(&c) << endl;
513 debug <<
"Algorithm with output:" << endl;
514 for(
int k = 0; k < obj->propertyCount(); k++)
516 QMetaProperty prop = obj->property(k);
518 QString propName = prop.name();
519 if(propName.startsWith(
QA_OUT))
521 propName = propName.remove(
QA_OUT);
522 debug << propName.rightJustified(30,
' ',
true) <<
"\t" << prop.read(&c) << endl;
526 debug <<
"------------------------------" << endl;
533 auto _printTree = [](decltype(tree) map)
535 foreach(
auto key, map.keys())
537 qInfo() <<
"key" << key->printName();
538 foreach(
auto value, map[key])
540 qInfo() <<
"\tvalue" << value->printName();
546 else _printTree(tree);
552 QMap<QAShrAlgorithm, QList<QAShrAlgorithm>> replacements;
553 foreach (
auto ptr, flatMap.uniqueKeys())
555 foreach(
auto child, flatMap[ptr])
560 replacements[ptr] << child;
565 bool some_changes =
true;
568 some_changes =
false;
569 foreach(
auto p1, replacements.keys())
572 auto p2 = replacements[p1].last();
574 if (replacements.contains(p2))
576 replacements[p1] += replacements.take(p2);
583 foreach(QList<QAShrAlgorithm> nodes, replacements.values())
587 foreach(
auto node, nodes)
589 node->setParallelExecution(
false);
597 for(
int k = 0; k < c.metaObject()->propertyCount(); k++)
599 QMetaProperty prop = c.metaObject()->property(k);
600 QString propName = prop.name();
601 QVariant propValue = prop.read(&c);
602 if(propValue.isValid() &&
603 (propName.startsWith(
QA_IN) || propName.startsWith(
QA_OUT) || propName.startsWith(
QA_PAR)))
604 properties.insert(propName, propValue);
606 return (stream << properties);
612 stream >> properties;
613 for(
int k = 0; k < c.metaObject()->propertyCount(); k++)
615 QMetaProperty prop = c.metaObject()->property(k);
616 QString propName = prop.name();
617 if(properties.contains(propName))
619 if (!prop.write(&c, properties.values(propName)))
621 qWarning() << c.
printName() <<
"Unable to write property value, report to the QAlgorithm developer";
Q_SLOT void serialExecution()
Start computing the algorithm tree on the calling thread.
Q_SIGNAL void raise(QString message) const
Signal emitted whenever an error occurs.
bool finished
Whether the algorithm finished to run and outputs are ready.
Q_SIGNAL void justStarted()
Signal emitted on algorithm's start.
QAShrAlgorithm findDescendant(const QAlgorithm *descendant) const
Find an descendant.
Q_SLOT void parallelExecution()
Start computing the algorithm tree on different threads.
Q_SLOT void abort(QString message="Unknown Error") const
Emit the given error signal.
bool started
Whether the algorithm started to run.
#define QA_OUT
Prefix for output properties.
static std::pair< QString, QVariant > makePropagationRules(std::initializer_list< std::pair< QString, QString >> lst)
Convenience method for writing PropagationRules.
QAShrAlgorithm findAncestor(const QAlgorithm *ancestor) const
Find an ancestor.
static void closeConnection(QAShrAlgorithm ancestor, QAShrAlgorithm descendant)
Disconnect two algorithms.
virtual void setParameters(const QAPropertyMap ¶meters)
Set parameters for the algorithm.
#define QA_PAR
Prefix for parameters.
QACompletionMap getAncestors() const
Get the value of ancestors.
QAShrAlgorithm operator>>(QAShrAlgorithm ancestor, QAShrAlgorithm descendant)
Creates connections like setConnection().
static bool checkConnection(const QAShrAlgorithm ancestor, const QAShrAlgorithm descendant)
Check if two algorithms are connected.
virtual bool getInput(QAShrAlgorithm parent)
Load inputs from parent's outputs.
QACompletionMap descendants
Map with descendants and their completion.
virtual void setup()
Set of instructions to set up the algorithm.
QAFlatRepresentation flattenTree(QAFlatRepresentation tree=QAFlatRepresentation()) const
Creates a flat representation of the algorithm tree.
QFutureWatcher< void > watcher
QMap< QAShrAlgorithm, QSet< QAShrAlgorithm > > QAFlatRepresentation
bool isStarted() const
Get the value of started.
QAlgorithm(QObject *parent=Q_NULLPTR)
Constructor.
virtual void run()=0
Core part of the algorithm, to be reimplemented in subclasses.
QAShrAlgorithm operator<<(QAShrAlgorithm descendant, QAShrAlgorithm ancestor)
Creates connections like setConnection().
bool isFinished() const
Get the value of finished.
static void improveTree(QAlgorithm *leaf)
Replace all removable connections, thus improving the tree performance.
static quint32 print_counter
QSharedPointer< QAlgorithm > QAShrAlgorithm
Abstract class that implements a generic algorithm.
void setFinished()
Set the algorithm as comleted and signals it.
static void setConnection(QAShrAlgorithm ancestor, QAShrAlgorithm descendant)
Connect two algorithms.
Q_SIGNAL void justFinished()
Signal emitted on algorithm's end.
QMap< QString, QVariant > QAPropertyMap
#define QA_IN
Prefix for input properties.
void printGraph(const QString &path=QString()) const
Create a GraphViz diagram of the algorithm tree.
QString printName() const
Returns name, memory address and class name of the algorithm.
bool allInputsReady() const
Checks if the algorithm is ready to run.
QMap< QAShrAlgorithm, bool > QACompletionMap
QACompletionMap getDescendants() const
Get the value of descendants.
QMultiMap< QString, QString > QAPropagationRules
void printTree(const QAFlatRepresentation &tree=QAFlatRepresentation()) const
Outputs a text representation of the algorithm tree.
void setStarted()
Set the algorithm as started and signals it.
QAShrAlgorithm findSharedThis() const
Find a shared pointer to this instance.
static bool isRemovableConnection(const QAShrAlgorithm p1, const QAShrAlgorithm p2)
Check if two algorithms are connected and the connection is removable.
QACompletionMap ancestors
Map with ancestors and their completion.
Q_SLOT void propagateExecution()
Execute descendants.