You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1153 lines
41 KiB
C++

// ***************************************************************** -*- C++ -*-
/*
* Copyright (C) 2004-2021 Exiv2 authors
* This program is part of the Exiv2 distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this f; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, 5th Floor, Boston, MA 02110-1301 USA.
*/
// *****************************************************************************
// included header files
#include "xmp_exiv2.hpp"
#include "types.hpp"
#include "error.hpp"
#include "value.hpp"
#include "properties.hpp"
// + standard includes
#include <iostream>
#include <algorithm>
#include <cassert>
#include <string>
// Adobe XMP Toolkit
#ifdef EXV_HAVE_XMP_TOOLKIT
# include <expat.h>
# define TXMP_STRING_TYPE std::string
# ifdef EXV_ADOBE_XMPSDK
# include <XMP.hpp>
# else
# include <XMPSDK.hpp>
# endif
# include <XMP.incl_cpp>
#endif // EXV_HAVE_XMP_TOOLKIT
#ifdef EXV_HAVE_XMP_TOOLKIT
// This anonymous namespace contains a class named XMLValidator, which uses
// libexpat to do a basic validation check on an XML document. This is to
// reduce the chance of hitting a bug in the (third-party) xmpsdk
// library. For example, it is easy to a trigger a stack overflow in xmpsdk
// with a deeply nested tree.
namespace {
using namespace Exiv2;
class XMLValidator {
size_t element_depth_ = 0;
size_t namespace_depth_ = 0;
// These fields are used to record whether an error occurred during
// parsing. Why do we need to store the error for later, rather
// than throw an exception immediately? Because expat is a C
// library, so it isn't designed to be able to handle exceptions
// thrown by the callback functions. Throwing exceptions during
// parsing is an example of one of the things that xmpsdk does
// wrong, leading to problems like https://github.com/Exiv2/exiv2/issues/1821.
bool haserror_ = false;
std::string errmsg_;
XML_Size errlinenum_ = 0;
XML_Size errcolnum_ = 0;
// Very deeply nested XML trees can cause a stack overflow in
// xmpsdk. They are also very unlikely to be valid XMP, so we
// error out if the depth exceeds this limit.
static const size_t max_recursion_limit_ = 1000;
const XML_Parser parser_;
public:
// Runs an XML parser on `buf`. Throws an exception if the XML is invalid.
static void check(const char* buf, size_t buflen) {
XMLValidator validator;
validator.check_internal(buf, buflen);
}
private:
// Private constructor, because this class is only constructed by
// the (static) check method.
XMLValidator() : parser_(XML_ParserCreateNS(0, '@')) {
if (!parser_) {
throw Error(kerXMPToolkitError, "Could not create expat parser");
}
}
~XMLValidator() {
XML_ParserFree(parser_);
}
void setError(const char* msg) {
const XML_Size errlinenum = XML_GetCurrentLineNumber(parser_);
const XML_Size errcolnum = XML_GetCurrentColumnNumber(parser_);
#ifndef SUPPRESS_WARNINGS
EXV_INFO << "Invalid XML at line " << errlinenum
<< ", column " << errcolnum
<< ": " << msg << "\n";
#endif
// If this is the first error, then save it.
if (!haserror_) {
haserror_ = true;
errmsg_ = msg;
errlinenum_ = errlinenum;
errcolnum_ = errcolnum;
}
}
void check_internal(const char* buf, size_t buflen) {
if (buflen > static_cast<size_t>(std::numeric_limits<int>::max())) {
throw Error(kerXMPToolkitError, "Buffer length is greater than INT_MAX");
}
XML_SetUserData(parser_, this);
XML_SetElementHandler(parser_, startElement_cb, endElement_cb);
XML_SetNamespaceDeclHandler(parser_, startNamespace_cb, endNamespace_cb);
XML_SetStartDoctypeDeclHandler(parser_, startDTD_cb);
const XML_Status result = XML_Parse(parser_, buf, static_cast<int>(buflen), true);
if (result == XML_STATUS_ERROR) {
setError(XML_ErrorString(XML_GetErrorCode(parser_)));
}
if (haserror_) {
throw XMP_Error(kXMPErr_BadXML, "Error in XMLValidator");
}
}
void startElement(const XML_Char*, const XML_Char**) noexcept {
if (element_depth_ > max_recursion_limit_) {
setError("Too deeply nested");
}
++element_depth_;
}
void endElement(const XML_Char*) noexcept {
if (element_depth_ > 0) {
--element_depth_;
} else {
setError("Negative depth");
}
}
void startNamespace(const XML_Char*, const XML_Char*) noexcept {
if (namespace_depth_ > max_recursion_limit_) {
setError("Too deeply nested");
}
++namespace_depth_;
}
void endNamespace(const XML_Char*) noexcept {
if (namespace_depth_ > 0) {
--namespace_depth_;
} else {
setError("Negative depth");
}
}
void startDTD(const XML_Char*, const XML_Char*, const XML_Char*, int) noexcept {
// DOCTYPE is used for XXE attacks.
setError("DOCTYPE not supported");
}
// This callback function is called by libexpat. It's a static wrapper
// around startElement().
static void XMLCALL startElement_cb(
void* userData, const XML_Char* name, const XML_Char* *attrs
) noexcept {
static_cast<XMLValidator*>(userData)->startElement(name, attrs);
}
// This callback function is called by libexpat. It's a static wrapper
// around endElement().
static void XMLCALL endElement_cb(void* userData, const XML_Char* name) noexcept {
static_cast<XMLValidator*>(userData)->endElement(name);
}
// This callback function is called by libexpat. It's a static wrapper
// around startNamespace().
static void XMLCALL startNamespace_cb(
void* userData, const XML_Char* prefix, const XML_Char* uri
) noexcept {
static_cast<XMLValidator*>(userData)->startNamespace(prefix, uri);
}
// This callback function is called by libexpat. It's a static wrapper
// around endNamespace().
static void XMLCALL endNamespace_cb(void* userData, const XML_Char* prefix) noexcept {
static_cast<XMLValidator*>(userData)->endNamespace(prefix);
}
static void XMLCALL startDTD_cb(
void *userData, const XML_Char *doctypeName, const XML_Char *sysid,
const XML_Char *pubid, int has_internal_subset
) noexcept {
static_cast<XMLValidator*>(userData)->startDTD(
doctypeName, sysid, pubid, has_internal_subset);
}
};
} // namespace
#endif // EXV_HAVE_XMP_TOOLKIT
// *****************************************************************************
// local declarations
namespace {
//! Unary predicate that matches an Xmpdatum by key
class FindXmpdatum {
public:
//! Constructor, initializes the object with key
explicit FindXmpdatum(const Exiv2::XmpKey& key) : key_(key.key())
{
}
/*!
@brief Returns true if prefix and property of the argument
Xmpdatum are equal to that of the object.
*/
bool operator()(const Exiv2::Xmpdatum& xmpdatum) const
{ return key_ == xmpdatum.key(); }
private:
std::string key_;
}; // class FindXmpdatum
#ifdef EXV_HAVE_XMP_TOOLKIT
//! Convert XMP Toolkit struct option bit to Value::XmpStruct
Exiv2::XmpValue::XmpStruct xmpStruct(const XMP_OptionBits& opt);
//! Convert Value::XmpStruct to XMP Toolkit array option bits
XMP_OptionBits xmpArrayOptionBits(Exiv2::XmpValue::XmpStruct xs);
//! Convert XMP Toolkit array option bits to array TypeId
Exiv2::TypeId arrayValueTypeId(const XMP_OptionBits& opt);
//! Convert XMP Toolkit array option bits to Value::XmpArrayType
Exiv2::XmpValue::XmpArrayType xmpArrayType(const XMP_OptionBits& opt);
//! Convert Value::XmpArrayType to XMP Toolkit array option bits
XMP_OptionBits xmpArrayOptionBits(Exiv2::XmpValue::XmpArrayType xat);
//! Convert XmpFormatFlags to XMP Toolkit format option bits
XMP_OptionBits xmpFormatOptionBits(Exiv2::XmpParser::XmpFormatFlags flags);
//! Print information about a parsed XMP node
void printNode(const std::string& schemaNs,
const std::string& propPath,
const std::string& propValue,
const XMP_OptionBits& opt);
//! Make an XMP key from a schema namespace and property path
Exiv2::XmpKey::UniquePtr makeXmpKey(const std::string& schemaNs,
const std::string& propPath);
#endif // EXV_HAVE_XMP_TOOLKIT
//! Helper class used to serialize critical sections
class AutoLock
{
public:
AutoLock(Exiv2::XmpParser::XmpLockFct xmpLockFct, void* pLockData)
: xmpLockFct_(xmpLockFct), pLockData_(pLockData)
{
if (xmpLockFct_) xmpLockFct_(pLockData_, true);
}
~AutoLock()
{
if (xmpLockFct_) xmpLockFct_(pLockData_, false);
}
private:
Exiv2::XmpParser::XmpLockFct xmpLockFct_;
void* pLockData_;
};
} // namespace
// *****************************************************************************
// class member definitions
namespace Exiv2 {
//! Internal Pimpl structure of class Xmpdatum.
struct Xmpdatum::Impl {
Impl(const XmpKey& key, const Value* pValue); //!< Constructor
Impl(const Impl& rhs); //!< Copy constructor
Impl& operator=(const Impl& rhs); //!< Assignment
// DATA
XmpKey::UniquePtr key_; //!< Key
Value::UniquePtr value_; //!< Value
};
Xmpdatum::Impl::Impl(const XmpKey& key, const Value* pValue)
: key_(key.clone())
{
if (pValue) value_ = pValue->clone();
}
Xmpdatum::Impl::Impl(const Impl& rhs)
{
if (rhs.key_.get() != nullptr)
key_ = rhs.key_->clone(); // deep copy
if (rhs.value_.get() != nullptr)
value_ = rhs.value_->clone(); // deep copy
}
Xmpdatum::Impl& Xmpdatum::Impl::operator=(const Impl& rhs)
{
if (this == &rhs) return *this;
key_.reset();
if (rhs.key_.get() != nullptr)
key_ = rhs.key_->clone(); // deep copy
value_.reset();
if (rhs.value_.get() != nullptr)
value_ = rhs.value_->clone(); // deep copy
return *this;
}
Xmpdatum::Xmpdatum(const XmpKey& key, const Value* pValue)
: p_(new Impl(key, pValue))
{
}
Xmpdatum::Xmpdatum(const Xmpdatum& rhs)
: Metadatum(rhs), p_(new Impl(*rhs.p_))
{
}
Xmpdatum& Xmpdatum::operator=(const Xmpdatum& rhs)
{
if (this == &rhs) return *this;
Metadatum::operator=(rhs);
*p_ = *rhs.p_;
return *this;
}
Xmpdatum::~Xmpdatum() = default;
std::string Xmpdatum::key() const
{
return p_->key_.get() == nullptr ? "" : p_->key_->key();
}
const char* Xmpdatum::familyName() const
{
return p_->key_.get() == nullptr ? "" : p_->key_->familyName();
}
std::string Xmpdatum::groupName() const
{
return p_->key_.get() == nullptr ? "" : p_->key_->groupName();
}
std::string Xmpdatum::tagName() const
{
return p_->key_.get() == nullptr ? "" : p_->key_->tagName();
}
std::string Xmpdatum::tagLabel() const
{
return p_->key_.get() == nullptr ? "" : p_->key_->tagLabel();
}
uint16_t Xmpdatum::tag() const
{
return p_->key_.get() == nullptr ? 0 : p_->key_->tag();
}
TypeId Xmpdatum::typeId() const
{
return p_->value_.get() == nullptr ? invalidTypeId : p_->value_->typeId();
}
const char* Xmpdatum::typeName() const
{
return TypeInfo::typeName(typeId());
}
long Xmpdatum::typeSize() const
{
return 0;
}
long Xmpdatum::count() const
{
return p_->value_.get() == nullptr ? 0 : p_->value_->count();
}
long Xmpdatum::size() const
{
return p_->value_.get() == nullptr ? 0 : p_->value_->size();
}
std::string Xmpdatum::toString() const
{
return p_->value_.get() == nullptr ? "" : p_->value_->toString();
}
std::string Xmpdatum::toString(long n) const
{
return p_->value_.get() == nullptr ? "" : p_->value_->toString(n);
}
long Xmpdatum::toLong(long n) const
{
return p_->value_.get() == nullptr ? -1 : static_cast<long>(p_->value_->toInt64(n));
}
float Xmpdatum::toFloat(long n) const
{
return p_->value_.get() == nullptr ? -1 : p_->value_->toFloat(n);
}
Rational Xmpdatum::toRational(long n) const
{
return p_->value_.get() == nullptr ? Rational(-1, 1) : p_->value_->toRational(n);
}
Value::UniquePtr Xmpdatum::getValue() const
{
return p_->value_.get() == nullptr ? nullptr : p_->value_->clone();
}
const Value& Xmpdatum::value() const
{
if (p_->value_.get() == nullptr)
throw Error(kerValueNotSet);
return *p_->value_;
}
long Xmpdatum::copy(byte* /*buf*/, ByteOrder /*byteOrder*/) const
{
throw Error(kerFunctionNotSupported, "Xmpdatum::copy");
return 0;
}
std::ostream& Xmpdatum::write(std::ostream& os, const ExifData*) const
{
return XmpProperties::printProperty(os, key(), value());
}
Xmpdatum& Xmpdatum::operator=(const std::string& value)
{
setValue(value);
return *this;
}
Xmpdatum& Xmpdatum::operator=(const Value& value)
{
setValue(&value);
return *this;
}
void Xmpdatum::setValue(const Value* pValue)
{
p_->value_.reset();
if (pValue) p_->value_ = pValue->clone();
}
int Xmpdatum::setValue(const std::string& value)
{
if (p_->value_.get() == nullptr) {
TypeId type = xmpText;
if (nullptr != p_->key_.get()) {
type = XmpProperties::propertyType(*p_->key_.get());
}
p_->value_ = Value::create(type);
}
return p_->value_->read(value);
}
Xmpdatum& XmpData::operator[](const std::string& key)
{
XmpKey xmpKey(key);
auto pos = findKey(xmpKey);
if (pos == end()) {
xmpMetadata_.push_back(Xmpdatum(xmpKey));
return xmpMetadata_.back();
}
return *pos;
}
int XmpData::add(const XmpKey& key, const Value* value)
{
return add(Xmpdatum(key, value));
}
int XmpData::add(const Xmpdatum& xmpDatum)
{
xmpMetadata_.push_back(xmpDatum);
return 0;
}
XmpData::const_iterator XmpData::findKey(const XmpKey& key) const
{
return std::find_if(xmpMetadata_.begin(), xmpMetadata_.end(),
FindXmpdatum(key));
}
XmpData::iterator XmpData::findKey(const XmpKey& key)
{
return std::find_if(xmpMetadata_.begin(), xmpMetadata_.end(),
FindXmpdatum(key));
}
void XmpData::clear()
{
xmpMetadata_.clear();
}
void XmpData::sortByKey()
{
std::sort(xmpMetadata_.begin(), xmpMetadata_.end(), cmpMetadataByKey);
}
XmpData::const_iterator XmpData::begin() const
{
return xmpMetadata_.begin();
}
XmpData::const_iterator XmpData::end() const
{
return xmpMetadata_.end();
}
bool XmpData::empty() const
{
return count() == 0;
}
long XmpData::count() const
{
return static_cast<long>(xmpMetadata_.size());
}
XmpData::iterator XmpData::begin()
{
return xmpMetadata_.begin();
}
XmpData::iterator XmpData::end()
{
return xmpMetadata_.end();
}
XmpData::iterator XmpData::erase(XmpData::iterator pos) {
return xmpMetadata_.erase(pos);
}
void XmpData::eraseFamily(XmpData::iterator& pos)
{
// https://github.com/Exiv2/exiv2/issues/521
// delete 'children' of XMP composites (XmpSeq and XmpBag)
// I build a StringVector of keys to remove
// Then I remove them with erase(....)
// erase() has nasty side effects on its argument
// The side effects are avoided by the two-step approach
// https://github.com/Exiv2/exiv2/issues/560
std::string key(pos->key());
std::vector<std::string> keys;
while ( pos != xmpMetadata_.end() ) {
if ( pos->key().find(key)==0 ) {
keys.push_back(pos->key());
pos++;
} else {
break;
}
}
// now erase the family!
for (auto&& k : keys) {
erase(findKey(Exiv2::XmpKey(k)));
}
}
bool XmpParser::initialized_ = false;
XmpParser::XmpLockFct XmpParser::xmpLockFct_ = nullptr;
void* XmpParser::pLockData_ = nullptr;
#ifdef EXV_HAVE_XMP_TOOLKIT
bool XmpParser::initialize(XmpParser::XmpLockFct xmpLockFct, void* pLockData)
{
if (!initialized_) {
xmpLockFct_ = xmpLockFct;
pLockData_ = pLockData;
initialized_ = SXMPMeta::Initialize();
#ifdef EXV_ADOBE_XMPSDK
SXMPMeta::RegisterNamespace("http://ns.adobe.com/lightroom/1.0/", "lr",NULL);
SXMPMeta::RegisterNamespace("http://rs.tdwg.org/dwc/index.htm", "dwc",NULL);
SXMPMeta::RegisterNamespace("http://purl.org/dc/terms/", "dcterms",NULL);
SXMPMeta::RegisterNamespace("http://www.digikam.org/ns/1.0/", "digiKam",NULL);
SXMPMeta::RegisterNamespace("http://www.digikam.org/ns/kipi/1.0/", "kipi",NULL);
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.0/", "MicrosoftPhoto",NULL);
SXMPMeta::RegisterNamespace("http://ns.acdsee.com/iptc/1.0/", "acdsee",NULL);
SXMPMeta::RegisterNamespace("http://iptc.org/std/Iptc4xmpExt/2008-02-29/", "iptcExt",NULL);
SXMPMeta::RegisterNamespace("http://ns.useplus.org/ldf/xmp/1.0/", "plus",NULL);
SXMPMeta::RegisterNamespace("http://ns.iview-multimedia.com/mediapro/1.0/", "mediapro",NULL);
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/expressionmedia/1.0/", "expressionmedia",NULL);
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.2/", "MP",NULL);
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.2/t/RegionInfo#", "MPRI",NULL);
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.2/t/Region#", "MPReg",NULL);
SXMPMeta::RegisterNamespace("http://ns.google.com/photos/1.0/panorama/", "GPano",NULL);
SXMPMeta::RegisterNamespace("http://www.metadataworkinggroup.com/schemas/regions/", "mwg-rs",NULL);
SXMPMeta::RegisterNamespace("http://www.metadataworkinggroup.com/schemas/keywords/", "mwg-kw",NULL);
SXMPMeta::RegisterNamespace("http://ns.adobe.com/xmp/sType/Area#", "stArea",NULL);
SXMPMeta::RegisterNamespace("http://cipa.jp/exif/1.0/", "exifEX",NULL);
SXMPMeta::RegisterNamespace("http://ns.adobe.com/camera-raw-saved-settings/1.0/", "crss",NULL);
SXMPMeta::RegisterNamespace("http://www.audio/", "audio",NULL);
SXMPMeta::RegisterNamespace("http://www.video/", "video",NULL);
#else
SXMPMeta::RegisterNamespace("http://ns.adobe.com/lightroom/1.0/", "lr");
SXMPMeta::RegisterNamespace("http://rs.tdwg.org/dwc/index.htm", "dwc");
SXMPMeta::RegisterNamespace("http://purl.org/dc/terms/", "dcterms");
SXMPMeta::RegisterNamespace("http://www.digikam.org/ns/1.0/", "digiKam");
SXMPMeta::RegisterNamespace("http://www.digikam.org/ns/kipi/1.0/", "kipi");
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.0/", "MicrosoftPhoto");
SXMPMeta::RegisterNamespace("http://ns.acdsee.com/iptc/1.0/", "acdsee");
SXMPMeta::RegisterNamespace("http://iptc.org/std/Iptc4xmpExt/2008-02-29/", "iptcExt");
SXMPMeta::RegisterNamespace("http://ns.useplus.org/ldf/xmp/1.0/", "plus");
SXMPMeta::RegisterNamespace("http://ns.iview-multimedia.com/mediapro/1.0/", "mediapro");
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/expressionmedia/1.0/", "expressionmedia");
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.2/", "MP");
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.2/t/RegionInfo#", "MPRI");
SXMPMeta::RegisterNamespace("http://ns.microsoft.com/photo/1.2/t/Region#", "MPReg");
SXMPMeta::RegisterNamespace("http://ns.google.com/photos/1.0/panorama/", "GPano");
SXMPMeta::RegisterNamespace("http://www.metadataworkinggroup.com/schemas/regions/", "mwg-rs");
SXMPMeta::RegisterNamespace("http://www.metadataworkinggroup.com/schemas/keywords/", "mwg-kw");
SXMPMeta::RegisterNamespace("http://ns.adobe.com/xmp/sType/Area#", "stArea");
SXMPMeta::RegisterNamespace("http://cipa.jp/exif/1.0/", "exifEX");
SXMPMeta::RegisterNamespace("http://ns.adobe.com/camera-raw-saved-settings/1.0/", "crss");
SXMPMeta::RegisterNamespace("http://www.audio/", "audio");
SXMPMeta::RegisterNamespace("http://www.video/", "video");
#endif
}
return initialized_;
}
#else
bool XmpParser::initialize(XmpParser::XmpLockFct, void* )
{
initialized_ = true;
return initialized_;
}
#endif
#ifdef EXV_HAVE_XMP_TOOLKIT
static XMP_Status nsDumper
( void* refCon
, XMP_StringPtr buffer
, XMP_StringLen bufferSize
) {
XMP_Status result = 0 ;
std::string out(buffer,bufferSize);
// remove blanks: http://stackoverflow.com/questions/83439/remove-spaces-from-stdstring-in-c
out.erase(std::remove_if(out.begin(), out.end(), isspace), out.end());
bool bURI = out.find("http://") != std::string::npos ;
bool bNS = out.find(':') != std::string::npos && !bURI;
// pop trailing ':' on a namespace
if ( bNS && !out.empty() ) {
std::size_t length = out.length();
if ( out[length-1] == ':' ) out = out.substr(0,length-1);
}
if ( bURI || bNS ) {
auto p = static_cast<std::map<std::string, std::string>*>(refCon);
std::map<std::string,std::string>& m = *p;
std::string b;
if ( bNS ) { // store the NS in dict[""]
m[b]=out;
} else if ( m.find(b) != m.end() ) { // store dict[uri] = dict[""]
m[m[b]]=out;
m.erase(b);
}
}
return result;
}
#endif
#ifdef EXV_HAVE_XMP_TOOLKIT
void XmpParser::registeredNamespaces(Exiv2::Dictionary& dict)
{
bool bInit = !initialized_;
try {
if (bInit) initialize();
SXMPMeta::DumpNamespaces(nsDumper,&dict);
if (bInit) terminate();
} catch (const XMP_Error& e) {
throw Error(kerXMPToolkitError, e.GetID(), e.GetErrMsg());
}
}
#else
void XmpParser::registeredNamespaces(Exiv2::Dictionary&){}
#endif
void XmpParser::terminate()
{
XmpProperties::unregisterNs();
if (initialized_) {
#ifdef EXV_HAVE_XMP_TOOLKIT
SXMPMeta::Terminate();
#endif
initialized_ = false;
}
}
#ifdef EXV_HAVE_XMP_TOOLKIT
void XmpParser::registerNs(const std::string& ns,
const std::string& prefix)
{
try {
initialize();
AutoLock autoLock(xmpLockFct_, pLockData_);
SXMPMeta::DeleteNamespace(ns.c_str());
#ifdef EXV_ADOBE_XMPSDK
SXMPMeta::RegisterNamespace(ns.c_str(), prefix.c_str(),NULL);
#else
SXMPMeta::RegisterNamespace(ns.c_str(), prefix.c_str());
#endif
}
catch (const XMP_Error& /* e */) {
// throw Error(kerXMPToolkitError, e.GetID(), e.GetErrMsg());
}
} // XmpParser::registerNs
#else
void XmpParser::registerNs(const std::string& /*ns*/,
const std::string& /*prefix*/)
{
initialize();
} // XmpParser::registerNs
#endif
void XmpParser::unregisterNs(const std::string& /*ns*/)
{
#ifdef EXV_HAVE_XMP_TOOLKIT
try {
// Throws XMP Toolkit error 8: Unimplemented method XMPMeta::DeleteNamespace
// SXMPMeta::DeleteNamespace(ns.c_str());
}
catch (const XMP_Error& e) {
throw Error(kerXMPToolkitError, e.GetID(), e.GetErrMsg());
}
#endif
} // XmpParser::unregisterNs
#ifdef EXV_HAVE_XMP_TOOLKIT
int XmpParser::decode( XmpData& xmpData,
const std::string& xmpPacket)
{ try {
xmpData.clear();
xmpData.setPacket(xmpPacket);
if (xmpPacket.empty()) return 0;
if (!initialize()) {
#ifndef SUPPRESS_WARNINGS
EXV_ERROR << "XMP toolkit initialization failed.\n";
#endif
return 2;
}
XMLValidator::check(xmpPacket.data(), xmpPacket.size());
SXMPMeta meta(xmpPacket.data(), static_cast<XMP_StringLen>(xmpPacket.size()));
SXMPIterator iter(meta);
std::string schemaNs, propPath, propValue;
XMP_OptionBits opt = 0;
while (iter.Next(&schemaNs, &propPath, &propValue, &opt)) {
printNode(schemaNs, propPath, propValue, opt);
if (XMP_PropIsAlias(opt)) {
throw Error(kerAliasesNotSupported, schemaNs, propPath, propValue);
continue;
}
if (XMP_NodeIsSchema(opt)) {
// Register unknown namespaces with Exiv2
// (Namespaces are automatically registered with the XMP Toolkit)
if (XmpProperties::prefix(schemaNs).empty()) {
std::string prefix;
bool ret = SXMPMeta::GetNamespacePrefix(schemaNs.c_str(), &prefix);
if (!ret) throw Error(kerSchemaNamespaceNotRegistered, schemaNs);
prefix = prefix.substr(0, prefix.size() - 1);
XmpProperties::registerNs(schemaNs, prefix);
}
continue;
}
auto key = makeXmpKey(schemaNs, propPath);
if (XMP_ArrayIsAltText(opt)) {
// Read Lang Alt property
auto val = std::make_unique<LangAltValue>();
XMP_Index count = meta.CountArrayItems(schemaNs.c_str(), propPath.c_str());
while (count-- > 0) {
// Get the text
bool haveNext = iter.Next(&schemaNs, &propPath, &propValue, &opt);
printNode(schemaNs, propPath, propValue, opt);
if ( !haveNext
|| !XMP_PropIsSimple(opt)
|| !XMP_PropHasLang(opt)) {
throw Error(kerDecodeLangAltPropertyFailed, propPath, opt);
}
const std::string text = propValue;
// Get the language qualifier
haveNext = iter.Next(&schemaNs, &propPath, &propValue, &opt);
printNode(schemaNs, propPath, propValue, opt);
if ( !haveNext
|| !XMP_PropIsSimple(opt)
|| !XMP_PropIsQualifier(opt)
|| propPath.substr(propPath.size() - 8, 8) != "xml:lang") {
throw Error(kerDecodeLangAltQualifierFailed, propPath, opt);
}
val->value_[propValue] = text;
}
xmpData.add(*key.get(), val.get());
continue;
}
if ( XMP_PropIsArray(opt)
&& !XMP_PropHasQualifiers(opt)
&& !XMP_ArrayIsAltText(opt)) {
// Check if all elements are simple
bool simpleArray = true;
SXMPIterator aIter(meta, schemaNs.c_str(), propPath.c_str());
std::string aSchemaNs, aPropPath, aPropValue;
XMP_OptionBits aOpt = 0;
while (aIter.Next(&aSchemaNs, &aPropPath, &aPropValue, &aOpt)) {
if (propPath == aPropPath) continue;
if ( !XMP_PropIsSimple(aOpt)
|| XMP_PropHasQualifiers(aOpt)
|| XMP_PropIsQualifier(aOpt)
|| XMP_NodeIsSchema(aOpt)
|| XMP_PropIsAlias(aOpt)) {
simpleArray = false;
break;
}
}
if (simpleArray) {
// Read the array into an XmpArrayValue
auto val = std::make_unique<XmpArrayValue>(arrayValueTypeId(opt));
XMP_Index count = meta.CountArrayItems(schemaNs.c_str(), propPath.c_str());
while (count-- > 0) {
iter.Next(&schemaNs, &propPath, &propValue, &opt);
printNode(schemaNs, propPath, propValue, opt);
val->read(propValue);
}
xmpData.add(*key.get(), val.get());
continue;
}
}
auto val = std::make_unique<XmpTextValue>();
if (XMP_PropIsStruct(opt) || XMP_PropIsArray(opt)) {
// Create a metadatum with only XMP options
val->setXmpArrayType(xmpArrayType(opt));
val->setXmpStruct(xmpStruct(opt));
xmpData.add(*key.get(), val.get());
continue;
}
if (XMP_PropIsSimple(opt) || XMP_PropIsQualifier(opt)) {
val->read(propValue);
xmpData.add(*key.get(), val.get());
continue;
}
// Don't let any node go by unnoticed
throw Error(kerUnhandledXmpNode, key->key(), opt);
} // iterate through all XMP nodes
return 0;
}
#ifndef SUPPRESS_WARNINGS
catch (const XMP_Error& e) {
EXV_ERROR << Error(kerXMPToolkitError, e.GetID(), e.GetErrMsg()) << "\n";
xmpData.clear();
return 3;
}
#else
catch (const XMP_Error&) {
xmpData.clear();
return 3;
}
#endif // SUPPRESS_WARNINGS
} // XmpParser::decode
#else
int XmpParser::decode( XmpData& xmpData,
const std::string& xmpPacket)
{
xmpData.clear();
if (!xmpPacket.empty()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "XMP toolkit support not compiled in.\n";
#endif
}
return 1;
} // XmpParser::decode
#endif // !EXV_HAVE_XMP_TOOLKIT
#ifdef EXV_HAVE_XMP_TOOLKIT
int XmpParser::encode( std::string& xmpPacket,
const XmpData& xmpData,
uint16_t formatFlags,
uint32_t padding)
{ try {
if (xmpData.empty()) {
xmpPacket.clear();
return 0;
}
if (!initialize()) {
#ifndef SUPPRESS_WARNINGS
EXV_ERROR << "XMP toolkit initialization failed.\n";
#endif
return 2;
}
// Register custom namespaces with XMP-SDK
for (auto&& i : XmpProperties::nsRegistry_) {
#ifdef EXIV2_DEBUG_MESSAGES
std::cerr << "Registering " << i.second.prefix_ << " : " << i.first << "\n";
#endif
registerNs(i.first, i.second.prefix_);
}
SXMPMeta meta;
for (auto&& i : xmpData) {
const std::string ns = XmpProperties::ns(i.groupName());
XMP_OptionBits options = 0;
if (i.typeId() == langAlt) {
// Encode Lang Alt property
const auto la = dynamic_cast<const LangAltValue*>(&i.value());
if (la == nullptr)
throw Error(kerEncodeLangAltPropertyFailed, i.key());
int idx = 1;
for (auto&& k : la->value_) {
if (!k.second.empty()) { // remove lang specs with no value
printNode(ns, i.tagName(), k.second, 0);
meta.AppendArrayItem(ns.c_str(), i.tagName().c_str(), kXMP_PropArrayIsAlternate,
k.second.c_str());
const std::string item = i.tagName() + "[" + toString(idx++) + "]";
meta.SetQualifier(ns.c_str(), item.c_str(), kXMP_NS_XML, "lang", k.first.c_str());
}
}
continue;
}
// Todo: Xmpdatum should have an XmpValue, not a Value
const auto val = dynamic_cast<const XmpValue*>(&i.value());
if (val == nullptr)
throw Error(kerInvalidKeyXmpValue, i.key(), i.typeName());
options = xmpArrayOptionBits(val->xmpArrayType())
| xmpArrayOptionBits(val->xmpStruct());
if (i.typeId() == xmpBag || i.typeId() == xmpSeq || i.typeId() == xmpAlt) {
printNode(ns, i.tagName(), "", options);
meta.SetProperty(ns.c_str(), i.tagName().c_str(), nullptr, options);
for (long idx = 0; idx < i.count(); ++idx) {
const std::string item = i.tagName() + "[" + toString(idx + 1) + "]";
printNode(ns, item, i.toString(idx), 0);
meta.SetProperty(ns.c_str(), item.c_str(), i.toString(idx).c_str());
}
continue;
}
if (i.typeId() == xmpText) {
if (i.count() == 0) {
printNode(ns, i.tagName(), "", options);
meta.SetProperty(ns.c_str(), i.tagName().c_str(), nullptr, options);
} else {
printNode(ns, i.tagName(), i.toString(0), options);
meta.SetProperty(ns.c_str(), i.tagName().c_str(), i.toString(0).c_str(), options);
}
continue;
}
// Don't let any Xmpdatum go by unnoticed
throw Error(kerUnhandledXmpdatum, i.tagName(), i.typeName());
}
std::string tmpPacket;
meta.SerializeToBuffer(&tmpPacket, xmpFormatOptionBits(static_cast<XmpFormatFlags>(formatFlags)), padding); // throws
xmpPacket = tmpPacket;
return 0;
}
#ifndef SUPPRESS_WARNINGS
catch (const XMP_Error& e) {
EXV_ERROR << Error(kerXMPToolkitError, e.GetID(), e.GetErrMsg()) << "\n";
return 3;
}
#else
catch (const XMP_Error&) {
return 3;
}
#endif // SUPPRESS_WARNINGS
} // XmpParser::decode
#else
int XmpParser::encode( std::string& /*xmpPacket*/,
const XmpData& xmpData,
uint16_t /*formatFlags*/,
uint32_t /*padding*/)
{
if (!xmpData.empty()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "XMP toolkit support not compiled in.\n";
#endif
}
return 1;
} // XmpParser::encode
#endif // !EXV_HAVE_XMP_TOOLKIT
} // namespace Exiv2
// *****************************************************************************
// local definitions
namespace {
#ifdef EXV_HAVE_XMP_TOOLKIT
Exiv2::XmpValue::XmpStruct xmpStruct(const XMP_OptionBits& opt)
{
Exiv2::XmpValue::XmpStruct var(Exiv2::XmpValue::xsNone);
if (XMP_PropIsStruct(opt)) {
var = Exiv2::XmpValue::xsStruct;
}
return var;
}
XMP_OptionBits xmpArrayOptionBits(Exiv2::XmpValue::XmpStruct xs)
{
XMP_OptionBits var(0);
switch (xs) {
case Exiv2::XmpValue::xsNone:
break;
case Exiv2::XmpValue::xsStruct:
XMP_SetOption(var, kXMP_PropValueIsStruct);
break;
}
return var;
}
Exiv2::TypeId arrayValueTypeId(const XMP_OptionBits& opt)
{
Exiv2::TypeId typeId(Exiv2::invalidTypeId);
if (XMP_PropIsArray(opt)) {
if (XMP_ArrayIsAlternate(opt)) typeId = Exiv2::xmpAlt;
else if (XMP_ArrayIsOrdered(opt)) typeId = Exiv2::xmpSeq;
else if (XMP_ArrayIsUnordered(opt)) typeId = Exiv2::xmpBag;
}
return typeId;
}
Exiv2::XmpValue::XmpArrayType xmpArrayType(const XMP_OptionBits& opt)
{
return Exiv2::XmpValue::xmpArrayType(arrayValueTypeId(opt));
}
XMP_OptionBits xmpArrayOptionBits(Exiv2::XmpValue::XmpArrayType xat)
{
XMP_OptionBits var(0);
switch (xat) {
case Exiv2::XmpValue::xaNone:
break;
case Exiv2::XmpValue::xaAlt:
XMP_SetOption(var, kXMP_PropValueIsArray);
XMP_SetOption(var, kXMP_PropArrayIsAlternate);
break;
case Exiv2::XmpValue::xaSeq:
XMP_SetOption(var, kXMP_PropValueIsArray);
XMP_SetOption(var, kXMP_PropArrayIsOrdered);
break;
case Exiv2::XmpValue::xaBag:
XMP_SetOption(var, kXMP_PropValueIsArray);
break;
}
return var;
}
#ifdef EXV_ADOBE_XMPSDK
#define kXMP_WriteAliasComments 0x0400UL
#endif
XMP_OptionBits xmpFormatOptionBits(Exiv2::XmpParser::XmpFormatFlags flags)
{
XMP_OptionBits var(0);
if (flags & Exiv2::XmpParser::omitPacketWrapper) var |= kXMP_OmitPacketWrapper;
if (flags & Exiv2::XmpParser::readOnlyPacket) var |= kXMP_ReadOnlyPacket;
if (flags & Exiv2::XmpParser::useCompactFormat) var |= kXMP_UseCompactFormat;
if (flags & Exiv2::XmpParser::includeThumbnailPad) var |= kXMP_IncludeThumbnailPad;
if (flags & Exiv2::XmpParser::exactPacketLength) var |= kXMP_ExactPacketLength;
if (flags & Exiv2::XmpParser::writeAliasComments) var |= kXMP_WriteAliasComments;
if (flags & Exiv2::XmpParser::omitAllFormatting) var |= kXMP_OmitAllFormatting;
return var;
}
#ifdef EXIV2_DEBUG_MESSAGES
void printNode(const std::string& schemaNs,
const std::string& propPath,
const std::string& propValue,
const XMP_OptionBits& opt)
{
static bool first = true;
if (first) {
first = false;
std::cout << "ashisabsals\n"
<< "lcqqtrgqlai\n";
}
enum { alia=0, sche, hasq, isqu, stru, arra,
abag, aseq, aalt, lang, simp, len };
std::string opts(len, '.');
if (XMP_PropIsAlias(opt)) opts[alia] = 'X';
if (XMP_NodeIsSchema(opt)) opts[sche] = 'X';
if (XMP_PropHasQualifiers(opt)) opts[hasq] = 'X';
if (XMP_PropIsQualifier(opt)) opts[isqu] = 'X';
if (XMP_PropIsStruct(opt)) opts[stru] = 'X';
if (XMP_PropIsArray(opt)) opts[arra] = 'X';
if (XMP_ArrayIsUnordered(opt)) opts[abag] = 'X';
if (XMP_ArrayIsOrdered(opt)) opts[aseq] = 'X';
if (XMP_ArrayIsAlternate(opt)) opts[aalt] = 'X';
if (XMP_ArrayIsAltText(opt)) opts[lang] = 'X';
if (XMP_PropIsSimple(opt)) opts[simp] = 'X';
std::cout << opts << " ";
if (opts[sche] == 'X') {
std::cout << "ns=" << schemaNs;
}
else {
std::cout << propPath << " = " << propValue;
}
std::cout << std::endl;
}
#else
void printNode(const std::string& ,
const std::string& ,
const std::string& ,
const XMP_OptionBits& )
{}
#endif // EXIV2_DEBUG_MESSAGES
Exiv2::XmpKey::UniquePtr makeXmpKey(const std::string& schemaNs,
const std::string& propPath)
{
std::string property;
std::string::size_type idx = propPath.find(':');
if (idx == std::string::npos) {
throw Exiv2::Error(Exiv2::kerPropertyNameIdentificationFailed, propPath, schemaNs);
}
// Don't worry about out_of_range, XMP parser takes care of this
property = propPath.substr(idx + 1);
std::string prefix = Exiv2::XmpProperties::prefix(schemaNs);
if (prefix.empty()) {
throw Exiv2::Error(Exiv2::kerNoPrefixForNamespace, propPath, schemaNs);
}
return std::make_unique<Exiv2::XmpKey>(prefix, property);
} // makeXmpKey
#endif // EXV_HAVE_XMP_TOOLKIT
} // namespace