22 #include <sys/types.h>
32 std::string undressed_file_name(
const std::string& fn) {
35 gSystem->ExpandPathName(tfn);
36 return std::string(tfn.Data());
40 int s_minPrintLevel = dd4hep::INFO;
42 std::string _clean_fname(
const std::string& filepath) {
44 std::string
const& temp =
getEnviron(filepath);
45 std::string temp2 = undressed_file_name( temp.empty() ? filepath : temp );
46 if ( strncmp(temp2.c_str(),
"file:",5)==0 )
return temp2.substr(5);
53 #include <xercesc/framework/LocalFileFormatTarget.hpp>
54 #include <xercesc/framework/StdOutFormatTarget.hpp>
55 #include <xercesc/framework/MemBufFormatTarget.hpp>
56 #include <xercesc/framework/MemBufInputSource.hpp>
57 #include <xercesc/sax/SAXParseException.hpp>
58 #include <xercesc/sax/EntityResolver.hpp>
59 #include <xercesc/sax/InputSource.hpp>
60 #include <xercesc/parsers/XercesDOMParser.hpp>
61 #include <xercesc/util/XMLEntityResolver.hpp>
62 #include <xercesc/util/PlatformUtils.hpp>
63 #include <xercesc/util/XercesDefs.hpp>
64 #include <xercesc/util/XMLUni.hpp>
65 #include <xercesc/util/XMLURL.hpp>
66 #include <xercesc/util/XMLString.hpp>
67 #include <xercesc/dom/DOM.hpp>
68 #include <xercesc/sax/ErrorHandler.hpp>
70 using namespace xercesc;
86 printout(DEBUG,
"DocumentErrorHandler",
"+++ Destructing the XercesC DOM-XML document error handler....");
95 void error(
const SAXParseException& e);
97 void fatalError(
const SAXParseException& e);
99 virtual bool handleError(
const DOMError& domError);
104 std::string err =
"DOM UNKNOWN: ";
105 switch (domError.getSeverity()) {
106 case DOMError::DOM_SEVERITY_WARNING:
107 err =
"DOM WARNING: ";
109 case DOMError::DOM_SEVERITY_ERROR:
112 case DOMError::DOM_SEVERITY_FATAL_ERROR:
118 printout(FATAL,
"DocumentErrorHandler",
"+++ %s %s: %s", err.c_str(),
120 DOMLocator* loc = domError.getLocation();
122 printout(FATAL,
"DocumentErrorHandler",
"+++ Location: Line:%d Column: %d",
123 int(loc->getLineNumber()),
int(loc->getColumnNumber()));
129 std::string m(
_toString(e.getMessage()));
130 if (m.find(
"The values for attribute 'name' must be names or name tokens") != std::string::npos
131 || m.find(
"The values for attribute 'ID' must be names or name tokens") != std::string::npos
132 || m.find(
"for attribute 'name' must be Name or Nmtoken") != std::string::npos
133 || m.find(
"for attribute 'ID' must be Name or Nmtoken") != std::string::npos
134 || m.find(
"for attribute 'name' is invalid Name or NMTOKEN value") != std::string::npos
135 || m.find(
"for attribute 'ID' is invalid Name or NMTOKEN value") != std::string::npos)
137 std::string sys(
_toString(e.getSystemId()));
138 printout(ERROR,
"XercesC",
"+++ Error at file \"%s\", Line %d Column: %d Message:%s",
139 sys.c_str(),
int(e.getLineNumber()),
int(e.getColumnNumber()), m.c_str());
143 std::string m(
_toString(e.getMessage()));
144 std::string sys(
_toString(e.getSystemId()));
145 printout(FATAL,
"XercesC",
"+++ FATAL Error at file \"%s\", Line %d Column: %d Message:%s",
146 sys.c_str(),
int(e.getLineNumber()),
int(e.getColumnNumber()), m.c_str());
152 class dd4hepDOMParser :
public XercesDOMParser {
157 class Resolver :
public XMLEntityResolver {
160 Resolver(dd4hepDOMParser* p) :
parser(p) {}
161 virtual ~Resolver() {}
162 virtual InputSource *resolveEntity(XMLResourceIdentifier *
id)
163 {
return parser->read_uri(
id); }
174 virtual ~dd4hepDOMParser() {
178 InputSource *read_uri(XMLResourceIdentifier *
id) {
180 std::string buf, systemID(
_toString(id->getSystemId()));
182 const XMLByte* input = (
const XMLByte*)XMLString::replicate(buf.c_str());
184 std::string baseURI(
_toString(id->getBaseURI()));
185 std::string schema(
_toString(id->getSchemaLocation()));
187 if ( s_minPrintLevel <= INFO ) {
188 printout(INFO,
"XercesC",
"+++ Resolved URI: sysID:%s uri:%s ns:%s schema:%s",
189 systemID.c_str(), baseURI.c_str(),
ns.c_str(), schema.c_str());
192 return new MemBufInputSource(input,buf.length(),systemID.c_str(),
true);
200 XercesDOMParser* make_parser(
UriReader* reader=0) {
201 XercesDOMParser*
parser =
new dd4hepDOMParser(reader);
202 parser->setValidationScheme(XercesDOMParser::Val_Auto);
203 parser->setValidationSchemaFullChecking(
true);
204 parser->setCreateEntityReferenceNodes(
false);
205 parser->setDoNamespaces(
false);
206 parser->setDoSchema(
true);
219 DOMImplementation *imp = DOMImplementationRegistry::getDOMImplementation(
Strng_t(
"LS"));
220 MemBufFormatTarget *tar =
new MemBufFormatTarget();
221 DOMLSOutput *out = imp->createLSOutput();
222 DOMLSSerializer *wrt = imp->createLSSerializer();
223 out->setByteStream(tar);
224 wrt->getDomConfig()->setParameter(
Strng_t(
"format-pretty-print"),
true);
225 wrt->write(doc, out);
226 os << tar->getRawBuffer() << std::endl << std::flush;
231 printout(ERROR,
"dumpTree",
"+++ Cannot dump invalid document.");
235 void dump_doc(DOMDocument* doc, std::ostream& os) {
252 std::string path = system_path(base);
253 std::string dir = ::dirname((
char*)path.c_str());
264 std::string path, dir = system_path(base);
265 TUri uri_base(dir.c_str()), uri_rel(fn.c_str());
266 TUrl url_base(dir.c_str());
267 path = TUri::MergePaths(uri_rel,uri_base);
268 TUri
final(path.c_str());
270 path = url_base.GetProtocol()+std::string(
"://")+
final.GetUri().Data();
271 if ( path[path.length()-1]==
'/' ) path = path.substr(0,path.length()-1);
277 DOMElement* elt = (DOMElement*)base.
ptr();
278 std::string path =
_toString(elt->getBaseURI());
279 if ( path[0] ==
'/' ) {
280 std::string tmp =
"file:"+path;
289 DOMElement* elt = (DOMElement*)base.
ptr();
296 XMLURL ref_url(elt->getBaseURI(), p);
301 return load(path, reader);
304 std::string b =
_toString(elt->getBaseURI());
306 printout(DEBUG,
"DocumentHandler",
"+++ URI exception: %s -> %s [%s]",b.c_str(),e.c_str(),exc.what());
309 std::string b =
_toString(elt->getBaseURI());
311 printout(DEBUG,
"DocumentHandler",
"+++ URI exception: %s -> %s",b.c_str(),e.c_str());
314 std::string buf, sys = system_path(base,fname);
316 std::string buf, sys, dir =
_toString(elt->getBaseURI());
318 dir = ::dirname((
char*)dir.c_str());
319 while( fn.substr(0,3) ==
"../" ) {
320 dir = ::dirname((
char*)dir.c_str());
323 sys = dir +
"/" + fn;
325 if ( reader->
load(sys, buf) ) {
327 Document doc =
parse(buf.c_str(), buf.length(), sys.c_str(), reader);
331 return parse(buf.c_str(), buf.length(), sys.c_str(), reader);
339 auto fname_clean = _clean_fname(fname);
341 printout(DEBUG,
"DocumentHandler",
"+++ Loading document URI: %s",fname_clean.c_str());
343 size_t idx = fname_clean.find(
':');
344 size_t idq = fname_clean.find(
'/');
345 if ( idq == std::string::npos ) idq = 0;
346 XMLURL xerurl = (
const XMLCh*)
Strng_t(idx==std::string::npos || idx>idq ?
"file:"+fname_clean : fname_clean);
347 std::string proto =
_toString(xerurl.getProtocolName());
349 printout(DEBUG,
"DocumentHandler",
"+++ protocol:%s path:%s",proto.c_str(), path.c_str());
353 std::unique_ptr < XercesDOMParser >
parser(make_parser(reader));
355 if ( !path.empty() ) {
356 parser->parse(path.c_str());
360 if ( reader && reader->
load(fname_clean, path) ) {
361 MemBufInputSource src((
const XMLByte*)path.c_str(), path.length(), fname.c_str(),
false);
363 return (XmlDocument*)
parser->adoptDocument();
365 return (XmlDocument*)0;
369 printout(ERROR,
"DocumentHandler",
"+++ Exception(XercesC): parse(path):%s",e.what());
371 parser->parse(fname.c_str());
375 printout(FATAL,
"DocumentHandler",
"+++ Exception(XercesC): parse(URI):%s",ex.what());
379 printout(DEBUG,
"DocumentHandler",
"+++ Document %s succesfully parsed with XercesC .....",path.c_str());
380 return (XmlDocument*)
parser->adoptDocument();
385 std::unique_ptr < XercesDOMParser >
parser(make_parser(rdr));
386 MemBufInputSource src((
const XMLByte*)bytes, length, sys_id,
false);
388 DOMDocument* doc =
parser->adoptDocument();
389 doc->setXmlStandalone(
true);
390 doc->setStrictErrorChecking(
true);
391 return (XmlDocument*) doc;
396 XMLFormatTarget *tar = 0;
397 DOMImplementation *imp = DOMImplementationRegistry::getDOMImplementation(
Strng_t(
"LS"));
398 DOMLSOutput *out = imp->createLSOutput();
399 DOMLSSerializer *wrt = imp->createLSSerializer();
402 tar =
new StdOutFormatTarget();
404 std::string fn = undressed_file_name(fname);
405 tar =
new LocalFileFormatTarget(
Strng_t(fn));
407 out->setByteStream(tar);
408 wrt->getDomConfig()->setParameter(
Strng_t(
"format-pretty-print"),
true);
409 wrt->write((xercesc::DOMDocument*) doc.
ptr(), out);
429 Xml(
void* ptr) : p(ptr) {}
430 Xml(
const void* ptr) : cp(ptr) {}
445 std::string fn, clean = _clean_fname(fname);
450 std::string bn =
Xml(elt.document()).
d->Value();
452 char drive[_MAX_DRIVE], dir[_MAX_DIR], file[_MAX_FNAME], ext[_MAX_EXT];
453 ::_splitpath(bn.c_str(),drive,dir,file,ext);
460 fn = ::dirname((
char*)bn.c_str());
463 fn += _clean_fname(fname);
465 if ( ::stat(fn.c_str(),&st)==0 )
467 else if ( ::stat(clean.c_str(),&st)==0 )
478 fn =
Xml(elt.document()).
d->Value();
480 return undressed_file_name(fn);
485 std::string clean = _clean_fname(fname);
487 printout(WARNING,
"DocumentHandler",
"+++ Loading document URI: %s %s",
488 fname.c_str(),
"[URI Resolution is not supported by TiXML]");
490 else if ( s_minPrintLevel <= INFO ) {
491 printout(INFO,
"DocumentHandler",
"+++ Loading document URI: %s [Resolved:'%s']",
492 fname.c_str(),clean.c_str());
499 if ( doc->
Error() ) {
500 printout(FATAL,
"DocumentHandler",
"+++ Error (TinyXML) parsing XML document:%s [%s]",
501 fname.c_str(), clean.c_str());
502 printout(FATAL,
"DocumentHandler",
"+++ Error (TinyXML) XML parsing error:%s",
504 printout(FATAL,
"DocumentHandler",
"+++ Document:%s Location Line:%d Column:%d",
506 except(
"dd4hep:XML",
"++ file:%s error:%s",clean.c_str(),doc->
ErrorDesc());
508 except(
"dd4hep:XML",
"++ Unknown error (TinyXML) while parsing:%s",fname.c_str());
512 printout(ERROR,
"DocumentHandler",
"+++ Exception (TinyXML): parse(path):%s",e.what());
515 if ( s_minPrintLevel <= INFO ) {
516 printout(INFO,
"DocumentHandler",
"+++ Document %s succesfully parsed with TinyXML .....",
519 return (XmlDocument*)doc;
527 std::string path = system_path(base, fname);
528 return load(path,reader);
534 printout(WARNING,
"DocumentHandler",
"+++ Parsing memory document %s",
535 "[URI Resolution is not supported by TiXML]");
540 size_t str_len = ::strlen(bytes);
543 if ( str_len+1 != len || bytes[str_len] != 0 || ::isspace(bytes[str_len-1]) ) {
544 std::unique_ptr<char[]> data(
new char[len+1]);
545 char* buff = data.get();
547 ::memcpy(buff, bytes, len+1);
549 for(
size_t i=len-1; i>0 && (buff[i]==0 || ::isspace(buff[i])); --i)
551 if ( 0 == doc->
Parse(buff) ) {
552 return (XmlDocument*)doc;
558 if ( 0 == doc->
Parse(bytes) ) {
559 return (XmlDocument*)doc;
561 if ( doc->
Error() ) {
562 printout(FATAL,
"DocumentHandler",
563 "+++ Error (TinyXML) while parsing XML string [%s]",
565 printout(FATAL,
"DocumentHandler",
566 "+++ XML Document error: %s Location Line:%d Column:%d",
568 throw std::runtime_error(std::string(
"dd4hep: ")+doc->
ErrorDesc());
570 throw std::runtime_error(
"dd4hep: Unknown error while parsing XML document string with TiXml.");
572 throw std::runtime_error(
"dd4hep: FAILED to parse invalid document string [NULL] with TiXml.");
575 printout(ERROR,
"DocumentHandler",
"+++ Exception (TinyXML): parse(string):%s",e.what());
583 std::string fn = undressed_file_name(fname);
584 FILE* file = fn.empty() ? stdout : ::fopen(fn.c_str(),
"w");
586 printout(ERROR,
"DocumentHandler",
"+++ Failed to open output file: %s",fname.c_str());
591 if ( !fn.empty() ) ::fclose(file);
623 int tmp = s_minPrintLevel;
624 s_minPrintLevel = level;
630 const char comment[] =
"\n"
631 " +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n"
632 " ++++ dd4hep generated alignment file using the ++++\n"
633 " ++++ dd4hep Detector description XML generator. ++++\n"
639 " ++++ M.Frank CERN/LHCb ++++\n"
640 " +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n ";
646 return load(fname, 0);
651 return load(base, fname, 0);
656 return parse(bytes, length,
"xml-memory-buffer", 0);
662 return system_path(base, fn);
667 std::string path = system_path(base,fname);
668 std::string dir = ::dirname((
char*)path.c_str());
674 std::string path = system_path(base);
675 std::string dir = ::dirname((
char*)path.c_str());
681 std::string top(tag);
682 std::string empty =
"<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n";
683 empty +=
"<" + top +
"/>\0\0";
694 return create(tag.c_str(), comment.c_str());