browsercore/core/network/networkdiskcache.cpp
author hgs
Tue, 29 Jun 2010 00:46:29 -0400
changeset 3 0954f5dd2cd0
permissions -rw-r--r--
201026
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
3
hgs
parents:
diff changeset
     1
/****************************************************************************
hgs
parents:
diff changeset
     2
**
hgs
parents:
diff changeset
     3
** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies).
hgs
parents:
diff changeset
     4
** All rights reserved.
hgs
parents:
diff changeset
     5
** Contact: Nokia Corporation (qt-info@nokia.com)
hgs
parents:
diff changeset
     6
**
hgs
parents:
diff changeset
     7
** This file is part of the QtNetwork module of the Qt Toolkit.
hgs
parents:
diff changeset
     8
**
hgs
parents:
diff changeset
     9
** $QT_BEGIN_LICENSE:LGPL$
hgs
parents:
diff changeset
    10
** Commercial Usage
hgs
parents:
diff changeset
    11
** Licensees holding valid Qt Commercial licenses may use this file in
hgs
parents:
diff changeset
    12
** accordance with the Qt Commercial License Agreement provided with the
hgs
parents:
diff changeset
    13
** Software or, alternatively, in accordance with the terms contained in
hgs
parents:
diff changeset
    14
** a written agreement between you and Nokia.
hgs
parents:
diff changeset
    15
**
hgs
parents:
diff changeset
    16
** GNU Lesser General Public License Usage
hgs
parents:
diff changeset
    17
** Alternatively, this file may be used under the terms of the GNU Lesser
hgs
parents:
diff changeset
    18
** General Public License version 2.1 as published by the Free Software
hgs
parents:
diff changeset
    19
** Foundation and appearing in the file LICENSE.LGPL included in the
hgs
parents:
diff changeset
    20
** packaging of this file.  Please review the following information to
hgs
parents:
diff changeset
    21
** ensure the GNU Lesser General Public License version 2.1 requirements
hgs
parents:
diff changeset
    22
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
hgs
parents:
diff changeset
    23
**
hgs
parents:
diff changeset
    24
** In addition, as a special exception, Nokia gives you certain additional
hgs
parents:
diff changeset
    25
** rights.  These rights are described in the Nokia Qt LGPL Exception
hgs
parents:
diff changeset
    26
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
hgs
parents:
diff changeset
    27
**
hgs
parents:
diff changeset
    28
** GNU General Public License Usage
hgs
parents:
diff changeset
    29
** Alternatively, this file may be used under the terms of the GNU
hgs
parents:
diff changeset
    30
** General Public License version 3.0 as published by the Free Software
hgs
parents:
diff changeset
    31
** Foundation and appearing in the file LICENSE.GPL included in the
hgs
parents:
diff changeset
    32
** packaging of this file.  Please review the following information to
hgs
parents:
diff changeset
    33
** ensure the GNU General Public License version 3.0 requirements will be
hgs
parents:
diff changeset
    34
** met: http://www.gnu.org/copyleft/gpl.html.
hgs
parents:
diff changeset
    35
**
hgs
parents:
diff changeset
    36
** If you have questions regarding the use of this file, please contact
hgs
parents:
diff changeset
    37
** Nokia at qt-info@nokia.com.
hgs
parents:
diff changeset
    38
** $QT_END_LICENSE$
hgs
parents:
diff changeset
    39
**
hgs
parents:
diff changeset
    40
****************************************************************************/
hgs
parents:
diff changeset
    41
hgs
parents:
diff changeset
    42
//#define QNETWORKDISKCACHE_DEBUG
hgs
parents:
diff changeset
    43
hgs
parents:
diff changeset
    44
#ifndef QT_NO_NETWORKDISKCACHE
hgs
parents:
diff changeset
    45
hgs
parents:
diff changeset
    46
#include "networkdiskcache.h"
hgs
parents:
diff changeset
    47
#include "networkdiskcache_p.h"
hgs
parents:
diff changeset
    48
#include "QtCore/qscopedpointer.h"
hgs
parents:
diff changeset
    49
hgs
parents:
diff changeset
    50
#include <qfile.h>
hgs
parents:
diff changeset
    51
#include <qdir.h>
hgs
parents:
diff changeset
    52
#include <qdatetime.h>
hgs
parents:
diff changeset
    53
#include <qdiriterator.h>
hgs
parents:
diff changeset
    54
#include <qurl.h>
hgs
parents:
diff changeset
    55
hgs
parents:
diff changeset
    56
#include <qdebug.h>
hgs
parents:
diff changeset
    57
hgs
parents:
diff changeset
    58
hgs
parents:
diff changeset
    59
#define MAX_COMPRESSION_SIZE (1024 * 1024 * 3)
hgs
parents:
diff changeset
    60
#define CACHE_SUBDIR_COUNT 16
hgs
parents:
diff changeset
    61
hgs
parents:
diff changeset
    62
QT_BEGIN_NAMESPACE
hgs
parents:
diff changeset
    63
hgs
parents:
diff changeset
    64
/*!
hgs
parents:
diff changeset
    65
    \class QNetworkDiskCache
hgs
parents:
diff changeset
    66
    \since 4.5
hgs
parents:
diff changeset
    67
    \inmodule QtNetwork
hgs
parents:
diff changeset
    68
hgs
parents:
diff changeset
    69
    \brief The QNetworkDiskCache class provides a very basic disk cache.
hgs
parents:
diff changeset
    70
hgs
parents:
diff changeset
    71
    QNetworkDiskCache stores each url in its own file inside of the
hgs
parents:
diff changeset
    72
    cacheDirectory using QDataStream.  Files with a text MimeType
hgs
parents:
diff changeset
    73
    are compressed using qCompress.  Each cache file starts with "cache_"
hgs
parents:
diff changeset
    74
    and ends in ".cache".  Data is written to disk only in insert()
hgs
parents:
diff changeset
    75
    and updateMetaData().
hgs
parents:
diff changeset
    76
hgs
parents:
diff changeset
    77
    Currently you can not share the same cache files with more then
hgs
parents:
diff changeset
    78
    one disk cache.
hgs
parents:
diff changeset
    79
hgs
parents:
diff changeset
    80
    QNetworkDiskCache by default limits the amount of space that the cache will
hgs
parents:
diff changeset
    81
    use on the system to 50MB.
hgs
parents:
diff changeset
    82
hgs
parents:
diff changeset
    83
    Note you have to set the cache directory before it will work.
hgs
parents:
diff changeset
    84
hgs
parents:
diff changeset
    85
    A network disk cache can be enabled by:
hgs
parents:
diff changeset
    86
hgs
parents:
diff changeset
    87
    \snippet doc/src/snippets/code/src_network_access_qnetworkdiskcache.cpp 0
hgs
parents:
diff changeset
    88
hgs
parents:
diff changeset
    89
    When sending requests, to control the preference of when to use the cache
hgs
parents:
diff changeset
    90
    and when to use the network, consider the following:
hgs
parents:
diff changeset
    91
hgs
parents:
diff changeset
    92
    \snippet doc/src/snippets/code/src_network_access_qnetworkdiskcache.cpp 1
hgs
parents:
diff changeset
    93
hgs
parents:
diff changeset
    94
    To check whether the response came from the cache or from the network, the
hgs
parents:
diff changeset
    95
    following can be applied:
hgs
parents:
diff changeset
    96
hgs
parents:
diff changeset
    97
    \snippet doc/src/snippets/code/src_network_access_qnetworkdiskcache.cpp 2
hgs
parents:
diff changeset
    98
*/
hgs
parents:
diff changeset
    99
