1
0
Fork 0
mirror of https://github.com/LadybirdBrowser/ladybird.git synced 2025-06-09 17:44:56 +09:00

LibCore: Add support for range-based for loops on LineIterators

This commit is contained in:
Sahan Fernando 2021-12-07 01:16:01 +11:00 committed by Ali Mohammad Pur
parent 6d948c1a92
commit 2c43eaa50c
Notes: sideshowbarker 2024-07-17 22:54:24 +09:00
3 changed files with 54 additions and 17 deletions

View file

@ -8,6 +8,14 @@
#include <LibTest/TestCase.h>
#include <unistd.h>
static bool files_have_same_contents(String filename1, String filename2)
{
auto file1 = Core::File::open(filename1, Core::OpenMode::ReadOnly).value();
auto file2 = Core::File::open(filename2, Core::OpenMode::ReadOnly).value();
auto contents1 = file1->read_all(), contents2 = file2->read_all();
return contents1 == contents2;
}
TEST_CASE(file_readline)
{
auto path = "long_lines.txt";
@ -26,23 +34,7 @@ TEST_CASE(file_readline)
}
file->close();
outputfile->close();
// Open files again for comparison since otherwise read_all returns empty (even when not closing the file)
file = Core::File::construct(path);
if (!file->open(Core::OpenMode::ReadOnly))
VERIFY_NOT_REACHED();
outputfile = Core::File::construct(output_path);
if (!outputfile->open(Core::OpenMode::ReadOnly))
VERIFY_NOT_REACHED();
auto inputData = file->read_all();
auto outputData = outputfile->read_all();
EXPECT(inputData.size() > 0);
EXPECT_EQ(inputData.size(), outputData.size());
// Compare char by char
for (size_t i = 0; i < inputData.size(); i++) {
EXPECT_EQ(inputData[i], outputData[i]);
}
VERIFY(files_have_same_contents(path, output_path));
}
TEST_CASE(file_get_read_position)
@ -76,3 +68,24 @@ TEST_CASE(file_get_read_position)
EXPECT_EQ(offset, 0);
}
}
TEST_CASE(file_lines_range)
{
auto path = "long_lines.txt";
auto file_or_error = Core::File::open(path, Core::OpenMode::ReadOnly);
if (file_or_error.is_error()) {
warnln("Failed to open {}: {}", path, file_or_error.error());
VERIFY_NOT_REACHED();
}
auto file = file_or_error.release_value();
auto output_path = "/tmp/output.txt";
auto outfile_or_error = Core::File::open(output_path, Core::OpenMode::WriteOnly);
auto outputfile = outfile_or_error.release_value();
for (auto line : file->lines()) {
outputfile->write(line);
outputfile->write("\n");
}
file->close();
outputfile->close();
VERIFY(files_have_same_contents(path, output_path));
}

View file

@ -319,4 +319,8 @@ LineIterator& LineIterator::operator++()
m_buffer = m_device->read_line();
return *this;
}
LineIterator LineRange::begin() { return m_device.line_begin(); }
LineIterator LineRange::end() { return m_device.line_end(); }
}

View file

@ -12,6 +12,8 @@
namespace Core {
class IODevice;
// This is not necessarily a valid iterator in all contexts,
// if we had concepts, this would be InputIterator, not Copyable, Movable.
class LineIterator {
@ -34,6 +36,20 @@ private:
String m_buffer;
};
class LineRange {
public:
LineRange() = delete;
explicit LineRange(IODevice& device)
: m_device(device)
{
}
LineIterator begin();
LineIterator end();
private:
IODevice& m_device;
};
enum class OpenMode : unsigned {
NotOpen = 0,
ReadOnly = 1,
@ -91,6 +107,10 @@ public:
LineIterator line_begin() & { return LineIterator(*this); }
LineIterator line_end() { return LineIterator(*this, true); }
LineRange lines()
{
return LineRange(*this);
}
protected:
explicit IODevice(Object* parent = nullptr);