/*************************************************************************** * This file is part of the Lime Report project * * Copyright (C) 2021 by Alexander Arin * * arin_a@bk.ru * * * ** GNU General Public License Usage ** * * * This library 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 3 of the License, or * * (at your option) any later version. * * You should have received a copy of the GNU General Public License * * along with this program. If not, see . * * * ** GNU Lesser General Public License ** * * * This library is free software: you can redistribute it and/or modify * * it under the terms of the GNU Lesser General Public License as * * published by the Free Software Foundation, either version 3 of the * * License, or (at your option) any later version. * * You should have received a copy of the GNU Lesser General Public * * License along with this library. * * If not, see . * * * * This library 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. * ****************************************************************************/ #include "lrdatasourcemanager.h" #include "lrdatadesignintf.h" #include #include #if QT_VERSION < 0x060000 #include #endif #include #include #include #include #ifdef BUILD_WITH_EASY_PROFILER #include "easy/profiler.h" #else # define EASY_BLOCK(...) # define EASY_END_BLOCK #endif namespace LimeReport{ DataNode::~DataNode() { for (int i = 0; im_childs.indexOf(const_cast(this)); } return 0; } void DataNode::clear() { for (int i=0; ichild(row); if (childNode){ return createIndex(row,column,childNode); } else return QModelIndex(); } QModelIndex DataSourceModel::parent(const QModelIndex& child) const { if (!child.isValid()) return QModelIndex(); DataNode* childNode = nodeFromIndex(child); if (!childNode) return QModelIndex(); DataNode* parentNode = childNode->parent(); if ((parentNode == m_rootNode) || (!parentNode)) return QModelIndex(); return createIndex(parentNode->row(),0,parentNode); } int DataSourceModel::rowCount(const QModelIndex& parent) const { if (!m_rootNode) return 0; DataNode* parentNode; if (parent.isValid()) parentNode = nodeFromIndex(parent); else parentNode = m_rootNode; return parentNode->childCount(); } int DataSourceModel::columnCount(const QModelIndex& parent) const { Q_UNUSED(parent) return 1; } QVariant DataSourceModel::data(const QModelIndex& index, int role) const { DataNode *node = nodeFromIndex(index); switch (role) { case Qt::DisplayRole: if (!node) return QVariant(); return node->name(); break; case Qt::DecorationRole : if (!node) return QIcon(); return node->icon(); break; default: return QVariant(); } } void DataSourceModel::setDataSourceManager(DataSourceManager* dataManager) { m_dataManager = dataManager; connect(m_dataManager,SIGNAL(datasourcesChanged()),this,SLOT(slotDatasourcesChanged())); updateModel(); } void DataSourceModel::slotDatasourcesChanged() { beginResetModel(); updateModel(); endResetModel(); } DataNode* DataSourceModel::nodeFromIndex(const QModelIndex& index) const { if (index.isValid()){ return static_cast(index.internalPointer()); } else return m_rootNode; } void DataSourceModel::fillFields(DataNode* parent) { foreach(QString name, m_dataManager->fieldNames(parent->name())){ parent->addChild(name, DataNode::Field,QIcon(":/report/images/field")); } } void DataSourceModel::updateModel() { QMap connections; m_rootNode->clear(); DataNode* ds = m_rootNode->addChild(tr("Datasources"),DataNode::DataSources,QIcon(":/report/images/databases")); foreach (QString name,m_dataManager->connectionNames()) { DataNode* connection = ds->addChild(name,DataNode::Connection,QIcon(":/report/images/database")); connections.insert(name,connection); } foreach (QString name, m_dataManager->dataSourceNames()) { DataNode* datasource; if (m_dataManager->isQuery(name)){ DataNode* connection = connections.value(m_dataManager->queryByName(name)->connectionName()); if (connection) datasource=connection->addChild(name,DataNode::Query,QIcon(":/report/images/table")); else datasource=ds->addChild(name,DataNode::Query); } else if (m_dataManager->isSubQuery(name)){ DataNode* connection = connections.value(m_dataManager->subQueryByName(name)->connectionName()); if (connection) datasource=connection->addChild(name,DataNode::SubQuery,QIcon(":/report/images/table")); else datasource=ds->addChild(name,DataNode::SubQuery); } else { datasource=ds->addChild(name,DataNode::Model,QIcon(":/report/images/table")); } fillFields(datasource); } DataNode* vars = m_rootNode->addChild(tr("Variables"),DataNode::Variables,QIcon(":/report/images/folder")); foreach (QString name, m_dataManager->variableNames()){ vars->addChild(name,DataNode::Variable,QIcon(":/report/images/value")); } vars = m_rootNode->addChild(tr("External variables"),DataNode::Variables,QIcon(":/report/images/folder")); foreach (QString name, m_dataManager->userVariableNames()){ vars->addChild(name,DataNode::Variable,QIcon(":/report/images/value")); } } DataSourceManager::DataSourceManager(QObject *parent) : QObject(parent), m_lastError(""), m_designTime(false), m_needUpdate(false), m_dbCredentialsProvider(0), m_hasChanges(false) { m_groupFunctionFactory.registerFunctionCreator(QLatin1String("COUNT"),new ConstructorGroupFunctionCreator); m_groupFunctionFactory.registerFunctionCreator(QLatin1String("SUM"),new ConstructorGroupFunctionCreator); m_groupFunctionFactory.registerFunctionCreator(QLatin1String("AVG"),new ConstructorGroupFunctionCreator); m_groupFunctionFactory.registerFunctionCreator(QLatin1String("MIN"),new ConstructorGroupFunctionCreator); m_groupFunctionFactory.registerFunctionCreator(QLatin1String("MAX"),new ConstructorGroupFunctionCreator); setSystemVariable(QLatin1String("#PAGE"),1,SecondPass); setSystemVariable(QLatin1String("#PAGE_COUNT"),0,SecondPass); setSystemVariable(QLatin1String("#IS_LAST_PAGEFOOTER"),false,FirstPass); setSystemVariable(QLatin1String("#IS_FIRST_PAGEFOOTER"),false,FirstPass); m_datasourcesModel.setDataSourceManager(this); connect(&m_reportVariables, SIGNAL(variableHasBeenAdded(QString)), this, SLOT(slotVariableHasBeenAdded(QString))); connect(&m_reportVariables, SIGNAL(variableHasBeenChanged(QString)), this, SLOT(slotVariableHasBeenChanged(QString))); connect(&m_userVariables, SIGNAL(variableHasBeenAdded(QString)), this, SLOT(slotVariableHasBeenAdded(QString))); connect(&m_userVariables, SIGNAL(variableHasBeenChanged(QString)), this, SLOT(slotVariableHasBeenChanged(QString))); } QString DataSourceManager::defaultDatabasePath() const { return m_defaultDatabasePath; } void DataSourceManager::setDefaultDatabasePath(const QString &defaultDatabasePath) { m_defaultDatabasePath = defaultDatabasePath; } QString DataSourceManager::putGroupFunctionsExpressions(QString expression) { if (m_groupFunctionsExpressionsMap.contains(expression)){ return QString::number(m_groupFunctionsExpressionsMap.value(expression)); } else { m_groupFunctionsExpressions.append(expression); m_groupFunctionsExpressionsMap.insert(expression, m_groupFunctionsExpressions.size()-1); return QString::number(m_groupFunctionsExpressions.size()-1); } } void DataSourceManager::clearGroupFuntionsExpressions() { m_groupFunctionsExpressionsMap.clear(); m_groupFunctionsExpressions.clear(); } QString DataSourceManager::getExpression(QString index) { bool ok = false; int i = index.toInt(&ok); if (ok && m_groupFunctionsExpressions.size()>i) return m_groupFunctionsExpressions.at(index.toInt()); else return ""; } bool DataSourceManager::designTime() const { return m_designTime; } void DataSourceManager::setDesignTime(bool designTime) { m_designTime = designTime; } DataSourceManager::~DataSourceManager() { clear(All); clearGroupFunction(); } void DataSourceManager::connectAllDatabases() { foreach(ConnectionDesc* conn,m_connections){ try{ connectConnection(conn); } catch (ReportError &e){ putError(e.what()); setLastError(e.what()); qDebug()<(ds),true); putHolder(name,holder); emit datasourcesChanged(); m_needUpdate = true; return ds; } void DataSourceManager::registerDbCredentialsProvider(IDbCredentialsProvider *provider) { m_dbCredentialsProvider = provider; } void DataSourceManager::addCallbackDatasource(ICallbackDatasource *datasource, const QString& name) { IDataSource* datasourceIntf = dynamic_cast(datasource); if (datasourceIntf){ IDataSourceHolder* holder = new CallbackDatasourceHolder(datasourceIntf,true); putHolder(name,holder); emit datasourcesChanged(); } } QSharedPointerDataSourceManager::previewSQL(const QString &connectionName, const QString &sqlText, QString masterDatasource) { QSqlDatabase db = QSqlDatabase::database(connectionName); if (db.isValid() && db.isOpen()){ QSqlQueryModel* model = new QSqlQueryModel(); QMap aliasesToParam; QString queryText = replaceVariables(sqlText,aliasesToParam); queryText = replaceFields(queryText,aliasesToParam,masterDatasource); QSqlQuery query(db); query.prepare(queryText); foreach(QString param,aliasesToParam.keys()){ QVariant value; if (param.contains(".")){ value = fieldData(aliasesToParam.value(param)); param=param.right(param.length()-param.indexOf('.')-1); } else { value = variable(aliasesToParam.value(param)); } if (value.isValid() || m_designTime) query.bindValue(':'+param,value); } query.exec(); model->setQuery(query); m_lastError = model->lastError().text(); putError(m_lastError); if (model->query().isActive()) return QSharedPointer(model); else return QSharedPointer(0); } if (!db.isOpen()){ m_lastError = tr("Connection \"%1\" is not open").arg(connectionName); putError(m_lastError); } return QSharedPointer(0); } void DataSourceManager::updateDatasourceModel() { m_datasourcesModel.updateModel(); emit datasourcesChanged(); m_needUpdate = false; } QString DataSourceManager::extractField(QString source) { if (source.contains('.')) { return source.right(source.length()-(source.indexOf('.')+1)); } return source; } QString DataSourceManager::replaceVariables(QString value){ #if QT_VERSION >= QT_VERSION_CHECK(5, 12, 3) QRegularExpression rx(Const::VARIABLE_RX); QRegularExpressionMatchIterator iter = rx.globalMatch(value); qsizetype pos = 0; QString result; while (iter.hasNext()) { QRegularExpressionMatch match = iter.next(); QString var = match.captured(0); var.remove("$V{"); var.remove("}"); result += value.mid(pos, match.capturedStart(0)); if (variable(var).isValid()){ result += variable(var).toString(); } else { result += QString(tr("Variable \"%1\" not found!").arg(var)); } pos = match.capturedEnd(0); } result += value.mid(pos); return result; // TODO: Qt6 port - done #else QRegExp rx(Const::VARIABLE_RX); if (value.contains(rx)){ int pos = -1; while ((pos=rx.indexIn(value))!=-1){ QString var=rx.cap(0); var.remove("$V{"); var.remove("}"); if (variable(var).isValid()){ value.replace(pos,rx.cap(0).length(),variable(var).toString()); } else { value.replace(pos,rx.cap(0).length(),QString(tr("Variable \"%1\" not found!").arg(var))); } } } return value; #endif return QString(); } QString DataSourceManager::replaceVariables(QString query, QMap &aliasesToParam) { #if QT_VERSION >= QT_VERSION_CHECK(5, 12, 3) QRegularExpression rx(Const::VARIABLE_RX); int curentAliasIndex = 0; if (query.contains(rx)){ int pos = -1; QRegularExpressionMatch match = rx.match(query); while ((pos=match.capturedStart())!=-1){ QString var=match.captured(0); var.remove("$V{"); var.remove("}"); if (!match.captured(1).isEmpty()){ if (aliasesToParam.contains(var)){ curentAliasIndex++; aliasesToParam.insert(var+"_v_alias"+QString::number(curentAliasIndex),var); var += "_v_alias"+QString::number(curentAliasIndex); } else { aliasesToParam.insert(var,var); } query.replace(pos,match.captured(0).length(),":"+var); } else { QString varName = match.captured(2).trimmed(); QString varParam = match.captured(3).trimmed(); if (!varName.isEmpty()){ if (!varParam.isEmpty() && varParam.compare("nobind") == 0 ){ query.replace(pos,match.captured(0).length(), variable(varName).toString()); } else { query.replace(pos,match.captured(0).length(), QString(tr("Unknown parameter \"%1\" for variable \"%2\" found!") .arg(varName) .arg(varParam)) ); } } else { query.replace(pos,match.captured(0).length(),QString(tr("Variable \"%1\" not found!").arg(var))); } } match = rx.match(query); } } // TODO: Qt6 port - done #else QRegExp rx(Const::VARIABLE_RX); int curentAliasIndex = 0; if (query.contains(rx)){ int pos = -1; while ((pos=rx.indexIn(query))!=-1){ QString var=rx.cap(0); var.remove("$V{"); var.remove("}"); if (!rx.cap(1).isEmpty()){ if (aliasesToParam.contains(var)){ curentAliasIndex++; aliasesToParam.insert(var+"_v_alias"+QString::number(curentAliasIndex),var); var += "_v_alias"+QString::number(curentAliasIndex); } else { aliasesToParam.insert(var,var); } query.replace(pos,rx.cap(0).length(),":"+var); } else { QString varName = rx.cap(2).trimmed(); QString varParam = rx.cap(3).trimmed(); if (!varName.isEmpty()){ if (!varParam.isEmpty() && varParam.compare("nobind") == 0 ){ query.replace(pos,rx.cap(0).length(), variable(varName).toString()); } else { query.replace(pos,rx.cap(0).length(), QString(tr("Unknown parameter \"%1\" for variable \"%2\" found!") .arg(varName) .arg(varParam)) ); } } else { query.replace(pos,rx.cap(0).length(),QString(tr("Variable \"%1\" not found!").arg(var))); } } } } #endif return query; } QString DataSourceManager::replaceFields(QString query, QMap &aliasesToParam, QString masterDatasource) { #if QT_VERSION >= QT_VERSION_CHECK(5, 12, 3) QRegularExpression rx(Const::FIELD_RX); int curentAliasIndex = 0; if (query.contains(rx)){ int pos = -1; QRegularExpressionMatch match = rx.match(query); while ((pos=match.capturedStart())!=-1){ QString field=match.captured(0); field.remove("$D{"); field.remove("}"); if (!aliasesToParam.contains(field)){ if (field.contains(".")) aliasesToParam.insert(field, field); else aliasesToParam.insert(field, masterDatasource+"."+field); } else { curentAliasIndex++; if (field.contains(".")) aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex), field); else aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex), masterDatasource+"."+field); field+="_f_alias"+QString::number(curentAliasIndex); } query.replace(pos,match.capturedLength(),":"+extractField(field)); match = rx.match(query); } } // TODO: Qt6 port - possible done #else QRegExp rx(Const::FIELD_RX); if (query.contains(rx)){ int curentAliasIndex=0; int pos; while ((pos=rx.indexIn(query))!=-1){ QString field=rx.cap(0); field.remove("$D{"); field.remove("}"); if (!aliasesToParam.contains(field)){ if (field.contains(".")) aliasesToParam.insert(field,field); else aliasesToParam.insert(field,masterDatasource+"."+field); } else { curentAliasIndex++; if (field.contains(".")) aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex),field); else aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex),masterDatasource+"."+field); field+="_f_alias"+QString::number(curentAliasIndex); } query.replace(pos,rx.cap(0).length(),":"+extractField(field)); } } return query; #endif return query; } void DataSourceManager::setReportVariable(const QString &name, const QVariant &value) { if (!containsVariable(name)){ addVariable(name,value); } else changeVariable(name,value); } void DataSourceManager::addQuery(const QString &name, const QString &sqlText, const QString &connectionName) { QueryDesc *queryDecs = new QueryDesc(name,sqlText,connectionName); putQueryDesc(queryDecs); putHolder(name,new QueryHolder(sqlText, connectionName, this)); m_hasChanges = true; m_varToDataSource.clear(); emit datasourcesChanged(); } void DataSourceManager::addSubQuery(const QString &name, const QString &sqlText, const QString &connectionName, const QString &masterDatasource) { SubQueryDesc *subQueryDesc = new SubQueryDesc(name.toLower(),sqlText,connectionName,masterDatasource); putSubQueryDesc(subQueryDesc); putHolder(name,new SubQueryHolder(sqlText, connectionName, masterDatasource, this)); m_hasChanges = true; m_varToDataSource.clear(); emit datasourcesChanged(); } void DataSourceManager::addProxy(const QString &name, const QString &master, const QString &detail, QList fields) { ProxyDesc *proxyDesc = new ProxyDesc(); proxyDesc->setName(name); proxyDesc->setMaster(master); proxyDesc->setDetail(detail); foreach(FieldsCorrelation correlation,fields){ proxyDesc->addFieldsCorrelation(correlation); } putProxyDesc(proxyDesc); putHolder(name,new ProxyHolder(proxyDesc, this)); m_hasChanges = true; emit datasourcesChanged(); } void DataSourceManager::addCSV(const QString& name, const QString& csvText, const QString &separator, bool firstRowIsHeader) { CSVDesc* csvDesc = new CSVDesc(name, csvText, separator, firstRowIsHeader); putCSVDesc(csvDesc); putHolder(name, new CSVHolder(*csvDesc, this)); m_hasChanges = true; emit datasourcesChanged(); } QString DataSourceManager::queryText(const QString &dataSourceName) { if (isQuery(dataSourceName)) return queryByName(dataSourceName)->queryText(); if (isSubQuery(dataSourceName)) return subQueryByName(dataSourceName)->queryText(); else return QString(); } QueryDesc *DataSourceManager::queryByName(const QString &datasourceName) { int queryIndex = queryIndexByName(datasourceName); if (queryIndex!=-1) return m_queries.at(queryIndex); return 0; } SubQueryDesc* DataSourceManager::subQueryByName(const QString &datasourceName) { int queryIndex = subQueryIndexByName(datasourceName); if (queryIndex!=-1) return m_subqueries.at(queryIndex); return 0; } ConnectionDesc* DataSourceManager::connectionByName(const QString &connectionName) { int queryIndex = connectionIndexByName(connectionName); if (queryIndex!=-1) return m_connections.at(queryIndex); return 0; } int DataSourceManager::queryIndexByName(const QString &dataSourceName) { for(int i=0;iqueryName().compare(dataSourceName,Qt::CaseInsensitive)==0) return i; } return -1; } int DataSourceManager::subQueryIndexByName(const QString &dataSourceName) { for(int i=0;iqueryName().compare(dataSourceName,Qt::CaseInsensitive)==0) return i; } return -1; } int DataSourceManager::proxyIndexByName(const QString &dataSourceName) { for(int i=0;iname().compare(dataSourceName,Qt::CaseInsensitive)==0) return i; } return -1; } int DataSourceManager::csvIndexByName(const QString &dataSourceName) { for(int i=0; i < m_csvs.count();++i){ CSVDesc* desc=m_csvs.at(i); if (desc->name().compare(dataSourceName,Qt::CaseInsensitive)==0) return i; } return -1; } int DataSourceManager::connectionIndexByName(const QString &connectionName) { for(int i=0;iname().compare(connectionName,Qt::CaseInsensitive)==0) return i; } return -1; } QList& DataSourceManager::conections() { return m_connections; } bool DataSourceManager::dataSourceIsValid(const QString &name) { if (m_datasources.value(name.toLower())) return !m_datasources.value(name.toLower())->isInvalid(); else throw ReportError(tr("Datasource \"%1\" not found!").arg(name)); } bool DataSourceManager::isQuery(const QString &dataSourceName) { return (queryByName(dataSourceName)); } QString DataSourceManager::connectionName(const QString &dataSourceName) { if (isQuery(dataSourceName)) return queryByName(dataSourceName)->connectionName(); if (isSubQuery(dataSourceName)) return subQueryByName(dataSourceName)->connectionName(); return QString(); } ProxyDesc *DataSourceManager::proxyByName(const QString &datasourceName) { int proxyIndex = proxyIndexByName(datasourceName); if (proxyIndex > -1) return m_proxies.at(proxyIndex); else return 0; } CSVDesc *DataSourceManager::csvByName(const QString &datasourceName) { int csvIndex = csvIndexByName(datasourceName); if (csvIndex > -1) return m_csvs.at(csvIndex); else return 0; } void DataSourceManager::removeDatasource(const QString &name) { if (m_datasources.contains(name)){ IDataSourceHolder *holder; holder=m_datasources.value(name); m_datasources.remove(name); delete holder; } if (isQuery(name)){ int queryIndex=queryIndexByName(name); delete m_queries.at(queryIndex); m_queries.removeAt(queryIndex); } if (isSubQuery(name)){ int queryIndex=subQueryIndexByName(name); delete m_subqueries.at(queryIndex); m_subqueries.removeAt(queryIndex); } if (isProxy(name)){ int proxyIndex=proxyIndexByName(name); delete m_proxies.at(proxyIndex); m_proxies.removeAt(proxyIndex); } if (isCSV(name)){ int csvIndex=csvIndexByName(name); delete m_csvs.at(csvIndex); m_csvs.removeAt(csvIndex); } invalidateLinkedDatasources(name); m_hasChanges = true; emit datasourcesChanged(); } void DataSourceManager::removeConnection(const QString &connectionName) { QList::iterator cit = m_connections.begin(); while( cit != m_connections.end() ){ if ( ((*cit)->name().compare(connectionName) == 0) ){ if ((*cit)->isInternal()) { QSqlDatabase db = QSqlDatabase::database(connectionName); db.close(); } QSqlDatabase::removeDatabase(connectionName); delete (*cit); cit = m_connections.erase(cit); } else { ++cit; } } m_hasChanges = true; emit datasourcesChanged(); } void DataSourceManager::addConnectionDesc(ConnectionDesc * connection) { if (!isConnection(connection->name())) { connect(connection,SIGNAL(nameChanged(QString,QString)),this,SLOT(slotConnectionRenamed(QString,QString))); m_connections.append(connection); m_hasChanges = true; if (connection->autoconnect()){ try{ connectConnection(connection); } catch(ReportError &exception) { qDebug()<name())); } } bool DataSourceManager::checkConnectionDesc(ConnectionDesc *connection) { if (connectConnection(connection)){ if (connection->isInternal()){ QSqlDatabase::removeDatabase(connection->name()); if (designTime()) emit datasourcesChanged(); } return true; } if (connection->isInternal()) QSqlDatabase::removeDatabase(connection->name()); return false; } void DataSourceManager::putHolder(const QString& name, IDataSourceHolder *dataSource) { if (!m_datasources.contains(name.toLower())){ m_datasources.insert( name.toLower(), dataSource ); } else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(name)); } void DataSourceManager::putQueryDesc(QueryDesc* queryDesc) { if (!containsDatasource(queryDesc->queryName())){ m_queries.append(queryDesc); connect(queryDesc, SIGNAL(queryTextChanged(QString,QString)), this, SLOT(slotQueryTextChanged(QString,QString))); } else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(queryDesc->queryName())); } void DataSourceManager::putSubQueryDesc(SubQueryDesc *subQueryDesc) { if (!containsDatasource(subQueryDesc->queryName())){ m_subqueries.append(subQueryDesc); connect(subQueryDesc, SIGNAL(queryTextChanged(QString,QString)), this, SLOT(slotQueryTextChanged(QString,QString))); } else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(subQueryDesc->queryName())); } void DataSourceManager::putProxyDesc(ProxyDesc *proxyDesc) { if (!containsDatasource(proxyDesc->name())){ m_proxies.append(proxyDesc); } else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(proxyDesc->name())); } void DataSourceManager::putCSVDesc(CSVDesc *csvDesc) { if (!containsDatasource(csvDesc->name())){ m_csvs.append(csvDesc); connect(csvDesc, SIGNAL(cvsTextChanged(QString, QString)), this, SLOT(slotCSVTextChanged(QString, QString))); } else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(csvDesc->name())); } bool DataSourceManager::initAndOpenDB(QSqlDatabase& db, ConnectionDesc& connectionDesc){ bool connected = false; db.setHostName(replaceVariables(connectionDesc.host())); db.setUserName(replaceVariables(connectionDesc.userName())); db.setPassword(replaceVariables(connectionDesc.password())); db.setDatabaseName(replaceVariables(connectionDesc.databaseName())); if (connectionDesc.port()!="") db.setPort(replaceVariables(connectionDesc.port()).toInt()); if (!connectionDesc.keepDBCredentials() && m_dbCredentialsProvider){ if (!m_dbCredentialsProvider->getUserName(connectionDesc.name()).isEmpty()) db.setUserName(m_dbCredentialsProvider->getUserName(connectionDesc.name())); if (!m_dbCredentialsProvider->getPassword(connectionDesc.name()).isEmpty()) db.setPassword(m_dbCredentialsProvider->getPassword(connectionDesc.name())); } QString dbName = replaceVariables(connectionDesc.databaseName()); if (connectionDesc.driver().compare("QSQLITE")==0){ if (!defaultDatabasePath().isEmpty()){ dbName = !QFileInfo(dbName).exists() ? defaultDatabasePath()+QFileInfo(dbName).fileName() : dbName; } if (QFileInfo(dbName).exists()){ db.setDatabaseName(dbName); } else { setLastError(tr("Database \"%1\" not found").arg(dbName)); return false; } } else { db.setDatabaseName(dbName); } connected=db.open(); if (!connected) setLastError(db.lastError().text()); return connected; } ReportSettings *DataSourceManager::reportSettings() const { return m_reportSettings; } void DataSourceManager::setReportSettings(ReportSettings *reportSettings) { m_reportSettings = reportSettings; } bool DataSourceManager::checkConnection(QSqlDatabase db){ QSqlQuery query("Select 1",db); return query.first(); } bool DataSourceManager::connectConnection(ConnectionDesc *connectionDesc) { bool connected = false; clearErrors(); QString lastError =""; foreach(QString datasourceName, dataSourceNames()){ dataSourceHolder(datasourceName)->clearErrors(); } if (!QSqlDatabase::contains(connectionDesc->name())){ QString dbError; { QSqlDatabase db = QSqlDatabase::addDatabase(connectionDesc->driver(),connectionDesc->name()); connectionDesc->setInternal(true); connected=initAndOpenDB(db, *connectionDesc); dbError = db.lastError().text(); } if (!connected){ if (!dbError.trimmed().isEmpty()) setLastError(dbError); QSqlDatabase::removeDatabase(connectionDesc->name()); return false; } } else { QSqlDatabase db = QSqlDatabase::database(connectionDesc->name()); if (!connectionDesc->isEqual(db) && connectionDesc->isInternal()){ db.close(); connected = initAndOpenDB(db, *connectionDesc); } else { connected = checkConnection(db); if (!connected && connectionDesc->isInternal()) connected = initAndOpenDB(db, *connectionDesc); } } if (!connected) { if (connectionDesc->isInternal()) QSqlDatabase::removeDatabase(connectionDesc->name()); return false; } else { foreach(QString datasourceName, dataSourceNames()){ if (isQuery(datasourceName)){ QueryHolder* qh = dynamic_cast(dataSourceHolder(datasourceName)); if (qh){ qh->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); invalidateChildren(datasourceName); } } } foreach(QString datasourceName, dataSourceNames()){ if (isProxy(datasourceName)){ ProxyHolder* ph = dynamic_cast(dataSourceHolder(datasourceName)); if (ph){ ph->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); } } } if (designTime()) emit datasourcesChanged(); } return true; } void DataSourceManager::clearReportVariables() { m_reportVariables.clearUserVariables(); } void DataSourceManager::connectAutoConnections() { foreach(ConnectionDesc* conn,m_connections){ if (conn->autoconnect()) { try { connectConnection(conn); } catch(ReportError &e){ setLastError(e.what()); putError(e.what()); qDebug()< DataSourceManager::childDatasources(const QString &parentDatasourceName) { QList result; foreach(QString datasourceName, dataSourceNames()){ if (isSubQuery(datasourceName)){ SubQueryHolder* sh = dynamic_cast(dataSourceHolder(datasourceName)); if (sh && sh->masterDatasource().compare(parentDatasourceName,Qt::CaseInsensitive)==0){ result.append(datasourceName); } } } return result; } void DataSourceManager::invalidateChildren(const QString &parentDatasourceName) { foreach(QString datasourceName, childDatasources(parentDatasourceName)){ SubQueryHolder* sh = dynamic_cast(dataSourceHolder(datasourceName)); if (sh) sh->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); invalidateChildren(datasourceName); } } bool DataSourceManager::containsDatasource(const QString &dataSourceName) { return m_datasources.contains(dataSourceName.toLower()); } bool DataSourceManager::isSubQuery(const QString &dataSourceName) { return subQueryIndexByName(dataSourceName) != -1; } bool DataSourceManager::isProxy(const QString &dataSourceName) { return proxyIndexByName(dataSourceName) != -1; } bool DataSourceManager::isCSV(const QString &datasourceName) { return csvIndexByName(datasourceName) != -1; } bool DataSourceManager::isConnection(const QString &connectionName) { return connectionIndexByName(connectionName) != -1; } bool DataSourceManager::isConnectionConnected(const QString &connectionName) { if (isConnection(connectionName) && QSqlDatabase::contains(connectionName)){ QSqlDatabase db = QSqlDatabase::database(connectionName); return db.isValid() && QSqlDatabase::database(connectionName).isOpen(); } return false; } bool DataSourceManager::connectConnection(const QString& connectionName) { return connectConnection(connectionByName(connectionName)); } void DataSourceManager::disconnectConnection(const QString& connectionName) { foreach(QString datasourceName, dataSourceNames()){ if (isQuery(datasourceName) || isSubQuery(datasourceName)){ QueryHolder* qh = dynamic_cast(dataSourceHolder(datasourceName)); if (qh && qh->connectionName().compare(connectionName, Qt::CaseInsensitive) == 0){ qh->invalidate(designTime() ? IDataSource::DESIGN_MODE : IDataSource::RENDER_MODE, true); qh->setLastError(tr("invalid connection")); } } } ConnectionDesc* connectionDesc = connectionByName(connectionName); if (connectionDesc->isInternal()){ { QSqlDatabase db = QSqlDatabase::database(connectionName); if (db.isOpen()) db.close(); } if (QSqlDatabase::contains(connectionName)) QSqlDatabase::removeDatabase(connectionName); } } IDataSource *DataSourceManager::dataSource(const QString &name) { IDataSourceHolder* holder = m_datasources.value(name.toLower()); if (holder) { if (holder->isInvalid()) { setLastError(name+" : "+holder->lastError()); return 0; } else { return holder->dataSource(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); } } else { setLastError(tr("Datasource \"%1\" not found!").arg(name)); return 0; } } IDataSourceHolder *DataSourceManager::dataSourceHolder(const QString &name) { if (m_datasources.value(name.toLower())) return m_datasources.value(name.toLower()); else return 0; } QStringList DataSourceManager::dataSourceNames() { QStringList result; for (int i=0;iconnectionName().compare(connectionName,Qt::CaseInsensitive)==0){ result.append(query->queryName()); } } foreach (QueryDesc* query, m_subqueries) { if (query->connectionName().compare(connectionName,Qt::CaseInsensitive)==0){ result.append(query->queryName()); } } return result; } QStringList DataSourceManager::connectionNames() { QStringList result; foreach(ConnectionDesc* conDesc,m_connections){ result.append(conDesc->name()); } return result; } QStringList DataSourceManager::fieldNames(const QString &datasourceName) { QStringList result; IDataSource* ds = dataSource(datasourceName); if (ds && !ds->isInvalid()){ for(int i=0; i < ds->columnCount(); i++){ result.append(ds->columnNameByIndex(i)); } result.sort(); } return result; } void DataSourceManager::addConnection(const QString &connectionName) { addConnectionDesc(new ConnectionDesc(QSqlDatabase::database(connectionName))); m_hasChanges = true; emit datasourcesChanged(); } QObject *DataSourceManager::createElement(const QString& collectionName, const QString&) { if (collectionName=="connections"){ ConnectionDesc* connection = new ConnectionDesc; m_connections.append(connection); return connection; } if (collectionName=="queries"){ QueryDesc* queryDesc = new QueryDesc; m_queries.append(queryDesc); return queryDesc; } if (collectionName=="subqueries"){ SubQueryDesc* subQueryDesc = new SubQueryDesc; m_subqueries.append(subQueryDesc); return subQueryDesc; } if (collectionName=="subproxies"){ ProxyDesc* proxyDesc = new ProxyDesc; m_proxies.append(proxyDesc); return proxyDesc; } if (collectionName=="variables"){ VarDesc* var = new VarDesc; m_tempVars.append(var); return var; } if (collectionName=="csvs"){ CSVDesc* csvDesc = new CSVDesc; m_csvs.append(csvDesc); return csvDesc; } return 0; } int DataSourceManager::elementsCount(const QString &collectionName) { if (collectionName=="connections"){ return m_connections.count(); } if (collectionName=="queries"){ return m_queries.count(); } if (collectionName=="subqueries"){ return m_subqueries.count(); } if (collectionName=="subproxies"){ return m_proxies.count(); } if (collectionName=="variables"){ return m_reportVariables.variablesCount(); } if (collectionName=="csvs"){ return m_csvs.count(); } return 0; } QObject* DataSourceManager::elementAt(const QString &collectionName, int index) { if (collectionName=="connections"){ return m_connections.at(index); } if (collectionName=="queries"){ return m_queries.at(index); } if (collectionName=="subqueries"){ return m_subqueries.at(index); } if (collectionName=="subproxies"){ return m_proxies.at(index); } if (collectionName=="variables"){ return m_reportVariables.variableAt(index); } if (collectionName=="csvs"){ return m_csvs.at(index); } return 0; } void DataSourceManager::collectionLoadFinished(const QString &collectionName) { EASY_BLOCK("DataSourceManager::collectionLoadFinished"); if (collectionName.compare("connections",Qt::CaseInsensitive) == 0){ } EASY_BLOCK("queryes"); if (collectionName.compare("queries",Qt::CaseInsensitive) == 0){ QMutableListIterator it(m_queries); while (it.hasNext()){ it.next(); if (!m_datasources.contains(it.value()->queryName().toLower())){ connect(it.value(), SIGNAL(queryTextChanged(QString,QString)), this, SLOT(slotQueryTextChanged(QString,QString))); putHolder(it.value()->queryName(),new QueryHolder(it.value()->queryText(), it.value()->connectionName(), this)); } else { delete it.value(); it.remove(); } } } EASY_END_BLOCK; EASY_BLOCK("subqueries") if (collectionName.compare("subqueries",Qt::CaseInsensitive) == 0){ QMutableListIterator it(m_subqueries); while (it.hasNext()){ it.next(); if (!m_datasources.contains(it.value()->queryName().toLower())){ connect(it.value(), SIGNAL(queryTextChanged(QString,QString)), this, SLOT(slotQueryTextChanged(QString,QString))); putHolder(it.value()->queryName(),new SubQueryHolder( it.value()->queryText(), it.value()->connectionName(), it.value()->master(), this) ); } else { delete it.value(); it.remove(); } } } EASY_END_BLOCK; EASY_BLOCK("subproxies"); if (collectionName.compare("subproxies",Qt::CaseInsensitive) == 0){ QMutableListIterator it(m_proxies); while (it.hasNext()){ it.next(); if (!m_datasources.contains(it.value()->name().toLower())){ putHolder(it.value()->name(),new ProxyHolder(it.value(), this)); } else { delete it.value(); it.remove(); } } } EASY_END_BLOCK; EASY_BLOCK("variables"); if (collectionName.compare("variables",Qt::CaseInsensitive) == 0){ foreach (VarDesc* item, m_tempVars) { if (!m_reportVariables.containsVariable(item->name())){ m_reportVariables.addVariable(item->name(),item->value(),VarDesc::Report,FirstPass); VarDesc* currentVar = m_reportVariables.variableByName(item->name()); currentVar->initFrom(item); } delete item; } m_tempVars.clear(); } EASY_END_BLOCK; if (collectionName.compare("csvs", Qt::CaseInsensitive) == 0){ QMutableListIterator it(m_csvs); while (it.hasNext()){ it.next(); if (!m_datasources.contains(it.value()->name().toLower())){ connect(it.value(), SIGNAL(cvsTextChanged(QString,QString)), this, SLOT(slotCSVTextChanged(QString,QString))); putHolder( it.value()->name(), new CSVHolder(*it.value(), this) ); } else { delete it.value(); it.remove(); } } } if (designTime()){ EASY_BLOCK("emit datasourcesChanged()"); emit datasourcesChanged(); EASY_END_BLOCK; } EASY_BLOCK("emit loadCollectionFinished(collectionName)"); emit loadCollectionFinished(collectionName); EASY_END_BLOCK; EASY_END_BLOCK; } void DataSourceManager::addVariable(const QString &name, const QVariant &value, VarDesc::VarType type, RenderPass pass) { if (type == VarDesc::User){ m_userVariables.addVariable(name, value, type, pass); } else { m_reportVariables.addVariable(name,value,type,pass); } if (designTime()){ EASY_BLOCK("DataSourceManager::addVariable emit ds changed"); emit datasourcesChanged(); EASY_END_BLOCK; } } void DataSourceManager::deleteVariable(const QString& name) { m_userVariables.deleteVariable(name); if (m_reportVariables.containsVariable(name)&&m_reportVariables.variableType(name)==VarDesc::Report){ m_reportVariables.deleteVariable(name); if (designTime()){ m_hasChanges = true; emit datasourcesChanged(); } } } void DataSourceManager::changeVariable(const QString& name,const QVariant& value) { if (m_userVariables.containsVariable(name)){ m_userVariables.changeVariable(name,value); } if (m_reportVariables.containsVariable(name)){ m_reportVariables.changeVariable(name,value); } } void DataSourceManager::setSystemVariable(const QString &name, const QVariant &value, RenderPass pass) { addVariable(name,value,VarDesc::System,pass); } void DataSourceManager::setLastError(const QString &value){ m_lastError = value; if (!value.isEmpty() && !m_errorsList.contains(value)) { m_errorsList.append(value); } } void DataSourceManager::invalidateLinkedDatasources(QString datasourceName) { foreach(QString name, dataSourceNames()){ if (isSubQuery(name)){ if (subQueryByName(name)->master().compare(datasourceName) == 0) dataSourceHolder(name)->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); } if (isProxy(name)){ ProxyDesc* proxy = proxyByName(name); if ((proxy->master().compare(datasourceName) == 0) || (proxy->child().compare(datasourceName) == 0)) dataSourceHolder(name)->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); } } } void DataSourceManager::slotConnectionRenamed(const QString &oldName, const QString &newName) { foreach(QueryDesc* query, m_queries){ if (query->connectionName().compare(oldName,Qt::CaseInsensitive)==0) query->setConnectionName(newName); } foreach(SubQueryDesc* query, m_subqueries){ if (query->connectionName().compare(oldName,Qt::CaseInsensitive)==0) query->setConnectionName(newName); } } void DataSourceManager::slotQueryTextChanged(const QString &queryName, const QString &queryText) { QueryHolder* holder = dynamic_cast(m_datasources.value(queryName)); if (holder){ holder->setQueryText(queryText); } m_varToDataSource.clear(); } void DataSourceManager::invalidateQueriesContainsVariable(const QString& variableName) { if (!variableIsSystem(variableName)){ if (m_varToDataSource.contains(variableName)){ foreach(QString datasourceName, m_varToDataSource.value(variableName)){ QueryHolder* holder = dynamic_cast(m_datasources.value(datasourceName)); if (holder) holder->invalidate(designTime() ? IDataSource::DESIGN_MODE : IDataSource::RENDER_MODE); } } else { QVector datasources; foreach (const QString& datasourceName, dataSourceNames()){ QueryHolder* holder = dynamic_cast(m_datasources.value(datasourceName)); if (holder){ #if QT_VERSION < 0x060000 QRegExp rx(QString(Const::NAMED_VARIABLE_RX).arg(variableName)); #else QRegularExpression rx(QString(Const::NAMED_VARIABLE_RX).arg(variableName)); #endif if (holder->queryText().contains(rx)){ holder->invalidate(designTime() ? IDataSource::DESIGN_MODE : IDataSource::RENDER_MODE); datasources.append(datasourceName); } } } m_varToDataSource.insert(variableName, datasources); } } } void DataSourceManager::slotVariableHasBeenAdded(const QString& variableName) { invalidateQueriesContainsVariable(variableName); if (variableType(variableName) == VarDesc::Report) m_hasChanges = true; } void DataSourceManager::slotVariableHasBeenChanged(const QString& variableName) { invalidateQueriesContainsVariable(variableName); if (variableType(variableName) == VarDesc::Report) m_hasChanges = true; } void DataSourceManager::slotCSVTextChanged(const QString &csvName, const QString &csvText) { CSVHolder* holder = dynamic_cast(m_datasources.value(csvName)); if (holder){ holder->setCSVText(csvText); } } void DataSourceManager::clear(ClearMethod method) { m_varToDataSource.clear(); DataSourcesMap::iterator dit; for( dit = m_datasources.begin(); dit != m_datasources.end(); ){ bool owned = (*dit)->isOwned() && (*dit)->isRemovable(); switch(method){ case All: invalidateLinkedDatasources(dit.key()); delete dit.value(); dit = m_datasources.erase(dit); break; default: if (owned){ invalidateLinkedDatasources(dit.key()); delete dit.value(); dit = m_datasources.erase(dit); } else { ++dit; } } } QList::iterator cit = m_connections.begin(); while( cit != m_connections.end() ){ if ( (*cit)->isInternal() ) QSqlDatabase::removeDatabase( (*cit)->name() ); delete (*cit); cit = m_connections.erase(cit); } //TODO: add smart pointes to collections foreach(QueryDesc *desc, m_queries) delete desc; foreach(SubQueryDesc* desc, m_subqueries) delete desc; foreach(ProxyDesc* desc, m_proxies) delete desc; m_queries.clear(); m_subqueries.clear(); m_proxies.clear(); // if (method == All) // clearUserVariables(); clearReportVariables(); emit cleared(); } void DataSourceManager::clearGroupFunction() { foreach(GroupFunction* gf ,m_groupFunctions.values()){ delete gf; } m_groupFunctions.clear(); } void DataSourceManager::clearGroupFunctionValues(const QString& bandObjectName) { foreach(GroupFunction* gf, m_groupFunctions.values(bandObjectName)){ gf->values().clear(); } } GroupFunction* DataSourceManager::addGroupFunction(const QString &name, const QString &expression, const QString &band, const QString& dataBand) { GroupFunction* gf = m_groupFunctionFactory.createGroupFunction(name,expression,dataBand,this); if (gf){ m_groupFunctions.insert(band,gf); } return gf; } GroupFunction *DataSourceManager::groupFunction(const QString &name, const QString &expression, const QString &band) { foreach(GroupFunction* gf,m_groupFunctions.values(band)){ if ((gf->name().compare(name,Qt::CaseInsensitive)==0)&& (gf->data().compare(expression,Qt::CaseInsensitive)==0) ) return gf; } return 0; } QList DataSourceManager::groupFunctionsByBand(const QString &band) { return m_groupFunctions.values(band); } void DataSourceManager::updateChildrenData(const QString &datasourceName) { foreach(SubQueryDesc* subquery,m_subqueries){ if (subquery->master().compare(datasourceName,Qt::CaseInsensitive)==0){ SubQueryHolder* holder=dynamic_cast(dataSourceHolder(subquery->queryName())); if (holder) holder->runQuery(); } } foreach(ProxyDesc* subproxy,m_proxies){ if(subproxy->master().compare(datasourceName,Qt::CaseInsensitive)==0){ ProxyHolder* holder = dynamic_cast(dataSourceHolder(subproxy->name())); holder->filterModel(); } } } QString DataSourceManager::extractDataSource(const QString &fieldName) { return fieldName.left(fieldName.indexOf('.')); } QString DataSourceManager::extractFieldName(const QString &fieldName) { return fieldName.right((fieldName.length()-fieldName.indexOf('.'))-1); } bool DataSourceManager::containsField(const QString &fieldName) { IDataSource* ds = dataSource(extractDataSource(fieldName)); if (ds) { return ds->columnIndexByName(extractFieldName(fieldName))!=-1; } return false; } bool DataSourceManager::containsVariable(const QString& variableName) { if (m_userVariables.containsVariable(variableName)) return true; return m_reportVariables.containsVariable(variableName); } void DataSourceManager::clearUserVariables() { m_userVariables.clearUserVariables(); m_reportVariables.clearUserVariables(); } QVariant DataSourceManager::fieldData(const QString &fieldName) { if (containsField(fieldName)){ IDataSource* ds = dataSource(extractDataSource(fieldName)); if (ds) return ds->data(extractFieldName(fieldName)); } return QVariant(); } QVariant DataSourceManager::fieldDataByRowIndex(const QString &fieldName, int rowIndex) { if (containsField(fieldName)){ IDataSource* ds = dataSource(extractDataSource(fieldName)); if (ds){ return ds->dataByRowIndex(extractFieldName(fieldName), rowIndex); } } return QVariant(); } QVariant DataSourceManager::fieldDataByKey(const QString& datasourceName, const QString& valueFieldName, const QString& keyFieldName, QVariant keyValue) { IDataSource* ds = dataSource(datasourceName); if (ds){ return ds->dataByKeyField(valueFieldName, keyFieldName, keyValue); } return QVariant(); } void DataSourceManager::reopenDatasource(const QString& datasourceName) { QueryHolder* qh = dynamic_cast(dataSourceHolder(datasourceName)); if (qh){ qh->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE); invalidateChildren(datasourceName); } } QVariant DataSourceManager::variable(const QString &variableName) { if (m_userVariables.containsVariable(variableName)) return m_userVariables.variable(variableName); return m_reportVariables.variable(variableName); } RenderPass DataSourceManager::variablePass(const QString &name) { if (m_userVariables.containsVariable(name)) return m_userVariables.variablePass(name); return m_reportVariables.variablePass(name); } bool DataSourceManager::variableIsSystem(const QString &name) { if (m_reportVariables.containsVariable(name)) return (m_reportVariables.variableType(name)==VarDesc::System); return false; } bool DataSourceManager::variableIsMandatory(const QString& name) { if (m_reportVariables.containsVariable(name)) return m_reportVariables.variableByName(name)->isMandatory(); return false; } void DataSourceManager::setVarableMandatory(const QString& name, bool value) { if (m_reportVariables.containsVariable(name)) m_reportVariables.variableByName(name)->setMandatory(value); } QStringList DataSourceManager::variableNames() { return m_reportVariables.variableNames(); } QStringList DataSourceManager::variableNamesByRenderPass(RenderPass pass) { QStringList result; foreach(QString variableName, m_reportVariables.variableNames()){ if (m_reportVariables.variablePass(variableName) == pass){ result.append(variableName); } } return result; } QStringList DataSourceManager::userVariableNames(){ return m_userVariables.variableNames(); } VarDesc::VarType DataSourceManager::variableType(const QString &name) { if (m_reportVariables.containsVariable(name)) return m_reportVariables.variableType(name); return VarDesc::User; } VariableDataType DataSourceManager::variableDataType(const QString& name) { if (m_reportVariables.containsVariable(name)) return m_reportVariables.variableByName(name)->dataType(); return Enums::Undefined; } void DataSourceManager::setVariableDataType(const QString& name, VariableDataType value) { if (m_reportVariables.containsVariable(name)) m_reportVariables.variableByName(name)->setDataType(value); } void DataSourceManager::setAllDatasourcesToFirst() { foreach(IDataSourceHolder* ds,m_datasources.values()) { if (ds->dataSource()) ds->dataSource()->first(); } } } //namespace LimeReport