hgs
parents:
diff changeset
   100
/*!
hgs
parents:
diff changeset
   101
    Creates a new disk cache. The \a parent argument is passed to
hgs
parents:
diff changeset
   102
    QAbstractNetworkCache's constructor.
hgs
parents:
diff changeset
   103
 */
hgs
parents:
diff changeset
   104
NetworkDiskCache::NetworkDiskCache(QObject *parent)
hgs
parents:
diff changeset
   105
    : QAbstractNetworkCache(*new NetworkDiskCachePrivate, parent)
hgs
parents:
diff changeset
   106
{
hgs
parents:
diff changeset
   107
}
hgs
parents:
diff changeset
   108
hgs
parents:
diff changeset
   109
/*!
hgs
parents:
diff changeset
   110
    Destroys the cache object.  This does not clear the disk cache.
hgs
parents:
diff changeset
   111
 */
hgs
parents:
diff changeset
   112
NetworkDiskCache::~NetworkDiskCache()
hgs
parents:
diff changeset
   113
{
hgs
parents:
diff changeset
   114
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   115
    QHashIterator<QIODevice*, QCacheItem*> it(d->inserting);
hgs
parents:
diff changeset
   116
    while (it.hasNext()) {
hgs
parents:
diff changeset
   117
        it.next();
hgs
parents:
diff changeset
   118
        delete it.value();
hgs
parents:
diff changeset
   119
    }
hgs
parents:
diff changeset
   120
}
hgs
parents:
diff changeset
   121
hgs
parents:
diff changeset
   122
/*!
hgs
parents:
diff changeset
   123
    Returns the location where cached files will be stored.
hgs
parents:
diff changeset
   124
*/
hgs
parents:
diff changeset
   125
QString NetworkDiskCache::cacheDirectory() const
hgs
parents:
diff changeset
   126
{
hgs
parents:
diff changeset
   127
    Q_D(const NetworkDiskCache);
hgs
parents:
diff changeset
   128
    return d->cacheDirectory;
hgs
parents:
diff changeset
   129
}
hgs
parents:
diff changeset
   130
hgs
parents:
diff changeset
   131
/*!
hgs
parents:
diff changeset
   132
    Sets the directory where cached files will be stored to \a cacheDir
hgs
parents:
diff changeset
   133
hgs
parents:
diff changeset
   134
    QNetworkDiskCache will create this directory if it does not exists.
hgs
parents:
diff changeset
   135
hgs
parents:
diff changeset
   136
    Prepared cache items will be stored in the new cache directory when
hgs
parents:
diff changeset
   137
    they are inserted.
hgs
parents:
diff changeset
   138
hgs
parents:
diff changeset
   139
    \sa QDesktopServices::CacheLocation
hgs
parents:
diff changeset
   140
*/
hgs
parents:
diff changeset
   141
void NetworkDiskCache::setCacheDirectory(const QString &cacheDir)
hgs
parents:
diff changeset
   142
{
hgs
parents:
diff changeset
   143
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   144
    qDebug() << "NetworkDiskCache::setCacheDirectory()" << cacheDir;
hgs
parents:
diff changeset
   145
#endif
hgs
parents:
diff changeset
   146
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   147
    if (cacheDir.isEmpty())
hgs
parents:
diff changeset
   148
        return;
hgs
parents:
diff changeset
   149
    d->cacheDirectory = cacheDir;
hgs
parents:
diff changeset
   150
    QDir cDir(d->cacheDirectory);
hgs
parents:
diff changeset
   151
    d->cacheDirectory = cDir.absolutePath();
hgs
parents:
diff changeset
   152
    if (!d->cacheDirectory.endsWith(QLatin1Char('/')))
hgs
parents:
diff changeset
   153
        d->cacheDirectory += QLatin1Char('/');
hgs
parents:
diff changeset
   154
hgs
parents:
diff changeset
   155
    QDir dir;
hgs
parents:
diff changeset
   156
    // Setup and create directories
hgs
parents:
diff changeset
   157
    if (!QFile::exists(d->cacheDirectory)) 
hgs
parents:
diff changeset
   158
    {
hgs
parents:
diff changeset
   159
        // ### make a static QDir function for this...
hgs
parents:
diff changeset
   160
        dir.mkpath(d->cacheDirectory);
hgs
parents:
diff changeset
   161
    }
hgs
parents:
diff changeset
   162
hgs
parents:
diff changeset
   163
    QString subDirectory;
hgs
parents:
diff changeset
   164
    for (int i = 0; i < CACHE_SUBDIR_COUNT; i++)
hgs
parents:
diff changeset
   165
    {
hgs
parents:
diff changeset
   166
        subDirectory = d->cacheDirectory + QString("%1").arg(i, 0, 16) + QLatin1Char('/');
hgs
parents:
diff changeset
   167
        if (!QFile::exists(subDirectory)) 
hgs
parents:
diff changeset
   168
        {
hgs
parents:
diff changeset
   169
    	    dir.mkpath(subDirectory);
hgs
parents:
diff changeset
   170
        }
hgs
parents:
diff changeset
   171
    }
hgs
parents:
diff changeset
   172
hgs
parents:
diff changeset
   173
    // For Temporary Prepared Directory
hgs
parents:
diff changeset
   174
    dir.mkpath(d->cacheDirectory + QLatin1String("prepared/"));
hgs
parents:
diff changeset
   175
}
hgs
parents:
diff changeset
   176
hgs
parents:
diff changeset
   177
/*!
hgs
parents:
diff changeset
   178
    \reimp
hgs
parents:
diff changeset
   179
*/
hgs
parents:
diff changeset
   180
qint64 NetworkDiskCache::cacheSize() const
hgs
parents:
diff changeset
   181
{
hgs
parents:
diff changeset
   182
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   183
    qDebug() << "NetworkDiskCache::cacheSize()";
hgs
parents:
diff changeset
   184
#endif
hgs
parents:
diff changeset
   185
    Q_D(const NetworkDiskCache);
hgs
parents:
diff changeset
   186
    if (d->cacheDirectory.isEmpty())
hgs
parents:
diff changeset
   187
        return 0;
hgs
parents:
diff changeset
   188
    if (d->currentCacheSize < 0) {
hgs
parents:
diff changeset
   189
        NetworkDiskCache *that = const_cast<NetworkDiskCache*>(this);
hgs
parents:
diff changeset
   190
        that->d_func()->currentCacheSize = that->expire();
hgs
parents:
diff changeset
   191
    }
hgs
parents:
diff changeset
   192
    return d->currentCacheSize;
hgs
parents:
diff changeset
   193
}
hgs
parents:
diff changeset
   194
