2020-10-02 08:22:40 +03:00
|
|
|
/*
|
2021-09-01 05:32:46 +03:00
|
|
|
* Copyright (c) 2020, Peter Elliott <pelliott@serenityos.org>
|
2021-04-22 23:40:43 +03:00
|
|
|
* Copyright (c) 2021, Idan Horowitz <idan.horowitz@serenityos.org>
|
2020-10-02 08:22:40 +03:00
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-10-02 08:22:40 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <AK/Span.h>
|
|
|
|
#include <AK/Stream.h>
|
2021-03-18 17:22:05 +03:00
|
|
|
#include <LibArchive/Tar.h>
|
2020-10-02 08:22:40 +03:00
|
|
|
|
2021-03-18 17:22:05 +03:00
|
|
|
namespace Archive {
|
2020-10-02 08:22:40 +03:00
|
|
|
|
2021-03-13 02:40:04 +03:00
|
|
|
class TarInputStream;
|
2020-10-02 08:22:40 +03:00
|
|
|
|
|
|
|
class TarFileStream : public InputStream {
|
|
|
|
public:
|
|
|
|
size_t read(Bytes) override;
|
|
|
|
bool unreliable_eof() const override;
|
|
|
|
|
|
|
|
bool read_or_error(Bytes) override;
|
|
|
|
bool discard_or_error(size_t count) override;
|
|
|
|
|
|
|
|
private:
|
2021-03-13 02:40:04 +03:00
|
|
|
TarFileStream(TarInputStream& stream);
|
|
|
|
TarInputStream& m_tar_stream;
|
2020-10-02 08:22:40 +03:00
|
|
|
int m_generation;
|
|
|
|
|
2021-03-13 02:40:04 +03:00
|
|
|
friend class TarInputStream;
|
2020-10-02 08:22:40 +03:00
|
|
|
};
|
|
|
|
|
2021-03-13 02:40:04 +03:00
|
|
|
class TarInputStream {
|
2020-10-02 08:22:40 +03:00
|
|
|
public:
|
2021-03-13 02:40:04 +03:00
|
|
|
TarInputStream(InputStream&);
|
2020-10-02 08:22:40 +03:00
|
|
|
void advance();
|
|
|
|
bool finished() const { return m_finished; }
|
2020-10-03 21:59:41 +03:00
|
|
|
bool valid() const;
|
2022-04-01 20:58:27 +03:00
|
|
|
TarFileHeader const& header() const { return m_header; }
|
2020-10-02 08:22:40 +03:00
|
|
|
TarFileStream file_contents();
|
|
|
|
|
2022-03-04 21:32:51 +03:00
|
|
|
template<VoidFunction<StringView, StringView> F>
|
|
|
|
ErrorOr<void> for_each_extended_header(F func);
|
|
|
|
|
2020-10-02 08:22:40 +03:00
|
|
|
private:
|
2021-03-18 17:22:05 +03:00
|
|
|
TarFileHeader m_header;
|
2020-10-02 08:22:40 +03:00
|
|
|
InputStream& m_stream;
|
|
|
|
unsigned long m_file_offset { 0 };
|
|
|
|
int m_generation { 0 };
|
|
|
|
bool m_finished { false };
|
|
|
|
|
|
|
|
friend class TarFileStream;
|
|
|
|
};
|
|
|
|
|
2021-03-13 02:40:04 +03:00
|
|
|
class TarOutputStream {
|
|
|
|
public:
|
|
|
|
TarOutputStream(OutputStream&);
|
2022-04-01 20:58:27 +03:00
|
|
|
void add_file(String const& path, mode_t, ReadonlyBytes);
|
|
|
|
void add_directory(String const& path, mode_t);
|
2021-03-13 02:40:04 +03:00
|
|
|
void finish();
|
|
|
|
|
|
|
|
private:
|
|
|
|
OutputStream& m_stream;
|
|
|
|
bool m_finished { false };
|
|
|
|
|
|
|
|
friend class TarFileStream;
|
|
|
|
};
|
|
|
|
|
2022-03-04 21:32:51 +03:00
|
|
|
template<VoidFunction<StringView, StringView> F>
|
|
|
|
inline ErrorOr<void> TarInputStream::for_each_extended_header(F func)
|
|
|
|
{
|
|
|
|
VERIFY(header().content_is_like_extended_header());
|
|
|
|
|
|
|
|
Archive::TarFileStream file_stream = file_contents();
|
|
|
|
|
|
|
|
ByteBuffer file_contents_buffer = TRY(ByteBuffer::create_zeroed(header().size()));
|
|
|
|
VERIFY(file_stream.read(file_contents_buffer) == header().size());
|
|
|
|
|
|
|
|
StringView file_contents { file_contents_buffer };
|
|
|
|
|
|
|
|
while (!file_contents.is_empty()) {
|
|
|
|
// Split off the length (until the first space).
|
|
|
|
Optional<size_t> length_end_index = file_contents.find(' ');
|
|
|
|
if (!length_end_index.has_value())
|
|
|
|
return Error::from_string_literal("Malformed extended header: No length found.");
|
|
|
|
Optional<unsigned int> length = file_contents.substring_view(0, length_end_index.value()).to_uint();
|
|
|
|
if (!length.has_value())
|
|
|
|
return Error::from_string_literal("Malformed extended header: Could not parse length.");
|
|
|
|
unsigned int remaining_length = length.value();
|
|
|
|
|
|
|
|
remaining_length -= length_end_index.value() + 1;
|
|
|
|
file_contents = file_contents.substring_view(length_end_index.value() + 1);
|
|
|
|
|
|
|
|
// Extract the header.
|
|
|
|
StringView header = file_contents.substring_view(0, remaining_length - 1);
|
|
|
|
file_contents = file_contents.substring_view(remaining_length - 1);
|
|
|
|
|
|
|
|
// Ensure that the header ends at the expected location.
|
|
|
|
if (file_contents.length() < 1 || !file_contents.starts_with('\n'))
|
|
|
|
return Error::from_string_literal("Malformed extended header: Header does not end at expected location.");
|
|
|
|
file_contents = file_contents.substring_view(1);
|
|
|
|
|
|
|
|
// Find the delimiting '='.
|
|
|
|
Optional<size_t> header_delimiter_index = header.find('=');
|
|
|
|
if (!header_delimiter_index.has_value())
|
|
|
|
return Error::from_string_literal("Malformed extended header: Header does not have a delimiter.");
|
|
|
|
StringView key = header.substring_view(0, header_delimiter_index.value());
|
|
|
|
StringView value = header.substring_view(header_delimiter_index.value() + 1);
|
|
|
|
|
|
|
|
func(key, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-10-02 08:22:40 +03:00
|
|
|
}
|