25 #ifndef __AMPS_FIELD_HPP__
26 #define __AMPS_FIELD_HPP__
28 #define _AMPS_SKIP_AMPSPLUSPLUS
30 #undef _AMPS_SKIP_AMPSPLUSPLUS
36 #define AMPS_UNSET_INDEX (size_t)-1
37 #define AMPS_TIMESTAMP_LEN 16
38 #define AMPS_TIMESTAMP_LEN_LONG 23
39 #define AMPS_MAX_BOOKMARK_LEN 42
41 #define AMPS_MAX_SUBID_LEN 1048576
43 #if (_MSC_VER >= 1400)
44 #define AMPS_snprintf(buf_, sz_, ...) _snprintf_s(buf_, sz_, _TRUNCATE, __VA_ARGS__)
45 #define AMPS_snprintf_amps_uint64_t(buf,sz,val) sprintf_s(buf,sz,"%I64u",val)
47 #define AMPS_snprintf_sizet(buf,sz,val) sprintf_s(buf,sz,"%lu",val)
49 #define AMPS_snprintf_sizet(buf,sz,val) sprintf_s(buf,sz,"%u",val)
52 #define AMPS_snprintf _snprintf
54 #define AMPS_snprintf_sizet(buf,sz,val) _sprintf(buf,sz,"%lu",val)
56 #define AMPS_snprintf_sizet(buf,sz,val) _sprintf(buf,sz,"%u",val)
60 #define AMPS_snprintf snprintf
61 #if defined(__x86_64__) || defined(__aarch64__)
62 #define AMPS_snprintf_amps_uint64_t(buf,sz,val) snprintf(buf,sz,"%lu", (unsigned long)val)
63 #define AMPS_snprintf_sizet(buf,sz,val) snprintf(buf,sz,"%lu",val)
65 #define AMPS_snprintf_amps_uint64_t(buf,sz,val) snprintf(buf,sz,"%llu",val)
66 #define AMPS_snprintf_sizet(buf,sz,val) snprintf(buf,sz,"%u",val)
92 Field() : _data(NULL), _len(0) {;}
93 Field(
const char* data_)
96 _len = ::strlen(data_);
98 Field(
const char* data_,
size_t len_)
114 Field(
const std::string& string_)
116 _data = string_.c_str();
117 _len = string_.length();
120 bool contains(
const char* searchString,
size_t len)
const
122 const char* dataEnd = _data + _len;
123 return std::search(_data, dataEnd, searchString, searchString +
len) != dataEnd;
136 operator std::string ()
const
138 return _len ? std::string(_data, _len) : std::string();
148 if ( _len == rhs_._len )
150 return ::memcmp(_data, rhs_._data, _len) == 0;
164 return (!_data && !rhs_);
166 return (_len == strlen(rhs_)) && (::strncmp(_data, rhs_, _len) == 0);
169 bool operator<(
const Field& rhs)
const;
178 if ( _len == rhs_._len )
180 return ::memcmp(_data, rhs_._data, _len) != 0;
192 return (_len != strlen(rhs_)) || (::memcmp(_data, rhs_, _len) != 0);
202 return rhs_.compare(0, rhs_.length(), _data, _len) != 0;
212 return rhs_.compare(0, rhs_.length(), _data, _len) == 0;
223 _data = (
char*)malloc(orig_._len);
224 ::memcpy(
static_cast<void*
>(
const_cast<char*
>(_data)),
225 orig_._data, orig_._len);
250 Field newField(_strdup(str_));
252 Field newField(strdup(str_));
286 void assign(
const char* ptr,
size_t len)
293 inline size_t hash_function()
const
296 const char* p_ = _data;
300 c = (
unsigned long) * p_;
309 size_t operator()(
const Field& f)
const
311 return f.hash_function();
314 bool operator()(
const Field& f1,
const Field& f2)
const
316 if (f1.len() < f2.len())
320 if (f1.len() > f2.len())
329 return ::memcmp(f1.data(), f2.data(), f2.len()) < 0;
334 static bool isTimestamp(
const Field& field_)
336 return (field_.len() >= AMPS_TIMESTAMP_LEN
337 && field_.len() <= AMPS_TIMESTAMP_LEN_LONG
338 && field_.data()[8] ==
'T');
341 static std::vector<Field> parseBookmarkList(
const Field& field_)
343 std::vector<Field> list;
344 const char* start = field_.data();
345 size_t remain = field_.len();
346 const char* comma = (
const char*)memchr((
const void*)start,
350 size_t len = (size_t)(comma - start);
353 #ifdef AMPS_USE_EMPLACE
354 list.emplace_back(Field(start,
len));
356 list.push_back(Field(start,
len));
360 remain = field_.len() - (size_t)(start - field_.data());
361 comma = (
const char*)memchr((
const void*)start,
366 #ifdef AMPS_USE_EMPLACE
367 list.emplace_back(Field(start, remain));
369 list.push_back(Field(start, remain));
376 static void parseBookmark(
const Field& field_,
377 amps_uint64_t& publisherId_,
378 amps_uint64_t& sequenceNumber_)
380 publisherId_ = sequenceNumber_ = (amps_uint64_t)0;
385 const char*
data = field_.data();
386 size_t len = field_.len();
388 if (isTimestamp(field_))
393 for ( ; i <
len; ++i)
395 if (!isdigit(
data[i]))
400 publisherId_ += (amps_uint64_t)(
data[i] -
'0');
405 publisherId_ = sequenceNumber_ = (amps_uint64_t)0;
408 for (i = i + 1; i <
len; ++i)
410 if (!isdigit(
data[i]))
414 sequenceNumber_ *= 10;
415 sequenceNumber_ += (amps_uint64_t)(
data[i] -
'0');
421 class BookmarkRange :
public Field
424 static bool isRange(
const Field& bookmark_)
426 return memchr(bookmark_.data(),
':', bookmark_.len()) != NULL;
430 : Field(), _start(), _end(), _open(AMPS_UNSET_INDEX)
431 , _capacity(AMPS_UNSET_INDEX)
436 BookmarkRange(
const Field& field_)
437 : Field(), _start(), _end(), _open(AMPS_UNSET_INDEX)
438 , _capacity(field_.len())
443 void set(
const Field& field_)
446 if (_data == field_.data() ||
operator==(field_))
455 bool foundSeparator =
false;
456 bool foundClose =
false;
457 for (
size_t i = 0; i < _len; ++i)
465 if (foundClose || _open != AMPS_UNSET_INDEX)
490 if (foundClose || _open == AMPS_UNSET_INDEX)
495 else if (foundSeparator)
499 _end.assign(_data + i, 0);
502 else if (!_start.data())
504 _start.assign(_data + i, 0);
511 if (foundSeparator || foundClose || _open == AMPS_UNSET_INDEX
517 foundSeparator =
true;
519 if (_start.len() == 0)
522 _start.assign(_start.data(), i - (
size_t)(_start.data() - _data));
530 if (foundClose || _open == AMPS_UNSET_INDEX || !_end.data())
541 _end.assign(_end.data(), i - (
size_t)(_end.data() - _data));
548 if (_end.data() && _end.len() == 0)
551 _end.assign(_end.data(), i - (
size_t)(_end.data() - _data));
554 else if (_start.data() && _start.len() == 0)
557 _start.assign(_start.data(), i - (
size_t)(_start.data() - _data));
569 if (_start.empty() || _end.empty())
587 _open = AMPS_UNSET_INDEX;
600 bool isStartInclusive()
const
602 return _data[_open] ==
'[';
607 void makeStartExclusive()
609 const_cast<char*
>(_data)[_open] =
'(';
614 bool isEndInclusive()
const
616 return _data[_len - 1] ==
']';
621 void makeEndExclusive()
623 const_cast<char*
>(_data)[_len - 1] =
')';
628 const Field& getStart()
const
633 const Field& getEnd()
const
640 void replaceStart(
const Field& start_,
bool makeExclusive_ =
false)
644 if (_capacity >= (start_.len() + _end.len() + 3))
646 char* data =
const_cast<char*
>(_data);
653 data[0] = _data[_open];
656 if ((
size_t)(_end.data() - _data - 2) < start_.len())
658 size_t newLen = start_.len() + _end.len() + 3;
662 for (
size_t dest = newLen - 1, src = _len - 1;
663 src > _start.len(); --src, --dest)
665 data[dest] = _data[src];
667 if (data[src] ==
':')
669 _end.assign(data + dest + 1, _end.len());
676 ::memcpy(
static_cast<void*
>(data + 1), start_.data(), start_.len());
677 _start.assign(data + 1, start_.len());
679 if ((
size_t)(_end.data() - _start.data()) > _start.len() + 1UL)
682 for (
size_t dest = _start.len() + 1, src = (
size_t)(_end.data() - data - 1);
683 src < _len; ++dest, ++src)
685 data[dest] = data[src];
686 if (data[src] ==
']' || data[src] ==
')')
688 _end.assign(data + _start.len() + 2, _end.len());
692 _len = _start.len() + _end.len() + 3;
699 _capacity = (4UL * AMPS_MAX_BOOKMARK_LEN + 6UL
700 >= start_.len() + _end.len() + 3UL)
701 ? (4UL * AMPS_MAX_BOOKMARK_LEN + 6UL)
702 : (start_.len() + _end.len() + 3UL);
703 char* data = (
char*)malloc(_capacity);
710 data[0] = _data[_open];
713 ::memcpy(
static_cast<void*
>(data + 1), start_.data(), start_.len());
714 _start.assign(data + 1, start_.len());
715 data[start_.len() + 1] =
':';
716 ::memcpy(
static_cast<void*
>(data + start_.len() + 2), _end.data(),
718 _end.assign(data + start_.len() + 2, _end.len());
719 size_t len = start_.len() + 3 + _end.len();
720 data[len - 1] = _data[_len - 1];
Core type and function declarations for the AMPS C client.
Field represents the value of a single field in a Message.
Definition: Field.hpp:87
bool operator==(const Field &rhs_) const
Comparison operator Returns ‘true’ if self and rhs are equivalent, ‘false’ otherwise.
Definition: Field.hpp:146
void clear()
Deletes the data associated with this Field, should only be used on Fields that were created as deepC...
Definition: Field.hpp:260
bool operator==(const char *rhs_) const
String comparison operator Returns ‘true’ if self and rhs are equivalent, ‘false’ otherwise.
Definition: Field.hpp:160
bool empty() const
Returns 'true' if empty, 'false' otherwise.
Definition: Field.hpp:128
void deepCopy(const Field &orig_)
Makes self a deep copy of the original field.
Definition: Field.hpp:218
size_t len() const
Returns the length of the data underlying this field.
Definition: Field.hpp:280
bool operator!=(const std::string &rhs_) const
String comparison operator Returns ‘true’ if self and rhs are not equivalent.
Definition: Field.hpp:200
bool operator!=(const Field &rhs_) const
Comparison operator Returns true if self and rhs are not equivalent.
Definition: Field.hpp:176
bool operator==(const std::string &rhs_) const
String comparison operator Returns ‘true’ if self and rhs are equivalent.
Definition: Field.hpp:210
static Field stringCopy(const char *str_)
Makes a copy of str_ in a new Field.
Definition: Field.hpp:247
const char * data() const
Returns the (non-null-terminated) data underlying this field.
Definition: Field.hpp:273
Field deepCopy() const
Makes a deep copy of self, returns it.
Definition: Field.hpp:237
bool operator!=(const char *rhs_) const
String comparison operator Returns true if self and rhs are not equivalent.
Definition: Field.hpp:190