hgs
parents:
diff changeset
   195
/*!
hgs
parents:
diff changeset
   196
    \reimp
hgs
parents:
diff changeset
   197
*/
hgs
parents:
diff changeset
   198
QIODevice *NetworkDiskCache::prepare(const QNetworkCacheMetaData &metaData)
hgs
parents:
diff changeset
   199
{
hgs
parents:
diff changeset
   200
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   201
    qDebug() << "NetworkDiskCache::prepare()" << metaData.url();
hgs
parents:
diff changeset
   202
#endif
hgs
parents:
diff changeset
   203
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   204
    if (!metaData.isValid() || !metaData.url().isValid() || !metaData.saveToDisk())
hgs
parents:
diff changeset
   205
        return 0;
hgs
parents:
diff changeset
   206
hgs
parents:
diff changeset
   207
    if (d->cacheDirectory.isEmpty()) {
hgs
parents:
diff changeset
   208
        qWarning() << "NetworkDiskCache::prepare() The cache directory is not set";
hgs
parents:
diff changeset
   209
        return 0;
hgs
parents:
diff changeset
   210
    }
hgs
parents:
diff changeset
   211
hgs
parents:
diff changeset
   212
    foreach (QNetworkCacheMetaData::RawHeader header, metaData.rawHeaders()) {
hgs
parents:
diff changeset
   213
        if (header.first.toLower() == "content-length") {
hgs
parents:
diff changeset
   214
            qint64 size = header.second.toInt();
hgs
parents:
diff changeset
   215
            if (size > (maximumCacheSize() * 3)/4)
hgs
parents:
diff changeset
   216
                return 0;
hgs
parents:
diff changeset
   217
            break;
hgs
parents:
diff changeset
   218
        }
hgs
parents:
diff changeset
   219
    }
hgs
parents:
diff changeset
   220
    QScopedPointer<QCacheItem> cacheItem(new QCacheItem);
hgs
parents:
diff changeset
   221
    cacheItem->metaData = metaData;
hgs
parents:
diff changeset
   222
hgs
parents:
diff changeset
   223
    QIODevice *device = 0;
hgs
parents:
diff changeset
   224
    if (cacheItem->canCompress()) {
hgs
parents:
diff changeset
   225
        cacheItem->data.open(QBuffer::ReadWrite);
hgs
parents:
diff changeset
   226
        device = &(cacheItem->data);
hgs
parents:
diff changeset
   227
    } else {
hgs
parents:
diff changeset
   228
        QString templateName = d->tmpCacheFileName();
hgs
parents:
diff changeset
   229
        QT_TRY {
hgs
parents:
diff changeset
   230
            cacheItem->file = new QTemporaryFile(templateName, &cacheItem->data);
hgs
parents:
diff changeset
   231
        } QT_CATCH(...) {
hgs
parents:
diff changeset
   232
            cacheItem->file = 0;
hgs
parents:
diff changeset
   233
        }
hgs
parents:
diff changeset
   234
        if (!cacheItem->file || !cacheItem->file->open()) {
hgs
parents:
diff changeset
   235
            qWarning() << "NetworkDiskCache::prepare() unable to open temporary file";
hgs
parents:
diff changeset
   236
            cacheItem.reset();
hgs
parents:
diff changeset
   237
            return 0;
hgs
parents:
diff changeset
   238
        }
hgs
parents:
diff changeset
   239
        cacheItem->writeHeader(cacheItem->file);
hgs
parents:
diff changeset
   240
        device = cacheItem->file;
hgs
parents:
diff changeset
   241
    }
hgs
parents:
diff changeset
   242
    d->inserting[device] = cacheItem.take();
hgs
parents:
diff changeset
   243
    return device;
hgs
parents:
diff changeset
   244
}
hgs
parents:
diff changeset
   245
hgs
parents:
diff changeset
   246
/*!
hgs
parents:
diff changeset
   247
    \reimp
hgs
parents:
diff changeset
   248
*/
hgs
parents:
diff changeset
   249
void NetworkDiskCache::insert(QIODevice *device)
hgs
parents:
diff changeset
   250
{
hgs
parents:
diff changeset
   251
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   252
    qDebug() << "NetworkDiskCache::insert()" << device;
hgs
parents:
diff changeset
   253
#endif
hgs
parents:
diff changeset
   254
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   255
    QHash<QIODevice*, QCacheItem*>::iterator it = d->inserting.find(device);
hgs
parents:
diff changeset
   256
    if (it == d->inserting.end()) {
hgs
parents:
diff changeset
   257
        qWarning() << "NetworkDiskCache::insert() called on a device we don't know about" << device;
hgs
parents:
diff changeset
   258
        return;
hgs
parents:
diff changeset
   259
    }
hgs
parents:
diff changeset
   260
hgs
parents:
diff changeset
   261
    d->storeItem(it.value());
hgs
parents:
diff changeset
   262
    delete it.value();
hgs
parents:
diff changeset
   263
    d->inserting.erase(it);
hgs
parents:
diff changeset
   264
}
hgs
parents:
diff changeset
   265
hgs
parents:
diff changeset
   266
// CRC32 implementation.
hgs
parents:
diff changeset
   267
// Could be made into new API QByteArray:qChecksum32()
hgs
parents:
diff changeset
   268
