mirror of
https://github.com/tomahawk-player/tomahawk.git
synced 2025-03-18 23:09:42 +01:00
Import Cloudstream from Clementine
This commit is contained in:
parent
6516321a86
commit
50370f7b53
@ -360,6 +360,7 @@ list(APPEND libSources
|
||||
sip/PeerInfo.cpp
|
||||
sip/SipStatusMessage.cpp
|
||||
|
||||
utils/Cloudstream.cpp
|
||||
utils/Json.cpp
|
||||
utils/TomahawkUtils.cpp
|
||||
utils/Logger.cpp
|
||||
|
193
src/libtomahawk/utils/Cloudstream.cpp
Normal file
193
src/libtomahawk/utils/Cloudstream.cpp
Normal file
@ -0,0 +1,193 @@
|
||||
/* This file is part of Clementine.
|
||||
Copyright 2012, David Sansome <me@davidsansome.com>
|
||||
|
||||
Clementine is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
Clementine is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with Clementine. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include "Cloudstream.h"
|
||||
|
||||
#include "Logger.h"
|
||||
|
||||
#include <QEventLoop>
|
||||
#include <QNetworkAccessManager>
|
||||
#include <QNetworkReply>
|
||||
#include <QNetworkRequest>
|
||||
|
||||
#include <taglib/id3v2framefactory.h>
|
||||
#include <taglib/mpegfile.h>
|
||||
|
||||
namespace {
|
||||
static const int kTaglibPrefixCacheBytes = 64 * 1024; // Should be enough.
|
||||
static const int kTaglibSuffixCacheBytes = 8 * 1024;
|
||||
}
|
||||
|
||||
CloudStream::CloudStream(const QUrl& url, const QString& filename,
|
||||
const long length, const QString& auth,
|
||||
QNetworkAccessManager* network)
|
||||
: url_(url),
|
||||
filename_(filename),
|
||||
encoded_filename_(filename_.toUtf8()),
|
||||
length_(length),
|
||||
auth_(auth),
|
||||
cursor_(0),
|
||||
network_(network),
|
||||
cache_(length),
|
||||
num_requests_(0) {}
|
||||
|
||||
TagLib::FileName CloudStream::name() const { return encoded_filename_.data(); }
|
||||
|
||||
bool CloudStream::CheckCache(int start, int end) {
|
||||
for (int i = start; i <= end; ++i) {
|
||||
if (!cache_.test(i)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
void CloudStream::FillCache(int start, TagLib::ByteVector data) {
|
||||
for (int i = 0; i < data.size(); ++i) {
|
||||
cache_.set(start + i, data[i]);
|
||||
}
|
||||
}
|
||||
|
||||
TagLib::ByteVector CloudStream::GetCached(int start, int end) {
|
||||
const uint size = end - start + 1;
|
||||
TagLib::ByteVector ret(size);
|
||||
for (int i = 0; i < size; ++i) {
|
||||
ret[i] = cache_.get(start + i);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
void CloudStream::Precache() {
|
||||
// For reading the tags of an MP3, TagLib tends to request:
|
||||
// 1. The first 1024 bytes
|
||||
// 2. Somewhere between the first 2KB and first 60KB
|
||||
// 3. The last KB or two.
|
||||
// 4. Somewhere in the first 64KB again
|
||||
//
|
||||
// OGG Vorbis may read the last 4KB.
|
||||
//
|
||||
// So, if we precache the first 64KB and the last 8KB we should be sorted :-)
|
||||
// Ideally, we would use bytes=0-655364,-8096 but Google Drive does not seem
|
||||
// to support multipart byte ranges yet so we have to make do with two
|
||||
// requests.
|
||||
|
||||
seek(0, TagLib::IOStream::Beginning);
|
||||
readBlock(kTaglibPrefixCacheBytes);
|
||||
seek(kTaglibSuffixCacheBytes, TagLib::IOStream::End);
|
||||
readBlock(kTaglibSuffixCacheBytes);
|
||||
clear();
|
||||
}
|
||||
|
||||
TagLib::ByteVector CloudStream::readBlock(ulong length) {
|
||||
const uint start = cursor_;
|
||||
const uint end = qMin(cursor_ + length - 1, length_ - 1);
|
||||
|
||||
if (end < start) {
|
||||
return TagLib::ByteVector();
|
||||
}
|
||||
|
||||
if (CheckCache(start, end)) {
|
||||
TagLib::ByteVector cached = GetCached(start, end);
|
||||
cursor_ += cached.size();
|
||||
return cached;
|
||||
}
|
||||
|
||||
QNetworkRequest request = QNetworkRequest(url_);
|
||||
if (!auth_.isEmpty()) {
|
||||
request.setRawHeader("Authorization", auth_.toUtf8());
|
||||
}
|
||||
request.setRawHeader("Range",
|
||||
QString("bytes=%1-%2").arg(start).arg(end).toUtf8());
|
||||
request.setAttribute(QNetworkRequest::CacheLoadControlAttribute,
|
||||
QNetworkRequest::AlwaysNetwork);
|
||||
|
||||
QNetworkReply* reply = network_->get(request);
|
||||
connect(reply, SIGNAL(sslErrors(QList<QSslError>)),
|
||||
SLOT(SSLErrors(QList<QSslError>)));
|
||||
++num_requests_;
|
||||
|
||||
QEventLoop loop;
|
||||
QObject::connect(reply, SIGNAL(finished()), &loop, SLOT(quit()));
|
||||
loop.exec();
|
||||
reply->deleteLater();
|
||||
|
||||
int code = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
||||
if (code >= 400) {
|
||||
tDebug() << "Error retrieving url to tag:" << url_;
|
||||
return TagLib::ByteVector();
|
||||
}
|
||||
|
||||
QByteArray data = reply->readAll();
|
||||
TagLib::ByteVector bytes(data.data(), data.size());
|
||||
cursor_ += data.size();
|
||||
|
||||
FillCache(start, bytes);
|
||||
return bytes;
|
||||
}
|
||||
|
||||
void CloudStream::writeBlock(const TagLib::ByteVector&) {
|
||||
tDebug() << Q_FUNC_INFO << "not implemented";
|
||||
}
|
||||
|
||||
void CloudStream::insert(const TagLib::ByteVector&, ulong, ulong) {
|
||||
tDebug() << Q_FUNC_INFO << "not implemented";
|
||||
}
|
||||
|
||||
void CloudStream::removeBlock(ulong, ulong) {
|
||||
tDebug() << Q_FUNC_INFO << "not implemented";
|
||||
}
|
||||
|
||||
bool CloudStream::readOnly() const {
|
||||
tDebug() << Q_FUNC_INFO;
|
||||
return true;
|
||||
}
|
||||
|
||||
bool CloudStream::isOpen() const { return true; }
|
||||
|
||||
void CloudStream::seek(long offset, TagLib::IOStream::Position p) {
|
||||
switch (p) {
|
||||
case TagLib::IOStream::Beginning:
|
||||
cursor_ = offset;
|
||||
break;
|
||||
|
||||
case TagLib::IOStream::Current:
|
||||
cursor_ = qMin(ulong(cursor_ + offset), length_);
|
||||
break;
|
||||
|
||||
case TagLib::IOStream::End:
|
||||
// This should really not have qAbs(), but OGG reading needs it.
|
||||
cursor_ = qMax(0UL, length_ - qAbs(offset));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void CloudStream::clear() { cursor_ = 0; }
|
||||
|
||||
long CloudStream::tell() const { return cursor_; }
|
||||
|
||||
long CloudStream::length() { return length_; }
|
||||
|
||||
void CloudStream::truncate(long) {
|
||||
tDebug() << Q_FUNC_INFO << "not implemented";
|
||||
}
|
||||
|
||||
void CloudStream::SSLErrors(const QList<QSslError>& errors) {
|
||||
foreach (const QSslError& error, errors) {
|
||||
tDebug() << error.error() << error.errorString();
|
||||
tDebug() << error.certificate();
|
||||
}
|
||||
}
|
82
src/libtomahawk/utils/Cloudstream.h
Normal file
82
src/libtomahawk/utils/Cloudstream.h
Normal file
@ -0,0 +1,82 @@
|
||||
/* This file is part of Clementine.
|
||||
Copyright 2012, David Sansome <me@davidsansome.com>
|
||||
|
||||
Clementine is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
Clementine is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with Clementine. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef GOOGLEDRIVESTREAM_H
|
||||
#define GOOGLEDRIVESTREAM_H
|
||||
|
||||
#include <QObject>
|
||||
#include <QList>
|
||||
#include <QSslError>
|
||||
#include <QUrl>
|
||||
|
||||
#include <google/sparsetable>
|
||||
#include <taglib/tiostream.h>
|
||||
|
||||
class QNetworkAccessManager;
|
||||
|
||||
class CloudStream : public QObject, public TagLib::IOStream {
|
||||
Q_OBJECT
|
||||
public:
|
||||
CloudStream(const QUrl& url, const QString& filename, const long length,
|
||||
const QString& auth, QNetworkAccessManager* network);
|
||||
|
||||
// Taglib::IOStream
|
||||
virtual TagLib::FileName name() const;
|
||||
virtual TagLib::ByteVector readBlock(ulong length);
|
||||
virtual void writeBlock(const TagLib::ByteVector&);
|
||||
virtual void insert(const TagLib::ByteVector&, ulong, ulong);
|
||||
virtual void removeBlock(ulong, ulong);
|
||||
virtual bool readOnly() const;
|
||||
virtual bool isOpen() const;
|
||||
virtual void seek(long offset, TagLib::IOStream::Position p);
|
||||
virtual void clear();
|
||||
virtual long tell() const;
|
||||
virtual long length();
|
||||
virtual void truncate(long);
|
||||
|
||||
google::sparsetable<char>::size_type cached_bytes() const {
|
||||
return cache_.num_nonempty();
|
||||
}
|
||||
|
||||
int num_requests() const { return num_requests_; }
|
||||
|
||||
// Use educated guess to request the bytes that TagLib will probably want.
|
||||
void Precache();
|
||||
|
||||
private:
|
||||
bool CheckCache(int start, int end);
|
||||
void FillCache(int start, TagLib::ByteVector data);
|
||||
TagLib::ByteVector GetCached(int start, int end);
|
||||
|
||||
private slots:
|
||||
void SSLErrors(const QList<QSslError>& errors);
|
||||
|
||||
private:
|
||||
const QUrl url_;
|
||||
const QString filename_;
|
||||
const QByteArray encoded_filename_;
|
||||
const ulong length_;
|
||||
const QString auth_;
|
||||
|
||||
int cursor_;
|
||||
QNetworkAccessManager* network_;
|
||||
|
||||
google::sparsetable<char> cache_;
|
||||
int num_requests_;
|
||||
};
|
||||
|
||||
#endif // GOOGLEDRIVESTREAM_H
|
Loading…
x
Reference in New Issue
Block a user