2007-07-25 20:21:06 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 by Dominik Riebeling
|
2007-08-14 22:47:01 +00:00
|
|
|
* $Id$
|
2007-07-25 20:21:06 +00:00
|
|
|
*
|
|
|
|
* All files in this archive are subject to the GNU General Public License.
|
|
|
|
* See the file COPYING in the source tree root for full license agreement.
|
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <QtCore>
|
|
|
|
#include <QtNetwork>
|
|
|
|
#include <QtDebug>
|
|
|
|
|
|
|
|
#include "httpget.h"
|
|
|
|
|
2008-04-13 18:43:51 +00:00
|
|
|
QDir HttpGet::m_globalCache; //< global cach path value for new objects
|
|
|
|
QUrl HttpGet::m_globalProxy; //< global proxy value for new objects
|
2008-05-17 19:36:54 +00:00
|
|
|
bool HttpGet::m_globalDumbCache = false; //< globally set cache "dumb" mode
|
2007-07-25 20:21:06 +00:00
|
|
|
|
|
|
|
HttpGet::HttpGet(QObject *parent)
|
|
|
|
: QObject(parent)
|
|
|
|
{
|
2007-08-15 13:28:15 +00:00
|
|
|
outputToBuffer = true;
|
2008-05-13 19:38:17 +00:00
|
|
|
m_cached = false;
|
2008-05-17 19:36:54 +00:00
|
|
|
m_dumbCache = m_globalDumbCache;
|
2007-08-14 22:47:01 +00:00
|
|
|
getRequest = -1;
|
2008-03-01 13:52:02 +00:00
|
|
|
// if a request is cancelled before a reponse is available return some
|
|
|
|
// hint about this in the http response instead of nonsense.
|
2008-05-13 19:38:17 +00:00
|
|
|
m_response = -1;
|
2008-03-01 13:52:02 +00:00
|
|
|
|
2008-03-05 21:12:24 +00:00
|
|
|
// default to global proxy / cache if not empty.
|
|
|
|
// proxy is automatically enabled, disable it by setting an empty proxy
|
|
|
|
// cache is enabled to be in line, can get disabled with setCache(bool)
|
|
|
|
if(!m_globalProxy.isEmpty())
|
|
|
|
setProxy(m_globalProxy);
|
|
|
|
m_usecache = false;
|
|
|
|
m_cachedir = m_globalCache;
|
2007-07-25 20:21:06 +00:00
|
|
|
connect(&http, SIGNAL(done(bool)), this, SLOT(httpDone(bool)));
|
|
|
|
connect(&http, SIGNAL(dataReadProgress(int, int)), this, SLOT(httpProgress(int, int)));
|
|
|
|
connect(&http, SIGNAL(requestFinished(int, bool)), this, SLOT(httpFinished(int, bool)));
|
|
|
|
connect(&http, SIGNAL(responseHeaderReceived(const QHttpResponseHeader&)), this, SLOT(httpResponseHeader(const QHttpResponseHeader&)));
|
2007-08-14 22:47:01 +00:00
|
|
|
connect(&http, SIGNAL(stateChanged(int)), this, SLOT(httpState(int)));
|
2007-08-15 13:28:15 +00:00
|
|
|
connect(&http, SIGNAL(requestStarted(int)), this, SLOT(httpStarted(int)));
|
2007-08-14 22:47:01 +00:00
|
|
|
|
|
|
|
connect(&http, SIGNAL(readyRead(const QHttpResponseHeader&)), this, SLOT(httpResponseHeader(const QHttpResponseHeader&)));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-04-13 18:43:51 +00:00
|
|
|
//! @brief set cache path
|
|
|
|
// @param d new directory to use as cache path
|
2007-08-27 17:40:35 +00:00
|
|
|
void HttpGet::setCache(QDir d)
|
|
|
|
{
|
|
|
|
m_cachedir = d;
|
2008-04-06 17:20:13 +00:00
|
|
|
bool result;
|
|
|
|
result = initializeCache(d);
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]"<< __func__ << "(QDir)" << d.absolutePath() << result;
|
2007-09-14 21:55:54 +00:00
|
|
|
m_usecache = result;
|
2007-08-27 17:40:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-04-13 18:43:51 +00:00
|
|
|
/** @brief enable / disable cache useage
|
|
|
|
* @param c set cache usage
|
|
|
|
*/
|
2007-08-27 17:40:35 +00:00
|
|
|
void HttpGet::setCache(bool c)
|
|
|
|
{
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "(bool) =" << c;
|
2007-08-27 17:40:35 +00:00
|
|
|
m_usecache = c;
|
2008-04-06 17:20:13 +00:00
|
|
|
// make sure cache is initialized
|
|
|
|
if(c)
|
|
|
|
m_usecache = initializeCache(m_cachedir);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool HttpGet::initializeCache(const QDir& d)
|
|
|
|
{
|
|
|
|
bool result;
|
|
|
|
QString p = d.absolutePath() + "/rbutil-cache";
|
|
|
|
if(QFileInfo(d.absolutePath()).isDir())
|
|
|
|
{
|
|
|
|
if(!QFileInfo(p).isDir())
|
|
|
|
result = d.mkdir("rbutil-cache");
|
2008-04-06 18:00:32 +00:00
|
|
|
else
|
|
|
|
result = true;
|
2008-04-06 17:20:13 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
result = false;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
2007-08-27 17:40:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-05-13 19:38:17 +00:00
|
|
|
/** @brief read all downloaded data into a buffer
|
|
|
|
* @return data
|
|
|
|
*/
|
2007-08-14 22:47:01 +00:00
|
|
|
QByteArray HttpGet::readAll()
|
|
|
|
{
|
|
|
|
return dataBuffer;
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-04-13 18:43:51 +00:00
|
|
|
/** @brief get http error
|
|
|
|
* @return http error
|
|
|
|
*/
|
2007-07-25 20:21:06 +00:00
|
|
|
QHttp::Error HttpGet::error()
|
|
|
|
{
|
|
|
|
return http.error();
|
|
|
|
}
|
|
|
|
|
2007-08-14 22:47:01 +00:00
|
|
|
|
2007-07-25 20:21:06 +00:00
|
|
|
void HttpGet::httpProgress(int read, int total)
|
|
|
|
{
|
|
|
|
emit dataReadProgress(read, total);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::setProxy(const QUrl &proxy)
|
|
|
|
{
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "(QUrl)" << proxy.toString();
|
2008-03-05 21:12:24 +00:00
|
|
|
m_proxy = proxy;
|
|
|
|
http.setProxy(m_proxy.host(), m_proxy.port(), m_proxy.userName(), m_proxy.password());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::setProxy(bool enable)
|
|
|
|
{
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "(bool)" << enable;
|
2008-03-05 21:12:24 +00:00
|
|
|
if(enable)
|
|
|
|
http.setProxy(m_proxy.host(), m_proxy.port(), m_proxy.userName(), m_proxy.password());
|
|
|
|
else
|
|
|
|
http.setProxy("", 0);
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::setFile(QFile *file)
|
|
|
|
{
|
|
|
|
outputFile = file;
|
2007-08-15 13:28:15 +00:00
|
|
|
outputToBuffer = false;
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "(QFile*)" << outputFile->fileName();
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::abort()
|
|
|
|
{
|
|
|
|
http.abort();
|
2007-08-15 13:28:15 +00:00
|
|
|
if(!outputToBuffer)
|
2007-08-14 22:47:01 +00:00
|
|
|
outputFile->close();
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool HttpGet::getFile(const QUrl &url)
|
|
|
|
{
|
|
|
|
if (!url.isValid()) {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] Error: Invalid URL" << endl;
|
2007-07-25 20:21:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (url.scheme() != "http") {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] Error: URL must start with 'http:'" << endl;
|
2007-07-25 20:21:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (url.path().isEmpty()) {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] Error: URL has no path" << endl;
|
2007-07-25 20:21:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
2007-08-14 22:47:01 +00:00
|
|
|
// if no output file was set write to buffer
|
2007-08-15 13:28:15 +00:00
|
|
|
if(!outputToBuffer) {
|
2007-08-14 22:47:01 +00:00
|
|
|
if (!outputFile->open(QIODevice::ReadWrite)) {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] Error: Cannot open " << qPrintable(outputFile->fileName())
|
2007-08-14 22:47:01 +00:00
|
|
|
<< " for writing: " << qPrintable(outputFile->errorString())
|
|
|
|
<< endl;
|
|
|
|
return false;
|
|
|
|
}
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] downloading" << url.toEncoded();
|
|
|
|
// create request
|
|
|
|
http.setHost(url.host(), url.port(80));
|
|
|
|
// construct query (if any)
|
|
|
|
QList<QPair<QString, QString> > qitems = url.queryItems();
|
|
|
|
if(url.hasQuery()) {
|
|
|
|
m_query = "?";
|
|
|
|
for(int i = 0; i < qitems.size(); i++)
|
|
|
|
m_query += QUrl::toPercentEncoding(qitems.at(i).first, "/") + "="
|
|
|
|
+ QUrl::toPercentEncoding(qitems.at(i).second, "/") + "&";
|
|
|
|
}
|
|
|
|
|
|
|
|
// create hash used for caching
|
|
|
|
m_hash = QCryptographicHash::hash(url.toEncoded(), QCryptographicHash::Md5).toHex();
|
|
|
|
m_path = QString(QUrl::toPercentEncoding(url.path(), "/"));
|
|
|
|
|
2008-05-17 19:36:54 +00:00
|
|
|
if(m_dumbCache || !m_usecache) {
|
2008-05-13 19:38:17 +00:00
|
|
|
getFileFinish();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// request HTTP header
|
|
|
|
connect(this, SIGNAL(headerFinished()), this, SLOT(getFileFinish()));
|
|
|
|
headRequest = http.head(m_path + m_query);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::getFileFinish()
|
|
|
|
{
|
|
|
|
m_cachefile = m_cachedir.absolutePath() + "/rbutil-cache/" + m_hash;
|
2007-08-27 17:40:35 +00:00
|
|
|
if(m_usecache) {
|
|
|
|
// check if the file is present in cache
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] cache ENABLED";
|
|
|
|
QFileInfo cachefile = QFileInfo(m_cachefile);
|
|
|
|
if(cachefile.isReadable()
|
|
|
|
&& cachefile.size() > 0
|
|
|
|
&& cachefile.lastModified() > m_serverTimestamp) {
|
|
|
|
|
|
|
|
qDebug() << "[HTTP] cached file found:" << m_cachefile;
|
|
|
|
|
2007-08-27 17:40:35 +00:00
|
|
|
getRequest = -1;
|
2008-05-13 19:38:17 +00:00
|
|
|
QFile c(m_cachefile);
|
2007-08-27 17:40:35 +00:00
|
|
|
if(!outputToBuffer) {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] copying cache file to output" << outputFile->fileName();
|
2007-08-27 17:40:35 +00:00
|
|
|
c.open(QIODevice::ReadOnly);
|
|
|
|
outputFile->open(QIODevice::ReadWrite);
|
|
|
|
outputFile->write(c.readAll());
|
|
|
|
outputFile->close();
|
|
|
|
c.close();
|
|
|
|
}
|
|
|
|
else {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] reading cache file into buffer";
|
2007-08-27 17:40:35 +00:00
|
|
|
c.open(QIODevice::ReadOnly);
|
|
|
|
dataBuffer = c.readAll();
|
|
|
|
c.close();
|
|
|
|
}
|
2008-05-13 19:38:17 +00:00
|
|
|
m_response = 200; // fake "200 OK" HTTP response
|
|
|
|
m_cached = true;
|
|
|
|
httpDone(false); // we're done now. Fake http "done" signal.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if(cachefile.isReadable())
|
|
|
|
qDebug() << "[HTTP] file in cache timestamp:" << cachefile.lastModified();
|
|
|
|
else
|
|
|
|
qDebug() << "[HTTP] file not in cache.";
|
|
|
|
qDebug() << "[HTTP] server file timestamp:" << m_serverTimestamp;
|
|
|
|
qDebug() << "[HTTP] downloading file to" << m_cachefile;
|
|
|
|
// unlink old cache file
|
|
|
|
if(cachefile.isReadable())
|
|
|
|
QFile(m_cachefile).remove();
|
2007-08-27 17:40:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
qDebug() << "[HTTP] cache DISABLED";
|
|
|
|
}
|
2008-03-05 21:12:24 +00:00
|
|
|
|
2007-08-15 13:28:15 +00:00
|
|
|
if(outputToBuffer) {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] downloading to buffer.";
|
|
|
|
getRequest = http.get(m_path + m_query);
|
2007-08-14 22:47:01 +00:00
|
|
|
}
|
|
|
|
else {
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] downloading to file:"
|
|
|
|
<< qPrintable(outputFile->fileName());
|
|
|
|
getRequest = http.get(m_path + m_query, outputFile);
|
2007-08-14 22:47:01 +00:00
|
|
|
}
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] GET request scheduled, id:" << getRequest;
|
2007-09-14 22:13:46 +00:00
|
|
|
|
2008-05-13 19:38:17 +00:00
|
|
|
return;
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
2007-08-14 22:47:01 +00:00
|
|
|
|
2007-07-25 20:21:06 +00:00
|
|
|
void HttpGet::httpDone(bool error)
|
|
|
|
{
|
|
|
|
if (error) {
|
2007-08-27 17:40:35 +00:00
|
|
|
qDebug() << "[HTTP] Error: " << qPrintable(http.errorString()) << httpResponse();
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
2007-08-15 14:15:24 +00:00
|
|
|
if(!outputToBuffer)
|
2007-08-14 22:47:01 +00:00
|
|
|
outputFile->close();
|
2007-08-27 17:40:35 +00:00
|
|
|
|
2008-05-13 19:38:17 +00:00
|
|
|
if(m_usecache && !m_cached) {
|
|
|
|
qDebug() << "[HTTP] creating cache file" << m_cachefile;
|
|
|
|
QFile c(m_cachefile);
|
2007-08-27 17:40:35 +00:00
|
|
|
c.open(QIODevice::ReadWrite);
|
|
|
|
if(!outputToBuffer) {
|
|
|
|
outputFile->open(QIODevice::ReadOnly | QIODevice::Truncate);
|
|
|
|
c.write(outputFile->readAll());
|
|
|
|
outputFile->close();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
c.write(dataBuffer);
|
|
|
|
|
|
|
|
c.close();
|
|
|
|
}
|
2008-05-13 19:38:17 +00:00
|
|
|
m_serverTimestamp = QDateTime();
|
|
|
|
// take care of concurring requests. If there is still one running,
|
|
|
|
// don't emit done(). That request will call this slot again.
|
|
|
|
if(http.currentId() == 0 && !http.hasPendingRequests())
|
|
|
|
emit done(error);
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::httpFinished(int id, bool error)
|
|
|
|
{
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "(int, bool) =" << id << error;
|
|
|
|
if(id == getRequest) {
|
|
|
|
dataBuffer = http.readAll();
|
|
|
|
|
|
|
|
emit requestFinished(id, error);
|
|
|
|
}
|
|
|
|
qDebug() << "[HTTP] hasPendingRequests =" << http.hasPendingRequests();
|
|
|
|
|
2007-07-25 20:21:06 +00:00
|
|
|
|
2008-05-13 19:38:17 +00:00
|
|
|
if(id == headRequest) {
|
|
|
|
QHttpResponseHeader h = http.lastResponse();
|
|
|
|
|
|
|
|
QString date = h.value("Last-Modified").simplified();
|
|
|
|
if(date.isEmpty()) {
|
|
|
|
m_serverTimestamp = QDateTime(); // no value = invalid
|
|
|
|
emit headerFinished();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// to successfully parse the date strip weekday and timezone
|
|
|
|
date.remove(0, date.indexOf(" ") + 1);
|
|
|
|
if(date.endsWith("GMT"))
|
|
|
|
date.truncate(date.indexOf(" GMT"));
|
|
|
|
// distinguish input formats (see RFC1945)
|
|
|
|
// RFC 850
|
|
|
|
if(date.contains("-"))
|
|
|
|
m_serverTimestamp = QDateTime::fromString(date, "dd-MMM-yy hh:mm:ss");
|
|
|
|
// asctime format
|
|
|
|
else if(date.at(0).isLetter())
|
|
|
|
m_serverTimestamp = QDateTime::fromString(date, "MMM d hh:mm:ss yyyy");
|
|
|
|
// RFC 822
|
|
|
|
else
|
|
|
|
m_serverTimestamp = QDateTime::fromString(date, "dd MMM yyyy hh:mm:ss");
|
|
|
|
qDebug() << "[HTTP] Header Request Date:" << date << ", parsed:" << m_serverTimestamp;
|
|
|
|
emit headerFinished();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(id == getRequest)
|
|
|
|
emit requestFinished(id, error);
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
2007-08-14 22:47:01 +00:00
|
|
|
void HttpGet::httpStarted(int id)
|
|
|
|
{
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "(int) =" << id;
|
|
|
|
qDebug() << "headRequest" << headRequest << "getRequest" << getRequest;
|
2007-08-14 22:47:01 +00:00
|
|
|
}
|
|
|
|
|
2007-07-25 20:21:06 +00:00
|
|
|
|
|
|
|
QString HttpGet::errorString()
|
|
|
|
{
|
|
|
|
return http.errorString();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::httpResponseHeader(const QHttpResponseHeader &resp)
|
|
|
|
{
|
2007-08-14 22:47:01 +00:00
|
|
|
// if there is a network error abort all scheduled requests for
|
|
|
|
// this download
|
2008-05-13 19:38:17 +00:00
|
|
|
m_response = resp.statusCode();
|
|
|
|
if(m_response != 200) {
|
|
|
|
qDebug() << "[HTTP] response error =" << m_response << resp.reasonPhrase();
|
2007-08-15 20:08:02 +00:00
|
|
|
http.abort();
|
|
|
|
}
|
|
|
|
// 301 -- moved permanently
|
2007-08-15 20:30:36 +00:00
|
|
|
// 302 -- found
|
2007-08-15 20:08:02 +00:00
|
|
|
// 303 -- see other
|
|
|
|
// 307 -- moved temporarily
|
|
|
|
// in all cases, header: location has the correct address so we can follow.
|
2008-05-13 19:38:17 +00:00
|
|
|
if(m_response == 301 || m_response == 302 || m_response == 303 || m_response == 307) {
|
2007-08-15 20:08:02 +00:00
|
|
|
// start new request with new url
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP] response =" << m_response << "- following";
|
|
|
|
getFile(resp.value("location") + m_query);
|
2007-08-15 20:08:02 +00:00
|
|
|
}
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int HttpGet::httpResponse()
|
|
|
|
{
|
2008-05-13 19:38:17 +00:00
|
|
|
return m_response;
|
2007-07-25 20:21:06 +00:00
|
|
|
}
|
2007-08-14 22:47:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
void HttpGet::httpState(int state)
|
|
|
|
{
|
|
|
|
QString s[] = {"Unconnected", "HostLookup", "Connecting", "Sending",
|
|
|
|
"Reading", "Connected", "Closing"};
|
|
|
|
if(state <= 6)
|
2008-05-13 19:38:17 +00:00
|
|
|
qDebug() << "[HTTP]" << __func__ << "() = " << s[state];
|
|
|
|
else qDebug() << "[HTTP]" << __func__ << "() = " << state;
|
2007-08-14 22:47:01 +00:00
|
|
|
}
|
|
|
|
|