TarStream.h 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. /*
  2. * Copyright (c) 2020, Peter Elliott <pelliott@serenityos.org>
  3. * Copyright (c) 2021, Idan Horowitz <idan.horowitz@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/MaybeOwned.h>
  9. #include <AK/Span.h>
  10. #include <AK/Stream.h>
  11. #include <LibArchive/Tar.h>
  12. namespace Archive {
  13. class TarInputStream;
  14. class TarFileStream : public Stream {
  15. public:
  16. virtual ErrorOr<Bytes> read_some(Bytes) override;
  17. virtual ErrorOr<size_t> write_some(ReadonlyBytes) override;
  18. virtual bool is_eof() const override;
  19. virtual bool is_open() const override { return true; }
  20. virtual void close() override {};
  21. private:
  22. TarFileStream(TarInputStream& stream);
  23. TarInputStream& m_tar_stream;
  24. int m_generation;
  25. friend class TarInputStream;
  26. };
  27. class TarInputStream {
  28. public:
  29. static ErrorOr<NonnullOwnPtr<TarInputStream>> construct(NonnullOwnPtr<Stream>);
  30. ErrorOr<void> advance();
  31. bool finished() const { return m_found_end_of_archive || m_stream->is_eof(); }
  32. ErrorOr<bool> valid() const;
  33. TarFileHeader const& header() const { return m_header; }
  34. TarFileStream file_contents();
  35. template<VoidFunction<StringView, StringView> F>
  36. ErrorOr<void> for_each_extended_header(F func);
  37. private:
  38. TarInputStream(NonnullOwnPtr<Stream>);
  39. ErrorOr<void> load_next_header();
  40. TarFileHeader m_header;
  41. NonnullOwnPtr<Stream> m_stream;
  42. unsigned long m_file_offset { 0 };
  43. int m_generation { 0 };
  44. bool m_found_end_of_archive { false };
  45. friend class TarFileStream;
  46. };
  47. class TarOutputStream {
  48. public:
  49. TarOutputStream(MaybeOwned<Stream>);
  50. ErrorOr<void> add_file(StringView path, mode_t, ReadonlyBytes);
  51. ErrorOr<void> add_link(StringView path, mode_t, StringView);
  52. ErrorOr<void> add_directory(StringView path, mode_t);
  53. ErrorOr<void> finish();
  54. private:
  55. MaybeOwned<Stream> m_stream;
  56. bool m_finished { false };
  57. friend class TarFileStream;
  58. };
  59. template<VoidFunction<StringView, StringView> F>
  60. inline ErrorOr<void> TarInputStream::for_each_extended_header(F func)
  61. {
  62. VERIFY(header().content_is_like_extended_header());
  63. Archive::TarFileStream file_stream = file_contents();
  64. auto header_size = TRY(header().size());
  65. ByteBuffer file_contents_buffer = TRY(ByteBuffer::create_zeroed(header_size));
  66. TRY(file_stream.read_until_filled(file_contents_buffer));
  67. StringView file_contents { file_contents_buffer };
  68. while (!file_contents.is_empty()) {
  69. // Split off the length (until the first space).
  70. Optional<size_t> length_end_index = file_contents.find(' ');
  71. if (!length_end_index.has_value())
  72. return Error::from_string_literal("Malformed extended header: No length found.");
  73. Optional<unsigned> length = file_contents.substring_view(0, length_end_index.value()).to_number<unsigned>();
  74. if (!length.has_value())
  75. return Error::from_string_literal("Malformed extended header: Could not parse length.");
  76. if (length_end_index.value() >= length.value())
  77. return Error::from_string_literal("Malformed extended header: Header length too short.");
  78. unsigned int remaining_length = length.value();
  79. remaining_length -= length_end_index.value() + 1;
  80. file_contents = file_contents.substring_view(length_end_index.value() + 1);
  81. if (file_contents.length() < remaining_length - 1)
  82. return Error::from_string_literal("Malformed extended header: Header length too large.");
  83. // Extract the header.
  84. StringView header = file_contents.substring_view(0, remaining_length - 1);
  85. file_contents = file_contents.substring_view(remaining_length - 1);
  86. // Ensure that the header ends at the expected location.
  87. if (file_contents.length() < 1 || !file_contents.starts_with('\n'))
  88. return Error::from_string_literal("Malformed extended header: Header does not end at expected location.");
  89. file_contents = file_contents.substring_view(1);
  90. // Find the delimiting '='.
  91. Optional<size_t> header_delimiter_index = header.find('=');
  92. if (!header_delimiter_index.has_value())
  93. return Error::from_string_literal("Malformed extended header: Header does not have a delimiter.");
  94. StringView key = header.substring_view(0, header_delimiter_index.value());
  95. StringView value = header.substring_view(header_delimiter_index.value() + 1);
  96. func(key, value);
  97. }
  98. return {};
  99. }
  100. }