mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			209 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			209 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|     This file is part of TON Blockchain Library.
 | |
| 
 | |
|     TON Blockchain Library is free software: you can redistribute it and/or modify
 | |
|     it under the terms of the GNU Lesser General Public License as published by
 | |
|     the Free Software Foundation, either version 2 of the License, or
 | |
|     (at your option) any later version.
 | |
| 
 | |
|     TON Blockchain Library is distributed in the hope that it will be useful,
 | |
|     but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|     GNU Lesser General Public License for more details.
 | |
| 
 | |
|     You should have received a copy of the GNU Lesser General Public License
 | |
|     along with TON Blockchain Library.  If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
|     Copyright 2017-2020 Telegram Systems LLP
 | |
| */
 | |
| #pragma once
 | |
| 
 | |
| #include "td/utils/common.h"
 | |
| #include "td/utils/format.h"
 | |
| #include "td/utils/logging.h"
 | |
| #include "td/utils/Slice.h"
 | |
| #include "td/utils/Status.h"
 | |
| 
 | |
| #include <cstring>
 | |
| #include <utility>
 | |
| 
 | |
| namespace td {
 | |
| 
 | |
| namespace detail {
 | |
| 
 | |
| template <class SliceT>
 | |
| class ParserImpl {
 | |
|  public:
 | |
|   explicit ParserImpl(SliceT data) : ptr_(data.begin()), end_(data.end()), status_() {
 | |
|   }
 | |
|   ParserImpl(ParserImpl &&other) : ptr_(other.ptr_), end_(other.end_), status_(std::move(other.status_)) {
 | |
|     other.clear();
 | |
|   }
 | |
|   ParserImpl &operator=(ParserImpl &&other) {
 | |
|     if (&other == this) {
 | |
|       return *this;
 | |
|     }
 | |
|     ptr_ = other.ptr_;
 | |
|     end_ = other.end_;
 | |
|     status_ = std::move(other.status_);
 | |
|     other.clear();
 | |
|     return *this;
 | |
|   }
 | |
|   ParserImpl(const ParserImpl &) = delete;
 | |
|   ParserImpl &operator=(const ParserImpl &) = delete;
 | |
|   ~ParserImpl() = default;
 | |
| 
 | |
|   bool empty() const {
 | |
|     return ptr_ == end_;
 | |
|   }
 | |
|   void clear() {
 | |
|     ptr_ = SliceT().begin();
 | |
|     end_ = ptr_;
 | |
|     status_ = Status::OK();
 | |
|   }
 | |
| 
 | |
|   SliceT read_till_nofail(char c) {
 | |
|     if (status_.is_error()) {
 | |
|       return SliceT();
 | |
|     }
 | |
|     auto till = static_cast<decltype(ptr_)>(std::memchr(ptr_, c, end_ - ptr_));
 | |
|     if (till == nullptr) {
 | |
|       till = end_;
 | |
|     }
 | |
|     SliceT result(ptr_, till);
 | |
|     ptr_ = till;
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   SliceT read_till_nofail(Slice str) {
 | |
|     if (status_.is_error()) {
 | |
|       return SliceT();
 | |
|     }
 | |
|     auto best_till = end_;
 | |
|     for (auto c : str) {
 | |
|       auto till = static_cast<decltype(ptr_)>(std::memchr(ptr_, c, end_ - ptr_));
 | |
|       if (till != nullptr && till < best_till) {
 | |
|         best_till = till;
 | |
|       }
 | |
|     }
 | |
|     SliceT result(ptr_, best_till);
 | |
|     ptr_ = best_till;
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   template <class F>
 | |
|   SliceT read_while(const F &f) {
 | |
|     auto save_ptr = ptr_;
 | |
|     while (ptr_ != end_ && f(*ptr_)) {
 | |
|       ptr_++;
 | |
|     }
 | |
|     return SliceT(save_ptr, ptr_);
 | |
|   }
 | |
|   SliceT read_all() {
 | |
|     auto save_ptr = ptr_;
 | |
|     ptr_ = end_;
 | |
|     return SliceT(save_ptr, ptr_);
 | |
|   }
 | |
| 
 | |
|   SliceT read_till(char c) {
 | |
|     if (status_.is_error()) {
 | |
|       return SliceT();
 | |
|     }
 | |
|     SliceT res = read_till_nofail(c);
 | |
|     if (ptr_ == end_ || ptr_[0] != c) {
 | |
|       status_ = Status::Error(PSLICE() << "Read till " << tag("char", c) << " failed");
 | |
|       return SliceT();
 | |
|     }
 | |
|     return res;
 | |
|   }
 | |
| 
 | |
|   char peek_char() {
 | |
|     if (ptr_ == end_) {
 | |
|       return 0;
 | |
|     }
 | |
|     return *ptr_;
 | |
|   }
 | |
| 
 | |
|   char *ptr() {
 | |
|     return ptr_;
 | |
|   }
 | |
| 
 | |
|   void skip_nofail(char c) {
 | |
|     if (ptr_ != end_ && ptr_[0] == c) {
 | |
|       ptr_++;
 | |
|     }
 | |
|   }
 | |
|   void skip(char c) {
 | |
|     if (status_.is_error()) {
 | |
|       return;
 | |
|     }
 | |
|     if (ptr_ == end_ || ptr_[0] != c) {
 | |
|       status_ = Status::Error(PSLICE() << "Skip " << tag("char", c) << " failed");
 | |
|       return;
 | |
|     }
 | |
|     ptr_++;
 | |
|   }
 | |
|   bool try_skip(char c) {
 | |
|     if (ptr_ != end_ && ptr_[0] == c) {
 | |
|       ptr_++;
 | |
|       return true;
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   void skip_till_not(Slice str) {
 | |
|     while (ptr_ != end_) {
 | |
|       if (std::memchr(str.data(), *ptr_, str.size()) == nullptr) {
 | |
|         break;
 | |
|       }
 | |
|       ptr_++;
 | |
|     }
 | |
|   }
 | |
|   void skip_whitespaces() {
 | |
|     skip_till_not(" \t\r\n");
 | |
|   }
 | |
|   SliceT read_word() {
 | |
|     skip_whitespaces();
 | |
|     return read_till_nofail(" \t\r\n");
 | |
|   }
 | |
| 
 | |
|   SliceT data() const {
 | |
|     return SliceT(ptr_, end_);
 | |
|   }
 | |
| 
 | |
|   Status &status() {
 | |
|     return status_;
 | |
|   }
 | |
| 
 | |
|   bool start_with(Slice prefix) const {
 | |
|     if (prefix.size() > static_cast<size_t>(end_ - ptr_)) {
 | |
|       return false;
 | |
|     }
 | |
|     return prefix == Slice(ptr_, prefix.size());
 | |
|   }
 | |
| 
 | |
|   bool skip_start_with(Slice prefix) {
 | |
|     if (start_with(prefix)) {
 | |
|       advance(prefix.size());
 | |
|       return true;
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   void advance(size_t diff) {
 | |
|     ptr_ += diff;
 | |
|     CHECK(ptr_ <= end_);
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   decltype(std::declval<SliceT>().begin()) ptr_;
 | |
|   decltype(std::declval<SliceT>().end()) end_;
 | |
|   Status status_;
 | |
| };
 | |
| 
 | |
| }  // namespace detail
 | |
| 
 | |
| using Parser = detail::ParserImpl<MutableSlice>;
 | |
| using ConstParser = detail::ParserImpl<Slice>;
 | |
| 
 | |
| }  // namespace td
 |