static const quint32 crc_tbl32[256] = {
hgs
parents:
diff changeset
   269
	0x00000000, 0x77073096, 0xee0e612c, 0x990951ba,
hgs
parents:
diff changeset
   270
	0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
hgs
parents:
diff changeset
   271
	0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
hgs
parents:
diff changeset
   272
	0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
hgs
parents:
diff changeset
   273
	0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de,
hgs
parents:
diff changeset
   274
	0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
hgs
parents:
diff changeset
   275
	0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec,
hgs
parents:
diff changeset
   276
	0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
hgs
parents:
diff changeset
   277
	0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
hgs
parents:
diff changeset
   278
	0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
hgs
parents:
diff changeset
   279
	0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940,
hgs
parents:
diff changeset
   280
	0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
hgs
parents:
diff changeset
   281
	0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116,
hgs
parents:
diff changeset
   282
	0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
hgs
parents:
diff changeset
   283
	0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
hgs
parents:
diff changeset
   284
	0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
hgs
parents:
diff changeset
   285
	0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a,
hgs
parents:
diff changeset
   286
	0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
hgs
parents:
diff changeset
   287
	0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818,
hgs
parents:
diff changeset
   288
	0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
hgs
parents:
diff changeset
   289
	0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
hgs
parents:
diff changeset
   290
	0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
hgs
parents:
diff changeset
   291
	0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c,
hgs
parents:
diff changeset
   292
	0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
hgs
parents:
diff changeset
   293
	0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2,
hgs
parents:
diff changeset
   294
	0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
hgs
parents:
diff changeset
   295
	0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
hgs
parents:
diff changeset
   296
	0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
hgs
parents:
diff changeset
   297
	0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086,
hgs
parents:
diff changeset
   298
	0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
hgs
parents:
diff changeset
   299
	0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4,
hgs
parents:
diff changeset
   300
	0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
hgs
parents:
diff changeset
   301
	0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
hgs
parents:
diff changeset
   302
	0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
hgs
parents:
diff changeset
   303
	0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8,
hgs
parents:
diff changeset
   304
	0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
hgs
parents:
diff changeset
   305
	0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe,
hgs
parents:
diff changeset
   306
	0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
hgs
parents:
diff changeset
   307
	0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
hgs
parents:
diff changeset
   308
	0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
hgs
parents:
diff changeset
   309
	0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252,
hgs
parents:
diff changeset
   310
	0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
hgs
parents:
diff changeset
   311
	0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60,
hgs
parents:
diff changeset
   312
	0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
hgs
parents:
diff changeset
   313
	0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
hgs
parents:
diff changeset
   314
	0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
hgs
parents:
diff changeset
   315
	0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04,
hgs
parents:
diff changeset
   316
	0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
hgs
parents:
diff changeset
   317
	0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a,
hgs
parents:
diff changeset
   318
	0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
hgs
parents:
diff changeset
   319
	0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
hgs
parents:
diff changeset
   320
	0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
hgs
parents:
diff changeset
   321
	0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e,
hgs
parents:
diff changeset
   322
	0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
hgs
parents:
diff changeset
   323
	0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c,
hgs
parents:
diff changeset
   324
	0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
hgs
parents:
diff changeset
   325
	0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
hgs
parents:
diff changeset
   326
	0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
hgs
parents:
diff changeset
   327
	0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0,
hgs
parents:
diff changeset
   328
	0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
hgs
parents:
diff changeset
   329
	0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6,
hgs
parents:
diff changeset
   330
	0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
hgs
parents:
diff changeset
   331
	0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
hgs
parents:
diff changeset
   332
	0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
hgs
parents:
diff changeset
   333
};
hgs
parents:
diff changeset
   334
	
hgs
parents:
diff changeset
   335
quint32 NetworkDiskCachePrivate::crc32(const char *data, uint len)
hgs
parents:
diff changeset
   336
{
hgs
parents:
diff changeset
   337
	const uchar *p = reinterpret_cast<const uchar *>(data);
hgs
parents:
diff changeset
   338
	const uchar *q = p + len;
hgs
parents:
diff changeset
   339
	const quint32 init = 0xFFFFFFFFL;
hgs
parents:
diff changeset
   340
	
hgs
parents:
diff changeset
   341
	quint32 crc32 = init;
hgs
parents:
diff changeset
   342
	while (p < q) {
hgs
parents:
diff changeset
   343
	    crc32 = (crc32 >> 8) ^ crc_tbl32[(crc32 ^ *p++) & 0xffL];
hgs
parents:
diff changeset
   344
	}
hgs
parents:
diff changeset
   345
	return crc32 ^ init ;
hgs
parents:
diff changeset
   346
}
hgs
parents:
diff changeset
   347
hgs
parents:
diff changeset
   348
void NetworkDiskCachePrivate::storeItem(QCacheItem *cacheItem)
hgs
parents:
diff changeset
   349
{
hgs
parents:
diff changeset
   350
    Q_Q(NetworkDiskCache);
hgs
parents:
diff changeset
   351
    
hgs
parents:
diff changeset
   352
    // just an idea of not caching anything more than 15k
hgs
parents:
diff changeset
   353
    //if (cacheItem->size() > (15*1024)) return;
hgs
parents:
diff changeset
   354
    
hgs
parents:
diff changeset
   355
    Q_ASSERT(cacheItem->metaData.saveToDisk());
hgs
parents:
diff changeset
   356
hgs
parents:
diff changeset
   357
    QString fileName = cacheFileName(cacheItem->metaData.url());
hgs
parents:
diff changeset
   358
    Q_ASSERT(!fileName.isEmpty());
hgs
parents:
diff changeset
   359
hgs
parents:
diff changeset
   360
    if (QFile::exists(fileName)) {
hgs
parents:
diff changeset
   361
        if (!QFile::remove(fileName)) {
hgs
parents:
diff changeset
   362
            qWarning() << "NetworkDiskCache: couldn't remove the cache file " << fileName;
hgs
parents:
diff changeset
   363
            return;
hgs
parents:
diff changeset
   364
        }
hgs
parents:
diff changeset
   365
    }
hgs
parents:
diff changeset
   366
hgs
parents:
diff changeset
   367
    if (currentCacheSize > 0)
hgs
parents:
diff changeset
   368
        currentCacheSize += 1024 + cacheItem->size();
hgs
parents:
diff changeset
   369
    currentCacheSize = q->expire();
hgs
parents:
diff changeset
   370
    if (!cacheItem->file) {
hgs
parents:
diff changeset
   371
        QString templateName = tmpCacheFileName();
hgs
parents:
diff changeset
   372
        cacheItem->file = new QTemporaryFile(templateName, &cacheItem->data);
hgs
parents:
diff changeset
   373
        if (cacheItem->file->open()) {
hgs
parents:
diff changeset
   374
            cacheItem->writeHeader(cacheItem->file);
hgs
parents:
diff changeset
   375
            cacheItem->writeCompressedData(cacheItem->file);
hgs
parents:
diff changeset
   376
        }
hgs
parents:
diff changeset
   377
    }
hgs
parents:
diff changeset
   378
hgs
parents:
diff changeset
   379
    if (cacheItem->file
hgs
parents:
diff changeset
   380
        && cacheItem->file->isOpen()
hgs
parents:
diff changeset
   381
        && cacheItem->file->error() == QFile::NoError) {
hgs
parents:
diff changeset
   382
        cacheItem->file->setAutoRemove(false);
hgs
parents:
diff changeset
   383
        // ### use atomic rename rather then remove & rename
hgs
parents:
diff changeset
   384
        if (cacheItem->file->rename(fileName))
hgs
parents:
diff changeset
   385
            currentCacheSize += cacheItem->file->size();
hgs
parents:
diff changeset
   386
        else
hgs
parents:
diff changeset
   387
            cacheItem->file->setAutoRemove(true);
hgs
parents:
diff changeset
   388
    }
hgs
parents:
diff changeset
   389
    if (cacheItem->metaData.url() == lastItem.metaData.url())
hgs
parents:
diff changeset
   390
        lastItem.reset();
hgs
parents:
diff changeset
   391
}
hgs
parents:
diff changeset
   392
