2020-02-09 01:12:59 +08:00
|
|
|
// Copyright (c) 2017-2020 Thomas Fussell
|
2017-06-12 22:34:35 +08:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, WRISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE
|
|
|
|
//
|
|
|
|
// @license: http://www.opensource.org/licenses/mit-license.php
|
|
|
|
// @author: see AUTHORS file
|
|
|
|
|
2017-06-16 06:10:27 +08:00
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
#include <xlnt/cell/cell.hpp>
|
2017-06-24 23:39:37 +08:00
|
|
|
#include <xlnt/packaging/manifest.hpp>
|
2017-06-16 06:10:27 +08:00
|
|
|
#include <xlnt/utils/optional.hpp>
|
2017-06-12 22:34:35 +08:00
|
|
|
#include <xlnt/workbook/streaming_workbook_reader.hpp>
|
2017-06-16 06:10:27 +08:00
|
|
|
#include <xlnt/workbook/workbook.hpp>
|
|
|
|
#include <xlnt/worksheet/worksheet.hpp>
|
2019-12-27 01:03:12 +08:00
|
|
|
#include <detail/implementations/workbook_impl.hpp>
|
|
|
|
#include <detail/serialization/open_stream.hpp>
|
|
|
|
#include <detail/serialization/vector_streambuf.hpp>
|
|
|
|
#include <detail/serialization/xlsx_consumer.hpp>
|
2017-06-12 22:34:35 +08:00
|
|
|
|
|
|
|
namespace xlnt {
|
|
|
|
|
2017-06-21 01:29:50 +08:00
|
|
|
streaming_workbook_reader::streaming_workbook_reader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-06-12 22:34:35 +08:00
|
|
|
streaming_workbook_reader::~streaming_workbook_reader()
|
|
|
|
{
|
2017-06-21 01:29:50 +08:00
|
|
|
close();
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
2017-06-15 03:43:25 +08:00
|
|
|
void streaming_workbook_reader::close()
|
2017-06-12 22:34:35 +08:00
|
|
|
{
|
2017-06-17 22:53:37 +08:00
|
|
|
if (consumer_)
|
|
|
|
{
|
|
|
|
consumer_.reset(nullptr);
|
2017-06-24 23:39:37 +08:00
|
|
|
stream_buffer_.reset(nullptr);
|
2017-06-17 22:53:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool streaming_workbook_reader::has_cell()
|
|
|
|
{
|
|
|
|
return consumer_->has_cell();
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
2017-06-16 06:10:27 +08:00
|
|
|
cell streaming_workbook_reader::read_cell()
|
2017-06-12 22:34:35 +08:00
|
|
|
{
|
2017-06-16 06:10:27 +08:00
|
|
|
return consumer_->read_cell();
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
2017-07-20 08:21:55 +08:00
|
|
|
bool streaming_workbook_reader::has_worksheet(const std::string &name)
|
2017-06-17 22:53:37 +08:00
|
|
|
{
|
2017-07-20 08:21:55 +08:00
|
|
|
auto titles = sheet_titles();
|
|
|
|
return std::find(titles.begin(), titles.end(), name) != titles.end();
|
2017-06-17 22:53:37 +08:00
|
|
|
}
|
|
|
|
|
2017-07-20 08:21:55 +08:00
|
|
|
void streaming_workbook_reader::begin_worksheet(const std::string &title)
|
2017-06-12 22:34:35 +08:00
|
|
|
{
|
2017-07-20 08:21:55 +08:00
|
|
|
if (!has_worksheet(title))
|
|
|
|
{
|
|
|
|
throw xlnt::exception("sheet not found");
|
|
|
|
}
|
|
|
|
|
|
|
|
worksheet_rel_id_ = workbook_->impl().sheet_title_rel_id_map_.at(title);
|
2017-06-24 23:39:37 +08:00
|
|
|
const auto workbook_rel = workbook_->manifest()
|
2019-12-27 01:03:12 +08:00
|
|
|
.relationship(path("/"), relationship_type::office_document);
|
2017-06-24 23:39:37 +08:00
|
|
|
const auto worksheet_rel = workbook_->manifest()
|
2019-12-27 01:03:12 +08:00
|
|
|
.relationship(workbook_rel.target().path(), worksheet_rel_id_);
|
2017-06-24 23:39:37 +08:00
|
|
|
|
2019-12-27 01:03:12 +08:00
|
|
|
auto rel_chain = std::vector<relationship>{workbook_rel, worksheet_rel};
|
2017-06-24 23:39:37 +08:00
|
|
|
|
|
|
|
const auto &manifest = consumer_->target_.manifest();
|
|
|
|
const auto part_path = manifest.canonicalize(rel_chain);
|
|
|
|
auto part_stream_buffer = consumer_->archive_->open(part_path);
|
|
|
|
part_stream_buffer_.swap(part_stream_buffer);
|
|
|
|
part_stream_.reset(new std::istream(part_stream_buffer_.get()));
|
|
|
|
parser_.reset(new xml::parser(*part_stream_, part_path.string()));
|
|
|
|
consumer_->parser_ = parser_.get();
|
|
|
|
|
2017-07-20 08:21:55 +08:00
|
|
|
consumer_->current_worksheet_ = nullptr;
|
|
|
|
|
|
|
|
for (auto &impl : workbook_->impl().worksheets_)
|
|
|
|
{
|
|
|
|
if (impl.title_ == title)
|
|
|
|
{
|
|
|
|
consumer_->current_worksheet_ = &impl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (consumer_->current_worksheet_ == nullptr)
|
|
|
|
{
|
|
|
|
throw xlnt::exception("sheet not found");
|
|
|
|
}
|
|
|
|
|
|
|
|
consumer_->read_worksheet_begin(worksheet_rel_id_);
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
2017-06-16 06:10:27 +08:00
|
|
|
worksheet streaming_workbook_reader::end_worksheet()
|
2017-06-12 22:34:35 +08:00
|
|
|
{
|
2017-07-20 08:21:55 +08:00
|
|
|
return consumer_->read_worksheet_end(worksheet_rel_id_);
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void streaming_workbook_reader::open(const std::vector<std::uint8_t> &data)
|
|
|
|
{
|
2017-06-24 23:39:37 +08:00
|
|
|
stream_buffer_.reset(new detail::vector_istreambuf(data));
|
|
|
|
stream_.reset(new std::istream(stream_buffer_.get()));
|
|
|
|
open(*stream_);
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void streaming_workbook_reader::open(const std::string &filename)
|
|
|
|
{
|
2017-06-24 23:39:37 +08:00
|
|
|
stream_.reset(new std::ifstream());
|
2017-07-12 11:40:55 +08:00
|
|
|
xlnt::detail::open_stream(static_cast<std::ifstream &>(*stream_), filename);
|
2017-06-24 23:39:37 +08:00
|
|
|
open(*stream_);
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
void streaming_workbook_reader::open(const std::wstring &filename)
|
|
|
|
{
|
2017-06-24 23:39:37 +08:00
|
|
|
stream_.reset(new std::ifstream());
|
2017-07-12 11:40:55 +08:00
|
|
|
xlnt::detail::open_stream(static_cast<std::ifstream &>(*stream_), filename);
|
2017-06-24 23:39:37 +08:00
|
|
|
open(*stream_);
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void streaming_workbook_reader::open(const xlnt::path &filename)
|
|
|
|
{
|
2017-06-24 23:39:37 +08:00
|
|
|
stream_.reset(new std::ifstream());
|
2017-07-12 11:40:55 +08:00
|
|
|
xlnt::detail::open_stream(static_cast<std::ifstream &>(*stream_), filename.string());
|
2017-06-24 23:39:37 +08:00
|
|
|
open(*stream_);
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void streaming_workbook_reader::open(std::istream &stream)
|
|
|
|
{
|
2017-06-16 06:10:27 +08:00
|
|
|
workbook_.reset(new workbook());
|
|
|
|
consumer_.reset(new detail::xlsx_consumer(*workbook_));
|
|
|
|
consumer_->open(stream);
|
2017-06-24 23:39:37 +08:00
|
|
|
|
|
|
|
const auto workbook_rel = workbook_->manifest()
|
2019-12-27 01:03:12 +08:00
|
|
|
.relationship(path("/"), relationship_type::office_document);
|
2017-06-24 23:39:37 +08:00
|
|
|
const auto workbook_path = workbook_rel.target().path();
|
2017-07-20 08:21:55 +08:00
|
|
|
}
|
2017-06-24 23:39:37 +08:00
|
|
|
|
2017-07-31 11:32:37 +08:00
|
|
|
void streaming_workbook_reader::open(std::unique_ptr<std::streambuf> &&buffer)
|
|
|
|
{
|
|
|
|
stream_buffer_.swap(buffer);
|
|
|
|
stream_.reset(new std::istream(stream_buffer_.get()));
|
|
|
|
open(*stream_);
|
|
|
|
}
|
|
|
|
|
2017-07-20 08:21:55 +08:00
|
|
|
std::vector<std::string> streaming_workbook_reader::sheet_titles()
|
|
|
|
{
|
|
|
|
return workbook_->sheet_titles();
|
2017-06-12 22:34:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace xlnt
|