mirror of
https://github.com/pocoproject/poco.git
synced 2025-01-08 11:02:17 +01:00
3b79a51722
This fixes compilation against libcxx 19, which removed the non-standard specializations for std::char_traits including the one for unsigned char, which would get instantiated for BLOB. See https://reviews.llvm.org/D138307 and https://reviews.llvm.org/D157058. Closes #4722.
355 lines
6.3 KiB
C++
355 lines
6.3 KiB
C++
//
|
|
// LOB.h
|
|
//
|
|
// Library: Data
|
|
// Package: DataCore
|
|
// Module: LOB
|
|
//
|
|
// Definition of the LOB class.
|
|
//
|
|
// Copyright (c) 2006, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#ifndef Data_LOB_INCLUDED
|
|
#define Data_LOB_INCLUDED
|
|
|
|
|
|
#include "Poco/Data/Data.h"
|
|
#include "Poco/SharedPtr.h"
|
|
#include "Poco/Dynamic/VarHolder.h"
|
|
#include "Poco/Exception.h"
|
|
#include <vector>
|
|
#include <algorithm>
|
|
|
|
|
|
namespace Poco {
|
|
namespace Data {
|
|
|
|
|
|
template <typename T>
|
|
class LOB
|
|
/// Representation of a Large OBject.
|
|
///
|
|
/// A LOB can hold arbitrary data.
|
|
/// The maximum size depends on the underlying database.
|
|
///
|
|
/// The LOBInputStream and LOBOutputStream classes provide
|
|
/// a convenient way to access the data in a LOB.
|
|
{
|
|
public:
|
|
using Iterator = typename std::vector<T>::const_iterator;
|
|
using ValueType = T;
|
|
using Container = std::vector<T>;
|
|
using ContentPtr = Poco::SharedPtr<Container>;
|
|
|
|
LOB(): _pContent(new std::vector<T>())
|
|
/// Creates an empty LOB.
|
|
{
|
|
}
|
|
|
|
LOB(const std::vector<T>& content):
|
|
_pContent(new std::vector<T>(content))
|
|
/// Creates the LOB, content is deep-copied.
|
|
{
|
|
}
|
|
|
|
LOB(const T* const pContent, std::size_t size):
|
|
_pContent(new std::vector<T>(pContent, pContent + size))
|
|
/// Creates the LOB by deep-copying pContent.
|
|
{
|
|
}
|
|
|
|
template<typename U = T>
|
|
LOB(const std::enable_if_t<std::is_same_v<T, U> && std::is_same_v<U, char>, std::string>& content):
|
|
_pContent(new std::vector<T>(content.begin(), content.end()))
|
|
/// Creates a LOB from a string.
|
|
{
|
|
}
|
|
|
|
template<typename U = T>
|
|
LOB(const std::enable_if_t<std::is_same_v<T, U> && std::is_same_v<U, wchar_t>, std::wstring>& content):
|
|
_pContent(new std::vector<T>(content.begin(), content.end()))
|
|
/// Creates a LOB from a string.
|
|
{
|
|
}
|
|
|
|
LOB(const LOB& other): _pContent(other._pContent)
|
|
/// Creates a LOB by copying another one.
|
|
{
|
|
}
|
|
|
|
LOB(LOB&& other) noexcept: _pContent(std::move(other._pContent))
|
|
{
|
|
}
|
|
|
|
~LOB()
|
|
/// Destroys the LOB.
|
|
{
|
|
}
|
|
|
|
LOB& operator = (const LOB& other)
|
|
/// Assignment operator.
|
|
{
|
|
LOB tmp(other);
|
|
swap(tmp);
|
|
return *this;
|
|
}
|
|
|
|
LOB& operator = (LOB&& other) noexcept
|
|
{
|
|
_pContent = std::move(other._pContent);
|
|
return *this;
|
|
}
|
|
|
|
bool operator == (const LOB& other) const
|
|
/// Compares for equality LOB by value.
|
|
{
|
|
return *_pContent == *other._pContent;
|
|
}
|
|
|
|
bool operator != (const LOB& other) const
|
|
/// Compares for inequality LOB by value.
|
|
{
|
|
return *_pContent != *other._pContent;
|
|
}
|
|
|
|
void swap(LOB& other) noexcept
|
|
/// Swaps the LOB with another one.
|
|
{
|
|
using std::swap;
|
|
swap(_pContent, other._pContent);
|
|
}
|
|
|
|
const std::vector<T>& content() const
|
|
/// Returns the content.
|
|
{
|
|
return *_pContent;
|
|
}
|
|
|
|
const T* rawContent() const
|
|
/// Returns the raw content.
|
|
///
|
|
/// If the LOB is empty, returns NULL.
|
|
{
|
|
if (_pContent->empty())
|
|
return 0;
|
|
else
|
|
return _pContent->data();
|
|
}
|
|
|
|
T* rawContent()
|
|
/// Returns the raw content.
|
|
///
|
|
/// If the LOB is empty, returns NULL.
|
|
{
|
|
if (_pContent->empty())
|
|
return 0;
|
|
else
|
|
return _pContent->data();
|
|
}
|
|
|
|
void assignVal(std::size_t count, const T& val)
|
|
/// Assigns raw content to internal storage.
|
|
{
|
|
ContentPtr tmp = new Container(count, val);
|
|
_pContent.swap(tmp);
|
|
}
|
|
|
|
void assignRaw(const T* ptr, std::size_t count)
|
|
/// Assigns raw content to internal storage.
|
|
{
|
|
poco_assert_dbg (ptr);
|
|
LOB tmp(ptr, count);
|
|
swap(tmp);
|
|
}
|
|
|
|
void appendRaw(const T* pChar, std::size_t count)
|
|
/// Assigns raw content to internal storage.
|
|
{
|
|
poco_assert_dbg (pChar);
|
|
_pContent->insert(_pContent->end(), pChar, pChar+count);
|
|
}
|
|
|
|
void reserve(std::size_t size)
|
|
/// Sets the capacity of the internal buffer.
|
|
{
|
|
_pContent->reserve(size);
|
|
}
|
|
|
|
void resize(std::size_t size)
|
|
/// Resizes the internal buffer.
|
|
{
|
|
_pContent->resize(size);
|
|
}
|
|
|
|
void clear(bool doCompact = false)
|
|
/// Clears the content of the blob.
|
|
/// If doCompact is true, trims the excess capacity.
|
|
{
|
|
_pContent->clear();
|
|
if (doCompact) compact();
|
|
}
|
|
|
|
void compact()
|
|
/// Trims the internal storage excess capacity.
|
|
{
|
|
std::vector<T>(*_pContent).swap(*_pContent);
|
|
}
|
|
|
|
Iterator begin() const
|
|
{
|
|
return _pContent->begin();
|
|
}
|
|
|
|
Iterator end() const
|
|
{
|
|
return _pContent->end();
|
|
}
|
|
|
|
std::size_t size() const
|
|
/// Returns the size of the LOB in bytes.
|
|
{
|
|
return static_cast<std::size_t>(_pContent->size());
|
|
}
|
|
|
|
std::size_t capacity() const
|
|
/// Returns the capacity of the underlying buffer.
|
|
{
|
|
return static_cast<std::size_t>(_pContent->capacity());
|
|
}
|
|
|
|
private:
|
|
ContentPtr _pContent;
|
|
};
|
|
|
|
|
|
using BLOB = LOB<unsigned char>;
|
|
using CLOB = LOB<char>;
|
|
using JSON = std::string;
|
|
|
|
//
|
|
// inlines
|
|
//
|
|
|
|
template <typename T>
|
|
inline void swap(LOB<T>& b1, LOB<T>& b2) noexcept
|
|
{
|
|
b1.swap(b2);
|
|
}
|
|
|
|
|
|
} } // namespace Poco::Data
|
|
|
|
|
|
namespace std
|
|
{
|
|
template<>
|
|
inline void swap<Poco::Data::BLOB>(Poco::Data::BLOB& b1, Poco::Data::BLOB& b2) noexcept
|
|
/// Full template specalization of std:::swap for BLOB
|
|
{
|
|
b1.swap(b2);
|
|
}
|
|
|
|
template<>
|
|
inline void swap<Poco::Data::CLOB>(Poco::Data::CLOB& c1, Poco::Data::CLOB& c2) noexcept
|
|
/// Full template specalization of std:::swap for CLOB
|
|
{
|
|
c1.swap(c2);
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// VarHolderImpl<LOB>
|
|
//
|
|
|
|
|
|
namespace Poco {
|
|
namespace Dynamic {
|
|
|
|
|
|
template <>
|
|
class VarHolderImpl<Poco::Data::BLOB>: public VarHolder
|
|
{
|
|
public:
|
|
VarHolderImpl(const Poco::Data::BLOB& val): _val(val)
|
|
{
|
|
}
|
|
|
|
~VarHolderImpl()
|
|
{
|
|
}
|
|
|
|
const std::type_info& type() const
|
|
{
|
|
return typeid(Poco::Data::BLOB);
|
|
}
|
|
|
|
void convert(std::string& val) const
|
|
{
|
|
val.assign(_val.begin(), _val.end());
|
|
}
|
|
|
|
VarHolder* clone(Placeholder<VarHolder>* pVarHolder = 0) const
|
|
{
|
|
return cloneHolder(pVarHolder, _val);
|
|
}
|
|
|
|
const Poco::Data::BLOB& value() const
|
|
{
|
|
return _val;
|
|
}
|
|
|
|
private:
|
|
VarHolderImpl();
|
|
Poco::Data::BLOB _val;
|
|
};
|
|
|
|
|
|
template <>
|
|
class VarHolderImpl<Poco::Data::CLOB>: public VarHolder
|
|
{
|
|
public:
|
|
VarHolderImpl(const Poco::Data::CLOB& val): _val(val)
|
|
{
|
|
}
|
|
|
|
~VarHolderImpl()
|
|
{
|
|
}
|
|
|
|
const std::type_info& type() const
|
|
{
|
|
return typeid(Poco::Data::CLOB);
|
|
}
|
|
|
|
void convert(std::string& val) const
|
|
{
|
|
val.assign(_val.begin(), _val.end());
|
|
}
|
|
|
|
VarHolder* clone(Placeholder<VarHolder>* pVarHolder = 0) const
|
|
{
|
|
return cloneHolder(pVarHolder, _val);
|
|
}
|
|
|
|
const Poco::Data::CLOB& value() const
|
|
{
|
|
return _val;
|
|
}
|
|
|
|
private:
|
|
VarHolderImpl();
|
|
Poco::Data::CLOB _val;
|
|
};
|
|
|
|
|
|
} } // namespace Poco::Dynamic
|
|
|
|
|
|
#endif // Data_LOB_INCLUDED
|