#include #include #include #include #include #include #include #include #include namespace xrpl { void extractTarLz4(boost::filesystem::path const& src, boost::filesystem::path const& dst) { if (!is_regular_file(src)) Throw("Invalid source file"); using archive_ptr = std::unique_ptr; archive_ptr const ar{archive_read_new(), [](struct archive* a) { archive_read_free(a); }}; if (!ar) Throw("Failed to allocate archive"); if (archive_read_support_format_tar(ar.get()) < ARCHIVE_OK) Throw(archive_error_string(ar.get())); if (archive_read_support_filter_lz4(ar.get()) < ARCHIVE_OK) Throw(archive_error_string(ar.get())); // Examples suggest this block size if (archive_read_open_filename(ar.get(), src.string().c_str(), 10240) < ARCHIVE_OK) { Throw(archive_error_string(ar.get())); } archive_ptr const aw{ archive_write_disk_new(), [](struct archive* a) { archive_write_free(a); }}; if (!aw) Throw("Failed to allocate archive"); if (archive_write_disk_set_options( aw.get(), ARCHIVE_EXTRACT_TIME | ARCHIVE_EXTRACT_PERM | ARCHIVE_EXTRACT_ACL | ARCHIVE_EXTRACT_FFLAGS) < ARCHIVE_OK) { Throw(archive_error_string(aw.get())); } if (archive_write_disk_set_standard_lookup(aw.get()) < ARCHIVE_OK) Throw(archive_error_string(aw.get())); int result = 0; struct archive_entry* entry = nullptr; while (true) { result = archive_read_next_header(ar.get(), &entry); if (result == ARCHIVE_EOF) break; if (result < ARCHIVE_OK) Throw(archive_error_string(ar.get())); archive_entry_set_pathname(entry, (dst / archive_entry_pathname(entry)).string().c_str()); if (archive_write_header(aw.get(), entry) < ARCHIVE_OK) Throw(archive_error_string(aw.get())); if (archive_entry_size(entry) > 0) { void const* buf = nullptr; size_t sz = 0; la_int64_t offset = 0; while (true) { result = archive_read_data_block(ar.get(), &buf, &sz, &offset); if (result == ARCHIVE_EOF) break; if (result < ARCHIVE_OK) Throw(archive_error_string(ar.get())); if (archive_write_data_block(aw.get(), buf, sz, offset) < ARCHIVE_OK) { Throw(archive_error_string(aw.get())); } } } if (archive_write_finish_entry(aw.get()) < ARCHIVE_OK) Throw(archive_error_string(aw.get())); } } } // namespace xrpl