Yoba Perl
hash.cpp
1 #include "yobaperl/hash.hpp"
2 #include "yobaperl/perl.hpp"
3 #include "yobaperl/scalar.hpp"
4 
5 namespace yoba {
6 
7 
8 
9 Hash::Hash(Perl & perl, HV * hv, bool increase_refcount)
10  : Variable(perl, reinterpret_cast<SV *>(hv), increase_refcount)
11 {
12 }
13 
14 
15 
16 I32 Hash::getSize() const
17 {
18  return HvUSEDKEYS(getHV());
19 }
20 
21 bool Hash::isEmpty() const
22 {
23  return getSize() == 0;
24 }
25 
26 bool Hash::isExists(const std::string & key) const
27 {
28  return hv_exists(getHV(), key.c_str(), key.length());
29 }
30 
31 Scalar Hash::get(const std::string & key)
32 {
33  if(isExists(key))
34  {
35  SV ** elements = hv_fetch(getHV(), key.c_str(), key.length(), /* lval */NULL);
36  YOBAPERL_ASSERT(elements);
37 
38  SV * element = *elements;
39  YOBAPERL_ASSERT(element);
40 
41  return Scalar(_perl, element, /* ref++ */true);
42  }
43  else
44  {
45  std::string error = "Hash entry not exists (key: " + key + ")";
46 
47  if(_perl.isExceptionsEnabled())
48  throw PerlException(_perl.getId(), error);
49 
50  if(_perl.isWarningsEnabled())
51  warn("%s", error.c_str());
52 
53  return _perl.newScalar();
54  }
55 }
56 
57 Hash & Hash::insert(const std::string & key, const Scalar & value)
58 {
59  _store(key, value.getSV());
60  return *this;
61 }
62 
63 Hash & Hash::insert(const std::pair<std::string, Scalar> & pair)
64 {
65  insert(pair.first, pair.second);
66  return *this;
67 }
68 
69 Hash & Hash::insert(const std::unordered_map<std::string, Scalar> & hashmap)
70 {
71  for(auto it = hashmap.cbegin(); it != hashmap.cend(); it++)
72  insert(it->first, it->second);
73  return *this;
74 }
75 
76 Scalar Hash::remove(const std::string & key)
77 {
78  if(isExists(key))
79  {
80  SV * result = hv_delete(getHV(), key.c_str(), static_cast<U32>(key.length()), /* flags */0);
81  YOBAPERL_ASSERT(result);
82  return Scalar(_perl, result, /* ref++ */true);
83  }
84  else
85  {
86  std::string error = "Hash entry not exists (key: " + key + ")";
87 
88  if(_perl.isExceptionsEnabled())
89  throw PerlException(_perl.getId(), error);
90 
91  if(_perl.isWarningsEnabled())
92  warn("%s", error.c_str());
93 
94  return _perl.newScalar();
95  }
96 }
97 
99 {
100  hv_clear(getHV());
101  return *this;
102 }
103 
104 std::string Hash::toString() const
105 {
106  std::stringstream ss;
107 
108  ss << "(";
109 
110  auto it = begin();
111  while(it)
112  {
113  ss << *it;
114  if(++it)
115  ss << ", ";
116  }
117 
118  ss << ")";
119 
120  return ss.str();
121 }
122 
123 std::unordered_map<std::string, Scalar> Hash::toMap() const
124 {
125  std::unordered_map<std::string, Scalar> result;
126  result.reserve(getSize());
127 
128  auto it = begin();
129  while(it)
130  {
131  result.insert((*it).toPair());
132  ++it;
133  }
134 
135  return result;
136 }
137 
139 {
140  return Scalar(_perl, newRV_inc(MUTABLE_SV(getHV())), false);
141 }
142 
144 {
145  Hash result = _perl.newHash();
146 
147  for(HashEntry entry : *this)
148  result.insert(entry.getKey(), entry.getValue().makeCopy());
149 
150  return result;
151 }
152 
154 {
155  return Iterator(*this);
156 }
157 
158 Hash::Iterator Hash::end() const
159 {
160  return Iterator(*this, /* is_null */true);
161 }
162 
163 
164 
165 HV * Hash::getHV() const
166 {
167  return reinterpret_cast<HV *>(getSV());
168 }
169 
170 
171 
172 Hash::operator bool() const
173 {
174  return !isEmpty();
175 }
176 
177 Scalar Hash::operator[] (const std::string & key) noexcept
178 {
179  return Scalar(_perl, *hv_fetch(getHV(), key.c_str(), key.length(), /* lval */NULL), /* ref++ */true);
180 }
181 
182 Scalar Hash::operator[] (const char * key) noexcept
183 {
184  return (*this)[std::string(key)];
185 }
186 
187 
188 
189 void Hash::_store(const std::string & key, SV * value)
190 {
191  SvREFCNT_inc_NN(value);
192  hv_store(getHV(), key.c_str(), key.length(), value, /* hash */0);
193 }
194 
195 I32 Hash::_interInit() const
196 {
197  return hv_iterinit(getHV());
198 }
199 
200 HE * Hash::_interNext() const
201 {
202  return hv_iternext(getHV());
203 }
204 
205 
206 
207 } // namespace yoba
SV * getSV() const
Get raw scalar.
Definition: variable.cpp:63
Hash makeCopy() const
Copy each element to new hash.
Definition: hash.cpp:143
Key-value pair.
Definition: hash_entry.hpp:17
Scalar get(const std::string &key)
Fetch element.
Definition: hash.cpp:31
Definition: array.cpp:5
std::string toString() const
Convert to C++ string.
Definition: hash.cpp:104
Hash(Perl &perl, HV *hv, bool increase_refcount)
Constructor.
Definition: hash.cpp:9
Hash & clear()
Remove all elements.
Definition: hash.cpp:98
bool isExists(const std::string &key) const
Check element exists.
Definition: hash.cpp:26
Hash reference
Definition: hash.hpp:22
Hash iterator.
std::string getKey() const
Get key string.
Definition: hash_entry.cpp:28
Hash & insert(const std::string &key, const Scalar &value)
Add element.
Definition: hash.cpp:57
HashIterator Iterator
Type alias.
Definition: hash.hpp:27
bool isEmpty() const
Check if hash is empty.
Definition: hash.cpp:21
Scalar makeCopy() const
Create copy.
Definition: scalar.cpp:187
Scalar makeRef() const
Take reference.
Definition: hash.cpp:138
Base class for perl variables.
Definition: variable.hpp:20
Scalar newScalar()
Create anonymous empty scalar.
Definition: perl.cpp:51
Main class.
Definition: perl.hpp:32
HV * getHV() const
Raw hash.
Definition: hash.cpp:165
Scalar getValue() const
Get value.
Definition: hash_entry.cpp:33
int getId() const
Perl object id.
Definition: perl.cpp:298
Iterator begin() const
Forward iterator.
Definition: hash.cpp:153
Perl exception.
std::unordered_map< std::string, Scalar > toMap() const
Convert to C++ hash map.
Definition: hash.cpp:123
Scalar remove(const std::string &key)
Remove element and return.
Definition: hash.cpp:76
I32 getSize() const
Elements count.
Definition: hash.cpp:16
Scalar reference
Definition: scalar.hpp:24
Scalar operator[](const std::string &key) noexcept
Unsafe version of get()
Definition: hash.cpp:177
Hash newHash()
Create anonymous empty hash.
Definition: perl.cpp:83