hgs
parents:
diff changeset
   393
/*!
hgs
parents:
diff changeset
   394
    \reimp
hgs
parents:
diff changeset
   395
*/
hgs
parents:
diff changeset
   396
bool NetworkDiskCache::remove(const QUrl &url)
hgs
parents:
diff changeset
   397
{
hgs
parents:
diff changeset
   398
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   399
    qDebug() << "NetworkDiskCache::remove()" << url;
hgs
parents:
diff changeset
   400
#endif
hgs
parents:
diff changeset
   401
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   402
hgs
parents:
diff changeset
   403
    // remove is also used to cancel insertions, not a common operation
hgs
parents:
diff changeset
   404
    QHashIterator<QIODevice*, QCacheItem*> it(d->inserting);
hgs
parents:
diff changeset
   405
    while (it.hasNext()) {
hgs
parents:
diff changeset
   406
        it.next();
hgs
parents:
diff changeset
   407
        QCacheItem *item = it.value();
hgs
parents:
diff changeset
   408
        if (item && item->metaData.url() == url) {
hgs
parents:
diff changeset
   409
            delete item;
hgs
parents:
diff changeset
   410
            d->inserting.remove(it.key());
hgs
parents:
diff changeset
   411
            return true;
hgs
parents:
diff changeset
   412
        }
hgs
parents:
diff changeset
   413
    }
hgs
parents:
diff changeset
   414
hgs
parents:
diff changeset
   415
    if (d->lastItem.metaData.url() == url)
hgs
parents:
diff changeset
   416
        d->lastItem.reset();
hgs
parents:
diff changeset
   417
    return d->removeFile(d->cacheFileName(url));
hgs
parents:
diff changeset
   418
}
hgs
parents:
diff changeset
   419
hgs
parents:
diff changeset
   420
/*!
hgs
parents:
diff changeset
   421
    Put all of the misc file removing into one function to be extra safe
hgs
parents:
diff changeset
   422
 */
hgs
parents:
diff changeset
   423
bool NetworkDiskCachePrivate::removeFile(const QString &file)
hgs
parents:
diff changeset
   424
{
hgs
parents:
diff changeset
   425
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   426
    qDebug() << "NetworkDiskCache::removeFile()" << file;
hgs
parents:
diff changeset
   427
#endif
hgs
parents:
diff changeset
   428
    if (file.isEmpty())
hgs
parents:
diff changeset
   429
        return false;
hgs
parents:
diff changeset
   430
    QFileInfo info(file);
hgs
parents:
diff changeset
   431
    QString fileName = info.fileName();
hgs
parents:
diff changeset
   432
    qint64 size = info.size();
hgs
parents:
diff changeset
   433
    if (QFile::remove(file)) {
hgs
parents:
diff changeset
   434
        currentCacheSize -= size;
hgs
parents:
diff changeset
   435
        return true;
hgs
parents:
diff changeset
   436
    }
hgs
parents:
diff changeset
   437
    return false;
hgs
parents:
diff changeset
   438
}
hgs
parents:
diff changeset
   439
hgs
parents:
diff changeset
   440
/*!
hgs
parents:
diff changeset
   441
    \reimp
hgs
parents:
diff changeset
   442
*/
hgs
parents:
diff changeset
   443
QNetworkCacheMetaData NetworkDiskCache::metaData(const QUrl &url)
hgs
parents:
diff changeset
   444
{
hgs
parents:
diff changeset
   445
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   446
    qDebug() << "NetworkDiskCache::metaData()" << url;
hgs
parents:
diff changeset
   447
#endif
hgs
parents:
diff changeset
   448
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   449
    if (d->lastItem.metaData.url() == url)
hgs
parents:
diff changeset
   450
        return d->lastItem.metaData;
hgs
parents:
diff changeset
   451
    return fileMetaData(d->cacheFileName(url));
hgs
parents:
diff changeset
   452
}
hgs
parents:
diff changeset
   453
hgs
parents:
diff changeset
   454
/*!
hgs
parents:
diff changeset
   455
    Returns the QNetworkCacheMetaData for the cache file \a fileName.
hgs
parents:
diff changeset
   456
hgs
parents:
diff changeset
   457
    If \a fileName is not a cache file QNetworkCacheMetaData will be invalid.
hgs
parents:
diff changeset
   458
 */
hgs
parents:
diff changeset
   459
QNetworkCacheMetaData NetworkDiskCache::fileMetaData(const QString &fileName) const
hgs
parents:
diff changeset
   460
{
hgs
parents:
diff changeset
   461
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   462
    qDebug() << "NetworkDiskCache::fileMetaData()" << fileName;
hgs
parents:
diff changeset
   463
#endif
hgs
parents:
diff changeset
   464
    Q_D(const NetworkDiskCache);
hgs
parents:
diff changeset
   465
    QFile file(fileName);
hgs
parents:
diff changeset
   466
    if (!file.open(QFile::ReadOnly))
hgs
parents:
diff changeset
   467
        return QNetworkCacheMetaData();
hgs
parents:
diff changeset
   468
    if (!d->lastItem.read(&file, false)) {
hgs
parents:
diff changeset
   469
        file.close();
hgs
parents:
diff changeset
   470
        NetworkDiskCachePrivate *that = const_cast<NetworkDiskCachePrivate*>(d);
hgs
parents:
diff changeset
   471
        that->removeFile(fileName);
hgs
parents:
diff changeset
   472
    }
hgs
parents:
diff changeset
   473
    return d->lastItem.metaData;
hgs
parents:
diff changeset
   474
}
hgs
parents:
diff changeset
   475
hgs
parents:
diff changeset
   476
/*!
hgs
parents:
diff changeset
   477
    \reimp
hgs
parents:
diff changeset
   478
*/
hgs
parents:
diff changeset
   479
