• Skip to content
  • Skip to link menu
KDE API Reference
  • KDE API Reference
  • kdenetwork API Reference
  • KDE Home
  • Contact Us
 

kget

  • sources
  • kde-4.12
  • kdenetwork
  • kget
  • transfer-plugins
  • metalink
metalinkhttp.cpp
Go to the documentation of this file.
1 /* This file is part of the KDE project
2 
3  Copyright (C) 2004 Dario Massarin <nekkar@libero.it>
4  Copyright (C) 2007 Manolo Valdes <nolis71cu@gmail.com>
5  Copyright (C) 2009 Matthias Fuchs <mat69@gmx.net>
6  Copyright (C) 2012 Aish Raj Dahal <dahalaishraj@gmail.com>
7 
8  This program is free software; you can redistribute it and/or
9  modify it under the terms of the GNU General Public
10  License as published by the Free Software Foundation; either
11  version 2 of the License, or (at your option) any later version.
12 */
13 
14 #include "metalinkhttp.h"
15 #include "metalinksettings.h"
16 #include "metalinkxml.h"
17 
18 #include "core/kget.h"
19 #include "core/transfergroup.h"
20 #include "core/download.h"
21 #include "core/transferdatasource.h"
22 #include "core/filemodel.h"
23 #include "core/urlchecker.h"
24 #include "core/verifier.h"
25 #include "core/signature.h"
26 
27 #ifdef HAVE_NEPOMUK
28  #include "core/nepomukhandler.h"
29  #include <Nepomuk2/Variant>
30 #endif //HAVE_NEPOMUK
31 
32 #include <KIconLoader>
33 #include <KIO/DeleteJob>
34 #include <KIO/NetAccess>
35 #include <KIO/RenameDialog>
36 #include <KLocale>
37 #include <KMessageBox>
38 #include <KDebug>
39 #include <KDialog>
40 #include <KStandardDirs>
41 
42 #include <QtCore/QFile>
43 #include <QtXml/QDomElement>
44 
49 QString base64ToHex(const QString& b64)
50 {
51  return QString(QByteArray::fromBase64(b64.toAscii()).toHex());
52 }
53 
54 MetalinkHttp::MetalinkHttp(TransferGroup * parent, TransferFactory * factory,
55  Scheduler * scheduler, const KUrl & source, const KUrl & dest,
56  KGetMetalink::MetalinkHttpParser *httpParser,
57  const QDomElement * e)
58  : AbstractMetalink(parent,factory,scheduler,source, dest, e) ,
59  m_signatureUrl(KUrl()),
60  m_httpparser(httpParser)
61 
62 {
63  m_httpparser->setParent(this);
64 }
65 
66 MetalinkHttp::~MetalinkHttp()
67 {
68 
69 }
70 
71 void MetalinkHttp::load(const QDomElement *element)
72 {
73  kDebug(5001);
74  Transfer::load(element);
75  DataSourceFactory * fac = new DataSourceFactory(this, m_dest);
76  m_dataSourceFactory.insert(m_dest, fac);
77 
78  connect(fac, SIGNAL(capabilitiesChanged()), this, SLOT(slotUpdateCapabilities()));
79  connect(fac, SIGNAL(dataSourceFactoryChange(Transfer::ChangesFlags)), this, SLOT(slotDataSourceFactoryChange(Transfer::ChangesFlags)));
80  connect(fac->verifier(), SIGNAL(verified(bool)), this, SLOT(slotVerified(bool)));
81  connect(fac->signature(), SIGNAL(verified(int)), this, SLOT(slotSignatureVerified()));
82  connect(fac, SIGNAL(log(QString,Transfer::LogLevel)), this, SLOT(setLog(QString,Transfer::LogLevel)));
83 
84  fac->load(element);
85 
86  if (fac->mirrors().isEmpty()) {
87  return;
88  }
89 
90  m_ready = true;
91 }
92 
93 void MetalinkHttp::save(const QDomElement &element)
94 {
95  kDebug(5001);
96  Transfer::save(element);
97  m_dataSourceFactory.begin().value()->save(element);
98 }
99 
100 void MetalinkHttp::startMetalink()
101 {
102  if (m_ready) {
103  foreach (DataSourceFactory *factory, m_dataSourceFactory) {
104  //specified number of files is downloaded simultanously
105  if (m_currentFiles < MetalinkSettings::simultanousFiles()) {
106  const Job::Status status = factory->status();
107 
108  //only start factories that should be downloaded
109  if (factory->doDownload() &&
110  (status != Job::Finished) &&
111  (status != Job::FinishedKeepAlive) &&
112  (status != Job::Running)) {
113  ++m_currentFiles;
114  factory->start();
115  }
116  } else {
117  break;
118  }
119  }
120  }
121 }
122 
123 void MetalinkHttp::start()
124 {
125  kDebug() << "metalinkhttp::start";
126 
127  if (!m_ready) {
128  setLinks();
129  setDigests();
130  if (metalinkHttpInit()) {
131  startMetalink();
132  }
133  } else {
134  startMetalink();
135  }
136 }
137 
138 void MetalinkHttp::setSignature(KUrl & dest, QByteArray & data, DataSourceFactory* dataFactory)
139 {
140  Q_UNUSED(dest);
141  dataFactory->signature()->setSignature(data,Signature::AsciiDetached);
142 
143 }
144 
145 void MetalinkHttp::slotSignatureVerified()
146 {
147  if (status() == Job::Finished) {
148  //see if some files are NotVerified
149  QStringList brokenFiles;
150  foreach (DataSourceFactory *factory, m_dataSourceFactory) {
151  if (m_fileModel) {
152  QModelIndex signatureVerified = m_fileModel->index(factory->dest(), FileItem::SignatureVerified);
153  m_fileModel->setData(signatureVerified, factory->signature()->status());
154  }
155  if (factory->doDownload() && (factory->verifier()->status() == Verifier::NotVerified)) {
156  brokenFiles.append(factory->dest().pathOrUrl());
157  }
158  }
159 
160  if (brokenFiles.count())
161  {
162  if (KMessageBox::warningYesNoCancelList(0,
163  i18n("The download could not be verified, try to repair it?"),
164  brokenFiles) == KMessageBox::Yes) {
165  if (repair()) {
166  KGet::addTransfer(m_metalinkxmlUrl);
167  //TODO Use a Notification instead. Check kget.h for how to use it.
168 
169  }
170  }
171  }
172  }
173 }
174 
175 bool MetalinkHttp::metalinkHttpInit()
176 {
177  kDebug() << "m_dest = " << m_dest;
178  const KUrl tempDest = KUrl(m_dest.directory());
179  KUrl dest = tempDest;
180  dest.addPath(m_dest.fileName());
181  kDebug() << "dest = " << dest;
182 
183  //sort the urls according to their priority (highest first)
184  qStableSort(m_linkheaderList);
185 
186  DataSourceFactory *dataFactory = new DataSourceFactory(this,dest);
187  dataFactory->setMaxMirrorsUsed(MetalinkSettings::mirrorsPerFile());
188 
189  connect(dataFactory, SIGNAL(capabilitiesChanged()), this, SLOT(slotUpdateCapabilities()));
190  connect(dataFactory, SIGNAL(dataSourceFactoryChange(Transfer::ChangesFlags)), this, SLOT(slotDataSourceFactoryChange(Transfer::ChangesFlags)));
191  connect(dataFactory->verifier(), SIGNAL(verified(bool)), this, SLOT(slotVerified(bool)));
192  connect(dataFactory->signature(), SIGNAL(verified(int)), this, SLOT(slotSignatureVerified()));
193  connect(dataFactory, SIGNAL(log(QString,Transfer::LogLevel)), this, SLOT(setLog(QString,Transfer::LogLevel)));
194 
195  //add the Mirrors Sources
196 
197  for(int i = 0; i < m_linkheaderList.size(); ++i) {
198  const KUrl url = m_linkheaderList[i].url;
199  if (url.isValid()) {
200  if (m_linkheaderList[i].pref) {
201  kDebug() << "found etag in a mirror" ;
202  KGetMetalink::MetalinkHttpParser* eTagCher = new KGetMetalink::MetalinkHttpParser(url) ;
203  if (eTagCher->getEtag() != m_httpparser->getEtag()) { //There is an ETag mismatch
204  continue ;
205  }
206  }
207 
208  dataFactory->addMirror(url, MetalinkSettings::connectionsPerUrl());
209  }
210  }
211 
212  //no datasource has been created, so remove the datasource factory
213  if (dataFactory->mirrors().isEmpty()) {
214  kDebug() << "data source factory being deleted" ;
215  delete dataFactory;
216  } else {
217  QHashIterator<QString, QString> itr(m_DigestList);
218  while(itr.hasNext()) {
219  itr.next();
220  kDebug() << itr.key() << ":" << itr.value() ;
221  }
222 
223  dataFactory->verifier()->addChecksums(m_DigestList);
224 
225  //Add OpenPGP signatures
226  if (m_signatureUrl != KUrl()) {
227  Download *signat_download = new Download(m_signatureUrl, QString(KStandardDirs::locateLocal("appdata", "metalinks/") + m_source.fileName()));
228  connect(signat_download, SIGNAL(finishedSuccessfully(KUrl,QByteArray)), SLOT(setSignature(KUrl,QByteArray)));
229  }
230  m_dataSourceFactory[dataFactory->dest()] = dataFactory;
231  }
232 
233  if (m_dataSourceFactory.size()) {
234  m_dest = dest;
235  }
236 
237  if (!m_dataSourceFactory.size()) {
238  //TODO make this via log in the future + do not display the KMessageBox
239  kWarning(5001) << "Download of" << m_source << "failed, no working URLs were found.";
240  KMessageBox::error(0, i18n("Download failed, no working URLs were found."), i18n("Error"));
241  setStatus(Job::Aborted);
242  setTransferChange(Tc_Status, true);
243  return false;
244  }
245 
246  m_ready = !m_dataSourceFactory.isEmpty();
247  slotUpdateCapabilities();
248 
249  return true;
250 }
251 
252 void MetalinkHttp::setLinks()
253 {
254  const QMultiMap<QString, QString>* headerInf = m_httpparser->getHeaderInfo();
255  const QList<QString> linkVals = headerInf->values("link");
256 
257  foreach (const QString link, linkVals) {
258  const KGetMetalink::HttpLinkHeader linkheader(link);
259 
260  if (linkheader.reltype == "duplicate") {
261  m_linkheaderList.append(linkheader);
262  }
263  else if (linkheader.reltype == "application/pgp-signature") {
264  m_signatureUrl = linkheader.url; //There will only be one signature
265  }
266  else if (linkheader.reltype == "application/metalink4+xml") {
267  m_metalinkxmlUrl = linkheader.url ; // There will only be one metalink xml (metainfo URL)
268  }
269  }
270 }
271 
272 void MetalinkHttp::deinit(Transfer::DeleteOptions options)
273 {
274  foreach (DataSourceFactory *factory, m_dataSourceFactory) {
275  if (options & Transfer::DeleteFiles) {
276  factory->deinit();
277  }
278  }
279 }
280 
281 void MetalinkHttp::setDigests()
282 {
283  const QMultiMap<QString, QString>* digestInfo = m_httpparser->getHeaderInfo();
284  const QList<QString> digestList = digestInfo->values("digest");
285 
286  foreach(const QString digest, digestList) {
287  const int eqDelimiter = digest.indexOf('=');
288  const QString digestType = MetalinkHttp::adaptDigestType(digest.left(eqDelimiter).trimmed());
289  const QString hexDigestValue = base64ToHex(digest.mid(eqDelimiter + 1).trimmed());
290 
291  m_DigestList.insertMulti(digestType,hexDigestValue);
292  }
293 }
294 
295 QString MetalinkHttp::adaptDigestType(const QString & hashType)
296 {
297  if (hashType == QString("SHA")) {
298  return QString("sha");
299  }
300  else if (hashType == QString("MD5")) {
301  return QString("md5");
302  }
303  else if (hashType == QString("SHA-256")) {
304  return QString("sha256");
305  }
306  else {
307  return hashType;
308  }
309 }
310 
311 #include "metalinkhttp.moc"
AbstractMetalink::m_ready
bool m_ready
Definition: abstractmetalink.h:104
KGet::addTransfer
static TransferHandler * addTransfer(KUrl srcUrl, QString destDir=QString(), QString suggestedFileName=QString(), QString groupName=QString(), bool start=false)
Adds a new transfer to the KGet.
Definition: kget.cpp:179
Transfer::ChangesFlags
int ChangesFlags
Definition: transfer.h:100
metalinkxml.h
Transfer::m_dest
KUrl m_dest
Definition: transfer.h:357
Signature::AsciiDetached
Definition: signature.h:53
TransferGroup
class TransferGroup:
Definition: transfergroup.h:46
Scheduler
Scheduler class: what handle all the jobs in kget.
Definition: scheduler.h:32
Job::Finished
The job is stopped, but this also indicates that it stopped because an error occurred.
Definition: job.h:47
Download
Definition: download.h:23
KGetMetalink::MetalinkHttpParser
Definition: metalinker.h:419
metalinksettings.h
AbstractMetalink::m_currentFiles
int m_currentFiles
Definition: abstractmetalink.h:102
Job::Status
Status
The status property describes the current job status.
Definition: job.h:42
AbstractMetalink::m_fileModel
FileModel * m_fileModel
Definition: abstractmetalink.h:101
Transfer::LogLevel
LogLevel
Definition: transfer.h:81
Job::FinishedKeepAlive
The job exited from its Running state successfully.
Definition: job.h:48
KGetMetalink::MetalinkHttpParser::getEtag
QString * getEtag()
Definition: metalinker.cpp:1345
DataSourceFactory
This class manages multiple DataSources and saves the received data to the file.
Definition: datasourcefactory.h:38
KGetMetalink::MetalinkHttpParser::getHeaderInfo
QMultiMap< QString, QString > * getHeaderInfo()
Definition: metalinker.cpp:1461
MetalinkHttp::MetalinkHttp
MetalinkHttp(TransferGroup *parent, TransferFactory *factory, Scheduler *scheduler, const KUrl &src, const KUrl &dest, KGetMetalink::MetalinkHttpParser *httpParser, const QDomElement *e=0)
Definition: metalinkhttp.cpp:54
DataSourceFactory::start
void start()
Definition: datasourcefactory.cpp:193
DataSourceFactory::mirrors
QHash< KUrl, QPair< bool, int > > mirrors() const
Return all mirrors, where bool defines if the mirror is used, while in defines the number of paralell...
Definition: datasourcefactory.cpp:565
AbstractMetalink::slotUpdateCapabilities
void slotUpdateCapabilities()
Definition: abstractmetalink.cpp:404
nepomukhandler.h
download.h
MetalinkHttp::~MetalinkHttp
~MetalinkHttp()
Definition: metalinkhttp.cpp:66
urlchecker.h
MetalinkHttp::start
void start()
Definition: metalinkhttp.cpp:123
MetalinkSettings::mirrorsPerFile
static int mirrorsPerFile()
Get MirrorsPerFile.
Definition: metalinksettings.h:73
Verifier::status
VerificationStatus status() const
Definition: verifier.cpp:206
Transfer::log
const QStringList log() const
Transfer history.
Transfer::Tc_Status
Definition: transfer.h:53
DataSourceFactory::setMaxMirrorsUsed
void setMaxMirrorsUsed(int maxMirrorsUsed)
Change the maximum number off mirrors that will be used for downloading, if the download started alre...
Definition: datasourcefactory.h:87
Job::Running
Definition: job.h:43
DataSourceFactory::addMirror
void addMirror(const KUrl &url, bool used, int numParalellConnections=1)
Add a mirror that can be used for downloading.
Definition: datasourcefactory.cpp:378
Transfer::load
virtual void load(const QDomElement *element)
Loads the transfer's info from the QDomElement.
Definition: transfer.cpp:244
AbstractMetalink::repair
bool repair(const KUrl &file=KUrl())
Tries to repair file.
Definition: abstractmetalink.cpp:260
MetalinkSettings::connectionsPerUrl
static int connectionsPerUrl()
Get ConnectionsPerUrl.
Definition: metalinksettings.h:104
DataSourceFactory::verifier
Verifier * verifier()
Definition: datasourcefactory.cpp:1222
Transfer::save
virtual void save(const QDomElement &element)
Saves this transfer to the given QDomNode.
Definition: transfer.cpp:230
DataSourceFactory::load
void load(const QDomElement *e)
Definition: datasourcefactory.cpp:952
Transfer::factory
TransferFactory * factory() const
Definition: transfer.h:272
transfergroup.h
signature.h
AbstractMetalink::slotDataSourceFactoryChange
void slotDataSourceFactoryChange(Transfer::ChangesFlags change)
Definition: abstractmetalink.cpp:55
base64ToHex
QString base64ToHex(const QString &b64)
Definition: metalinkhttp.cpp:49
verifier.h
Transfer::capabilitiesChanged
void capabilitiesChanged()
Emitted when the capabilities of the Transfer change.
MetalinkHttp::slotSignatureVerified
void slotSignatureVerified()
Definition: metalinkhttp.cpp:145
transferdatasource.h
Job::Aborted
The job is stopped.
Definition: job.h:45
Transfer::setTransferChange
virtual void setTransferChange(ChangesFlags change, bool postEvent=false)
Makes the TransferHandler associated with this transfer know that a change in this transfer has occur...
Definition: transfer.cpp:338
AbstractMetalink
Definition: abstractmetalink.h:22
metalinkhttp.h
AbstractMetalink::m_dataSourceFactory
QHash< KUrl, DataSourceFactory * > m_dataSourceFactory
Definition: abstractmetalink.h:103
DataSourceFactory::dest
KUrl dest() const
Definition: datasourcefactory.h:76
MetalinkHttp::save
void save(const QDomElement &element)
Definition: metalinkhttp.cpp:93
Job::status
Status status() const
Definition: job.h:93
Transfer::dest
const KUrl & dest() const
Definition: transfer.h:149
FileModel::setData
bool setData(const QModelIndex &index, const QVariant &value, int role=Qt::EditRole)
Definition: filemodel.cpp:390
kget.h
Signature::status
VerificationStatus status() const
Definition: signature.cpp:124
KGetMetalink::HttpLinkHeader
Definition: metalinker.h:478
Verifier::addChecksums
void addChecksums(const QHash< QString, QString > &checksums)
Add multiple checksums that will later be used in the verification process.
Definition: verifier.cpp:519
Transfer::setStatus
void setStatus(Job::Status jobStatus, const QString &text=QString(), const QPixmap &pix=QPixmap())
Sets the Job status to jobStatus, the status text to text and the status pixmap to pix...
Definition: transfer.cpp:292
FileItem::SignatureVerified
Definition: filemodel.h:49
AbstractMetalink::slotVerified
void slotVerified(bool isVerified)
Definition: abstractmetalink.cpp:198
Signature::setSignature
void setSignature(const QByteArray &signature, SignatureType type)
Definition: signature.cpp:146
Transfer::DeleteFiles
Definition: transfer.h:97
DataSourceFactory::doDownload
bool doDownload() const
Returns whether the datasourcefactory should download the file or not, true by default.
Definition: datasourcefactory.h:141
MetalinkHttp::load
void load(const QDomElement *element)
Definition: metalinkhttp.cpp:71
MetalinkHttp::deinit
void deinit(Transfer::DeleteOptions options)
Definition: metalinkhttp.cpp:272
TransferFactory
TransferFactory.
Definition: transferfactory.h:52
Transfer::m_source
KUrl m_source
Definition: transfer.h:356
MetalinkSettings::simultanousFiles
static int simultanousFiles()
Get SimultanousFiles.
Definition: metalinksettings.h:42
DataSourceFactory::deinit
void deinit()
Deletes the created (downloadInitialized() is true) file if the download was not finished Does not de...
Definition: datasourcefactory.cpp:112
Transfer::setLog
void setLog(const QString &message, Transfer::LogLevel level=Log_Info)
Set Transfer history.
Definition: transfer.cpp:201
DataSourceFactory::status
Job::Status status() const
Definition: datasourcefactory.h:152
Verifier::NotVerified
Definition: verifier.h:79
DataSourceFactory::signature
Signature * signature()
Definition: datasourcefactory.cpp:1231
FileModel::index
QModelIndex index(int row, int column, const QModelIndex &parent=QModelIndex()) const
Definition: filemodel.cpp:436
filemodel.h
This file is part of the KDE documentation.
Documentation copyright © 1996-2014 The KDE developers.
Generated on Tue Oct 14 2014 22:53:17 by doxygen 1.8.7 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.

kget

Skip menu "kget"
  • Main Page
  • Namespace List
  • Namespace Members
  • Alphabetical List
  • Class List
  • Class Hierarchy
  • Class Members
  • File List
  • File Members

kdenetwork API Reference

Skip menu "kdenetwork API Reference"
  • kget
  • kopete
  •   kopete
  •   libkopete
  • krdc
  • krfb

Search



Report problems with this website to our bug tracking system.
Contact the specific authors with questions and comments about the page contents.

KDE® and the K Desktop Environment® logo are registered trademarks of KDE e.V. | Legal