read_until.hpp 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149
  1. //
  2. // impl/read_until.hpp
  3. // ~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2015 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. //
  10. #ifndef BOOST_ASIO_IMPL_READ_UNTIL_HPP
  11. #define BOOST_ASIO_IMPL_READ_UNTIL_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include <algorithm>
  16. #include <string>
  17. #include <vector>
  18. #include <utility>
  19. #include <boost/asio/buffer.hpp>
  20. #include <boost/asio/buffers_iterator.hpp>
  21. #include <boost/asio/detail/bind_handler.hpp>
  22. #include <boost/asio/detail/handler_alloc_helpers.hpp>
  23. #include <boost/asio/detail/handler_cont_helpers.hpp>
  24. #include <boost/asio/detail/handler_invoke_helpers.hpp>
  25. #include <boost/asio/detail/handler_type_requirements.hpp>
  26. #include <boost/asio/detail/limits.hpp>
  27. #include <boost/asio/detail/throw_error.hpp>
  28. #include <boost/asio/detail/push_options.hpp>
  29. namespace boost {
  30. namespace asio {
  31. template <typename SyncReadStream, typename Allocator>
  32. inline std::size_t read_until(SyncReadStream& s,
  33. boost::asio::basic_streambuf<Allocator>& b, char delim)
  34. {
  35. boost::system::error_code ec;
  36. std::size_t bytes_transferred = read_until(s, b, delim, ec);
  37. boost::asio::detail::throw_error(ec, "read_until");
  38. return bytes_transferred;
  39. }
  40. template <typename SyncReadStream, typename Allocator>
  41. std::size_t read_until(SyncReadStream& s,
  42. boost::asio::basic_streambuf<Allocator>& b, char delim,
  43. boost::system::error_code& ec)
  44. {
  45. std::size_t search_position = 0;
  46. for (;;)
  47. {
  48. // Determine the range of the data to be searched.
  49. typedef typename boost::asio::basic_streambuf<
  50. Allocator>::const_buffers_type const_buffers_type;
  51. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  52. const_buffers_type buffers = b.data();
  53. iterator begin = iterator::begin(buffers);
  54. iterator start_pos = begin + search_position;
  55. iterator end = iterator::end(buffers);
  56. // Look for a match.
  57. iterator iter = std::find(start_pos, end, delim);
  58. if (iter != end)
  59. {
  60. // Found a match. We're done.
  61. ec = boost::system::error_code();
  62. return iter - begin + 1;
  63. }
  64. else
  65. {
  66. // No match. Next search can start with the new data.
  67. search_position = end - begin;
  68. }
  69. // Check if buffer is full.
  70. if (b.size() == b.max_size())
  71. {
  72. ec = error::not_found;
  73. return 0;
  74. }
  75. // Need more data.
  76. std::size_t bytes_to_read = read_size_helper(b, 65536);
  77. b.commit(s.read_some(b.prepare(bytes_to_read), ec));
  78. if (ec)
  79. return 0;
  80. }
  81. }
  82. template <typename SyncReadStream, typename Allocator>
  83. inline std::size_t read_until(SyncReadStream& s,
  84. boost::asio::basic_streambuf<Allocator>& b, const std::string& delim)
  85. {
  86. boost::system::error_code ec;
  87. std::size_t bytes_transferred = read_until(s, b, delim, ec);
  88. boost::asio::detail::throw_error(ec, "read_until");
  89. return bytes_transferred;
  90. }
  91. namespace detail
  92. {
  93. // Algorithm that finds a subsequence of equal values in a sequence. Returns
  94. // (iterator,true) if a full match was found, in which case the iterator
  95. // points to the beginning of the match. Returns (iterator,false) if a
  96. // partial match was found at the end of the first sequence, in which case
  97. // the iterator points to the beginning of the partial match. Returns
  98. // (last1,false) if no full or partial match was found.
  99. template <typename Iterator1, typename Iterator2>
  100. std::pair<Iterator1, bool> partial_search(
  101. Iterator1 first1, Iterator1 last1, Iterator2 first2, Iterator2 last2)
  102. {
  103. for (Iterator1 iter1 = first1; iter1 != last1; ++iter1)
  104. {
  105. Iterator1 test_iter1 = iter1;
  106. Iterator2 test_iter2 = first2;
  107. for (;; ++test_iter1, ++test_iter2)
  108. {
  109. if (test_iter2 == last2)
  110. return std::make_pair(iter1, true);
  111. if (test_iter1 == last1)
  112. {
  113. if (test_iter2 != first2)
  114. return std::make_pair(iter1, false);
  115. else
  116. break;
  117. }
  118. if (*test_iter1 != *test_iter2)
  119. break;
  120. }
  121. }
  122. return std::make_pair(last1, false);
  123. }
  124. } // namespace detail
  125. template <typename SyncReadStream, typename Allocator>
  126. std::size_t read_until(SyncReadStream& s,
  127. boost::asio::basic_streambuf<Allocator>& b, const std::string& delim,
  128. boost::system::error_code& ec)
  129. {
  130. std::size_t search_position = 0;
  131. for (;;)
  132. {
  133. // Determine the range of the data to be searched.
  134. typedef typename boost::asio::basic_streambuf<
  135. Allocator>::const_buffers_type const_buffers_type;
  136. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  137. const_buffers_type buffers = b.data();
  138. iterator begin = iterator::begin(buffers);
  139. iterator start_pos = begin + search_position;
  140. iterator end = iterator::end(buffers);
  141. // Look for a match.
  142. std::pair<iterator, bool> result = detail::partial_search(
  143. start_pos, end, delim.begin(), delim.end());
  144. if (result.first != end)
  145. {
  146. if (result.second)
  147. {
  148. // Full match. We're done.
  149. ec = boost::system::error_code();
  150. return result.first - begin + delim.length();
  151. }
  152. else
  153. {
  154. // Partial match. Next search needs to start from beginning of match.
  155. search_position = result.first - begin;
  156. }
  157. }
  158. else
  159. {
  160. // No match. Next search can start with the new data.
  161. search_position = end - begin;
  162. }
  163. // Check if buffer is full.
  164. if (b.size() == b.max_size())
  165. {
  166. ec = error::not_found;
  167. return 0;
  168. }
  169. // Need more data.
  170. std::size_t bytes_to_read = read_size_helper(b, 65536);
  171. b.commit(s.read_some(b.prepare(bytes_to_read), ec));
  172. if (ec)
  173. return 0;
  174. }
  175. }
  176. #if defined(BOOST_ASIO_HAS_BOOST_REGEX)
  177. template <typename SyncReadStream, typename Allocator>
  178. inline std::size_t read_until(SyncReadStream& s,
  179. boost::asio::basic_streambuf<Allocator>& b, const boost::regex& expr)
  180. {
  181. boost::system::error_code ec;
  182. std::size_t bytes_transferred = read_until(s, b, expr, ec);
  183. boost::asio::detail::throw_error(ec, "read_until");
  184. return bytes_transferred;
  185. }
  186. template <typename SyncReadStream, typename Allocator>
  187. std::size_t read_until(SyncReadStream& s,
  188. boost::asio::basic_streambuf<Allocator>& b, const boost::regex& expr,
  189. boost::system::error_code& ec)
  190. {
  191. std::size_t search_position = 0;
  192. for (;;)
  193. {
  194. // Determine the range of the data to be searched.
  195. typedef typename boost::asio::basic_streambuf<
  196. Allocator>::const_buffers_type const_buffers_type;
  197. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  198. const_buffers_type buffers = b.data();
  199. iterator begin = iterator::begin(buffers);
  200. iterator start_pos = begin + search_position;
  201. iterator end = iterator::end(buffers);
  202. // Look for a match.
  203. boost::match_results<iterator,
  204. typename std::vector<boost::sub_match<iterator> >::allocator_type>
  205. match_results;
  206. if (regex_search(start_pos, end, match_results, expr,
  207. boost::match_default | boost::match_partial))
  208. {
  209. if (match_results[0].matched)
  210. {
  211. // Full match. We're done.
  212. ec = boost::system::error_code();
  213. return match_results[0].second - begin;
  214. }
  215. else
  216. {
  217. // Partial match. Next search needs to start from beginning of match.
  218. search_position = match_results[0].first - begin;
  219. }
  220. }
  221. else
  222. {
  223. // No match. Next search can start with the new data.
  224. search_position = end - begin;
  225. }
  226. // Check if buffer is full.
  227. if (b.size() == b.max_size())
  228. {
  229. ec = error::not_found;
  230. return 0;
  231. }
  232. // Need more data.
  233. std::size_t bytes_to_read = read_size_helper(b, 65536);
  234. b.commit(s.read_some(b.prepare(bytes_to_read), ec));
  235. if (ec)
  236. return 0;
  237. }
  238. }
  239. #endif // defined(BOOST_ASIO_HAS_BOOST_REGEX)
  240. template <typename SyncReadStream, typename Allocator, typename MatchCondition>
  241. std::size_t read_until(SyncReadStream& s,
  242. boost::asio::basic_streambuf<Allocator>& b,
  243. MatchCondition match_condition, boost::system::error_code& ec,
  244. typename enable_if<is_match_condition<MatchCondition>::value>::type*)
  245. {
  246. std::size_t search_position = 0;
  247. for (;;)
  248. {
  249. // Determine the range of the data to be searched.
  250. typedef typename boost::asio::basic_streambuf<
  251. Allocator>::const_buffers_type const_buffers_type;
  252. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  253. const_buffers_type buffers = b.data();
  254. iterator begin = iterator::begin(buffers);
  255. iterator start_pos = begin + search_position;
  256. iterator end = iterator::end(buffers);
  257. // Look for a match.
  258. std::pair<iterator, bool> result = match_condition(start_pos, end);
  259. if (result.second)
  260. {
  261. // Full match. We're done.
  262. ec = boost::system::error_code();
  263. return result.first - begin;
  264. }
  265. else if (result.first != end)
  266. {
  267. // Partial match. Next search needs to start from beginning of match.
  268. search_position = result.first - begin;
  269. }
  270. else
  271. {
  272. // No match. Next search can start with the new data.
  273. search_position = end - begin;
  274. }
  275. // Check if buffer is full.
  276. if (b.size() == b.max_size())
  277. {
  278. ec = error::not_found;
  279. return 0;
  280. }
  281. // Need more data.
  282. std::size_t bytes_to_read = read_size_helper(b, 65536);
  283. b.commit(s.read_some(b.prepare(bytes_to_read), ec));
  284. if (ec)
  285. return 0;
  286. }
  287. }
  288. template <typename SyncReadStream, typename Allocator, typename MatchCondition>
  289. inline std::size_t read_until(SyncReadStream& s,
  290. boost::asio::basic_streambuf<Allocator>& b, MatchCondition match_condition,
  291. typename enable_if<is_match_condition<MatchCondition>::value>::type*)
  292. {
  293. boost::system::error_code ec;
  294. std::size_t bytes_transferred = read_until(s, b, match_condition, ec);
  295. boost::asio::detail::throw_error(ec, "read_until");
  296. return bytes_transferred;
  297. }
  298. namespace detail
  299. {
  300. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  301. class read_until_delim_op
  302. {
  303. public:
  304. read_until_delim_op(AsyncReadStream& stream,
  305. boost::asio::basic_streambuf<Allocator>& streambuf,
  306. char delim, ReadHandler& handler)
  307. : stream_(stream),
  308. streambuf_(streambuf),
  309. delim_(delim),
  310. start_(0),
  311. search_position_(0),
  312. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  313. {
  314. }
  315. #if defined(BOOST_ASIO_HAS_MOVE)
  316. read_until_delim_op(const read_until_delim_op& other)
  317. : stream_(other.stream_),
  318. streambuf_(other.streambuf_),
  319. delim_(other.delim_),
  320. start_(other.start_),
  321. search_position_(other.search_position_),
  322. handler_(other.handler_)
  323. {
  324. }
  325. read_until_delim_op(read_until_delim_op&& other)
  326. : stream_(other.stream_),
  327. streambuf_(other.streambuf_),
  328. delim_(other.delim_),
  329. start_(other.start_),
  330. search_position_(other.search_position_),
  331. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  332. {
  333. }
  334. #endif // defined(BOOST_ASIO_HAS_MOVE)
  335. void operator()(const boost::system::error_code& ec,
  336. std::size_t bytes_transferred, int start = 0)
  337. {
  338. const std::size_t not_found = (std::numeric_limits<std::size_t>::max)();
  339. std::size_t bytes_to_read;
  340. switch (start_ = start)
  341. {
  342. case 1:
  343. for (;;)
  344. {
  345. {
  346. // Determine the range of the data to be searched.
  347. typedef typename boost::asio::basic_streambuf<
  348. Allocator>::const_buffers_type const_buffers_type;
  349. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  350. const_buffers_type buffers = streambuf_.data();
  351. iterator begin = iterator::begin(buffers);
  352. iterator start_pos = begin + search_position_;
  353. iterator end = iterator::end(buffers);
  354. // Look for a match.
  355. iterator iter = std::find(start_pos, end, delim_);
  356. if (iter != end)
  357. {
  358. // Found a match. We're done.
  359. search_position_ = iter - begin + 1;
  360. bytes_to_read = 0;
  361. }
  362. // No match yet. Check if buffer is full.
  363. else if (streambuf_.size() == streambuf_.max_size())
  364. {
  365. search_position_ = not_found;
  366. bytes_to_read = 0;
  367. }
  368. // Need to read some more data.
  369. else
  370. {
  371. // Next search can start with the new data.
  372. search_position_ = end - begin;
  373. bytes_to_read = read_size_helper(streambuf_, 65536);
  374. }
  375. }
  376. // Check if we're done.
  377. if (!start && bytes_to_read == 0)
  378. break;
  379. // Start a new asynchronous read operation to obtain more data.
  380. stream_.async_read_some(streambuf_.prepare(bytes_to_read),
  381. BOOST_ASIO_MOVE_CAST(read_until_delim_op)(*this));
  382. return; default:
  383. streambuf_.commit(bytes_transferred);
  384. if (ec || bytes_transferred == 0)
  385. break;
  386. }
  387. const boost::system::error_code result_ec =
  388. (search_position_ == not_found)
  389. ? error::not_found : ec;
  390. const std::size_t result_n =
  391. (ec || search_position_ == not_found)
  392. ? 0 : search_position_;
  393. handler_(result_ec, result_n);
  394. }
  395. }
  396. //private:
  397. AsyncReadStream& stream_;
  398. boost::asio::basic_streambuf<Allocator>& streambuf_;
  399. char delim_;
  400. int start_;
  401. std::size_t search_position_;
  402. ReadHandler handler_;
  403. };
  404. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  405. inline void* asio_handler_allocate(std::size_t size,
  406. read_until_delim_op<AsyncReadStream,
  407. Allocator, ReadHandler>* this_handler)
  408. {
  409. return boost_asio_handler_alloc_helpers::allocate(
  410. size, this_handler->handler_);
  411. }
  412. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  413. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  414. read_until_delim_op<AsyncReadStream,
  415. Allocator, ReadHandler>* this_handler)
  416. {
  417. boost_asio_handler_alloc_helpers::deallocate(
  418. pointer, size, this_handler->handler_);
  419. }
  420. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  421. inline bool asio_handler_is_continuation(
  422. read_until_delim_op<AsyncReadStream,
  423. Allocator, ReadHandler>* this_handler)
  424. {
  425. return this_handler->start_ == 0 ? true
  426. : boost_asio_handler_cont_helpers::is_continuation(
  427. this_handler->handler_);
  428. }
  429. template <typename Function, typename AsyncReadStream, typename Allocator,
  430. typename ReadHandler>
  431. inline void asio_handler_invoke(Function& function,
  432. read_until_delim_op<AsyncReadStream,
  433. Allocator, ReadHandler>* this_handler)
  434. {
  435. boost_asio_handler_invoke_helpers::invoke(
  436. function, this_handler->handler_);
  437. }
  438. template <typename Function, typename AsyncReadStream, typename Allocator,
  439. typename ReadHandler>
  440. inline void asio_handler_invoke(const Function& function,
  441. read_until_delim_op<AsyncReadStream,
  442. Allocator, ReadHandler>* this_handler)
  443. {
  444. boost_asio_handler_invoke_helpers::invoke(
  445. function, this_handler->handler_);
  446. }
  447. } // namespace detail
  448. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  449. BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  450. void (boost::system::error_code, std::size_t))
  451. async_read_until(AsyncReadStream& s,
  452. boost::asio::basic_streambuf<Allocator>& b, char delim,
  453. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  454. {
  455. // If you get an error on the following line it means that your handler does
  456. // not meet the documented type requirements for a ReadHandler.
  457. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  458. detail::async_result_init<
  459. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  460. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  461. detail::read_until_delim_op<AsyncReadStream,
  462. Allocator, BOOST_ASIO_HANDLER_TYPE(ReadHandler,
  463. void (boost::system::error_code, std::size_t))>(
  464. s, b, delim, init.handler)(
  465. boost::system::error_code(), 0, 1);
  466. return init.result.get();
  467. }
  468. namespace detail
  469. {
  470. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  471. class read_until_delim_string_op
  472. {
  473. public:
  474. read_until_delim_string_op(AsyncReadStream& stream,
  475. boost::asio::basic_streambuf<Allocator>& streambuf,
  476. const std::string& delim, ReadHandler& handler)
  477. : stream_(stream),
  478. streambuf_(streambuf),
  479. delim_(delim),
  480. start_(0),
  481. search_position_(0),
  482. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  483. {
  484. }
  485. #if defined(BOOST_ASIO_HAS_MOVE)
  486. read_until_delim_string_op(const read_until_delim_string_op& other)
  487. : stream_(other.stream_),
  488. streambuf_(other.streambuf_),
  489. delim_(other.delim_),
  490. start_(other.start_),
  491. search_position_(other.search_position_),
  492. handler_(other.handler_)
  493. {
  494. }
  495. read_until_delim_string_op(read_until_delim_string_op&& other)
  496. : stream_(other.stream_),
  497. streambuf_(other.streambuf_),
  498. delim_(BOOST_ASIO_MOVE_CAST(std::string)(other.delim_)),
  499. start_(other.start_),
  500. search_position_(other.search_position_),
  501. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  502. {
  503. }
  504. #endif // defined(BOOST_ASIO_HAS_MOVE)
  505. void operator()(const boost::system::error_code& ec,
  506. std::size_t bytes_transferred, int start = 0)
  507. {
  508. const std::size_t not_found = (std::numeric_limits<std::size_t>::max)();
  509. std::size_t bytes_to_read;
  510. switch (start_ = start)
  511. {
  512. case 1:
  513. for (;;)
  514. {
  515. {
  516. // Determine the range of the data to be searched.
  517. typedef typename boost::asio::basic_streambuf<
  518. Allocator>::const_buffers_type const_buffers_type;
  519. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  520. const_buffers_type buffers = streambuf_.data();
  521. iterator begin = iterator::begin(buffers);
  522. iterator start_pos = begin + search_position_;
  523. iterator end = iterator::end(buffers);
  524. // Look for a match.
  525. std::pair<iterator, bool> result = detail::partial_search(
  526. start_pos, end, delim_.begin(), delim_.end());
  527. if (result.first != end && result.second)
  528. {
  529. // Full match. We're done.
  530. search_position_ = result.first - begin + delim_.length();
  531. bytes_to_read = 0;
  532. }
  533. // No match yet. Check if buffer is full.
  534. else if (streambuf_.size() == streambuf_.max_size())
  535. {
  536. search_position_ = not_found;
  537. bytes_to_read = 0;
  538. }
  539. // Need to read some more data.
  540. else
  541. {
  542. if (result.first != end)
  543. {
  544. // Partial match. Next search needs to start from beginning of
  545. // match.
  546. search_position_ = result.first - begin;
  547. }
  548. else
  549. {
  550. // Next search can start with the new data.
  551. search_position_ = end - begin;
  552. }
  553. bytes_to_read = read_size_helper(streambuf_, 65536);
  554. }
  555. }
  556. // Check if we're done.
  557. if (!start && bytes_to_read == 0)
  558. break;
  559. // Start a new asynchronous read operation to obtain more data.
  560. stream_.async_read_some(streambuf_.prepare(bytes_to_read),
  561. BOOST_ASIO_MOVE_CAST(read_until_delim_string_op)(*this));
  562. return; default:
  563. streambuf_.commit(bytes_transferred);
  564. if (ec || bytes_transferred == 0)
  565. break;
  566. }
  567. const boost::system::error_code result_ec =
  568. (search_position_ == not_found)
  569. ? error::not_found : ec;
  570. const std::size_t result_n =
  571. (ec || search_position_ == not_found)
  572. ? 0 : search_position_;
  573. handler_(result_ec, result_n);
  574. }
  575. }
  576. //private:
  577. AsyncReadStream& stream_;
  578. boost::asio::basic_streambuf<Allocator>& streambuf_;
  579. std::string delim_;
  580. int start_;
  581. std::size_t search_position_;
  582. ReadHandler handler_;
  583. };
  584. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  585. inline void* asio_handler_allocate(std::size_t size,
  586. read_until_delim_string_op<AsyncReadStream,
  587. Allocator, ReadHandler>* this_handler)
  588. {
  589. return boost_asio_handler_alloc_helpers::allocate(
  590. size, this_handler->handler_);
  591. }
  592. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  593. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  594. read_until_delim_string_op<AsyncReadStream,
  595. Allocator, ReadHandler>* this_handler)
  596. {
  597. boost_asio_handler_alloc_helpers::deallocate(
  598. pointer, size, this_handler->handler_);
  599. }
  600. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  601. inline bool asio_handler_is_continuation(
  602. read_until_delim_string_op<AsyncReadStream,
  603. Allocator, ReadHandler>* this_handler)
  604. {
  605. return this_handler->start_ == 0 ? true
  606. : boost_asio_handler_cont_helpers::is_continuation(
  607. this_handler->handler_);
  608. }
  609. template <typename Function, typename AsyncReadStream,
  610. typename Allocator, typename ReadHandler>
  611. inline void asio_handler_invoke(Function& function,
  612. read_until_delim_string_op<AsyncReadStream,
  613. Allocator, ReadHandler>* this_handler)
  614. {
  615. boost_asio_handler_invoke_helpers::invoke(
  616. function, this_handler->handler_);
  617. }
  618. template <typename Function, typename AsyncReadStream,
  619. typename Allocator, typename ReadHandler>
  620. inline void asio_handler_invoke(const Function& function,
  621. read_until_delim_string_op<AsyncReadStream,
  622. Allocator, ReadHandler>* this_handler)
  623. {
  624. boost_asio_handler_invoke_helpers::invoke(
  625. function, this_handler->handler_);
  626. }
  627. } // namespace detail
  628. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  629. BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  630. void (boost::system::error_code, std::size_t))
  631. async_read_until(AsyncReadStream& s,
  632. boost::asio::basic_streambuf<Allocator>& b, const std::string& delim,
  633. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  634. {
  635. // If you get an error on the following line it means that your handler does
  636. // not meet the documented type requirements for a ReadHandler.
  637. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  638. detail::async_result_init<
  639. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  640. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  641. detail::read_until_delim_string_op<AsyncReadStream,
  642. Allocator, BOOST_ASIO_HANDLER_TYPE(ReadHandler,
  643. void (boost::system::error_code, std::size_t))>(
  644. s, b, delim, init.handler)(
  645. boost::system::error_code(), 0, 1);
  646. return init.result.get();
  647. }
  648. #if defined(BOOST_ASIO_HAS_BOOST_REGEX)
  649. namespace detail
  650. {
  651. template <typename AsyncReadStream, typename Allocator,
  652. typename RegEx, typename ReadHandler>
  653. class read_until_expr_op
  654. {
  655. public:
  656. read_until_expr_op(AsyncReadStream& stream,
  657. boost::asio::basic_streambuf<Allocator>& streambuf,
  658. const boost::regex& expr, ReadHandler& handler)
  659. : stream_(stream),
  660. streambuf_(streambuf),
  661. expr_(expr),
  662. start_(0),
  663. search_position_(0),
  664. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  665. {
  666. }
  667. #if defined(BOOST_ASIO_HAS_MOVE)
  668. read_until_expr_op(const read_until_expr_op& other)
  669. : stream_(other.stream_),
  670. streambuf_(other.streambuf_),
  671. expr_(other.expr_),
  672. start_(other.start_),
  673. search_position_(other.search_position_),
  674. handler_(other.handler_)
  675. {
  676. }
  677. read_until_expr_op(read_until_expr_op&& other)
  678. : stream_(other.stream_),
  679. streambuf_(other.streambuf_),
  680. expr_(other.expr_),
  681. start_(other.start_),
  682. search_position_(other.search_position_),
  683. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  684. {
  685. }
  686. #endif // defined(BOOST_ASIO_HAS_MOVE)
  687. void operator()(const boost::system::error_code& ec,
  688. std::size_t bytes_transferred, int start = 0)
  689. {
  690. const std::size_t not_found = (std::numeric_limits<std::size_t>::max)();
  691. std::size_t bytes_to_read;
  692. switch (start_ = start)
  693. {
  694. case 1:
  695. for (;;)
  696. {
  697. {
  698. // Determine the range of the data to be searched.
  699. typedef typename boost::asio::basic_streambuf<
  700. Allocator>::const_buffers_type const_buffers_type;
  701. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  702. const_buffers_type buffers = streambuf_.data();
  703. iterator begin = iterator::begin(buffers);
  704. iterator start_pos = begin + search_position_;
  705. iterator end = iterator::end(buffers);
  706. // Look for a match.
  707. boost::match_results<iterator,
  708. typename std::vector<boost::sub_match<iterator> >::allocator_type>
  709. match_results;
  710. bool match = regex_search(start_pos, end, match_results, expr_,
  711. boost::match_default | boost::match_partial);
  712. if (match && match_results[0].matched)
  713. {
  714. // Full match. We're done.
  715. search_position_ = match_results[0].second - begin;
  716. bytes_to_read = 0;
  717. }
  718. // No match yet. Check if buffer is full.
  719. else if (streambuf_.size() == streambuf_.max_size())
  720. {
  721. search_position_ = not_found;
  722. bytes_to_read = 0;
  723. }
  724. // Need to read some more data.
  725. else
  726. {
  727. if (match)
  728. {
  729. // Partial match. Next search needs to start from beginning of
  730. // match.
  731. search_position_ = match_results[0].first - begin;
  732. }
  733. else
  734. {
  735. // Next search can start with the new data.
  736. search_position_ = end - begin;
  737. }
  738. bytes_to_read = read_size_helper(streambuf_, 65536);
  739. }
  740. }
  741. // Check if we're done.
  742. if (!start && bytes_to_read == 0)
  743. break;
  744. // Start a new asynchronous read operation to obtain more data.
  745. stream_.async_read_some(streambuf_.prepare(bytes_to_read),
  746. BOOST_ASIO_MOVE_CAST(read_until_expr_op)(*this));
  747. return; default:
  748. streambuf_.commit(bytes_transferred);
  749. if (ec || bytes_transferred == 0)
  750. break;
  751. }
  752. const boost::system::error_code result_ec =
  753. (search_position_ == not_found)
  754. ? error::not_found : ec;
  755. const std::size_t result_n =
  756. (ec || search_position_ == not_found)
  757. ? 0 : search_position_;
  758. handler_(result_ec, result_n);
  759. }
  760. }
  761. //private:
  762. AsyncReadStream& stream_;
  763. boost::asio::basic_streambuf<Allocator>& streambuf_;
  764. RegEx expr_;
  765. int start_;
  766. std::size_t search_position_;
  767. ReadHandler handler_;
  768. };
  769. template <typename AsyncReadStream, typename Allocator,
  770. typename RegEx, typename ReadHandler>
  771. inline void* asio_handler_allocate(std::size_t size,
  772. read_until_expr_op<AsyncReadStream,
  773. Allocator, RegEx, ReadHandler>* this_handler)
  774. {
  775. return boost_asio_handler_alloc_helpers::allocate(
  776. size, this_handler->handler_);
  777. }
  778. template <typename AsyncReadStream, typename Allocator,
  779. typename RegEx, typename ReadHandler>
  780. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  781. read_until_expr_op<AsyncReadStream,
  782. Allocator, RegEx, ReadHandler>* this_handler)
  783. {
  784. boost_asio_handler_alloc_helpers::deallocate(
  785. pointer, size, this_handler->handler_);
  786. }
  787. template <typename AsyncReadStream, typename Allocator,
  788. typename RegEx, typename ReadHandler>
  789. inline bool asio_handler_is_continuation(
  790. read_until_expr_op<AsyncReadStream,
  791. Allocator, RegEx, ReadHandler>* this_handler)
  792. {
  793. return this_handler->start_ == 0 ? true
  794. : boost_asio_handler_cont_helpers::is_continuation(
  795. this_handler->handler_);
  796. }
  797. template <typename Function, typename AsyncReadStream, typename Allocator,
  798. typename RegEx, typename ReadHandler>
  799. inline void asio_handler_invoke(Function& function,
  800. read_until_expr_op<AsyncReadStream,
  801. Allocator, RegEx, ReadHandler>* this_handler)
  802. {
  803. boost_asio_handler_invoke_helpers::invoke(
  804. function, this_handler->handler_);
  805. }
  806. template <typename Function, typename AsyncReadStream, typename Allocator,
  807. typename RegEx, typename ReadHandler>
  808. inline void asio_handler_invoke(const Function& function,
  809. read_until_expr_op<AsyncReadStream,
  810. Allocator, RegEx, ReadHandler>* this_handler)
  811. {
  812. boost_asio_handler_invoke_helpers::invoke(
  813. function, this_handler->handler_);
  814. }
  815. } // namespace detail
  816. template <typename AsyncReadStream, typename Allocator, typename ReadHandler>
  817. BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  818. void (boost::system::error_code, std::size_t))
  819. async_read_until(AsyncReadStream& s,
  820. boost::asio::basic_streambuf<Allocator>& b, const boost::regex& expr,
  821. BOOST_ASIO_MOVE_ARG(ReadHandler) handler)
  822. {
  823. // If you get an error on the following line it means that your handler does
  824. // not meet the documented type requirements for a ReadHandler.
  825. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  826. detail::async_result_init<
  827. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  828. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  829. detail::read_until_expr_op<AsyncReadStream, Allocator,
  830. boost::regex, BOOST_ASIO_HANDLER_TYPE(ReadHandler,
  831. void (boost::system::error_code, std::size_t))>(
  832. s, b, expr, init.handler)(
  833. boost::system::error_code(), 0, 1);
  834. return init.result.get();
  835. }
  836. #endif // defined(BOOST_ASIO_HAS_BOOST_REGEX)
  837. namespace detail
  838. {
  839. template <typename AsyncReadStream, typename Allocator,
  840. typename MatchCondition, typename ReadHandler>
  841. class read_until_match_op
  842. {
  843. public:
  844. read_until_match_op(AsyncReadStream& stream,
  845. boost::asio::basic_streambuf<Allocator>& streambuf,
  846. MatchCondition match_condition, ReadHandler& handler)
  847. : stream_(stream),
  848. streambuf_(streambuf),
  849. match_condition_(match_condition),
  850. start_(0),
  851. search_position_(0),
  852. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(handler))
  853. {
  854. }
  855. #if defined(BOOST_ASIO_HAS_MOVE)
  856. read_until_match_op(const read_until_match_op& other)
  857. : stream_(other.stream_),
  858. streambuf_(other.streambuf_),
  859. match_condition_(other.match_condition_),
  860. start_(other.start_),
  861. search_position_(other.search_position_),
  862. handler_(other.handler_)
  863. {
  864. }
  865. read_until_match_op(read_until_match_op&& other)
  866. : stream_(other.stream_),
  867. streambuf_(other.streambuf_),
  868. match_condition_(other.match_condition_),
  869. start_(other.start_),
  870. search_position_(other.search_position_),
  871. handler_(BOOST_ASIO_MOVE_CAST(ReadHandler)(other.handler_))
  872. {
  873. }
  874. #endif // defined(BOOST_ASIO_HAS_MOVE)
  875. void operator()(const boost::system::error_code& ec,
  876. std::size_t bytes_transferred, int start = 0)
  877. {
  878. const std::size_t not_found = (std::numeric_limits<std::size_t>::max)();
  879. std::size_t bytes_to_read;
  880. switch (start_ = start)
  881. {
  882. case 1:
  883. for (;;)
  884. {
  885. {
  886. // Determine the range of the data to be searched.
  887. typedef typename boost::asio::basic_streambuf<
  888. Allocator>::const_buffers_type const_buffers_type;
  889. typedef boost::asio::buffers_iterator<const_buffers_type> iterator;
  890. const_buffers_type buffers = streambuf_.data();
  891. iterator begin = iterator::begin(buffers);
  892. iterator start_pos = begin + search_position_;
  893. iterator end = iterator::end(buffers);
  894. // Look for a match.
  895. std::pair<iterator, bool> result = match_condition_(start_pos, end);
  896. if (result.second)
  897. {
  898. // Full match. We're done.
  899. search_position_ = result.first - begin;
  900. bytes_to_read = 0;
  901. }
  902. // No match yet. Check if buffer is full.
  903. else if (streambuf_.size() == streambuf_.max_size())
  904. {
  905. search_position_ = not_found;
  906. bytes_to_read = 0;
  907. }
  908. // Need to read some more data.
  909. else
  910. {
  911. if (result.first != end)
  912. {
  913. // Partial match. Next search needs to start from beginning of
  914. // match.
  915. search_position_ = result.first - begin;
  916. }
  917. else
  918. {
  919. // Next search can start with the new data.
  920. search_position_ = end - begin;
  921. }
  922. bytes_to_read = read_size_helper(streambuf_, 65536);
  923. }
  924. }
  925. // Check if we're done.
  926. if (!start && bytes_to_read == 0)
  927. break;
  928. // Start a new asynchronous read operation to obtain more data.
  929. stream_.async_read_some(streambuf_.prepare(bytes_to_read),
  930. BOOST_ASIO_MOVE_CAST(read_until_match_op)(*this));
  931. return; default:
  932. streambuf_.commit(bytes_transferred);
  933. if (ec || bytes_transferred == 0)
  934. break;
  935. }
  936. const boost::system::error_code result_ec =
  937. (search_position_ == not_found)
  938. ? error::not_found : ec;
  939. const std::size_t result_n =
  940. (ec || search_position_ == not_found)
  941. ? 0 : search_position_;
  942. handler_(result_ec, result_n);
  943. }
  944. }
  945. //private:
  946. AsyncReadStream& stream_;
  947. boost::asio::basic_streambuf<Allocator>& streambuf_;
  948. MatchCondition match_condition_;
  949. int start_;
  950. std::size_t search_position_;
  951. ReadHandler handler_;
  952. };
  953. template <typename AsyncReadStream, typename Allocator,
  954. typename MatchCondition, typename ReadHandler>
  955. inline void* asio_handler_allocate(std::size_t size,
  956. read_until_match_op<AsyncReadStream,
  957. Allocator, MatchCondition, ReadHandler>* this_handler)
  958. {
  959. return boost_asio_handler_alloc_helpers::allocate(
  960. size, this_handler->handler_);
  961. }
  962. template <typename AsyncReadStream, typename Allocator,
  963. typename MatchCondition, typename ReadHandler>
  964. inline void asio_handler_deallocate(void* pointer, std::size_t size,
  965. read_until_match_op<AsyncReadStream,
  966. Allocator, MatchCondition, ReadHandler>* this_handler)
  967. {
  968. boost_asio_handler_alloc_helpers::deallocate(
  969. pointer, size, this_handler->handler_);
  970. }
  971. template <typename AsyncReadStream, typename Allocator,
  972. typename MatchCondition, typename ReadHandler>
  973. inline bool asio_handler_is_continuation(
  974. read_until_match_op<AsyncReadStream,
  975. Allocator, MatchCondition, ReadHandler>* this_handler)
  976. {
  977. return this_handler->start_ == 0 ? true
  978. : boost_asio_handler_cont_helpers::is_continuation(
  979. this_handler->handler_);
  980. }
  981. template <typename Function, typename AsyncReadStream, typename Allocator,
  982. typename MatchCondition, typename ReadHandler>
  983. inline void asio_handler_invoke(Function& function,
  984. read_until_match_op<AsyncReadStream,
  985. Allocator, MatchCondition, ReadHandler>* this_handler)
  986. {
  987. boost_asio_handler_invoke_helpers::invoke(
  988. function, this_handler->handler_);
  989. }
  990. template <typename Function, typename AsyncReadStream, typename Allocator,
  991. typename MatchCondition, typename ReadHandler>
  992. inline void asio_handler_invoke(const Function& function,
  993. read_until_match_op<AsyncReadStream,
  994. Allocator, MatchCondition, ReadHandler>* this_handler)
  995. {
  996. boost_asio_handler_invoke_helpers::invoke(
  997. function, this_handler->handler_);
  998. }
  999. } // namespace detail
  1000. template <typename AsyncReadStream, typename Allocator,
  1001. typename MatchCondition, typename ReadHandler>
  1002. BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
  1003. void (boost::system::error_code, std::size_t))
  1004. async_read_until(AsyncReadStream& s,
  1005. boost::asio::basic_streambuf<Allocator>& b,
  1006. MatchCondition match_condition, BOOST_ASIO_MOVE_ARG(ReadHandler) handler,
  1007. typename enable_if<is_match_condition<MatchCondition>::value>::type*)
  1008. {
  1009. // If you get an error on the following line it means that your handler does
  1010. // not meet the documented type requirements for a ReadHandler.
  1011. BOOST_ASIO_READ_HANDLER_CHECK(ReadHandler, handler) type_check;
  1012. detail::async_result_init<
  1013. ReadHandler, void (boost::system::error_code, std::size_t)> init(
  1014. BOOST_ASIO_MOVE_CAST(ReadHandler)(handler));
  1015. detail::read_until_match_op<AsyncReadStream, Allocator,
  1016. MatchCondition, BOOST_ASIO_HANDLER_TYPE(ReadHandler,
  1017. void (boost::system::error_code, std::size_t))>(
  1018. s, b, match_condition, init.handler)(
  1019. boost::system::error_code(), 0, 1);
  1020. return init.result.get();
  1021. }
  1022. } // namespace asio
  1023. } // namespace boost
  1024. #include <boost/asio/detail/pop_options.hpp>
  1025. #endif // BOOST_ASIO_IMPL_READ_UNTIL_HPP