QIODevice *NetworkDiskCache::data(const QUrl &url)
hgs
parents:
diff changeset
   480
{
hgs
parents:
diff changeset
   481
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   482
    qDebug() << "NetworkDiskCache::data()" << url;
hgs
parents:
diff changeset
   483
#endif
hgs
parents:
diff changeset
   484
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   485
    QScopedPointer<QBuffer> buffer;
hgs
parents:
diff changeset
   486
    if (!url.isValid())
hgs
parents:
diff changeset
   487
        return 0;
hgs
parents:
diff changeset
   488
    if (d->lastItem.metaData.url() == url && d->lastItem.data.isOpen()) {
hgs
parents:
diff changeset
   489
        buffer.reset(new QBuffer);
hgs
parents:
diff changeset
   490
        buffer->setData(d->lastItem.data.data());
hgs
parents:
diff changeset
   491
    } else {
hgs
parents:
diff changeset
   492
        QScopedPointer<QFile> file(new QFile(d->cacheFileName(url)));
hgs
parents:
diff changeset
   493
        if (!file->open(QFile::ReadOnly | QIODevice::Unbuffered))
hgs
parents:
diff changeset
   494
            return 0;
hgs
parents:
diff changeset
   495
hgs
parents:
diff changeset
   496
        if (!d->lastItem.read(file.data(), true)) {
hgs
parents:
diff changeset
   497
            file->close();
hgs
parents:
diff changeset
   498
            remove(url);
hgs
parents:
diff changeset
   499
            return 0;
hgs
parents:
diff changeset
   500
        }
hgs
parents:
diff changeset
   501
        if (d->lastItem.data.isOpen()) {
hgs
parents:
diff changeset
   502
            // compressed
hgs
parents:
diff changeset
   503
            buffer.reset(new QBuffer);
hgs
parents:
diff changeset
   504
            buffer->setData(d->lastItem.data.data());
hgs
parents:
diff changeset
   505
        } else {
hgs
parents:
diff changeset
   506
            buffer.reset(new QBuffer);
hgs
parents:
diff changeset
   507
            // ### verify that QFile uses the fd size and not the file name
hgs
parents:
diff changeset
   508
            qint64 size = file->size() - file->pos();
hgs
parents:
diff changeset
   509
            const uchar *p = 0;
hgs
parents:
diff changeset
   510
#ifndef Q_OS_WINCE
hgs
parents:
diff changeset
   511
            p = file->map(file->pos(), size);
hgs
parents:
diff changeset
   512
#endif
hgs
parents:
diff changeset
   513
            if (p) {
hgs
parents:
diff changeset
   514
                buffer->setData((const char *)p, size);
hgs
parents:
diff changeset
   515
                file.take()->setParent(buffer.data());
hgs
parents:
diff changeset
   516
            } else {
hgs
parents:
diff changeset
   517
                buffer->setData(file->readAll());
hgs
parents:
diff changeset
   518
            }
hgs
parents:
diff changeset
   519
        }
hgs
parents:
diff changeset
   520
    }
hgs
parents:
diff changeset
   521
    buffer->open(QBuffer::ReadOnly);
hgs
parents:
diff changeset
   522
    return buffer.take();
hgs
parents:
diff changeset
   523
}
hgs
parents:
diff changeset
   524
hgs
parents:
diff changeset
   525
/*!
hgs
parents:
diff changeset
   526
    \reimp
hgs
parents:
diff changeset
   527
*/
hgs
parents:
diff changeset
   528
void NetworkDiskCache::updateMetaData(const QNetworkCacheMetaData &metaData)
hgs
parents:
diff changeset
   529
{
hgs
parents:
diff changeset
   530
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   531
    qDebug() << "NetworkDiskCache::updateMetaData()" << metaData.url();
hgs
parents:
diff changeset
   532
#endif
hgs
parents:
diff changeset
   533
    QUrl url = metaData.url();
hgs
parents:
diff changeset
   534
    QIODevice *oldDevice = data(url);
hgs
parents:
diff changeset
   535
    if (!oldDevice) {
hgs
parents:
diff changeset
   536
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   537
        qDebug() << "NetworkDiskCache::updateMetaData(), no device!";
hgs
parents:
diff changeset
   538
#endif
hgs
parents:
diff changeset
   539
        return;
hgs
parents:
diff changeset
   540
    }
hgs
parents:
diff changeset
   541
hgs
parents:
diff changeset
   542
    QIODevice *newDevice = prepare(metaData);
hgs
parents:
diff changeset
   543
    if (!newDevice) {
hgs
parents:
diff changeset
   544
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   545
        qDebug() << "NetworkDiskCache::updateMetaData(), no new device!" << url;
hgs
parents:
diff changeset
   546
#endif
hgs
parents:
diff changeset
   547
        return;
hgs
parents:
diff changeset
   548
    }
hgs
parents:
diff changeset
   549
    char data[1024];
hgs
parents:
diff changeset
   550
    while (!oldDevice->atEnd()) {
hgs
parents:
diff changeset
   551
        qint64 s = oldDevice->read(data, 1024);
hgs
parents:
diff changeset
   552
        newDevice->write(data, s);
hgs
parents:
diff changeset
   553
    }
hgs
parents:
diff changeset
   554
    delete oldDevice;
hgs
parents:
diff changeset
   555
    insert(newDevice);
hgs
parents:
diff changeset
   556
}
hgs
parents:
diff changeset
   557
hgs
parents:
diff changeset
   558
/*!
hgs
parents:
diff changeset
   559
    Returns the current maximum size for the disk cache.
hgs
parents:
diff changeset
   560
hgs
parents:
diff changeset
   561
    \sa setMaximumCacheSize()
hgs
parents:
diff changeset
   562
 */
hgs
parents:
diff changeset
   563
qint64 NetworkDiskCache::maximumCacheSize() const
hgs
parents:
diff changeset
   564
{
hgs
parents:
diff changeset
   565
    Q_D(const NetworkDiskCache);
hgs
parents:
diff changeset
   566
    return d->maximumCacheSize;
hgs
parents:
diff changeset
   567
}
hgs
parents:
diff changeset
   568
hgs
parents:
diff changeset
   569
/*!
hgs
parents:
diff changeset
   570
    Sets the maximum size of the disk cache to be \a size.
hgs
parents:
diff changeset
   571
hgs
parents:
diff changeset
   572
    If the new size is smaller then the current cache size then the cache will call expire().
hgs
parents:
diff changeset
   573
hgs
parents:
diff changeset
   574
    \sa maximumCacheSize()
hgs
parents:
diff changeset
   575
 */
hgs
parents:
diff changeset
   576
void NetworkDiskCache::setMaximumCacheSize(qint64 size)
hgs
parents:
diff changeset
   577
{
hgs
parents:
diff changeset
   578
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   579
    bool expireCache = (size < d->maximumCacheSize);
hgs
parents:
diff changeset
   580
    d->maximumCacheSize = size;
hgs
parents:
diff changeset
   581
    if (expireCache)
hgs
parents:
diff changeset
   582
        d->currentCacheSize = expire();
hgs
parents:
diff changeset
   583
}
hgs
parents:
diff changeset
   584
hgs
parents:
diff changeset
   585
