| #include "ccnx-name.h" |
| #include <boost/lexical_cast.hpp> |
| #include <ctype.h> |
| #include <boost/algorithm/string/join.hpp> |
| |
| namespace Ccnx{ |
| CcnxCharbufPtr CcnxCharbuf::Null; |
| |
| void |
| CcnxCharbuf::init(ccn_charbuf *buf) |
| { |
| if (buf != NULL) |
| { |
| m_buf = ccn_charbuf_create(); |
| ccn_charbuf_reserve(m_buf, buf->length); |
| memcpy(m_buf->buf, buf->buf, buf->length); |
| m_buf->length = buf->length; |
| } |
| } |
| |
| CcnxCharbuf::CcnxCharbuf() |
| : m_buf(NULL) |
| { |
| m_buf = ccn_charbuf_create(); |
| } |
| |
| CcnxCharbuf::CcnxCharbuf(ccn_charbuf *buf) |
| : m_buf(NULL) |
| { |
| init(buf); |
| } |
| |
| CcnxCharbuf::CcnxCharbuf(const CcnxCharbuf &other) |
| : m_buf (NULL) |
| { |
| init(other.m_buf); |
| } |
| |
| CcnxCharbuf::~CcnxCharbuf() |
| { |
| ccn_charbuf_destroy(&m_buf); |
| } |
| |
| Name::Name() |
| { |
| } |
| |
| Name::Name(const string &name) |
| { |
| stringstream ss(name); |
| string compStr; |
| bool first = true; |
| while(getline(ss, compStr, '/')) |
| { |
| // discard the first empty comp before the first '/' |
| if (first) |
| { |
| first = false; |
| continue; |
| } |
| Bytes comp(compStr.begin(), compStr.end()); |
| m_comps.push_back(comp); |
| } |
| } |
| |
| Name::Name(const vector<Bytes> &comps) |
| { |
| m_comps = comps; |
| } |
| |
| Name::Name(const Name &other) |
| { |
| m_comps = other.m_comps; |
| } |
| |
| Name::Name(const unsigned char *data, const ccn_indexbuf *comps) |
| { |
| for (int i = 0; i < comps->n - 1; i++) |
| { |
| const unsigned char *compPtr; |
| size_t size; |
| ccn_name_comp_get(data, comps, i, &compPtr, &size); |
| Bytes comp; |
| readRaw(comp, compPtr, size); |
| m_comps.push_back(comp); |
| } |
| } |
| |
| Name::Name (const unsigned char *buf, const size_t length) |
| { |
| ccn_indexbuf *idx = ccn_indexbuf_create(); |
| const ccn_charbuf namebuf = { length, length, const_cast<unsigned char *> (buf) }; |
| ccn_name_split (&namebuf, idx); |
| |
| const unsigned char *compPtr = NULL; |
| size_t size = 0; |
| int i = 0; |
| while (ccn_name_comp_get(namebuf.buf, idx, i, &compPtr, &size) == 0) |
| { |
| Bytes comp; |
| readRaw (comp, compPtr, size); |
| m_comps.push_back(comp); |
| i++; |
| } |
| ccn_indexbuf_destroy(&idx); |
| } |
| |
| |
| Name & |
| Name::operator=(const Name &other) |
| { |
| m_comps = other.m_comps; |
| return *this; |
| } |
| bool |
| Name::operator==(const string &str) const |
| { |
| return this->toString() == str; |
| } |
| |
| bool |
| Name::operator!=(const string &str) const |
| { |
| return !(*this == str); |
| } |
| |
| Name |
| operator+(const Name &n1, const Name &n2) |
| { |
| vector<Bytes> comps = n1.m_comps; |
| copy(n2.m_comps.begin(), n2.m_comps.end(), back_inserter(comps)); |
| return Name(comps); |
| } |
| |
| string |
| Name::toString() const |
| { |
| stringstream ss(stringstream::out); |
| ss << *this; |
| return ss.str(); |
| } |
| |
| CcnxCharbufPtr |
| Name::toCcnxCharbuf() const |
| { |
| CcnxCharbufPtr ptr(new CcnxCharbuf()); |
| ccn_charbuf *cbuf = ptr->getBuf(); |
| ccn_name_init(cbuf); |
| int size = m_comps.size(); |
| for (int i = 0; i < size; i++) |
| { |
| ccn_name_append(cbuf, head(m_comps[i]), m_comps[i].size()); |
| } |
| return ptr; |
| } |
| |
| Name & |
| Name::appendComp(const Bytes &comp) |
| { |
| m_comps.push_back(comp); |
| return *this; |
| } |
| |
| Name & |
| Name::appendComp(const string &compStr) |
| { |
| Bytes comp(compStr.begin(), compStr.end()); |
| appendComp(comp); |
| return *this; |
| } |
| |
| Bytes |
| Name::getComp(int index) const |
| { |
| if (index >= m_comps.size()) |
| { |
| boost::throw_exception(NameException() << error_info_str("Index out of range: " + boost::lexical_cast<string>(index))); |
| } |
| return m_comps[index]; |
| } |
| |
| string |
| Name::getCompAsString(int index) const |
| { |
| Bytes comp = getComp(index); |
| stringstream ss(stringstream::out); |
| int size = comp.size(); |
| for (int i = 0; i < size; i++) |
| { |
| unsigned char ch = comp[i]; |
| if (isprint(ch)) |
| { |
| ss << (char) ch; |
| } |
| else |
| { |
| ss << "%" << hex << setfill('0') << setw(2) << ch; |
| } |
| } |
| |
| return ss.str(); |
| } |
| |
| Name |
| Name::getPartialName(int start, int n) const |
| { |
| int size = m_comps.size(); |
| if (start < 0 || start >= size || n > 0 && start + n > size) |
| { |
| stringstream ss(stringstream::out); |
| ss << "getPartialName() parameter out of range! "; |
| ss << "start = " << start; |
| ss << "n = " << n; |
| ss << "size = " << size; |
| boost::throw_exception(NameException() << error_info_str(ss.str())); |
| } |
| |
| vector<Bytes> comps; |
| int end; |
| if (n > 0) |
| { |
| end = start + n; |
| } |
| else |
| { |
| end = size; |
| } |
| |
| for (int i = start; i < end; i++) |
| { |
| comps.push_back(m_comps[i]); |
| } |
| |
| return Name(comps); |
| } |
| |
| ostream & |
| operator <<(ostream &os, const Name &name) |
| { |
| int size = name.size(); |
| vector<string> strComps; |
| for (int i = 0; i < size; i++) |
| { |
| strComps.push_back(name.getCompAsString(i)); |
| } |
| string joined = boost::algorithm::join(strComps, "/"); |
| os << "/" << joined; |
| return os; |
| } |
| |
| bool |
| operator ==(const Name &n1, const Name &n2) |
| { |
| stringstream ss1(stringstream::out); |
| stringstream ss2(stringstream::out); |
| ss1 << n1; |
| ss2 << n2; |
| return ss1.str() == ss2.str(); |
| } |
| |
| bool |
| operator !=(const Name &n1, const Name &n2) |
| { |
| return !(n1 == n2); |
| } |
| |
| bool |
| operator <(const Name &n1, const Name &n2) |
| { |
| stringstream ss1(stringstream::out); |
| stringstream ss2(stringstream::out); |
| ss1 << n1; |
| ss2 << n2; |
| return ss1.str() < ss2.str(); |
| } |
| |
| |
| } // Ccnx |