Skip to content

Commit f499e70

Browse files
committed
Rework how decoding is set up to support a wider variety of inputs
1 parent 5aa0c15 commit f499e70

File tree

4 files changed

+276
-154
lines changed

4 files changed

+276
-154
lines changed

CHANGES.md

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
### New features
66
- `bencode::data` (and `bencode::basic_data`, etc) now support `operator []` and
77
`at` member functions to get list/dictionary elements
8+
- Decoding functions now accept a pointer plus length as input
89

910
### Breaking changes
1011
- To decode only the next bencode object in a string or stream, you must now

README.md

+9-5
Original file line numberDiff line numberDiff line change
@@ -54,18 +54,20 @@ However, you can [customize this](#bringing-your-own-variant) if you like.
5454

5555
### Decoding
5656

57-
Decoding bencoded data is simple. Just call `decode`, which will return a `data`
58-
object that you can operate on:
57+
Decoding bencoded data is simple. Just call `decode` with a string or some other
58+
container holding character data. This will return a `data` object that you can
59+
operate on:
5960

6061
```c++
6162
bencode::data data = bencode::decode("i42e");
6263
auto value = std::get<bencode::integer>(data);
6364
```
6465

65-
`decode` also has an overload that takes an iterator pair:
66+
`decode` also has overloads that takes an iterator pair or a pointer and length:
6667

6768
```c++
68-
auto data = bencode::decode(foo.begin(), foo.end());
69+
auto data1 = bencode::decode(foo.begin(), foo.end());
70+
auto data2 = bencode::decode(c_str, std::strlen(c_str));
6971
```
7072

7173
Finally, you can pass an `std::istream` directly to `decode`. By default, this
@@ -96,7 +98,9 @@ auto data2 = bencode::decode_some(input); // contains "foo"
9698
```
9799
98100
When calling `decode_some` with an iterator pair, it will update the value of
99-
the "begin" iterator in-place to point to where the parsing left off.
101+
the "begin" iterator in-place to point to where the parsing left off. Similary,
102+
calling `decode_some` with a pointer or pointer/length, it will update the
103+
pointer's value in-place.
100104
101105
#### Views
102106

include/bencode.hpp

+35-2
Original file line numberDiff line numberDiff line change
@@ -335,6 +335,17 @@ namespace bencode {
335335
template<typename T>
336336
inline constexpr bool is_view_v = is_view<T>::value;
337337

338+
template<typename, typename = std::void_t<>>
339+
struct is_iterable : std::false_type {};
340+
341+
template<typename T>
342+
struct is_iterable<T, std::void_t<
343+
decltype(std::begin(std::declval<T&>()), std::end(std::declval<T&>()))
344+
>> : std::true_type {};
345+
346+
template<typename T>
347+
inline constexpr bool is_iterable_v = is_iterable<T>::value;
348+
338349
template<typename Integer>
339350
inline void check_overflow(Integer value, Integer digit) {
340351
using limits = std::numeric_limits<Integer>;
@@ -607,9 +618,21 @@ namespace bencode {
607618
return detail::do_decode<Data>(b, end, true);
608619
}
609620

621+
template<typename Data, typename String,
622+
std::enable_if_t<detail::is_iterable_v<String> &&
623+
!std::is_array_v<String>, bool> = true>
624+
inline Data basic_decode(const String &s) {
625+
return basic_decode<Data>(std::begin(s), std::end(s));
626+
}
627+
628+
template<typename Data>
629+
inline Data basic_decode(const char *s) {
630+
return basic_decode<Data>(s, s + std::strlen(s));
631+
}
632+
610633
template<typename Data>
611-
inline Data basic_decode(const string_view &s) {
612-
return basic_decode<Data>(s.begin(), s.end());
634+
inline Data basic_decode(const char *s, std::size_t length) {
635+
return basic_decode<Data>(s, s + length);
613636
}
614637

615638
template<typename Data>
@@ -622,6 +645,16 @@ namespace bencode {
622645
return detail::do_decode<Data>(begin, end, false);
623646
}
624647

648+
template<typename Data>
649+
inline Data basic_decode_some(const char *&s) {
650+
return basic_decode_some<Data>(s, s + std::strlen(s));
651+
}
652+
653+
template<typename Data>
654+
inline Data basic_decode_some(const char *&s, std::size_t length) {
655+
return basic_decode_some<Data>(s, s + length);
656+
}
657+
625658
template<typename Data>
626659
inline Data basic_decode_some(std::istream &s, eof_behavior e = check_eof) {
627660
return detail::do_decode<Data>(s, e, false);

0 commit comments

Comments
 (0)