/*!
hgs
parents:
diff changeset
   586
    Cleans the cache so that its size is under the maximum cache size.
hgs
parents:
diff changeset
   587
    Returns the current size of the cache.
hgs
parents:
diff changeset
   588
hgs
parents:
diff changeset
   589
    When the current size of the cache is greater than the maximumCacheSize()
hgs
parents:
diff changeset
   590
    older cache files are removed until the total size is less then 90% of
hgs
parents:
diff changeset
   591
    maximumCacheSize() starting with the oldest ones first using the file
hgs
parents:
diff changeset
   592
    creation date to determine how old a cache file is.
hgs
parents:
diff changeset
   593
hgs
parents:
diff changeset
   594
    Subclasses can reimplement this function to change the order that cache
hgs
parents:
diff changeset
   595
    files are removed taking into account information in the application
hgs
parents:
diff changeset
   596
    knows about that QNetworkDiskCache does not, for example the number of times
hgs
parents:
diff changeset
   597
    a cache is accessed.
hgs
parents:
diff changeset
   598
hgs
parents:
diff changeset
   599
    Note: cacheSize() calls expire if the current cache size is unknown.
hgs
parents:
diff changeset
   600
hgs
parents:
diff changeset
   601
    \sa maximumCacheSize(), fileMetaData()
hgs
parents:
diff changeset
   602
 */
hgs
parents:
diff changeset
   603
qint64 NetworkDiskCache::expire()
hgs
parents:
diff changeset
   604
{
hgs
parents:
diff changeset
   605
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   606
    if (d->currentCacheSize >= 0 && d->currentCacheSize < maximumCacheSize())
hgs
parents:
diff changeset
   607
        return d->currentCacheSize;
hgs
parents:
diff changeset
   608
hgs
parents:
diff changeset
   609
    if (cacheDirectory().isEmpty()) {
hgs
parents:
diff changeset
   610
        qWarning() << "NetworkDiskCache::expire() The cache directory is not set";
hgs
parents:
diff changeset
   611
        return 0;
hgs
parents:
diff changeset
   612
    }
hgs
parents:
diff changeset
   613
hgs
parents:
diff changeset
   614
    QDir::Filters filters = QDir::AllDirs | QDir:: Files | QDir::NoDotAndDotDot;
hgs
parents:
diff changeset
   615
    QDirIterator it(cacheDirectory(), filters, QDirIterator::Subdirectories);
hgs
parents:
diff changeset
   616
hgs
parents:
diff changeset
   617
    QMultiMap<QDateTime, QString> cacheItems;
hgs
parents:
diff changeset
   618
    qint64 totalSize = 0;
hgs
parents:
diff changeset
   619
    while (it.hasNext()) {
hgs
parents:
diff changeset
   620
        QString path = it.next();
hgs
parents:
diff changeset
   621
        QFileInfo info = it.fileInfo();
hgs
parents:
diff changeset
   622
        QString fileName = info.fileName();
hgs
parents:
diff changeset
   623
        cacheItems.insert(info.created(), path);
hgs
parents:
diff changeset
   624
        totalSize += info.size();
hgs
parents:
diff changeset
   625
    }
hgs
parents:
diff changeset
   626
hgs
parents:
diff changeset
   627
    int removedFiles = 0;
hgs
parents:
diff changeset
   628
    qint64 goal = (maximumCacheSize() * 9) / 10;
hgs
parents:
diff changeset
   629
    QMultiMap<QDateTime, QString>::const_iterator i = cacheItems.constBegin();
hgs
parents:
diff changeset
   630
    while (i != cacheItems.constEnd()) {
hgs
parents:
diff changeset
   631
        if (totalSize < goal)
hgs
parents:
diff changeset
   632
            break;
hgs
parents:
diff changeset
   633
        QString name = i.value();
hgs
parents:
diff changeset
   634
        QFile file(name);
hgs
parents:
diff changeset
   635
        qint64 size = file.size();
hgs
parents:
diff changeset
   636
        file.remove();
hgs
parents:
diff changeset
   637
        totalSize -= size;
hgs
parents:
diff changeset
   638
        ++removedFiles;
hgs
parents:
diff changeset
   639
        ++i;
hgs
parents:
diff changeset
   640
    }
hgs
parents:
diff changeset
   641
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   642
    if (removedFiles > 0) {
hgs
parents:
diff changeset
   643
        qDebug() << "NetworkDiskCache::expire()"
hgs
parents:
diff changeset
   644
                << "Removed:" << removedFiles
hgs
parents:
diff changeset
   645
                << "Kept:" << cacheItems.count() - removedFiles;
hgs
parents:
diff changeset
   646
    }
hgs
parents:
diff changeset
   647
#endif
hgs
parents:
diff changeset
   648
    if (removedFiles > 0)
hgs
parents:
diff changeset
   649
        d->lastItem.reset();
hgs
parents:
diff changeset
   650
    return totalSize;
hgs
parents:
diff changeset
   651
}
hgs
parents:
diff changeset
   652
hgs
parents:
diff changeset
   653
/*!
hgs
parents:
diff changeset
   654
    \reimp
hgs
parents:
diff changeset
   655
*/
hgs
parents:
diff changeset
   656
void NetworkDiskCache::clear()
hgs
parents:
diff changeset
   657
{
hgs
parents:
diff changeset
   658
#if defined(QNETWORKDISKCACHE_DEBUG)
hgs
parents:
diff changeset
   659
    qDebug() << "NetworkDiskCache::clear()";
hgs
parents:
diff changeset
   660
#endif
hgs
parents:
diff changeset
   661
    Q_D(NetworkDiskCache);
hgs
parents:
diff changeset
   662
    qint64 size = d->maximumCacheSize;
hgs
parents:
diff changeset
   663
    d->maximumCacheSize = 0;
hgs
parents:
diff changeset
   664
    d->currentCacheSize = expire();
hgs
parents:
diff changeset
   665
    d->maximumCacheSize = size;
hgs
parents:
diff changeset
   666
}
hgs
parents:
diff changeset
   667
hgs
parents:
diff changeset
   668
QByteArray NetworkDiskCachePrivate::generateId(const QUrl &url) const
hgs
parents:
diff changeset
   669
{
hgs
parents:
diff changeset
   670
    QUrl cleanUrl = url;
hgs
parents:
diff changeset
   671
    cleanUrl.setPassword(QString());
hgs
parents:
diff changeset
   672
    cleanUrl.setFragment(QString());
hgs
parents:
diff changeset
   673
    QByteArray blob = cleanUrl.toEncoded();
hgs
parents:
diff changeset
   674
hgs
parents:
diff changeset
   675
    QByteArray hash;
hgs
parents:
diff changeset
   676
    hash.setNum(crc32(blob.data(), blob.length()), 16);
hgs
parents:
diff changeset
   677
    return hash;
hgs
parents:
diff changeset
   678
}
hgs
parents:
diff changeset
   679
hgs
parents:
diff changeset
   680
QString NetworkDiskCachePrivate::tmpCacheFileName() const
hgs
parents:
diff changeset
   681
{
hgs
parents:
diff changeset
   682
    return cacheDirectory + QLatin1String("prepared/") + QLatin1String("XXXXXX");
hgs
parents:
diff changeset
   683
}
hgs
parents:
diff changeset
   684
hgs
parents:
diff changeset
   685
