GUI webcache tool. Implement save to file

This commit is contained in:
Jean-Francois Dockes 2021-03-22 18:46:18 +01:00
parent f614b28687
commit f45f5a0fd2
2 changed files with 82 additions and 29 deletions

View File

@ -1,4 +1,4 @@
/* Copyright (C) 2016-2020 J.F.Dockes /* Copyright (C) 2016-2021 J.F.Dockes
* This program is free software; you can redistribute it and/or modify * This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or * the Free Software Foundation; either version 2 of the License, or
@ -38,6 +38,7 @@
#include "rclmain_w.h" #include "rclmain_w.h"
#include "smallut.h" #include "smallut.h"
#include "log.h" #include "log.h"
#include "copyfile.h"
using namespace std; using namespace std;
@ -51,19 +52,23 @@ public:
string mimetype; string mimetype;
string date; string date;
string size; string size;
// Only useful in the filtered list. Corresponding index in the
// complete one
int allidx{-1};
}; };
class WebcacheModelInternal { class WebcacheModelInternal {
public: public:
std::shared_ptr<WebStore> cache; std::unique_ptr<WebStore> cache;
// The complete list
vector<CEnt> all; vector<CEnt> all;
// The entries matching the current filter (displayed)
vector<CEnt> disp; vector<CEnt> disp;
}; };
WebcacheModel::WebcacheModel(QObject *parent) WebcacheModel::WebcacheModel(QObject *parent)
: QAbstractTableModel(parent), m(new WebcacheModelInternal()) : QAbstractTableModel(parent), m(new WebcacheModelInternal())
{ {
//qDebug() << "WebcacheModel::WebcacheModel()";
reload(); reload();
} }
WebcacheModel::~WebcacheModel() WebcacheModel::~WebcacheModel()
@ -73,38 +78,42 @@ WebcacheModel::~WebcacheModel()
void WebcacheModel::reload() void WebcacheModel::reload()
{ {
m->cache = std::shared_ptr<WebStore>(new WebStore(theconfig)); int idx = 0;
m->all.clear(); m->all.clear();
m->disp.clear(); m->disp.clear();
if (!(m->cache = std::unique_ptr<WebStore>(new WebStore(theconfig)))) {
goto out;
}
if (m->cache) { bool eof;
bool eof; m->cache->cc()->rewind(eof);
m->cache->cc()->rewind(eof); while (!eof) {
while (!eof) { string udi, sdic;
string udi, sdic; m->cache->cc()->getCurrent(udi, sdic);
m->cache->cc()->getCurrent(udi, sdic); if (!udi.empty()) {
ConfSimple dic(sdic); ConfSimple dic(sdic);
string mime, url, mtime, fbytes; string mime, url, mtime, fbytes;
dic.get("mimetype", mime); dic.get("mimetype", mime);
dic.get("url", url); dic.get("url", url);
dic.get("fmtime", mtime); dic.get("fmtime", mtime);
dic.get("fbytes", fbytes); dic.get("fbytes", fbytes);
if (!udi.empty()) { CEnt entry(udi, url, mime);
CEnt entry(udi, url, mime); entry.allidx = idx++;
if (!mtime.empty()) { if (!mtime.empty()) {
time_t clck = atoll(mtime.c_str()); time_t clck = atoll(mtime.c_str());
entry.date = utf8datestring("%c", localtime(&clck)); entry.date = utf8datestring("%c", localtime(&clck));
}
if (!fbytes.empty()) {
entry.size = displayableBytes(atoll(fbytes.c_str()));
}
m->all.push_back(entry);
m->disp.push_back(entry);
} }
if (!m->cache->cc()->next(eof)) if (!fbytes.empty()) {
break; entry.size = displayableBytes(atoll(fbytes.c_str()));
}
m->all.push_back(entry);
m->disp.push_back(entry);
} }
if (!m->cache->cc()->next(eof))
break;
} }
out:
emit dataChanged(createIndex(0,0), createIndex(1, m->all.size())); emit dataChanged(createIndex(0,0), createIndex(1, m->all.size()));
} }
@ -122,6 +131,34 @@ string WebcacheModel::getURL(unsigned int idx)
return m->disp[idx].url; return m->disp[idx].url;
} }
string WebcacheModel::getData(unsigned int idx)
{
LOGDEB0("WebcacheModel::getData: idx " << idx << "\n");
if (idx > m->disp.size() || !m->cache) {
LOGERR("WebcacheModel::getData: idx > m->disp.size()" << m->disp.size() << "\n");
return string();
}
// Get index in the "all" list
auto allidx = m->disp[idx].allidx;
LOGDEB0("WebcacheModel::getData: allidx " << allidx << "\n");
if (allidx < 0 || allidx >= int(m->all.size())) {
LOGERR("WebcacheModel::getData: allidx > m->all.size()" << m->all.size() << "\n");
return string();
}
string udi = m->all[allidx].udi;
// Compute the instance for this udi (in case we are not erasing older instances).
int instance = 0;
for (int i = 0; i < allidx; i++) {
if (m->all[i].udi == udi) {
instance++;
}
}
string dic, data;
m->cache->cc()->get(udi, dic, &data, instance);
LOGDEB0("WebcacheModel::getData: got " << data.size() << " bytes of data\n");
return data;
}
int WebcacheModel::rowCount(const QModelIndex&) const int WebcacheModel::rowCount(const QModelIndex&) const
{ {
//qDebug() << "WebcacheModel::rowCount(): " << m->disp.size(); //qDebug() << "WebcacheModel::rowCount(): " << m->disp.size();
@ -172,15 +209,15 @@ QVariant WebcacheModel::data(const QModelIndex& index, int role) const
void WebcacheModel::setSearchFilter(const QString& _txt) void WebcacheModel::setSearchFilter(const QString& _txt)
{ {
SimpleRegexp re( SimpleRegexp re(qs2utf8s(_txt), SimpleRegexp::SRE_NOSUB|SimpleRegexp::SRE_ICASE);
qs2utf8s(_txt), SimpleRegexp::SRE_NOSUB|SimpleRegexp::SRE_ICASE);
m->disp.clear(); m->disp.clear();
for (unsigned int i = 0; i < m->all.size(); i++) { for (unsigned int i = 0; i < m->all.size(); i++) {
if (re(m->all[i].url)) { if (re(m->all[i].url)) {
m->disp.push_back(m->all[i]); m->disp.push_back(m->all[i]);
m->disp.back().allidx = i;
} else { } else {
LOGDEB1(" WebcacheMOdel::filter: match failed. exp" << LOGDEB1("WebcacheModel::filter: match failed. exp" <<
qs2utf8s(_txt) << "data" << m->all[i].url); qs2utf8s(_txt) << "data" << m->all[i].url);
} }
} }
@ -249,6 +286,7 @@ void WebcacheEdit::createPopupMenu(const QPoint& pos)
QMenu *popup = new QMenu(this); QMenu *popup = new QMenu(this);
if (selsz == 1) { if (selsz == 1) {
popup->addAction(tr("Copy URL"), this, SLOT(copyURL())); popup->addAction(tr("Copy URL"), this, SLOT(copyURL()));
popup->addAction(tr("Save to File"), this, SLOT(saveToFile()));
} }
if (m_recoll) { if (m_recoll) {
RclMain::IndexerState ixstate = m_recoll->indexerState(); RclMain::IndexerState ixstate = m_recoll->indexerState();
@ -300,6 +338,19 @@ void WebcacheEdit::copyURL()
} }
} }
void WebcacheEdit::saveToFile()
{
QModelIndexList selection = tableview->selectionModel()->selectedRows();
if (selection.size() != 1)
return;
string data = m_model->getData(selection[0].row());
QString qfn = myGetFileName(false, "Saving webcache data");
string reason;
if (!stringtofile(data, qs2utf8s(qfn).c_str(), reason)) {
QMessageBox::warning(0, "Recoll", tr("File creation failed: ") + u8s2qs(reason));
}
}
void WebcacheEdit::saveColState() void WebcacheEdit::saveColState()
{ {
//qDebug() << "void WebcacheEdit::saveColState()"; //qDebug() << "void WebcacheEdit::saveColState()";

View File

@ -1,4 +1,4 @@
/* Copyright (C) 2016 J.F.Dockes /* Copyright (C) 2016-2021 J.F.Dockes
* This program is free software; you can redistribute it and/or modify * This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or * the Free Software Foundation; either version 2 of the License, or
@ -40,11 +40,12 @@ public:
virtual int rowCount (const QModelIndex& = QModelIndex()) const; virtual int rowCount (const QModelIndex& = QModelIndex()) const;
virtual int columnCount(const QModelIndex& = QModelIndex()) const; virtual int columnCount(const QModelIndex& = QModelIndex()) const;
virtual QVariant headerData (int col, Qt::Orientation orientation, virtual QVariant headerData (int col, Qt::Orientation orientation,
int role = Qt::DisplayRole) const; int role = Qt::DisplayRole) const;
virtual QVariant data(const QModelIndex& index, virtual QVariant data(const QModelIndex& index,
int role = Qt::DisplayRole ) const; int role = Qt::DisplayRole ) const;
bool deleteIdx(unsigned int idx); bool deleteIdx(unsigned int idx);
std::string getURL(unsigned int idx); std::string getURL(unsigned int idx);
std::string getData(unsigned int idx);
public slots: public slots:
void setSearchFilter(const QString&); void setSearchFilter(const QString&);
@ -66,6 +67,7 @@ public slots:
void createPopupMenu(const QPoint&); void createPopupMenu(const QPoint&);
void deleteSelected(); void deleteSelected();
void copyURL(); void copyURL();
void saveToFile();
protected: protected:
void closeEvent(QCloseEvent *); void closeEvent(QCloseEvent *);
private: private: