Maemo: Use project file's INSTALLS variable for packaging.

This commit is contained in:
ck
2010-06-16 08:57:13 +02:00
parent 72bd632ca4
commit f96efa07e1
4 changed files with 359 additions and 93 deletions

View File

@@ -32,22 +32,31 @@
#include "maemopackagecreationstep.h" #include "maemopackagecreationstep.h"
#include "maemotoolchain.h" #include "maemotoolchain.h"
#include <qt4projectmanager/profilereader.h>
#include <qt4projectmanager/qt4buildconfiguration.h> #include <qt4projectmanager/qt4buildconfiguration.h>
#include <qt4projectmanager/qt4project.h> #include <qt4projectmanager/qt4project.h>
#include <qt4projectmanager/qt4target.h> #include <qt4projectmanager/qt4target.h>
#include <QtCore/QDir> #include <profile.h>
#include <prowriter.h>
#include <QtCore/QCryptographicHash>
#include <QtCore/QFile>
#include <QtCore/QFileInfo> #include <QtCore/QFileInfo>
namespace { namespace {
const char * const MODIFIED_KEY QString pathVar(const QString &var)
= "Qt4ProjectManager.BuildStep.MaemoPackage.Modified"; {
const char * const REMOTE_EXE_DIR_KEY return var + QLatin1String(".path");
= "Qt4ProjectManager.BuildStep.MaemoPackage.RemoteExeDir"; }
const char * const LOCAL_FILES_KEY
= "Qt4ProjectManager.BuildStep.MaemoPackage.LocalFiles"; QString filesVar(const QString &var)
const char * const REMOTE_DIRS_KEY {
= "Qt4ProjectManager.BuildStep.MaemoPackage.RemoteDirs"; return var + QLatin1String(".files");
}
const QLatin1String InstallsVar("INSTALLS");
const QLatin1String TargetVar("target");
} }
namespace Qt4ProjectManager { namespace Qt4ProjectManager {
@@ -56,43 +65,161 @@ namespace Internal {
MaemoPackageContents::MaemoPackageContents(MaemoPackageCreationStep *packageStep) MaemoPackageContents::MaemoPackageContents(MaemoPackageCreationStep *packageStep)
: QAbstractTableModel(packageStep), : QAbstractTableModel(packageStep),
m_packageStep(packageStep), m_packageStep(packageStep),
m_modified(true) m_proFileOption(new ProFileOption),
m_proFileReader(new ProFileReader(m_proFileOption.data())),
m_modified(false),
m_proFile(0)
{ {
} }
MaemoPackageContents::~MaemoPackageContents() {}
bool MaemoPackageContents::init()
{
return m_proFile ? true : buildModel();
}
bool MaemoPackageContents::buildModel() const
{
m_deployables.clear();
const Qt4ProFileNode * const proFileNode = m_packageStep
->qt4BuildConfiguration()->qt4Target()->qt4Project()->rootProjectNode();
if (m_proFileName.isEmpty()) {
m_proFileName = proFileNode->path();
m_proDir = QFileInfo(m_proFileName).dir();
}
resetProFileContents();
if (!m_proFile)
return false;
const QStringList elemList = m_proFileReader->values(InstallsVar, m_proFile);
bool targetFound = false;
foreach (const QString &elem, elemList) {
const QStringList paths
= m_proFileReader->values(pathVar(elem), m_proFile);
if (paths.count() != 1) {
qWarning("Error: Variable %s has %d values.",
qPrintable(pathVar(elem)), paths.count());
continue;
}
const QStringList files
= m_proFileReader->values(filesVar(elem), m_proFile);
if (files.isEmpty() && elem != TargetVar) {
qWarning("Error: Variable %s has no RHS.",
qPrintable(filesVar(elem)));
continue;
}
if (elem == TargetVar) {
m_deployables.prepend(Deployable(m_packageStep->localExecutableFilePath(),
paths.first()));
targetFound = true;
} else {
foreach (const QString &file, files)
m_deployables << Deployable(cleanPath(file), paths.first());
}
}
if (!targetFound) {
const QString remoteDir = proFileNode->projectType() == LibraryTemplate
? QLatin1String("/usr/local/lib")
: QLatin1String("/usr/local/bin");
m_deployables.prepend(Deployable(m_packageStep->localExecutableFilePath(),
remoteDir));
QString errorString;
if (!readProFileContents(&errorString)) {
qWarning("Error reading .pro file: %s", qPrintable(errorString));
return false;
}
addValueToProFile(pathVar(TargetVar), remoteDir);
addValueToProFile(InstallsVar, TargetVar);
if (!writeProFileContents(&errorString)) {
qWarning("Error writing .pro file: %s", qPrintable(errorString));
return false;
}
}
m_modified = true;
return true;
}
MaemoPackageContents::Deployable MaemoPackageContents::deployableAt(int row) const MaemoPackageContents::Deployable MaemoPackageContents::deployableAt(int row) const
{ {
Q_ASSERT(row >= 0 && row < rowCount()); Q_ASSERT(row >= 0 && row < rowCount());
return row == 0 return m_deployables.at(row);
? Deployable(m_packageStep->localExecutableFilePath(),
remoteExecutableDir())
: m_deployables.at(row - 1);
} }
bool MaemoPackageContents::addDeployable(const Deployable &deployable) bool MaemoPackageContents::addDeployable(const Deployable &deployable,
QString *error)
{ {
if (m_deployables.contains(deployable) || deployableAt(0) == deployable) if (m_deployables.contains(deployable) || deployableAt(0) == deployable)
return false; return false;
if (!readProFileContents(error))
return false;
QCryptographicHash elemHash(QCryptographicHash::Md5);
elemHash.addData(deployable.localFilePath.toUtf8());
const QString elemName = QString::fromAscii(elemHash.result().toHex());
addFileToProFile(filesVar(elemName), deployable.localFilePath);
addValueToProFile(pathVar(elemName), deployable.remoteDir);
addValueToProFile(InstallsVar, elemName);
if (!writeProFileContents(error))
return false;
beginInsertRows(QModelIndex(), rowCount(), rowCount()); beginInsertRows(QModelIndex(), rowCount(), rowCount());
m_deployables << deployable; m_deployables << deployable;
endInsertRows(); endInsertRows();
m_modified = true;
return true; return true;
} }
void MaemoPackageContents::removeDeployableAt(int row) bool MaemoPackageContents::removeDeployableAt(int row, QString *error)
{ {
Q_ASSERT(row > 0 && row < rowCount()); Q_ASSERT(row > 0 && row < rowCount());
const Deployable &deployable = deployableAt(row);
const QString elemToRemove = findInstallsElem(deployable);
if (elemToRemove.isEmpty()) {
*error = tr("Inconsistent model: Deployable not found in .pro file.");
return false;
}
if (!readProFileContents(error))
return false;
const QString filesVarName = filesVar(elemToRemove);
const bool isOnlyElem
= m_proFileReader->values(filesVarName, m_proFile).count() == 1;
bool success
= removeFileFromProFile(filesVarName, deployable.localFilePath);
if (success && isOnlyElem) {
success = removeValueFromProFile(pathVar(elemToRemove),
deployable.remoteDir);
if (success)
success = removeValueFromProFile(InstallsVar, elemToRemove);
}
if (!success) {
*error = tr("Could not remove deployable from .pro file.");
return false;
}
if (!writeProFileContents(error))
return false;
beginRemoveRows(QModelIndex(), row, row); beginRemoveRows(QModelIndex(), row, row);
m_deployables.removeAt(row - 1); m_deployables.removeAt(row - 1);
endRemoveRows(); endRemoveRows();
m_modified = true; return true;
} }
int MaemoPackageContents::rowCount(const QModelIndex &parent) const int MaemoPackageContents::rowCount(const QModelIndex &parent) const
{ {
return parent.isValid() ? 0 : m_deployables.count() + 1; if (!m_proFile)
buildModel();
return parent.isValid() ? 0 : m_deployables.count();
} }
int MaemoPackageContents::columnCount(const QModelIndex &parent) const int MaemoPackageContents::columnCount(const QModelIndex &parent) const
@@ -128,12 +255,33 @@ bool MaemoPackageContents::setData(const QModelIndex &index,
|| role != Qt::EditRole) || role != Qt::EditRole)
return false; return false;
const QString &remoteDir = value.toString(); QString error;
if (index.row() == 0) if (!readProFileContents(&error)) {
m_remoteExecutableDir = remoteDir; qWarning(qPrintable(error));
else return false;
m_deployables[index.row() - 1].remoteDir = remoteDir; }
m_modified = true;
Deployable &deployable = m_deployables[index.row()];
const QString elemToChange = findInstallsElem(deployable);
if (elemToChange.isEmpty()) {
qWarning("Error: Inconsistent model. "
"INSTALLS element not found in .pro file");
return false;
}
const QString pathElem = pathVar(elemToChange);
if (!removeValueFromProFile(pathElem, deployable.remoteDir)) {
qWarning("Error: Could not change remote path in .pro file.");
return false;
}
const QString &newRemoteDir = value.toString();
addValueToProFile(pathElem, newRemoteDir);
if (!writeProFileContents(&error)) {
qWarning(qPrintable(error));
return false;
}
deployable.remoteDir = newRemoteDir;
emit dataChanged(index, index); emit dataChanged(index, index);
return true; return true;
} }
@@ -146,57 +294,146 @@ QVariant MaemoPackageContents::headerData(int section,
return section == 0 ? tr("Local File Path") : tr("Remote Directory"); return section == 0 ? tr("Local File Path") : tr("Remote Directory");
} }
QVariantMap MaemoPackageContents::toMap() const
{
QVariantMap map;
map.insert(MODIFIED_KEY, m_modified);
map.insert(REMOTE_EXE_DIR_KEY, m_remoteExecutableDir);
QDir dir;
QStringList localFiles;
QStringList remoteDirs;
foreach (const Deployable &p, m_deployables) {
localFiles << dir.fromNativeSeparators(p.localFilePath);
remoteDirs << p.remoteDir;
}
map.insert(LOCAL_FILES_KEY, localFiles);
map.insert(REMOTE_DIRS_KEY, remoteDirs);
return map;
}
void MaemoPackageContents::fromMap(const QVariantMap &map)
{
m_modified = map.value(MODIFIED_KEY).toBool();
m_remoteExecutableDir = map.value(REMOTE_EXE_DIR_KEY).toString();
const QStringList localFiles = map.value(LOCAL_FILES_KEY).toStringList();
const QStringList remoteDirs = map.value(REMOTE_DIRS_KEY).toStringList();
if (localFiles.count() != remoteDirs.count())
qWarning("%s: serialized data inconsistent", Q_FUNC_INFO);
QDir dir;
const int count = qMin(localFiles.count(), remoteDirs.count());
for (int i = 0; i < count; ++i) {
m_deployables << Deployable(dir.toNativeSeparators(localFiles.at(i)),
remoteDirs.at(i));
}
}
QString MaemoPackageContents::remoteExecutableDir() const
{
if (m_remoteExecutableDir.isEmpty()) {
const Qt4ProjectType projectType
= m_packageStep->qt4BuildConfiguration()->qt4Target()->qt4Project()
->rootProjectNode()->projectType();
m_remoteExecutableDir = projectType == LibraryTemplate
? QLatin1String("/usr/local/lib")
: QLatin1String("/usr/local/bin");
}
return m_remoteExecutableDir;
}
QString MaemoPackageContents::remoteExecutableFilePath() const QString MaemoPackageContents::remoteExecutableFilePath() const
{ {
return remoteExecutableDir() + '/' + m_packageStep->executableFileName(); if (!m_proFile)
buildModel();
return deployableAt(0).remoteDir + '/' + m_packageStep->executableFileName();
}
bool MaemoPackageContents::readProFileContents(QString *error) const
{
if (!m_proFileLines.isEmpty())
return true;
QFile proFileOnDisk(m_proFileName);
if (!proFileOnDisk.open(QIODevice::ReadOnly)) {
*error = tr("Project file '%1' could not be opened for reading.")
.arg(m_proFileName);
return false;
}
const QString proFileContents
= QString::fromLatin1(proFileOnDisk.readAll());
if (proFileOnDisk.error() != QFile::NoError) {
*error = tr("Project file '%1' could not be read.")
.arg(m_proFileName);
return false;
}
m_proFileLines = proFileContents.split('\n');
return true;
}
bool MaemoPackageContents::writeProFileContents(QString *error) const
{
QFile proFileOnDisk(m_proFileName);
if (!proFileOnDisk.open(QIODevice::WriteOnly)) {
*error = tr("Project file '%1' could not be opened for writing.")
.arg(m_proFileName);
resetProFileContents();
return false;
}
// TODO: Disconnect and reconnect FS watcher here.
proFileOnDisk.write(m_proFileLines.join("\n").toLatin1());
proFileOnDisk.close();
if (proFileOnDisk.error() != QFile::NoError) {
*error = tr("Project file '%1' could not be written.")
.arg(m_proFileName);
resetProFileContents();
return false;
}
m_modified = true;
return true;
}
QString MaemoPackageContents::cleanPath(const QString &relFileName) const
{
// I'd rather use QDir::cleanPath(), but that doesn't work well
// enough for redundant ".." dirs.
return QFileInfo(m_proFile->directoryName() + '/'
+ relFileName).canonicalFilePath();
}
QString MaemoPackageContents::findInstallsElem(const Deployable &deployable) const
{
const QStringList elems = m_proFileReader->values(InstallsVar, m_proFile);
foreach (const QString &elem, elems) {
const QStringList elemPaths
= m_proFileReader->values(pathVar(elem), m_proFile);
if (elemPaths.count() != 1 || elemPaths.first() != deployable.remoteDir)
continue;
if (elem == TargetVar)
return elem;
const QStringList elemFiles
= m_proFileReader->values(filesVar(elem), m_proFile);
foreach (const QString &file, elemFiles) {
if (cleanPath(file) == deployable.localFilePath)
return elem;
}
}
return QString();
}
void MaemoPackageContents::addFileToProFile(const QString &var,
const QString &absFilePath)
{
ProWriter::addFiles(m_proFile, &m_proFileLines, m_proDir,
QStringList(absFilePath), var);
parseProFile(ParseFromLines);
}
void MaemoPackageContents::addValueToProFile(const QString &var,
const QString &value) const
{
ProWriter::addVarValues(m_proFile, &m_proFileLines, m_proDir,
QStringList(value), var);
parseProFile(ParseFromLines);
}
bool MaemoPackageContents::removeFileFromProFile(const QString &var,
const QString &absFilePath)
{
const bool success = ProWriter::removeFiles(m_proFile, &m_proFileLines,
m_proDir, QStringList(absFilePath),
QStringList(var)).isEmpty();
if (success)
parseProFile(ParseFromLines);
else
resetProFileContents();
return success;
}
bool MaemoPackageContents::removeValueFromProFile(const QString &var,
const QString &value)
{
const bool success = ProWriter::removeVarValues(m_proFile,
&m_proFileLines, m_proDir, QStringList(value),
QStringList(var)).isEmpty();
if (success)
parseProFile(ParseFromLines);
else
resetProFileContents();
return success;
}
void MaemoPackageContents::parseProFile(ParseType type) const
{
if (type == ParseFromLines) {
m_proFile = m_proFileReader->parsedProFile(m_proFileName,
m_proFileLines.join("\n"));
} else {
m_proFile = m_proFileReader->readProFile(m_proFileName)
? m_proFileReader->proFileFor(m_proFileName) : 0;
}
}
void MaemoPackageContents::resetProFileContents() const
{
m_proFileLines.clear();
parseProFile(ParseFromFile);
if (!m_proFile)
qWarning("Fatal: Could not parse .pro file '%s'.",
qPrintable(m_proFileName));
} }
} // namespace Qt4ProjectManager } // namespace Qt4ProjectManager

