//
// MessagePack for C++ static resolution routine
//
// Copyright (C) 2008 FURUHASHI Sadayuki
//
//    Licensed under the Apache License, Version 2.0 (the "License");
//    you may not use this file except in compliance with the License.
//    You may obtain a copy of the License at
//
//        http://www.apache.org/licenses/LICENSE-2.0
//
//    Unless required by applicable law or agreed to in writing, software
//    distributed under the License is distributed on an "AS IS" BASIS,
//    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//    See the License for the specific language governing permissions and
//    limitations under the License.
//
#ifndef MSGPACK_TYPE_RAW_HPP__
#define MSGPACK_TYPE_RAW_HPP__

#include "msgpack/object.hpp"
#include <string.h>
#include <string>

namespace msgpack {

namespace type {

struct raw_ref {
	raw_ref() : ptr(NULL), size(0) {}
	raw_ref(const char* p, uint32_t s) : ptr(p), size(s) {}

	const char* ptr;
	uint32_t size;

	std::string str() { return std::string(ptr, size); }

	bool operator== (const raw_ref& x)
	{
		return size == x.size && memcmp(ptr, x.ptr, size) == 0;
	}

	bool operator!= (const raw_ref& x)
	{
		return !(*this != x);
	}

	bool operator< (const raw_ref& x)
	{
		if(size == x.size) { return memcmp(ptr, x.ptr, size) < 0; }
		else { return size < x.size; }
	}

	bool operator> (const raw_ref& x)
	{
		if(size == x.size) { return memcmp(ptr, x.ptr, size) > 0; }
		else { return size > x.size; }
	}
};

}  // namespace type


inline type::raw_ref& operator>> (object o, type::raw_ref& v)
{
	if(o.type != type::RAW) { throw type_error(); }
	v.ptr  = o.via.ref.ptr;
	v.size = o.via.ref.size;
	return v;
}


inline std::string& operator>> (object o, std::string& v)
{
	type::raw_ref r;
	o >> r;
	v.assign(r.ptr, r.size);
	return v;
}


template <typename Stream>
inline packer<Stream>& operator<< (packer<Stream>& o, const type::raw_ref& v)
{
	o.pack_raw(v.size);
	o.pack_raw_body(v.ptr, v.size);
	return o;
}


template <typename Stream>
inline packer<Stream>& operator<< (packer<Stream>& o, const std::string& v)
{
	o.pack_raw(v.size());
	o.pack_raw_body(v.data(), v.size());
	return o;
}


}  // namespace msgpack

#endif /* msgpack/type/raw.hpp */