1 //
2 // ip/basic_resolver_iterator.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 
11 #ifndef ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP
12 #define ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP
13 
14 
15 #include "asio/detail/config.hpp"
16 #include <cstddef>
17 #include <cstring>
18 #include <iterator>
19 #include <string>
20 #include <vector>
21 #include "asio/detail/shared_ptr.hpp"
22 #include "asio/detail/socket_ops.hpp"
23 #include "asio/detail/socket_types.hpp"
24 #include "asio/ip/basic_resolver_entry.hpp"
25 
26 
27 #include "asio/detail/push_options.hpp"
28 
29 namespace asio {
30 namespace ip {
31 
32 /// An iterator over the entries produced by a resolver.
33 /**
34  * The asio::ip::basic_resolver_iterator class template is used to define
35  * iterators over the results returned by a resolver.
36  *
37  * The iterator's value_type, obtained when the iterator is dereferenced, is:
38  * @code const basic_resolver_entry<InternetProtocol> @endcode
39  *
40  * @par Thread Safety
41  * @e Distinct @e objects: Safe.@n
42  * @e Shared @e objects: Unsafe.
43  */
44 template <typename InternetProtocol>
45 class basic_resolver_iterator
46 {
47 public:
48   /// The type used for the distance between two iterators.
49   typedef std::ptrdiff_t difference_type;
50 
51   /// The type of the value pointed to by the iterator.
52   typedef basic_resolver_entry<InternetProtocol> value_type;
53 
54   /// The type of the result of applying operator->() to the iterator.
55   typedef const basic_resolver_entry<InternetProtocol>* pointer;
56 
57   /// The type of the result of applying operator*() to the iterator.
58   typedef const basic_resolver_entry<InternetProtocol>& reference;
59 
60   /// The iterator category.
61   typedef std::forward_iterator_tag iterator_category;
62 
63   /// Default constructor creates an end iterator.
64   basic_resolver_iterator()
65     : index_(0)
66   {
67   }
68 
69   /// Create an iterator from an addrinfo list returned by getaddrinfo.
70   static basic_resolver_iterator create(
71       asio::detail::addrinfo_type* address_info,
72       const std::string& host_name, const std::string& service_name)
73   {
74     basic_resolver_iterator iter;
75     if (!address_info)
76       return iter;
77 
78     std::string actual_host_name = host_name;
79     if (address_info->ai_canonname)
80       actual_host_name = address_info->ai_canonname;
81 
82     iter.values_.reset(new values_type);
83 
84     while (address_info)
85     {
86       if (address_info->ai_family == ASIO_OS_DEF(AF_INET)
87           || address_info->ai_family == ASIO_OS_DEF(AF_INET6))
88       {
89         using namespace std; // For memcpy.
90         typename InternetProtocol::endpoint endpoint;
91         endpoint.resize(static_cast<std::size_t>(address_info->ai_addrlen));
92         memcpy(endpoint.data(), address_info->ai_addr,
93             address_info->ai_addrlen);
94         iter.values_->push_back(
95             basic_resolver_entry<InternetProtocol>(endpoint,
96               actual_host_name, service_name));
97       }
98       address_info = address_info->ai_next;
99     }
100 
101     return iter;
102   }
103 
104   /// Create an iterator from an endpoint, host name and service name.
105   static basic_resolver_iterator create(
106       const typename InternetProtocol::endpoint& endpoint,
107       const std::string& host_name, const std::string& service_name)
108   {
109     basic_resolver_iterator iter;
110     iter.values_.reset(new values_type);
111     iter.values_->push_back(
112         basic_resolver_entry<InternetProtocol>(
113           endpoint, host_name, service_name));
114     return iter;
115   }
116 
117   /// Create an iterator from a sequence of endpoints, host and service name.
118   template <typename EndpointIterator>
119   static basic_resolver_iterator create(
120       EndpointIterator begin, EndpointIterator end,
121       const std::string& host_name, const std::string& service_name)
122   {
123     basic_resolver_iterator iter;
124     if (begin != end)
125     {
126       iter.values_.reset(new values_type);
127       for (EndpointIterator ep_iter = begin; ep_iter != end; ++ep_iter)
128       {
129         iter.values_->push_back(
130             basic_resolver_entry<InternetProtocol>(
131               *ep_iter, host_name, service_name));
132       }
133     }
134     return iter;
135   }
136 
137 
138   /// Dereference an iterator.
139   const basic_resolver_entry<InternetProtocol>& operator*() const
140   {
141     return dereference();
142   }
143 
144   /// Dereference an iterator.
145   const basic_resolver_entry<InternetProtocol>* operator->() const
146   {
147     return &dereference();
148   }
149 
150   /// Increment operator (prefix).
151   basic_resolver_iterator& operator++()
152   {
153     increment();
154     return *this;
155   }
156 
157   /// Increment operator (postfix).
158   basic_resolver_iterator operator++(int)
159   {
160     basic_resolver_iterator tmp(*this);
161     ++*this;
162     return tmp;
163   }
164 
165   /// Test two iterators for equality.
166   friend bool operator==(const basic_resolver_iterator& a,
167       const basic_resolver_iterator& b)
168   {
169     return a.equal(b);
170   }
171 
172   /// Test two iterators for inequality.
173   friend bool operator!=(const basic_resolver_iterator& a,
174       const basic_resolver_iterator& b)
175   {
176     return !a.equal(b);
177   }
178 
179 private:
180   void increment()
181   {
182     if (++index_ == values_->size())
183     {
184       // Reset state to match a default constructed end iterator.
185       values_.reset();
186       index_ = 0;
187     }
188   }
189 
190   bool equal(const basic_resolver_iterator& other) const
191   {
192     if (!values_ && !other.values_)
193       return true;
194     if (values_ != other.values_)
195       return false;
196     return index_ == other.index_;
197   }
198 
199   const basic_resolver_entry<InternetProtocol>& dereference() const
200   {
201     return (*values_)[index_];
202   }
203 
204   typedef std::vector<basic_resolver_entry<InternetProtocol> > values_type;
205   asio::detail::shared_ptr<values_type> values_;
206   std::size_t index_;
207 };
208 
209 } // namespace ip
210 } // namespace asio
211 
212 #include "asio/detail/pop_options.hpp"
213 
214 #endif // ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP
215