View File

@@ -31,14 +31,21 @@
#define MAEMOPACKAGECONTENTS_H #define MAEMOPACKAGECONTENTS_H
#include <QtCore/QAbstractTableModel> #include <QtCore/QAbstractTableModel>
#include <QtCore/QDir>
#include <QtCore/QList> #include <QtCore/QList>
#include <QtCore/QScopedPointer>
#include <QtCore/QString> #include <QtCore/QString>
#include <QtCore/QVariantMap> #include <QtCore/QStringList>
QT_BEGIN_NAMESPACE
class ProFile;
class ProFileOption;
QT_END_NAMESPACE
namespace Qt4ProjectManager { namespace Qt4ProjectManager {
namespace Internal { namespace Internal {
class MaemoPackageCreationStep; class MaemoPackageCreationStep;
class ProFileReader;
class MaemoPackageContents : public QAbstractTableModel class MaemoPackageContents : public QAbstractTableModel
{ {
@@ -60,15 +67,15 @@ public:
}; };
MaemoPackageContents(MaemoPackageCreationStep *packageStep); MaemoPackageContents(MaemoPackageCreationStep *packageStep);
~MaemoPackageContents();
bool init();
virtual int rowCount(const QModelIndex &parent = QModelIndex()) const; virtual int rowCount(const QModelIndex &parent = QModelIndex()) const;
QVariantMap toMap() const;
void fromMap(const QVariantMap &map);
Deployable deployableAt(int row) const; Deployable deployableAt(int row) const;
bool addDeployable(const Deployable &deployable); bool addDeployable(const Deployable &deployable, QString *error);
void removeDeployableAt(int row); bool removeDeployableAt(int row, QString *error);
bool isModified() const { return m_modified; } bool isModified() const { return m_modified; }
void setUnModified() { m_modified = false; } void setUnModified() { m_modified = false; }
QString remoteExecutableFilePath() const; QString remoteExecutableFilePath() const;
@@ -83,13 +90,31 @@ private:
virtual bool setData(const QModelIndex &index, const QVariant &value, virtual bool setData(const QModelIndex &index, const QVariant &value,
int role = Qt::EditRole); int role = Qt::EditRole);
QString remoteExecutableDir() const; bool buildModel() const;
void resetProFileContents() const;
bool readProFileContents(QString *error) const;
bool writeProFileContents(QString *error) const;
QString cleanPath(const QString &relFileName) const;
QString findInstallsElem(const Deployable &deployable) const;
void addFileToProFile(const QString &var, const QString &absFilePath);
void addValueToProFile(const QString &var, const QString &value) const;
bool removeFileFromProFile(const QString &var, const QString &absFilePath);
bool removeValueFromProFile(const QString &var, const QString &value);
enum ParseType { ParseFromFile, ParseFromLines };
void parseProFile(ParseType type) const;
private:
const MaemoPackageCreationStep * const m_packageStep; const MaemoPackageCreationStep * const m_packageStep;
QList<Deployable> m_deployables; QScopedPointer<ProFileOption> m_proFileOption;
bool m_modified; QScopedPointer<ProFileReader> m_proFileReader;
mutable QString m_remoteExecutableDir; mutable QList<Deployable> m_deployables;
mutable bool m_modified;
mutable ProFile *m_proFile;
mutable QStringList m_proFileLines; // TODO: FS watcher
mutable QString m_proFileName;
mutable QDir m_proDir;
}; };
} // namespace Qt4ProjectManager } // namespace Qt4ProjectManager

View File

@@ -95,12 +95,11 @@ QVariantMap MaemoPackageCreationStep::toMap() const
{ {
QVariantMap map(ProjectExplorer::BuildStep::toMap()); QVariantMap map(ProjectExplorer::BuildStep::toMap());
map.insert(PackagingEnabledKey, m_packagingEnabled); map.insert(PackagingEnabledKey, m_packagingEnabled);
return map.unite(m_packageContents->toMap()); return map;
} }
bool MaemoPackageCreationStep::fromMap(const QVariantMap &map) bool MaemoPackageCreationStep::fromMap(const QVariantMap &map)
{ {
m_packageContents->fromMap(map);
m_packagingEnabled = map.value(PackagingEnabledKey, true).toBool(); m_packagingEnabled = map.value(PackagingEnabledKey, true).toBool();
return ProjectExplorer::BuildStep::fromMap(map); return ProjectExplorer::BuildStep::fromMap(map);
} }

View File

@@ -109,9 +109,9 @@ void MaemoPackageCreationWidget::addFile()
deployable(QDir::toNativeSeparators(QFileInfo(localFile).absoluteFilePath()), deployable(QDir::toNativeSeparators(QFileInfo(localFile).absoluteFilePath()),
"/"); "/");
MaemoPackageContents * const contents = m_step->packageContents(); MaemoPackageContents * const contents = m_step->packageContents();
if (!contents->addDeployable(deployable)) { QString errorString;
QMessageBox::information(this, tr("File already in package"), if (!contents->addDeployable(deployable, &errorString)) {
tr("You have already added this file.")); QMessageBox::information(this, tr("Error adding file"), errorString);
} else { } else {
const QModelIndex newIndex const QModelIndex newIndex
= contents->index(contents->rowCount() - 1, 1); = contents->index(contents->rowCount() - 1, 1);
@@ -128,8 +128,13 @@ void MaemoPackageCreationWidget::removeFile()
if (selectedRows.isEmpty()) if (selectedRows.isEmpty())
return; return;
const int row = selectedRows.first().row(); const int row = selectedRows.first().row();
if (row != 0) if (row != 0) {
m_step->packageContents()->removeDeployableAt(row); QString errorString;
if (!m_step->packageContents()->removeDeployableAt(row, &errorString)) {
QMessageBox::information(this, tr("Error removing file"),
errorString);
}
}
} }
void MaemoPackageCreationWidget::enableOrDisableRemoveButton() void MaemoPackageCreationWidget::enableOrDisableRemoveButton()