mirror of
https://github.com/qpdf/qpdf.git
synced 2024-12-22 19:08:59 +00:00
1318 lines
30 KiB
C++
1318 lines
30 KiB
C++
#include <qpdf/QPDFJob.hh>
|
|
|
|
#include <regex>
|
|
|
|
#include <qpdf/QPDFLogger.hh>
|
|
#include <qpdf/QTC.hh>
|
|
#include <qpdf/QUtil.hh>
|
|
|
|
void
|
|
QPDFJob::Config::checkConfiguration()
|
|
{
|
|
o.checkConfiguration();
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::inputFile(std::string const& filename)
|
|
{
|
|
if (o.m->infilename == nullptr) {
|
|
o.m->infilename = QUtil::make_shared_cstr(filename);
|
|
} else {
|
|
usage("input file has already been given");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::emptyInput()
|
|
{
|
|
if (o.m->infilename == nullptr) {
|
|
// Various places in QPDFJob.cc know that the empty string for infile means empty. We set it
|
|
// to something other than a null pointer as an indication that some input source has been
|
|
// specified. This approach means that passing "" as the argument to inputFile in job JSON,
|
|
// or equivalently using "" as a positional command-line argument would be the same as
|
|
// --empty. This probably isn't worth blocking or coding around.
|
|
o.m->infilename = QUtil::make_shared_cstr("");
|
|
} else {
|
|
usage("empty input can't be used since input file has already been given");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::outputFile(std::string const& filename)
|
|
{
|
|
if ((o.m->outfilename == nullptr) && (!o.m->replace_input)) {
|
|
o.m->outfilename = QUtil::make_shared_cstr(filename);
|
|
} else {
|
|
usage("output file has already been given");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::replaceInput()
|
|
{
|
|
if ((o.m->outfilename == nullptr) && (!o.m->replace_input)) {
|
|
o.m->replace_input = true;
|
|
} else {
|
|
usage("replace-input can't be used since output file has already been given");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::allowWeakCrypto()
|
|
{
|
|
o.m->allow_weak_crypto = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::check()
|
|
{
|
|
o.m->check = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::checkLinearization()
|
|
{
|
|
o.m->check_linearization = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::coalesceContents()
|
|
{
|
|
o.m->coalesce_contents = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::collate()
|
|
{
|
|
return collate("");
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::collate(std::string const& parameter)
|
|
{
|
|
if (parameter.empty()) {
|
|
o.m->collate.push_back(1);
|
|
return this;
|
|
}
|
|
size_t pos = 0;
|
|
// Parse a,b,c
|
|
while (true) {
|
|
auto end = parameter.find(',', pos);
|
|
auto n = parameter.substr(pos, end);
|
|
if (n.empty()) {
|
|
usage("--collate: trailing comma");
|
|
}
|
|
o.m->collate.push_back(QIntC::to_size(QUtil::string_to_uint(n.c_str())));
|
|
if (end == std::string::npos) {
|
|
break;
|
|
}
|
|
pos = end + 1;
|
|
}
|
|
if (o.m->collate.empty()) {
|
|
o.m->collate.push_back(1);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::compressStreams(std::string const& parameter)
|
|
{
|
|
o.m->compress_streams_set = true;
|
|
o.m->compress_streams = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::compressionLevel(std::string const& parameter)
|
|
{
|
|
o.m->compression_level = QUtil::string_to_int(parameter.c_str());
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::copyEncryption(std::string const& parameter)
|
|
{
|
|
o.m->encryption_file = parameter;
|
|
o.m->copy_encryption = true;
|
|
o.m->encrypt = false;
|
|
o.m->decrypt = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::decrypt()
|
|
{
|
|
o.m->decrypt = true;
|
|
o.m->encrypt = false;
|
|
o.m->copy_encryption = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::deterministicId()
|
|
{
|
|
o.m->deterministic_id = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::encryptionFilePassword(std::string const& parameter)
|
|
{
|
|
o.m->encryption_file_password = QUtil::make_shared_cstr(parameter);
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::externalizeInlineImages()
|
|
{
|
|
o.m->externalize_inline_images = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::filteredStreamData()
|
|
{
|
|
o.m->show_filtered_stream_data = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::flattenAnnotations(std::string const& parameter)
|
|
{
|
|
o.m->flatten_annotations = true;
|
|
if (parameter == "screen") {
|
|
o.m->flatten_annotations_forbidden |= an_no_view;
|
|
} else if (parameter == "print") {
|
|
o.m->flatten_annotations_required |= an_print;
|
|
} else if (parameter != "all") {
|
|
usage("invalid flatten-annotations option");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::flattenRotation()
|
|
{
|
|
o.m->flatten_rotation = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::forceVersion(std::string const& parameter)
|
|
{
|
|
o.m->force_version = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::generateAppearances()
|
|
{
|
|
o.m->generate_appearances = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::ignoreXrefStreams()
|
|
{
|
|
o.m->ignore_xref_streams = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::iiMinBytes(std::string const& parameter)
|
|
{
|
|
o.m->ii_min_bytes = QUtil::string_to_uint(parameter.c_str());
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::isEncrypted()
|
|
{
|
|
o.m->check_is_encrypted = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::json()
|
|
{
|
|
return json("");
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::json(std::string const& parameter)
|
|
{
|
|
if (parameter.empty() || (parameter == "latest")) {
|
|
o.m->json_version = JSON::LATEST;
|
|
} else {
|
|
o.m->json_version = QUtil::string_to_int(parameter.c_str());
|
|
}
|
|
if ((o.m->json_version < 1) || (o.m->json_version > JSON::LATEST)) {
|
|
usage(std::string("unsupported json version ") + parameter);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jsonKey(std::string const& parameter)
|
|
{
|
|
o.m->json_keys.insert(parameter);
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jsonObject(std::string const& parameter)
|
|
{
|
|
o.m->json_objects.insert(parameter);
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jsonStreamData(std::string const& parameter)
|
|
{
|
|
o.m->json_stream_data_set = true;
|
|
if (parameter == "none") {
|
|
o.m->json_stream_data = qpdf_sj_none;
|
|
} else if (parameter == "inline") {
|
|
o.m->json_stream_data = qpdf_sj_inline;
|
|
} else if (parameter == "file") {
|
|
o.m->json_stream_data = qpdf_sj_file;
|
|
} else {
|
|
usage("invalid json-streams option");
|
|
}
|
|
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jsonStreamPrefix(std::string const& parameter)
|
|
{
|
|
o.m->json_stream_prefix = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jsonInput()
|
|
{
|
|
o.m->json_input = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jsonOutput(std::string const& parameter)
|
|
{
|
|
o.m->json_output = true;
|
|
json(parameter);
|
|
if (!o.m->json_stream_data_set) {
|
|
// No need to set json_stream_data_set -- that indicates explicit use of --json-stream-data.
|
|
o.m->json_stream_data = qpdf_sj_inline;
|
|
}
|
|
if (!o.m->decode_level_set) {
|
|
o.m->decode_level = qpdf_dl_none;
|
|
}
|
|
o.m->json_keys.insert("qpdf");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::updateFromJson(std::string const& parameter)
|
|
{
|
|
o.m->update_from_json = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::testJsonSchema()
|
|
{
|
|
o.m->test_json_schema = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::keepFilesOpen(std::string const& parameter)
|
|
{
|
|
o.m->keep_files_open_set = true;
|
|
o.m->keep_files_open = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::keepFilesOpenThreshold(std::string const& parameter)
|
|
{
|
|
o.m->keep_files_open_threshold = QUtil::string_to_uint(parameter.c_str());
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::keepInlineImages()
|
|
{
|
|
o.m->keep_inline_images = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::linearize()
|
|
{
|
|
o.m->linearize = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::linearizePass1(std::string const& parameter)
|
|
{
|
|
o.m->linearize_pass1 = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::listAttachments()
|
|
{
|
|
o.m->list_attachments = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::minVersion(std::string const& parameter)
|
|
{
|
|
o.m->min_version = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::newlineBeforeEndstream()
|
|
{
|
|
o.m->newline_before_endstream = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::noOriginalObjectIds()
|
|
{
|
|
o.m->suppress_original_object_id = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::noWarn()
|
|
{
|
|
o.m->suppress_warnings = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::normalizeContent(std::string const& parameter)
|
|
{
|
|
o.m->normalize_set = true;
|
|
o.m->normalize = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::oiMinArea(std::string const& parameter)
|
|
{
|
|
o.m->oi_min_area = QUtil::string_to_uint(parameter.c_str());
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::oiMinHeight(std::string const& parameter)
|
|
{
|
|
o.m->oi_min_height = QUtil::string_to_uint(parameter.c_str());
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::oiMinWidth(std::string const& parameter)
|
|
{
|
|
o.m->oi_min_width = QUtil::string_to_uint(parameter.c_str());
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::optimizeImages()
|
|
{
|
|
o.m->optimize_images = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::password(std::string const& parameter)
|
|
{
|
|
o.m->password = QUtil::make_shared_cstr(parameter);
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::passwordIsHexKey()
|
|
{
|
|
o.m->password_is_hex_key = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::preserveUnreferenced()
|
|
{
|
|
o.m->preserve_unreferenced_objects = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::preserveUnreferencedResources()
|
|
{
|
|
o.m->remove_unreferenced_page_resources = QPDFJob::re_no;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::progress()
|
|
{
|
|
o.m->progress = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::qdf()
|
|
{
|
|
o.m->qdf_mode = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::rawStreamData()
|
|
{
|
|
o.m->show_raw_stream_data = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::recompressFlate()
|
|
{
|
|
o.m->recompress_flate_set = true;
|
|
o.m->recompress_flate = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::removeAttachment(std::string const& parameter)
|
|
{
|
|
o.m->attachments_to_remove.push_back(parameter);
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::removePageLabels()
|
|
{
|
|
o.m->remove_page_labels = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::reportMemoryUsage()
|
|
{
|
|
o.m->report_mem_usage = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::requiresPassword()
|
|
{
|
|
o.m->check_requires_password = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::removeRestrictions()
|
|
{
|
|
o.m->remove_restrictions = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showAttachment(std::string const& parameter)
|
|
{
|
|
o.m->attachment_to_show = parameter;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showEncryption()
|
|
{
|
|
o.m->show_encryption = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showEncryptionKey()
|
|
{
|
|
o.m->show_encryption_key = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showLinearization()
|
|
{
|
|
o.m->show_linearization = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showNpages()
|
|
{
|
|
o.m->show_npages = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showPages()
|
|
{
|
|
o.m->show_pages = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showXref()
|
|
{
|
|
o.m->show_xref = true;
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::splitPages()
|
|
{
|
|
return splitPages("");
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::splitPages(std::string const& parameter)
|
|
{
|
|
int n = (parameter.empty() ? 1 : QUtil::string_to_int(parameter.c_str()));
|
|
o.m->split_pages = n;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::staticAesIv()
|
|
{
|
|
o.m->static_aes_iv = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::staticId()
|
|
{
|
|
o.m->static_id = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::suppressPasswordRecovery()
|
|
{
|
|
o.m->suppress_password_recovery = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::suppressRecovery()
|
|
{
|
|
o.m->suppress_recovery = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::verbose()
|
|
{
|
|
o.m->verbose = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::warningExit0()
|
|
{
|
|
o.m->warnings_exit_zero = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::withImages()
|
|
{
|
|
o.m->show_page_images = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::passwordFile(std::string const& parameter)
|
|
{
|
|
std::list<std::string> lines;
|
|
if (parameter == "-") {
|
|
QTC::TC("qpdf", "QPDFJob_config password stdin");
|
|
lines = QUtil::read_lines_from_file(std::cin);
|
|
} else {
|
|
QTC::TC("qpdf", "QPDFJob_config password file");
|
|
lines = QUtil::read_lines_from_file(parameter.c_str());
|
|
}
|
|
if (lines.size() >= 1) {
|
|
o.m->password = QUtil::make_shared_cstr(lines.front());
|
|
|
|
if (lines.size() > 1) {
|
|
*QPDFLogger::defaultLogger()->getError()
|
|
<< this->o.m->message_prefix << ": WARNING: all but the first line of"
|
|
<< " the password file are ignored\n";
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::passwordMode(std::string const& parameter)
|
|
{
|
|
if (parameter == "bytes") {
|
|
o.m->password_mode = QPDFJob::pm_bytes;
|
|
} else if (parameter == "hex-bytes") {
|
|
o.m->password_mode = QPDFJob::pm_hex_bytes;
|
|
} else if (parameter == "unicode") {
|
|
o.m->password_mode = QPDFJob::pm_unicode;
|
|
} else if (parameter == "auto") {
|
|
o.m->password_mode = QPDFJob::pm_auto;
|
|
} else {
|
|
usage("invalid password-mode option");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::streamData(std::string const& parameter)
|
|
{
|
|
o.m->stream_data_set = true;
|
|
if (parameter == "compress") {
|
|
o.m->stream_data_mode = qpdf_s_compress;
|
|
} else if (parameter == "preserve") {
|
|
o.m->stream_data_mode = qpdf_s_preserve;
|
|
} else if (parameter == "uncompress") {
|
|
o.m->stream_data_mode = qpdf_s_uncompress;
|
|
} else {
|
|
usage("invalid stream-data option");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::decodeLevel(std::string const& parameter)
|
|
{
|
|
o.m->decode_level_set = true;
|
|
if (parameter == "none") {
|
|
o.m->decode_level = qpdf_dl_none;
|
|
} else if (parameter == "generalized") {
|
|
o.m->decode_level = qpdf_dl_generalized;
|
|
} else if (parameter == "specialized") {
|
|
o.m->decode_level = qpdf_dl_specialized;
|
|
} else if (parameter == "all") {
|
|
o.m->decode_level = qpdf_dl_all;
|
|
} else {
|
|
usage("invalid option");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::objectStreams(std::string const& parameter)
|
|
{
|
|
o.m->object_stream_set = true;
|
|
if (parameter == "disable") {
|
|
o.m->object_stream_mode = qpdf_o_disable;
|
|
} else if (parameter == "preserve") {
|
|
o.m->object_stream_mode = qpdf_o_preserve;
|
|
} else if (parameter == "generate") {
|
|
o.m->object_stream_mode = qpdf_o_generate;
|
|
} else {
|
|
usage("invalid object stream mode");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::removeUnreferencedResources(std::string const& parameter)
|
|
{
|
|
if (parameter == "auto") {
|
|
o.m->remove_unreferenced_page_resources = QPDFJob::re_auto;
|
|
} else if (parameter == "yes") {
|
|
o.m->remove_unreferenced_page_resources = QPDFJob::re_yes;
|
|
} else if (parameter == "no") {
|
|
o.m->remove_unreferenced_page_resources = QPDFJob::re_no;
|
|
} else {
|
|
usage("invalid value for --remove-unreferenced-page-resources");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::showObject(std::string const& parameter)
|
|
{
|
|
QPDFJob::parse_object_id(parameter, o.m->show_trailer, o.m->show_obj, o.m->show_gen);
|
|
o.m->require_outfile = false;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::jobJsonFile(std::string const& parameter)
|
|
{
|
|
try {
|
|
o.initializeFromJson(QUtil::read_file_into_string(parameter.c_str()), true);
|
|
} catch (std::exception& e) {
|
|
throw std::runtime_error(
|
|
"error with job-json file " + std::string(parameter) + ": " + e.what() + "\nRun " +
|
|
this->o.m->message_prefix + " --job-json-help for information on the file format.");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::rotate(std::string const& parameter)
|
|
{
|
|
o.parseRotationParameter(parameter);
|
|
return this;
|
|
}
|
|
|
|
std::shared_ptr<QPDFJob::CopyAttConfig>
|
|
QPDFJob::Config::copyAttachmentsFrom()
|
|
{
|
|
return std::shared_ptr<CopyAttConfig>(new CopyAttConfig(this));
|
|
}
|
|
|
|
QPDFJob::CopyAttConfig::CopyAttConfig(Config* c) :
|
|
config(c)
|
|
{
|
|
}
|
|
|
|
QPDFJob::CopyAttConfig*
|
|
QPDFJob::CopyAttConfig::file(std::string const& parameter)
|
|
{
|
|
this->caf.path = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::CopyAttConfig*
|
|
QPDFJob::CopyAttConfig::prefix(std::string const& parameter)
|
|
{
|
|
this->caf.prefix = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::CopyAttConfig*
|
|
QPDFJob::CopyAttConfig::password(std::string const& parameter)
|
|
{
|
|
this->caf.password = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::CopyAttConfig::endCopyAttachmentsFrom()
|
|
{
|
|
if (this->caf.path.empty()) {
|
|
usage("copy attachments: no file specified");
|
|
}
|
|
this->config->o.m->attachments_to_copy.push_back(this->caf);
|
|
return this->config;
|
|
}
|
|
|
|
QPDFJob::AttConfig::AttConfig(Config* c) :
|
|
config(c)
|
|
{
|
|
}
|
|
|
|
std::shared_ptr<QPDFJob::AttConfig>
|
|
QPDFJob::Config::addAttachment()
|
|
{
|
|
return std::shared_ptr<AttConfig>(new AttConfig(this));
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::file(std::string const& parameter)
|
|
{
|
|
this->att.path = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::key(std::string const& parameter)
|
|
{
|
|
this->att.key = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::filename(std::string const& parameter)
|
|
{
|
|
this->att.filename = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::creationdate(std::string const& parameter)
|
|
{
|
|
if (!QUtil::pdf_time_to_qpdf_time(parameter)) {
|
|
usage(std::string(parameter) + " is not a valid PDF timestamp");
|
|
}
|
|
this->att.creationdate = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::moddate(std::string const& parameter)
|
|
{
|
|
if (!QUtil::pdf_time_to_qpdf_time(parameter)) {
|
|
usage(std::string(parameter) + " is not a valid PDF timestamp");
|
|
}
|
|
this->att.moddate = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::mimetype(std::string const& parameter)
|
|
{
|
|
if (parameter.find('/') == std::string::npos) {
|
|
usage("mime type should be specified as type/subtype");
|
|
}
|
|
this->att.mimetype = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::description(std::string const& parameter)
|
|
{
|
|
this->att.description = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::AttConfig*
|
|
QPDFJob::AttConfig::replace()
|
|
{
|
|
this->att.replace = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::AttConfig::endAddAttachment()
|
|
{
|
|
static std::string now = QUtil::qpdf_time_to_pdf_time(QUtil::get_current_qpdf_time());
|
|
if (this->att.path.empty()) {
|
|
usage("add attachment: no file specified");
|
|
}
|
|
std::string last_element = QUtil::path_basename(this->att.path);
|
|
if (last_element.empty()) {
|
|
usage("file for --add-attachment may not be empty");
|
|
}
|
|
if (this->att.filename.empty()) {
|
|
this->att.filename = last_element;
|
|
}
|
|
if (this->att.key.empty()) {
|
|
this->att.key = last_element;
|
|
}
|
|
if (this->att.creationdate.empty()) {
|
|
this->att.creationdate = now;
|
|
}
|
|
if (this->att.moddate.empty()) {
|
|
this->att.moddate = now;
|
|
}
|
|
|
|
this->config->o.m->attachments_to_add.push_back(this->att);
|
|
return this->config;
|
|
}
|
|
|
|
QPDFJob::PagesConfig::PagesConfig(Config* c) :
|
|
config(c)
|
|
{
|
|
}
|
|
|
|
std::shared_ptr<QPDFJob::PagesConfig>
|
|
QPDFJob::Config::pages()
|
|
{
|
|
if (!o.m->page_specs.empty()) {
|
|
usage("--pages may only be specified one time");
|
|
}
|
|
return std::shared_ptr<PagesConfig>(new PagesConfig(this));
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::PagesConfig::endPages()
|
|
{
|
|
auto n_specs = config->o.m->page_specs.size();
|
|
if (n_specs == 0) {
|
|
usage("--pages: no page specifications given");
|
|
}
|
|
return this->config;
|
|
}
|
|
|
|
QPDFJob::PagesConfig*
|
|
QPDFJob::PagesConfig::pageSpec(
|
|
std::string const& filename, std::string const& range, char const* password)
|
|
{
|
|
this->config->o.m->page_specs.emplace_back(filename, password, range);
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::PagesConfig*
|
|
QPDFJob::PagesConfig::file(std::string const& arg)
|
|
{
|
|
this->config->o.m->page_specs.emplace_back(arg, nullptr, "");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::PagesConfig*
|
|
QPDFJob::PagesConfig::range(std::string const& arg)
|
|
{
|
|
if (config->o.m->page_specs.empty()) {
|
|
QTC::TC("qpdf", "QPDFJob misplaced page range");
|
|
usage("in --range must follow a file name");
|
|
}
|
|
auto& last = config->o.m->page_specs.back();
|
|
if (!last.range.empty()) {
|
|
QTC::TC("qpdf", "QPDFJob duplicated range");
|
|
usage("--range already specified for this file");
|
|
}
|
|
last.range = arg;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::PagesConfig*
|
|
QPDFJob::PagesConfig::password(std::string const& arg)
|
|
{
|
|
if (config->o.m->page_specs.empty()) {
|
|
QTC::TC("qpdf", "QPDFJob misplaced pages password");
|
|
usage("in --pages, --password must follow a file name");
|
|
}
|
|
auto& last = config->o.m->page_specs.back();
|
|
if (last.password) {
|
|
QTC::TC("qpdf", "QPDFJob duplicated pages password");
|
|
usage("--password already specified for this file");
|
|
}
|
|
last.password = QUtil::make_shared_cstr(arg);
|
|
return this;
|
|
}
|
|
|
|
std::shared_ptr<QPDFJob::UOConfig>
|
|
QPDFJob::Config::overlay()
|
|
{
|
|
o.m->overlay.emplace_back("overlay");
|
|
o.m->under_overlay = &o.m->overlay.back();
|
|
return std::shared_ptr<UOConfig>(new UOConfig(this));
|
|
}
|
|
|
|
std::shared_ptr<QPDFJob::UOConfig>
|
|
QPDFJob::Config::underlay()
|
|
{
|
|
o.m->underlay.emplace_back("underlay");
|
|
o.m->under_overlay = &o.m->underlay.back();
|
|
return std::shared_ptr<UOConfig>(new UOConfig(this));
|
|
}
|
|
|
|
QPDFJob::UOConfig::UOConfig(Config* c) :
|
|
config(c)
|
|
{
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::UOConfig::endUnderlayOverlay()
|
|
{
|
|
if (config->o.m->under_overlay->filename.empty()) {
|
|
usage(config->o.m->under_overlay->which + " file not specified");
|
|
}
|
|
config->o.m->under_overlay = nullptr;
|
|
return this->config;
|
|
}
|
|
|
|
QPDFJob::UOConfig*
|
|
QPDFJob::UOConfig::file(std::string const& parameter)
|
|
{
|
|
if (!config->o.m->under_overlay->filename.empty()) {
|
|
usage(config->o.m->under_overlay->which + " file already specified");
|
|
} else {
|
|
config->o.m->under_overlay->filename = parameter;
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::UOConfig*
|
|
QPDFJob::UOConfig::to(std::string const& parameter)
|
|
{
|
|
config->o.parseNumrange(parameter.c_str(), 0);
|
|
config->o.m->under_overlay->to_nr = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::UOConfig*
|
|
QPDFJob::UOConfig::from(std::string const& parameter)
|
|
{
|
|
if (!parameter.empty()) {
|
|
config->o.parseNumrange(parameter.c_str(), 0);
|
|
}
|
|
config->o.m->under_overlay->from_nr = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::UOConfig*
|
|
QPDFJob::UOConfig::repeat(std::string const& parameter)
|
|
{
|
|
if (!parameter.empty()) {
|
|
config->o.parseNumrange(parameter.c_str(), 0);
|
|
}
|
|
config->o.m->under_overlay->repeat_nr = parameter;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::UOConfig*
|
|
QPDFJob::UOConfig::password(std::string const& parameter)
|
|
{
|
|
config->o.m->under_overlay->password = QUtil::make_shared_cstr(parameter);
|
|
return this;
|
|
}
|
|
|
|
std::shared_ptr<QPDFJob::EncConfig>
|
|
QPDFJob::Config::encrypt(
|
|
int keylen, std::string const& user_password, std::string const& owner_password)
|
|
{
|
|
o.m->keylen = keylen;
|
|
if (keylen == 256) {
|
|
o.m->use_aes = true;
|
|
}
|
|
o.m->user_password = user_password;
|
|
o.m->owner_password = owner_password;
|
|
return std::shared_ptr<EncConfig>(new EncConfig(this));
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::Config::setPageLabels(const std::vector<std::string>& specs)
|
|
{
|
|
static std::regex page_label_re(R"(^(z|r?\d+):([DaArR])?(?:/(\d+)?(?:/(.+)?)?)?$)");
|
|
o.m->page_label_specs.clear();
|
|
for (auto const& spec: specs) {
|
|
std::smatch match;
|
|
if (!std::regex_match(spec, match, page_label_re)) {
|
|
usage("page label spec must be n:[D|a|A|r|R][/start[/prefix]]");
|
|
}
|
|
auto first_page_str = match[1].str();
|
|
int first_page;
|
|
if (first_page_str == "z") {
|
|
first_page = -1;
|
|
} else if (first_page_str.at(0) == 'r') {
|
|
first_page = -QUtil::string_to_int(first_page_str.substr(1).c_str());
|
|
} else {
|
|
first_page = QUtil::string_to_int(first_page_str.c_str());
|
|
}
|
|
auto label_type_ch = match[2].matched ? match[2].str().at(0) : '\0';
|
|
qpdf_page_label_e label_type;
|
|
switch (label_type_ch) {
|
|
case 'D':
|
|
label_type = pl_digits;
|
|
break;
|
|
case 'a':
|
|
label_type = pl_alpha_lower;
|
|
break;
|
|
case 'A':
|
|
label_type = pl_alpha_upper;
|
|
break;
|
|
case 'r':
|
|
label_type = pl_roman_lower;
|
|
break;
|
|
case 'R':
|
|
label_type = pl_roman_upper;
|
|
break;
|
|
default:
|
|
label_type = pl_none;
|
|
}
|
|
|
|
auto start_num = match[3].matched ? QUtil::string_to_int(match[3].str().c_str()) : 1;
|
|
if (start_num < 1) {
|
|
usage("starting page number must be >= 1");
|
|
}
|
|
auto prefix = match[4].matched ? match[4].str() : "";
|
|
// We can't check ordering until we know how many pages there are, so that is delayed until
|
|
// near the end.
|
|
o.m->page_label_specs.emplace_back(first_page, label_type, start_num, prefix);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig::EncConfig(Config* c) :
|
|
config(c)
|
|
{
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::EncConfig::endEncrypt()
|
|
{
|
|
if (config->o.m->keylen == 0) {
|
|
usage("encryption key length is required");
|
|
}
|
|
config->o.m->encrypt = true;
|
|
config->o.m->decrypt = false;
|
|
config->o.m->copy_encryption = false;
|
|
return this->config;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::allowInsecure()
|
|
{
|
|
config->o.m->allow_insecure = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::accessibility(std::string const& parameter)
|
|
{
|
|
config->o.m->r3_accessibility = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::extract(std::string const& parameter)
|
|
{
|
|
if (config->o.m->keylen == 40) {
|
|
config->o.m->r2_extract = (parameter == "y");
|
|
} else {
|
|
config->o.m->r3_extract = (parameter == "y");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::print(std::string const& parameter)
|
|
{
|
|
if (config->o.m->keylen == 40) {
|
|
config->o.m->r2_print = (parameter == "y");
|
|
} else if (parameter == "full") {
|
|
config->o.m->r3_print = qpdf_r3p_full;
|
|
} else if (parameter == "low") {
|
|
config->o.m->r3_print = qpdf_r3p_low;
|
|
} else if (parameter == "none") {
|
|
config->o.m->r3_print = qpdf_r3p_none;
|
|
} else {
|
|
usage("invalid print option");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::modify(std::string const& parameter)
|
|
{
|
|
if (config->o.m->keylen == 40) {
|
|
config->o.m->r2_modify = (parameter == "y");
|
|
} else if (parameter == "all") {
|
|
config->o.m->r3_assemble = true;
|
|
config->o.m->r3_annotate_and_form = true;
|
|
config->o.m->r3_form_filling = true;
|
|
config->o.m->r3_modify_other = true;
|
|
} else if (parameter == "annotate") {
|
|
config->o.m->r3_assemble = true;
|
|
config->o.m->r3_annotate_and_form = true;
|
|
config->o.m->r3_form_filling = true;
|
|
config->o.m->r3_modify_other = false;
|
|
} else if (parameter == "form") {
|
|
config->o.m->r3_assemble = true;
|
|
config->o.m->r3_annotate_and_form = false;
|
|
config->o.m->r3_form_filling = true;
|
|
config->o.m->r3_modify_other = false;
|
|
} else if (parameter == "assembly") {
|
|
config->o.m->r3_assemble = true;
|
|
config->o.m->r3_annotate_and_form = false;
|
|
config->o.m->r3_form_filling = false;
|
|
config->o.m->r3_modify_other = false;
|
|
} else if (parameter == "none") {
|
|
config->o.m->r3_assemble = false;
|
|
config->o.m->r3_annotate_and_form = false;
|
|
config->o.m->r3_form_filling = false;
|
|
config->o.m->r3_modify_other = false;
|
|
} else {
|
|
usage("invalid modify option");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::cleartextMetadata()
|
|
{
|
|
config->o.m->cleartext_metadata = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::assemble(std::string const& parameter)
|
|
{
|
|
config->o.m->r3_assemble = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::annotate(std::string const& parameter)
|
|
{
|
|
if (config->o.m->keylen == 40) {
|
|
config->o.m->r2_annotate = (parameter == "y");
|
|
} else {
|
|
config->o.m->r3_annotate_and_form = (parameter == "y");
|
|
}
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::form(std::string const& parameter)
|
|
{
|
|
config->o.m->r3_form_filling = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::modifyOther(std::string const& parameter)
|
|
{
|
|
config->o.m->r3_modify_other = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::useAes(std::string const& parameter)
|
|
{
|
|
config->o.m->use_aes = (parameter == "y");
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::forceV4()
|
|
{
|
|
config->o.m->force_V4 = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::EncConfig*
|
|
QPDFJob::EncConfig::forceR5()
|
|
{
|
|
config->o.m->force_R5 = true;
|
|
return this;
|
|
}
|
|
|
|
QPDFJob::PageLabelsConfig::PageLabelsConfig(Config* c) :
|
|
config(c)
|
|
{
|
|
}
|
|
|
|
QPDFJob::Config*
|
|
QPDFJob::PageLabelsConfig::endSetPageLabels()
|
|
{
|
|
return this->config;
|
|
}
|