@@ -335,12 +335,16 @@ namespace bencode {
335
335
};
336
336
337
337
template <typename T>
338
- concept sequence = iterable<T> && !std::convertible_to<T, std::string_view>;
338
+ concept stringish = iterable<T> && requires(T &t) {
339
+ std::size (t);
340
+ requires std::same_as<std::iter_value_t <decltype (std::begin (t))>, char >;
341
+ };
339
342
340
343
template <typename T>
341
- concept mapping = sequence <T> && requires {
344
+ concept mapping = iterable <T> && requires {
342
345
typename T::key_type;
343
346
typename T::mapped_type;
347
+ requires stringish<typename T::key_type>;
344
348
};
345
349
346
350
template <std::integral Integer>
@@ -736,28 +740,42 @@ namespace bencode {
736
740
}
737
741
738
742
template <detail::output_iterator_ref Iter>
739
- inline void encode (Iter &&iter, integer value) {
743
+ inline void encode_to (Iter &&iter, integer value) {
740
744
*iter++ = u8' i' ;
741
745
detail::write_integer (iter, value);
742
746
*iter++ = u8' e' ;
743
747
}
744
748
749
+ template <detail::output_iterator_ref Iter, detail::stringish Str>
750
+ requires (!std::is_array_v<Str>)
751
+ inline void encode_to(Iter &&iter, const Str &value) {
752
+ detail::write_integer (iter, std::size (value));
753
+ *iter++ = u8' :' ;
754
+ std::copy (std::begin (value), std::end (value), iter);
755
+ }
756
+
745
757
template <detail::output_iterator_ref Iter>
746
- inline void encode (Iter &&iter, const string_view & value) {
747
- detail::write_integer (iter, value. size () );
758
+ inline void encode_to (Iter &&iter, const char * value, std:: size_t length ) {
759
+ detail::write_integer (iter, length );
748
760
*iter++ = u8' :' ;
749
- std::copy (value.begin (), value.end (), iter);
761
+ std::copy (value, value + length, iter);
762
+ }
763
+
764
+ template <detail::output_iterator_ref Iter, std::size_t N>
765
+ inline void encode_to (Iter &&iter, const char (&value)[N]) {
766
+ // Don't write the null terminator.
767
+ encode_to (std::forward<Iter>(iter), value, N - 1 );
750
768
}
751
769
752
- template <detail::output_iterator_ref Iter, detail::sequence Seq>
753
- void encode (Iter &&iter, const Seq &value) {
770
+ template <detail::output_iterator_ref Iter, detail::iterable Seq>
771
+ void encode_to (Iter &&iter, const Seq &value) {
754
772
detail::list_encoder e (iter);
755
773
for (auto &&i : value)
756
774
e.add (i);
757
775
}
758
776
759
777
template <detail::output_iterator_ref Iter, detail::mapping Map>
760
- void encode (Iter &&iter, const Map &value) {
778
+ void encode_to (Iter &&iter, const Map &value) {
761
779
detail::dict_encoder e (iter);
762
780
for (auto &&i : value)
763
781
e.add (i.first , i.second );
@@ -771,7 +789,7 @@ namespace bencode {
771
789
772
790
template <typename T>
773
791
void operator ()(T &&operand) const {
774
- encode (iter, std::forward<T>(operand));
792
+ encode_to (iter, std::forward<T>(operand));
775
793
}
776
794
private:
777
795
Iter &iter;
@@ -781,38 +799,38 @@ namespace bencode {
781
799
template <detail::output_iterator_ref Iter,
782
800
template <typename ...> typename Variant, typename I, typename S,
783
801
template <typename ...> typename L, template <typename ...> typename D>
784
- void encode (Iter &&iter, const basic_data<Variant, I, S, L, D> &value) {
802
+ void encode_to (Iter &&iter, const basic_data<Variant, I, S, L, D> &value) {
785
803
variant_traits<Variant>::visit (detail::encode_visitor (iter), value);
786
804
}
787
805
788
806
namespace detail {
789
807
template <detail::output_iterator_ref Iter>
790
808
template <typename T>
791
809
inline list_encoder<Iter> & list_encoder<Iter>::add(T &&value) {
792
- encode (iter, std::forward<T>(value));
810
+ encode_to (iter, std::forward<T>(value));
793
811
return *this ;
794
812
}
795
813
796
814
template <detail::output_iterator_ref Iter>
797
815
template <typename T>
798
816
inline dict_encoder<Iter> &
799
817
dict_encoder<Iter>::add(const string_view &key, T &&value) {
800
- encode (iter, key);
801
- encode (iter, std::forward<T>(value));
818
+ encode_to (iter, key);
819
+ encode_to (iter, std::forward<T>(value));
802
820
return *this ;
803
821
}
804
822
}
805
823
806
- template <typename T>
807
- std::string encode (T &&t) {
824
+ template <typename ... T>
825
+ std::string encode (T &&... t) {
808
826
std::stringstream ss;
809
- encode (std::ostreambuf_iterator (ss), std::forward<T>(t));
827
+ encode_to (std::ostreambuf_iterator (ss), std::forward<T>(t)... );
810
828
return ss.str ();
811
829
}
812
830
813
- template <typename T>
814
- void encode (std::ostream &os, T &&t) {
815
- encode (std::ostreambuf_iterator (os), std::forward<T>(t));
831
+ template <typename ... T>
832
+ void encode_to (std::ostream &os, T &&... t) {
833
+ encode_to (std::ostreambuf_iterator (os), std::forward<T>(t)... );
816
834
}
817
835
818
836
}
0 commit comments