summaryrefslogtreecommitdiff
path: root/somebar/src/line_buffer.hpp
blob: a5497bf5b6f3aaec7ba928016f2578c0c6685834 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
// somebar - dwl bar
// See LICENSE file for copyright and license details.

#pragma once
#include <array>
#include <algorithm>
#include <sys/types.h>

// reads data from Reader, and passes complete lines to Consumer.
template<size_t BufSize>
class LineBuffer {
	using Iterator = typename std::array<char, BufSize>::iterator;
	std::array<char, BufSize> _buffer;
	Iterator _bufferedTo;
	Iterator _consumedTo;
	bool _discardLine {false};
public:
	LineBuffer()
		: _bufferedTo {_buffer.begin()}
		, _consumedTo {_buffer.begin()}
	{
	}

	template<typename Reader, typename Consumer>
	ssize_t readLines(const Reader& reader, const Consumer& consumer)
	{
		while (true) {
			auto bytesRead = reader(_bufferedTo, _buffer.end() - _bufferedTo);
			if (bytesRead <= 0) {
				return bytesRead;
			}
			_bufferedTo += bytesRead;
			dispatchLines(consumer);
			resetBuffer();
		}
	}
private:
	template<typename Consumer>
	void dispatchLines(const Consumer& consumer)
	{
		while (true) {
			auto separator = std::find(_consumedTo, _bufferedTo, '\n');
			if (separator == _bufferedTo) {
				break;
			}
			size_t lineLength = separator - _consumedTo;
			if (!_discardLine) {
				consumer(_consumedTo, lineLength);
			}
			_consumedTo = separator + 1;
			_discardLine = false;
		}
	}

	void resetBuffer()
	{
		size_t bytesRemaining = _bufferedTo - _consumedTo;
		if (bytesRemaining == _buffer.size()) {
			// line too long
			_discardLine = true;
			_consumedTo = _buffer.begin();
			_bufferedTo = _buffer.begin();
		} else {
			// move the last partial message to the front of the buffer, so a full-sized
			// message will fit
			std::copy(_consumedTo, _bufferedTo, _buffer.begin());
			_consumedTo = _buffer.begin();
			_bufferedTo = _consumedTo + bytesRemaining;
		}
	}
};