/*************************************************************************** * Copyright (C) 2016-2019 by Ilya Kotov * * forkotov02@ya.ru * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This program 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 this program; if not, write to the * * Free Software Foundation, Inc., * * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * ***************************************************************************/ #include #include "archiveinputdevice.h" ArchiveInputDevice::ArchiveInputDevice(const QString &url, QObject *parent) : QIODevice(parent) { m_archive = nullptr; m_entry = nullptr; QString filePath = url.section("#", -1); QString archivePath = url; archivePath.remove(QRegExp("^.+://")); archivePath.remove(QRegExp("#.+$")); m_archive = archive_read_new(); archive_read_support_filter_all(m_archive); archive_read_support_format_all(m_archive); int r = archive_read_open_filename(m_archive, archivePath.toLocal8Bit().constData(), 10240); if (r != ARCHIVE_OK) { qWarning("ArchiveInputDevice: unable to open file '%s', libarchive error: %s", qPrintable(archivePath), archive_error_string(m_archive)); return; } while (archive_read_next_header(m_archive, &m_entry) == ARCHIVE_OK) { QString pathName = QString::fromLocal8Bit(archive_entry_pathname(m_entry)); if(!pathName.startsWith("/")) pathName.prepend("/"); if(archive_entry_filetype(m_entry) == AE_IFREG && filePath == pathName) { open(QIODevice::ReadOnly); m_buffer.open(QBuffer::ReadWrite); break; } archive_read_data_skip(m_archive); } m_close_libarchive = true; } ArchiveInputDevice::ArchiveInputDevice(archive *a, archive_entry *e, QObject *parent) : QIODevice(parent) { m_archive = a; m_entry = e; open(QIODevice::ReadOnly); m_buffer.open(QBuffer::ReadWrite); m_close_libarchive = false; } ArchiveInputDevice::~ArchiveInputDevice() { if(m_close_libarchive && m_archive) { archive_read_close(m_archive); archive_read_free(m_archive); m_archive = nullptr; } } bool ArchiveInputDevice::seek(qint64 pos) { QIODevice::seek(pos); if(pos > m_buffer.size()) { char tmp[1024]; qint64 delta = pos - m_buffer.size(); while (delta > 0) { qint64 r = qMin(qint64(sizeof(tmp)), delta); r = archive_read_data(m_archive, tmp, r); if(r > 0) { m_buffer.buffer().append(tmp, r); delta -= r; continue; } else if(r < 0) { qWarning("ArchiveInputDevice: seeking failed; libarchive error: %s", archive_error_string(m_archive)); } return false; } } return m_buffer.seek(pos); } qint64 ArchiveInputDevice::size() const { return archive_entry_size(m_entry); } qint64 ArchiveInputDevice::readData(char *data, qint64 maxSize) { if(m_buffer.pos() + maxSize > m_buffer.size()) { qint64 l = m_buffer.pos() + maxSize - m_buffer.size(); char tmp[l]; int r = archive_read_data(m_archive, tmp, l); if(r > 0) m_buffer.buffer().append(tmp, r); else if(r < 0) qWarning("ArchiveInputDevice: reading failed; libarchive error: %s", archive_error_string(m_archive)); } return m_buffer.read(data, maxSize); } qint64 ArchiveInputDevice::writeData(const char *, qint64) { return -1; }