QString NetworkDiskCachePrivate::cacheFileName(const QUrl &url) const
hgs
parents:
diff changeset
   686
{
hgs
parents:
diff changeset
   687
    if (!url.isValid())
hgs
parents:
diff changeset
   688
        return QString();
hgs
parents:
diff changeset
   689
hgs
parents:
diff changeset
   690
    // Directories were already created during setup phase.
hgs
parents:
diff changeset
   691
    QString subDirectory;
hgs
parents:
diff changeset
   692
    QByteArray filenameID;
hgs
parents:
diff changeset
   693
    filenameID = generateId(url);
hgs
parents:
diff changeset
   694
    subDirectory = cacheDirectory + QLatin1Char(filenameID.at(0)) + QLatin1Char('/');
hgs
parents:
diff changeset
   695
hgs
parents:
diff changeset
   696
    return  subDirectory + QLatin1String(filenameID);
hgs
parents:
diff changeset
   697
}
hgs
parents:
diff changeset
   698
hgs
parents:
diff changeset
   699
/*!
hgs
parents:
diff changeset
   700
    We compress small text and JavaScript files.
hgs
parents:
diff changeset
   701
 */
hgs
parents:
diff changeset
   702
bool QCacheItem::canCompress() const
hgs
parents:
diff changeset
   703
{
hgs
parents:
diff changeset
   704
    bool sizeOk = false;
hgs
parents:
diff changeset
   705
    bool typeOk = false;
hgs
parents:
diff changeset
   706
    foreach (QNetworkCacheMetaData::RawHeader header, metaData.rawHeaders()) {
hgs
parents:
diff changeset
   707
        if (header.first.toLower() == "content-length") {
hgs
parents:
diff changeset
   708
            qint64 size = header.second.toLongLong();
hgs
parents:
diff changeset
   709
            if (size > MAX_COMPRESSION_SIZE)
hgs
parents:
diff changeset
   710
                return false;
hgs
parents:
diff changeset
   711
            else
hgs
parents:
diff changeset
   712
                sizeOk = true;
hgs
parents:
diff changeset
   713
        }
hgs
parents:
diff changeset
   714
hgs
parents:
diff changeset
   715
        if (header.first.toLower() == "content-type") {
hgs
parents:
diff changeset
   716
            QByteArray type = header.second;
hgs
parents:
diff changeset
   717
            if (type.startsWith("text/")
hgs
parents:
diff changeset
   718
                    || (type.startsWith("application/")
hgs
parents:
diff changeset
   719
                        && (type.endsWith("javascript") || type.endsWith("ecmascript"))))
hgs
parents:
diff changeset
   720
                typeOk = true;
hgs
parents:
diff changeset
   721
            else
hgs
parents:
diff changeset
   722
                return false;
hgs
parents:
diff changeset
   723
        }
hgs
parents:
diff changeset
   724
        if (sizeOk && typeOk)
hgs
parents:
diff changeset
   725
            return true;
hgs
parents:
diff changeset
   726
    }
hgs
parents:
diff changeset
   727
    return false;
hgs
parents:
diff changeset
   728
}
hgs
parents:
diff changeset
   729
hgs
parents:
diff changeset
   730
enum
hgs
parents:
diff changeset
   731
{
hgs
parents:
diff changeset
   732
    CacheMagic = 0xe8,
hgs
parents:
diff changeset
   733
    CurrentCacheVersion = 7
hgs
parents:
diff changeset
   734
};
hgs
parents:
diff changeset
   735
hgs
parents:
diff changeset
   736
void QCacheItem::writeHeader(QFile *device) const
hgs
parents:
diff changeset
   737
{
hgs
parents:
diff changeset
   738
    QDataStream out(device);
hgs
parents:
diff changeset
   739
hgs
parents:
diff changeset
   740
    out << qint32(CacheMagic);
hgs
parents:
diff changeset
   741
    out << qint32(CurrentCacheVersion);
hgs
parents:
diff changeset
   742
    out << metaData;
hgs
parents:
diff changeset
   743
    bool compressed = canCompress();
hgs
parents:
diff changeset
   744
    out << compressed;
hgs
parents:
diff changeset
   745
}
hgs
parents:
diff changeset
   746
hgs
parents:
diff changeset
   747
void QCacheItem::writeCompressedData(QFile *device) const
hgs
parents:
diff changeset
   748
{
hgs
parents:
diff changeset
   749
    QDataStream out(device);
hgs
parents:
diff changeset
   750
hgs
parents:
diff changeset
   751
    out << qCompress(data.data());
hgs
parents:
diff changeset
   752
}
hgs
parents:
diff changeset
   753
hgs
parents:
diff changeset
   754
/*!
hgs
parents:
diff changeset
   755
    Returns false if the file is a cache file,
hgs
parents:
diff changeset
   756
    but is an older version and should be removed otherwise true.
hgs
parents:
diff changeset
   757
 */
hgs
parents:
diff changeset
   758
bool QCacheItem::read(QFile *device, bool readData)
hgs
parents:
diff changeset
   759
{
hgs
parents:
diff changeset
   760
    reset();
hgs
parents:
diff changeset
   761
hgs
parents:
diff changeset
   762
    QDataStream in(device);
hgs
parents:
diff changeset
   763
hgs
parents:
diff changeset
   764
    qint32 marker;
hgs
parents:
diff changeset
   765
    qint32 v;
hgs
parents:
diff changeset
   766
    in >> marker;
hgs
parents:
diff changeset
   767
    in >> v;
hgs
parents:
diff changeset
   768
    if (marker != CacheMagic)
hgs
parents:
diff changeset
   769
        return true;
hgs
parents:
diff changeset
   770
hgs
parents:
diff changeset
   771
    // If the cache magic is correct, but the version is not we should remove it
hgs
parents:
diff changeset
   772
    if (v != CurrentCacheVersion)
hgs
parents:
diff changeset
   773
        return false;
hgs
parents:
diff changeset
   774
hgs
parents:
diff changeset
   775
    bool compressed;
hgs
parents:
diff changeset
   776
    QByteArray dataBA;
hgs
parents:
diff changeset
   777
    in >> metaData;
hgs
parents:
diff changeset
   778
    in >> compressed;
hgs
parents:
diff changeset
   779
    if (readData && compressed) {
hgs
parents:
diff changeset
   780
        in >> dataBA;
hgs
parents:
diff changeset
   781
        data.setData(qUncompress(dataBA));
hgs
parents:
diff changeset
   782
        data.open(QBuffer::ReadOnly);
hgs
parents:
diff changeset
   783
    }
hgs
parents:
diff changeset
   784
    return metaData.isValid();
hgs
parents:
diff changeset
   785
}
hgs
parents:
diff changeset
   786
hgs
parents:
diff changeset
   787
QT_END_NAMESPACE
hgs
parents:
diff changeset
   788
hgs
parents:
diff changeset
   789
#endif // QT_NO_NETWORKDISKCACHE