Annotation of parser3/src/classes/xdoc.C, revision 1.154

1.1       parser      1: /** @file
1.2       parser      2:        Parser: @b xdoc parser class.
1.1       parser      3: 
1.148     paf         4:        Copyright (c) 2001-2005 ArtLebedev Group (http://www.artlebedev.com)
1.85      paf         5:        Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru)
1.96      paf         6: */
1.1       parser      7: 
1.111     paf         8: #include "pa_config_includes.h"
                      9: 
1.1       parser     10: #ifdef XML
1.96      paf        11: 
1.154   ! misha      12: static const char * const IDENT_XDOC_C="$Date: 2007/02/28 19:36:31 $";
1.82      paf        13: 
1.69      paf        14: #include "libxml/tree.h"
1.133     paf        15: #include "libxml/HTMLtree.h"
1.69      paf        16: #include "libxslt/xsltInternals.h"
                     17: #include "libxslt/transform.h"
                     18: #include "libxslt/xsltutils.h"
1.90      paf        19: #include "libxslt/variables.h"
1.132     paf        20: #include "libxslt/imports.h"
1.111     paf        21: 
                     22: #include "pa_vmethod_frame.h"
                     23: 
                     24: #include "pa_stylesheet_manager.h"
                     25: #include "pa_request.h"
                     26: #include "pa_vxdoc.h"
                     27: #include "pa_charset.h"
                     28: #include "pa_vfile.h"
1.115     paf        29: #include "pa_xml_exception.h"
1.111     paf        30: #include "xnode.h"
                     31: 
1.1       parser     32: // defines
                     33: 
                     34: #define XDOC_CLASS_NAME "xdoc"
                     35: 
                     36: #define XDOC_OUTPUT_METHOD_OPTION_NAME "method"
                     37: #define XDOC_OUTPUT_METHOD_OPTION_VALUE_XML "xml"
                     38: #define XDOC_OUTPUT_METHOD_OPTION_VALUE_HTML "html"
                     39: #define XDOC_OUTPUT_METHOD_OPTION_VALUE_TEXT "text"
                     40: 
                     41: // class
                     42: 
1.111     paf        43: class MXdoc: public MXnode {
1.1       parser     44: public: // VStateless_class
1.149     paf        45:        Value* create_new_value(Pool&, HashStringValue&) { return new VXdoc(); }
1.1       parser     46: 
                     47: public:
1.111     paf        48:        MXdoc();
1.1       parser     49: 
                     50: };
                     51: 
1.111     paf        52: // global variable
                     53: 
                     54: DECLARE_CLASS_VAR(xdoc, new MXdoc, 0);
1.61      paf        55: 
1.138     paf        56: // helper classes
                     57: 
                     58: class xmlOutputBuffer_auto_ptr {
                     59: public:
                     60:        explicit xmlOutputBuffer_auto_ptr(xmlOutputBuffer *_APtr = 0) 
                     61:                : _Owns(_APtr != 0), _Ptr(_APtr) {}
                     62:        xmlOutputBuffer_auto_ptr(const xmlOutputBuffer_auto_ptr& _Y) 
                     63:                : _Owns(_Y._Owns), _Ptr(_Y.release()) {}
                     64:        xmlOutputBuffer_auto_ptr& operator=(const xmlOutputBuffer_auto_ptr& _Y) 
                     65:                {if (this != &_Y)
                     66:                        {if (_Ptr != _Y.get())
                     67:                                {if (_Owns && _Ptr)
                     68:                                        xmlOutputBufferClose(_Ptr);
                     69:                                _Owns = _Y._Owns; }
                     70:                        else if (_Y._Owns)
                     71:                                _Owns = true;
                     72:                        _Ptr = _Y.release(); }
                     73:                return (*this); }
                     74:        ~xmlOutputBuffer_auto_ptr()
                     75:                {if (_Owns && _Ptr)
                     76:                        xmlOutputBufferClose(_Ptr); }
                     77:        xmlOutputBuffer& operator*() const 
                     78:                {return (*get()); }
                     79:        xmlOutputBuffer *operator->() const 
                     80:                {return (get()); }
                     81:        xmlOutputBuffer *get() const 
                     82:                {return (_Ptr); }
                     83:        xmlOutputBuffer *release() const 
                     84:                {((xmlOutputBuffer_auto_ptr *)this)->_Owns = false;
                     85:                return (_Ptr); }
                     86: private:
                     87:        bool _Owns;
                     88:        xmlOutputBuffer *_Ptr;
                     89: };
                     90: 
1.65      paf        91: class xsltTransformContext_auto_ptr {
                     92: public:
1.67      paf        93:        explicit xsltTransformContext_auto_ptr(xsltTransformContext *_APtr = 0) 
                     94:                : _Owns(_APtr != 0), _Ptr(_APtr) {}
1.65      paf        95:        xsltTransformContext_auto_ptr(const xsltTransformContext_auto_ptr& _Y) 
                     96:                : _Owns(_Y._Owns), _Ptr(_Y.release()) {}
                     97:        xsltTransformContext_auto_ptr& operator=(const xsltTransformContext_auto_ptr& _Y) 
                     98:                {if (this != &_Y)
                     99:                        {if (_Ptr != _Y.get())
                    100:                                {if (_Owns && _Ptr)
                    101:                                        xsltFreeTransformContext(_Ptr);
                    102:                                _Owns = _Y._Owns; }
                    103:                        else if (_Y._Owns)
                    104:                                _Owns = true;
                    105:                        _Ptr = _Y.release(); }
                    106:                return (*this); }
                    107:        ~xsltTransformContext_auto_ptr()
                    108:                {if (_Owns && _Ptr)
                    109:                        xsltFreeTransformContext(_Ptr); }
                    110:        xsltTransformContext& operator*() const 
                    111:                {return (*get()); }
                    112:        xsltTransformContext *operator->() const 
                    113:                {return (get()); }
                    114:        xsltTransformContext *get() const 
                    115:                {return (_Ptr); }
                    116:        xsltTransformContext *release() const 
                    117:                {((xsltTransformContext_auto_ptr *)this)->_Owns = false;
                    118:                return (_Ptr); }
                    119: private:
                    120:        bool _Owns;
                    121:        xsltTransformContext *_Ptr;
                    122: };
1.61      paf       123: 
1.65      paf       124: class xsltStylesheet_auto_ptr {
                    125: public:
1.67      paf       126:        explicit xsltStylesheet_auto_ptr(xsltStylesheet *_APtr = 0) 
                    127:                : _Owns(_APtr != 0), _Ptr(_APtr) {}
1.65      paf       128:        xsltStylesheet_auto_ptr(const xsltStylesheet_auto_ptr& _Y) 
                    129:                : _Owns(_Y._Owns), _Ptr(_Y.release()) {}
                    130:        xsltStylesheet_auto_ptr& operator=(const xsltStylesheet_auto_ptr& _Y) 
                    131:                {if (this != &_Y)
                    132:                        {if (_Ptr != _Y.get())
                    133:                                {if (_Owns && _Ptr)
                    134:                                        xsltFreeStylesheet(_Ptr);
                    135:                                _Owns = _Y._Owns; }
                    136:                        else if (_Y._Owns)
                    137:                                _Owns = true;
                    138:                        _Ptr = _Y.release(); }
                    139:                return (*this); }
                    140:        ~xsltStylesheet_auto_ptr()
                    141:                {if (_Owns && _Ptr)
                    142:                        xsltFreeStylesheet(_Ptr); }
                    143:        xsltStylesheet& operator*() const 
                    144:                {return (*get()); }
                    145:        xsltStylesheet *operator->() const 
                    146:                {return (get()); }
                    147:        xsltStylesheet *get() const 
                    148:                {return (_Ptr); }
                    149:        xsltStylesheet *release() const 
                    150:                {((xsltStylesheet_auto_ptr *)this)->_Owns = false;
                    151:                return (_Ptr); }
                    152: private:
                    153:        bool _Owns;
                    154:        xsltStylesheet *_Ptr;
                    155: };
1.61      paf       156: 
1.1       parser    157: // methods
                    158: 
1.149     paf       159: static void writeNode(Request& r, VXdoc& xdoc, xmlNode* node) {
                    160:        if(!node)
1.154   ! misha     161:                throw Exception(PARSER_RUNTIME,
1.149     paf       162:                        0,
                    163:                        "error creating node"); // OOM, bad name, things like that
1.54      paf       164: 
                    165:        // write out result
1.149     paf       166:        r.write_no_lang(xdoc.wrap(*node));
                    167: }
                    168: 
                    169: struct IdsIteratorInfo {
                    170:        xmlChar *elementId;
                    171:        xmlNode *element;
                    172: };
                    173: 
                    174: /* Hash Scanner function for pa_getElementById */
                    175: extern "C" void // switching to calling convetion of libxml
                    176: idsHashScanner (void *payload, void *data, xmlChar *name) {
                    177:        IdsIteratorInfo *priv = (IdsIteratorInfo *)data;
                    178: 
                    179:        if (priv->element == NULL && xmlStrEqual (name, priv->elementId))
                    180:        {
                    181:                xmlNode* parent=((xmlID *)payload)->attr->parent;
                    182:                assert(parent);
                    183:                priv->element=parent;
                    184:        }
                    185: }
                    186: 
                    187: static xmlNode*
                    188: pa_getElementById(xmlDoc& xmldoc, xmlChar* elementId) {
                    189:        xmlHashTable *ids = (xmlHashTable *)xmldoc.ids;
                    190:        IdsIteratorInfo iter={elementId, NULL};
                    191:        xmlHashScan(ids, idsHashScanner, &iter);
                    192:        return iter.element;
                    193: }
                    194: 
                    195: /*
                    196: static xmlNode *
                    197: pa_importNode (xmlDoc& xmldoc, xmlNode& importedNode, bool deep) {
                    198:        xmlNode *result = NULL;
                    199: 
                    200:        switch (importedNode.type) {
                    201:        case XML_ATTRIBUTE_NODE:
                    202:                result = (xmlNode *)xmlCopyProp(xmldoc, (xmlAttr *)importedNode);
                    203:                result.parent=0; // no idea
                    204:                break;
                    205:        case XML_DOCUMENT_FRAG_NODE:
                    206:        case XML_ELEMENT_NODE:
                    207:        case XML_ENTITY_REF_NODE:
                    208:        case XML_PI_NODE:
                    209:        case XML_TEXT_NODE:
                    210:        case XML_CDATA_SECTION_NODE:
                    211:        case XML_COMMENT_NODE:
                    212:                result = xmlCopyNode (importedNode->n, deep);
                    213:                xmlSetTreeDoc (result, priv->n);
                    214:                break;
                    215:        default:
                    216:                *exc = GDOME_NOT_SUPPORTED_ERR;
                    217:        }
                    218: 
                    219:        return result;
1.54      paf       220: }
1.149     paf       221: */
1.54      paf       222: 
1.16      parser    223: // Element createElement(in DOMString tagName) raises(DOMException);
1.111     paf       224: static void _createElement(Request& r, MethodParams& params) {
                    225:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       226:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    227: 
1.149     paf       228:        xmlChar* tagName=as_xmlchar(r, params, 0, "tagName must be string");
1.16      parser    229: 
1.149     paf       230:        xmlNode *node=xmlNewDocNode(&xmldoc, NULL, tagName, NULL);
                    231:        writeNode(r, vdoc, node);
1.16      parser    232: }
                    233: 
1.113     paf       234: // Element createElementNS(in DOMString namespaceURI, in DOMString qualifiedName) raises(DOMException);
1.112     paf       235: static void _createElementNS(Request& r, MethodParams& params) {
                    236:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       237:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.112     paf       238: 
                    239:        // namespaceURI;localName
1.149     paf       240:        xmlChar* namespaceURI=as_xmlchar(r, params, 0, "namespaceURI must be string");
                    241:        xmlChar* qualifiedName=as_xmlchar(r, params, 1, "qualifiedName must be string");
                    242: 
                    243:        xmlChar* prefix=0;
                    244:        xmlChar* localName=xmlSplitQName2(qualifiedName, &prefix);
1.112     paf       245: 
1.149     paf       246:        xmlNode *node;
                    247:        if(localName) {
                    248:                xmlNs& ns=pa_xmlMapNs(xmldoc, namespaceURI, prefix);
                    249:                node=xmlNewDocNode(&xmldoc, &ns, localName, NULL);
                    250:        } else
                    251:                node=xmlNewDocNode(&xmldoc, NULL, qualifiedName/*unqualified, actually*/, NULL);
                    252:        writeNode(r, vdoc, node);
1.112     paf       253: }
                    254: 
1.16      parser    255: // DocumentFragment createDocumentFragment()
1.111     paf       256: static void _createDocumentFragment(Request& r, MethodParams&) {
                    257:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       258:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    259: 
1.149     paf       260:        xmlNode *node=xmlNewDocFragment(&xmldoc);
                    261:        writeNode(r, vdoc, node);
1.16      parser    262: }
                    263: 
                    264: // Text createTextNode(in DOMString data);
1.111     paf       265: static void _createTextNode(Request& r, MethodParams& params) {
                    266:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       267:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    268: 
1.149     paf       269:        xmlChar* data=as_xmlchar(r, params, 0, "data must be string");
1.16      parser    270: 
1.149     paf       271:        xmlNode *node=xmlNewDocText(&xmldoc, data);
                    272:        writeNode(r, vdoc, node);
1.16      parser    273: }
                    274: 
                    275: // Comment createComment(in DOMString data)
1.111     paf       276: static void _createComment(Request& r, MethodParams& params) {
                    277:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.16      parser    278: 
1.149     paf       279:        xmlChar* data=as_xmlchar(r, params, 0, "data must be string");
1.16      parser    280: 
1.149     paf       281:        xmlNode *node=xmlNewComment(data);
                    282:        writeNode(r, vdoc, node);
1.16      parser    283: }
                    284: 
                    285: // CDATASection createCDATASection(in DOMString data) raises(DOMException);
1.111     paf       286: static void _createCDATASection(Request& r, MethodParams& params) {
                    287:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       288:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    289: 
1.149     paf       290:        xmlChar* data=as_xmlchar(r, params, 0, "data must be string");
1.16      parser    291: 
1.149     paf       292:        xmlNode *node=xmlNewCDataBlock(&xmldoc, data, strlen((const char*)data));
                    293:        writeNode(r, vdoc, node);
1.16      parser    294: }
                    295: 
                    296: // ProcessingInstruction createProcessingInstruction(in DOMString target,in DOMString data) raises(DOMException);
1.111     paf       297: static void _createProcessingInstruction(Request& r, MethodParams& params) {
                    298:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       299:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    300: 
1.149     paf       301:        xmlChar* target=as_xmlchar(r, params, 0, "data must be string");
                    302:        xmlChar* data=as_xmlchar(r, params, 1, "data must be string");
1.16      parser    303: 
1.149     paf       304:        xmlNode *node=xmlNewDocPI(&xmldoc, target, data);
                    305:        writeNode(r, vdoc, node);
1.16      parser    306: }
                    307: 
                    308: // Attr createAttribute(in DOMString name) raises(DOMException);
1.111     paf       309: static void _createAttribute(Request& r, MethodParams& params) {
                    310:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       311:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    312: 
1.149     paf       313:        xmlChar* name=as_xmlchar(r, params, 0, "name must be string");
1.16      parser    314: 
1.149     paf       315:        xmlNode *node=(xmlNode*)xmlNewDocProp(&xmldoc, name, 0);
                    316:        writeNode(r, vdoc, node);
1.16      parser    317: }
1.112     paf       318: 
1.113     paf       319: // Attr createAttributeNS(in DOMString namespaceURI, in DOMString qualifiedName) raises(DOMException);
1.112     paf       320: static void _createAttributeNS(Request& r, MethodParams& params) {
                    321:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       322:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.112     paf       323: 
1.149     paf       324:        xmlChar* namespaceURI=as_xmlchar(r, params, 0, "namespaceURI must be string");
                    325:        xmlChar* qualifiedName=as_xmlchar(r, params, 1, "name must be string");
                    326: 
                    327:        xmlChar* prefix=0;
                    328:        xmlChar* localName=xmlSplitQName2(qualifiedName, &prefix);
                    329: 
                    330:        xmlNode *node;
                    331:        if(localName) {
                    332:                xmlNs& ns=pa_xmlMapNs(xmldoc, namespaceURI, prefix);
                    333:                node=(xmlNode*)xmlNewDocProp(&xmldoc, localName, NULL);
                    334:                xmlSetNs(node, &ns);
                    335:        } else
                    336:                node=(xmlNode*)xmlNewDocProp(&xmldoc, qualifiedName/*unqualified, actually*/, NULL);
                    337:        writeNode(r, vdoc, node);
1.112     paf       338: }
                    339: 
1.16      parser    340: // EntityReference createEntityReference(in DOMString name) raises(DOMException);
1.111     paf       341: static void _createEntityReference(Request& r, MethodParams& params) {
                    342:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       343:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    344: 
1.149     paf       345:        xmlChar* name=as_xmlchar(r, params, 0, "name must be string");
1.16      parser    346: 
1.149     paf       347:        xmlNode *node=xmlNewReference(&xmldoc, name);
                    348:        writeNode(r, vdoc, node);
1.99      paf       349: }
                    350: 
                    351: 
1.111     paf       352: static void _getElementById(Request& r, MethodParams& params) {
                    353:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       354:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.16      parser    355: 
1.54      paf       356:        // elementId
1.149     paf       357:        xmlChar* elementId=as_xmlchar(r, params, 0, "elementID must be string");
1.1       parser    358: 
1.149     paf       359:        if(xmlNode *node=pa_getElementById(xmldoc, elementId)) {
1.54      paf       360:                // write out result
1.149     paf       361:                writeNode(r, vdoc, node);
                    362:        }
1.54      paf       363: }
1.79      paf       364: 
1.111     paf       365: static void _importNode(Request& r, MethodParams& params) {
                    366:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.149     paf       367:        xmlDoc& xmldoc=vdoc.get_xmldoc();
1.79      paf       368: 
1.149     paf       369:        xmlNode& importedNode=
1.111     paf       370:                as_node(params, 0, "importedNode must be node");
1.79      paf       371:        bool deep=
1.111     paf       372:                params.as_bool(1, "deep must be bool", r);
1.79      paf       373: 
1.149     paf       374:        xmlNode *node=xmlDocCopyNode(&importedNode, &xmldoc, deep?1: 0);
1.79      paf       375:        // write out result
1.149     paf       376:        writeNode(r, vdoc, node);
1.79      paf       377: }
1.54      paf       378: /*
                    379: GdomeElement *gdome_doc_createElementNS (GdomeDocument *self, GdomeDOMString *namespaceURI, GdomeDOMString *qualifiedName, GdomeException *exc);
                    380: GdomeAttr *gdome_doc_createAttributeNS (GdomeDocument *self, GdomeDOMString *namespaceURI, GdomeDOMString *qualifiedName, GdomeException *exc);
                    381: */
1.1       parser    382: 
1.111     paf       383: static void _create(Request& r, MethodParams& params) {
                    384:        Charset& source_charset=r.charsets.source();
                    385:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.1       parser    386: 
1.111     paf       387:        Value& param=params[params.count()-1];
1.149     paf       388:        xmlDoc* xmldoc;
1.111     paf       389:        bool set_encoding=false;
1.95      paf       390:        if(param.get_junction()) { // {<?xml?>...}
1.111     paf       391:                Temp_lang temp_lang(r, String::L_XML);
1.91      paf       392:                const String& xml=r.process_to_string(param);
1.83      paf       393: 
1.127     paf       394:                const char* cstr=xml.cstr(String::L_UNSPECIFIED, 0, &r.charsets);
1.149     paf       395:                xmldoc=xmlParseMemory(cstr, strlen(cstr));
1.111     paf       396:                //printf("document=0x%p\n", document);
1.149     paf       397:                if(!xmldoc || xmlHaveGenericErrors())
                    398:                        throw XmlException(0);
1.111     paf       399: 
                    400:                // must be last action in if, see after if}
1.149     paf       401:        } else { // [localName]
1.153     misha     402:                if(const String* value = param.get_string()){
                    403:                        xmlChar* localName=r.transcode(*value);
1.120     paf       404: #if 0
1.153     misha     405:                        GdomeDocumentType *documentType=gdome_di_createDocumentType (
                    406:                                docimpl, 
                    407:                                r.transcode(qualifiedName), 
                    408:                                0/*publicId*/, 
                    409:                                0/*systemId*/, 
                    410:                                &exc);
                    411:                        if(!documentType || exc || xmlHaveGenericErrors())
                    412:                                throw Exception(
                    413:                                        method_name, 
                    414:                                        exc);
                    415:                        /// +xalan createXMLDecl ?
1.120     paf       416: #endif
1.151     misha     417:                        xmldoc=xmlNewDoc(0);
                    418:                        if(!xmldoc || xmlHaveGenericErrors())
                    419:                                throw XmlException(0);
                    420:                        xmlNode* node=xmlNewChild((xmlNode*)xmldoc, NULL, localName, NULL);
                    421:                        if(!node || xmlHaveGenericErrors())
                    422:                                throw XmlException(0);
1.103     paf       423: 
1.151     misha     424:                        set_encoding=true;
                    425:                        // must be last action in if, see after if}
1.153     misha     426:                } else {
                    427:                        VFile* vfile=param.as_vfile(String::L_UNSPECIFIED);
                    428:                        xmldoc=xmlParseMemory(vfile->value_ptr(), vfile->value_size());
                    429:                        if(!xmldoc || xmlHaveGenericErrors())
                    430:                                throw XmlException(0);
1.151     misha     431:                }
1.111     paf       432:        }
                    433:        // must be first action after if}
                    434:        // replace any previous parsed source
1.149     paf       435:        vdoc.set_xmldoc(r.charsets, *xmldoc); 
1.95      paf       436:        
                    437:        // URI 
1.111     paf       438:        const char* URI_cstr;
                    439:        if(params.count()>1) { // absolute(param)
                    440:                const String& URI=params.as_string(0, "URI must be string");
1.149     paf       441:                URI_cstr=r.absolute(URI).cstr();
1.95      paf       442:        } else // default = disk path to requested document
1.111     paf       443:                URI_cstr=r.request_info.path_translated;
1.95      paf       444:        if(URI_cstr)
1.149     paf       445:                xmldoc->URL=source_charset.transcode_buf2xchar(URI_cstr, strlen(URI_cstr));
1.95      paf       446: 
1.111     paf       447:        if(set_encoding) {
                    448:                const char* source_charset_name=source_charset.NAME().cstr();
1.149     paf       449:                xmldoc->encoding=source_charset.transcode_buf2xchar(source_charset_name, strlen(source_charset_name));
1.111     paf       450:        }
1.9       parser    451: }
                    452: 
1.111     paf       453: static void _load(Request& r, MethodParams& params) {
                    454:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.9       parser    455: 
1.54      paf       456:        // filespec
1.122     paf       457:        const String* uri=&params.as_string(0, "uri must be string");
                    458:        const char* uri_cstr;
                    459:        if(uri->pos("://")==STRING_NOT_FOUND) // disk path
                    460:                uri_cstr=r.absolute(*uri).cstr(String::L_FILE_SPEC);
                    461:        else // xxx:// 
                    462:                uri_cstr=uri->cstr(String::L_AS_IS); // leave as-is for xmlParseFile to handle
1.105     paf       463: 
1.122     paf       464:        /// todo!! add SAFE MODE!!
1.149     paf       465:        xmlDoc* xmldoc=xmlParseFile(uri_cstr);
                    466:        if(!xmldoc || xmlHaveGenericErrors())
                    467:                throw XmlException(uri);
                    468:        
1.111     paf       469:        // must be first action after if}
                    470:        // replace any previous parsed source
1.149     paf       471:        vdoc.set_xmldoc(r.charsets, *xmldoc); 
1.9       parser    472: }
                    473: 
1.111     paf       474: static void param_option_over_output_option(
                    475:                                            HashStringValue& param_options, const char* option_name,
                    476:                                            const String*& output_option) {
1.114     paf       477:        if(Value* value=param_options.get(String::Body(option_name)))
1.61      paf       478:                output_option=&value->as_string();
1.46      paf       479: }
1.111     paf       480: static void param_option_over_output_option(
                    481:                                            HashStringValue& param_options, const char* option_name,
1.143     paf       482:                                            int& output_option) {
1.114     paf       483:        if(Value* value=param_options.get(String::Body(option_name))) {
1.46      paf       484:                const String& s=value->as_string();
                    485:                if(s=="yes")
1.143     paf       486:                        output_option=1;
1.46      paf       487:                else if(s=="no")
1.143     paf       488:                        output_option=0;
1.46      paf       489:                else
1.154   ! misha     490:                        throw Exception(PARSER_RUNTIME,
1.46      paf       491:                                &s,
                    492:                                "%s must be either 'yes' or 'no'", option_name);
                    493:        }
                    494: }
                    495: 
1.100     paf       496: /// @test valid_options check
1.111     paf       497: static void prepare_output_options(Request& r,
                    498:                                   MethodParams& params, size_t index,
                    499:                                   VXdoc::Output_options& oo) {
1.46      paf       500: /*
                    501: <xsl:output
                    502:   !method = "xml" | "html" | "text" | qname-but-not-ncname 
                    503:   !version = nmtoken 
                    504:   !encoding = string 
                    505:   !omit-xml-declaration = "yes" | "no"
                    506:   !standalone = "yes" | "no"
                    507:   !doctype-public = string 
                    508:   !doctype-system = string 
                    509:   cdata-section-elements = qnames 
                    510:   !indent = "yes" | "no"
                    511:   !media-type = string /> 
1.61      paf       512: */
1.1       parser    513: 
1.46      paf       514:        // configuring with options from parameter...
1.111     paf       515:        if(params.count()>index) {
                    516:                Value& voptions=params.as_no_junction(index, "options must be string");
1.1       parser    517:                if(voptions.is_defined()) {
1.111     paf       518:                        if(HashStringValue *options=voptions.get_hash()) {
1.1       parser    519:                                // $.method[xml|html|text]
1.114     paf       520:                                if(Value* vmethod=options->get(String::Body(XDOC_OUTPUT_METHOD_OPTION_NAME)))
1.61      paf       521:                                        oo.method=&vmethod->as_string();
1.1       parser    522: 
1.46      paf       523:                                // $.version[1.0]
1.111     paf       524:                                param_option_over_output_option(*options, "version", oo.version);
1.1       parser    525:                                // $.encoding[windows-1251|...]
1.111     paf       526:                                param_option_over_output_option(*options, "encoding", oo.encoding);
1.46      paf       527:                                // $.omit-xml-declaration[yes|no]
1.111     paf       528:                                param_option_over_output_option(*options, "omit-xml-declaration", oo.omitXmlDeclaration);
1.46      paf       529:                                // $.standalone[yes|no]
1.111     paf       530:                                param_option_over_output_option(*options, "standalone", oo.standalone);
1.46      paf       531:                                // $.indent[yes|no]
1.111     paf       532:                                param_option_over_output_option(*options, "indent", oo.indent);
1.46      paf       533:                                // $.media-type[text/{html|xml|plain}]
1.111     paf       534:                                param_option_over_output_option(*options, "media-type", oo.mediaType);                           
1.19      parser    535:                        }
1.1       parser    536:                }
                    537:        }
                    538: 
1.46      paf       539:        // default encoding from pool
1.61      paf       540:        if(!oo.encoding)
1.111     paf       541:                oo.encoding=new String(r.charsets.source().NAME(), String::L_TAINTED);
1.46      paf       542:        // default method=xml
                    543:        if(!oo.method)
1.111     paf       544:                oo.method=new String(XDOC_OUTPUT_METHOD_OPTION_VALUE_XML);
1.61      paf       545:        // default mediaType = depending on method
                    546:        if(!oo.mediaType) {
                    547:                if(*oo.method==XDOC_OUTPUT_METHOD_OPTION_VALUE_XML)
1.111     paf       548:                        oo.mediaType=new String("text/xml");
1.61      paf       549:                else if(*oo.method==XDOC_OUTPUT_METHOD_OPTION_VALUE_HTML)
1.111     paf       550:                        oo.mediaType=new String("text/html");
1.61      paf       551:                else // XDOC_OUTPUT_METHOD_OPTION_VALUE_TEXT & all others
1.111     paf       552:                        oo.mediaType=new String("text/plain");
1.61      paf       553:        }
                    554: }
1.46      paf       555: 
1.111     paf       556: struct Xdoc2buf_result {
                    557:        char* str;
                    558:        size_t length;
                    559: };
                    560: static Xdoc2buf_result xdoc2buf(Request& r, VXdoc& vdoc, 
                    561:                                         MethodParams& params, int index,
1.63      paf       562:                                         VXdoc::Output_options& oo,
1.140     paf       563:                                         const String* file_spec,
                    564:                                         bool use_source_charset_to_render_and_client_charset_to_write_to_header=false) {
1.111     paf       565:        Xdoc2buf_result result;
                    566:        prepare_output_options(r, params, index,
1.61      paf       567:                oo);
                    568: 
1.140     paf       569:        const char* render_encoding;
                    570:        const char* header_encoding;
                    571:        if(use_source_charset_to_render_and_client_charset_to_write_to_header) {
                    572:                render_encoding=r.charsets.source().NAME_CSTR();
                    573:                header_encoding=r.charsets.client().NAME_CSTR();
                    574:        } else {
                    575:                header_encoding=render_encoding=oo.encoding->cstr();
                    576:        }
                    577: 
                    578:        xmlCharEncodingHandler *renderer=xmlFindCharEncodingHandler(render_encoding);
                    579:        if(!renderer)
1.154   ! misha     580:                throw Exception(PARSER_RUNTIME,
1.111     paf       581:                        0,
1.140     paf       582:                        "encoding '%s' not supported", render_encoding);
                    583:        // UTF-8 renderer contains empty input/output converters, 
1.61      paf       584:        // which is wrong for xmlOutputBufferCreateIO
1.140     paf       585:        // while zero renderer goes perfectly 
                    586:        if(strcmp(render_encoding, "UTF-8")==0)
                    587:                renderer=0;
1.61      paf       588: 
1.140     paf       589:        xmlOutputBuffer_auto_ptr outputBuffer(xmlAllocOutputBuffer(renderer));
1.138     paf       590: 
1.65      paf       591:        xsltStylesheet_auto_ptr stylesheet(xsltNewStylesheet());
                    592:        if(!stylesheet.get())
1.87      paf       593:                throw Exception(0,
1.111     paf       594:                        0,
1.61      paf       595:                        "xsltNewStylesheet failed");
                    596: 
1.143     paf       597:        #define OOSTRING2STYLE(name) \
1.149     paf       598:                stylesheet->name=oo.name?BAD_CAST xmlMemStrdup((const char*)r.transcode(*oo.name)):0
1.143     paf       599:        #define OOBOOL2STYLE(name) \
                    600:                if(oo.name>=0) stylesheet->name=oo.name
1.61      paf       601: 
1.143     paf       602:        OOSTRING2STYLE(method);
                    603:        OOSTRING2STYLE(encoding);
                    604:        OOSTRING2STYLE(mediaType);
                    605: //     OOSTRING2STYLE(doctypeSystem);
                    606: //     OOSTRING2STYLE(doctypePublic);
                    607:        OOBOOL2STYLE(indent);
                    608:        OOSTRING2STYLE(version);
                    609:        OOBOOL2STYLE(standalone);
                    610:        OOBOOL2STYLE(omitXmlDeclaration);
1.61      paf       611: 
1.149     paf       612:        xmlDoc& xmldoc=vdoc.get_xmldoc();
                    613:        xmldoc.encoding=BAD_CAST xmlMemStrdup(render_encoding);
1.142     paf       614:        if(header_encoding)
                    615:                stylesheet->encoding=BAD_CAST xmlMemStrdup(header_encoding);
1.149     paf       616:        if(xsltSaveResultTo(outputBuffer.get(), &xmldoc, stylesheet.get())<0
                    617:                || xmlHaveGenericErrors())
                    618:                throw XmlException(0);
1.61      paf       619: 
1.138     paf       620:        // write out result
                    621:        char *gnome_str;  size_t gnome_length;
                    622:        if(outputBuffer->conv) {
                    623:                gnome_length=outputBuffer->conv->use;
                    624:                gnome_str=(char *)outputBuffer->conv->content;
                    625:        } else {
                    626:                gnome_length=outputBuffer->buffer->use;
                    627:                gnome_str=(char *)outputBuffer->buffer->content;
                    628:        }
                    629: 
                    630:        if((result.length=gnome_length)) {
                    631:                result.str=pa_strdup(gnome_str, gnome_length);
                    632:        } else
                    633:                result.str=0;
1.118     paf       634: 
1.63      paf       635:        if(file_spec)
1.111     paf       636:                file_write(*file_spec,
1.138     paf       637:                        gnome_str, gnome_length, 
1.111     paf       638:                        true/*as_text*/);
                    639: 
                    640:        return result;
1.1       parser    641: }
                    642: 
1.111     paf       643: static void _file(Request& r, MethodParams& params) {
                    644:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.63      paf       645:        VXdoc::Output_options oo(vdoc.output_options);
1.111     paf       646:        Xdoc2buf_result buf=xdoc2buf(r, vdoc, params, 0, 
1.63      paf       647:                oo,
1.111     paf       648:                0/*not to file, to memory*/);
1.63      paf       649:        // write out result
1.111     paf       650:        r.write_no_lang(String(buf.str, buf.length));
1.1       parser    651: 
1.58      paf       652:        // write out result
1.111     paf       653:        VFile& vfile=*new VFile;
                    654:        VHash& vhcontent_type=*new VHash;
                    655:        vhcontent_type.hash().put(
                    656:                value_name, 
                    657:                new VString(*oo.mediaType));
                    658:        vhcontent_type.hash().put(
1.126     paf       659:                String::Body("charset"), 
1.111     paf       660:                new VString(*oo.encoding));
                    661: 
                    662:        vfile.set(false/*tainted*/, buf.str?buf.str:""/*to distinguish from stat-ed file*/, buf.length, 
                    663:                0/*file_name*/, &vhcontent_type);
1.58      paf       664:        r.write_no_lang(vfile);
1.63      paf       665: }
                    666: 
1.111     paf       667: static void _save(Request& r, MethodParams& params) {
                    668:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.63      paf       669: 
1.111     paf       670:        const String& file_spec=r.absolute(params.as_string(0, "file name must be string"));
1.63      paf       671:        
                    672:        VXdoc::Output_options oo(vdoc.output_options);
1.111     paf       673:        xdoc2buf(r, vdoc, params, 1, 
1.63      paf       674:                oo,
1.111     paf       675:                &file_spec);
1.63      paf       676: }
                    677: 
1.111     paf       678: static void _string(Request& r, MethodParams& params) {
                    679:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.63      paf       680:        VXdoc::Output_options oo(vdoc.output_options);
1.111     paf       681:        Xdoc2buf_result buf=xdoc2buf(r, vdoc, params, 0, 
1.63      paf       682:                oo,
1.140     paf       683:                0/*not to file, to memory*/,
                    684:                true/*use source charset to render, client charset to put to header*/);
1.63      paf       685:        // write out result
1.145     paf       686:        r.write_no_lang(String(String::Body(buf.str), String::L_AS_IS));
1.1       parser    687: }
1.58      paf       688: 
1.111     paf       689: #ifndef DOXYGEN
                    690: struct Add_xslt_param_info {
                    691:        Request* r;
1.149     paf       692:        Array<const xmlChar*>* strings;
                    693:        const xmlChar** current_transform_param;
1.111     paf       694: };
                    695: #endif
                    696: static void add_xslt_param(
                    697:                           HashStringValue::key_type attribute, 
                    698:                           HashStringValue::value_type meaning, 
                    699:                           Add_xslt_param_info* info) {
1.149     paf       700:        xmlChar* s;
                    701:        *info->current_transform_param++=(s=info->r->transcode(attribute)); *info->strings+=s;
                    702:        *info->current_transform_param++=(s=info->r->transcode(meaning->as_string())); *info->strings+=s;
1.111     paf       703: }
1.132     paf       704: 
1.111     paf       705: static VXdoc& _transform(Request& r, const String* stylesheet_source, 
1.149     paf       706:                                                   VXdoc& vdoc, xsltStylesheetPtr stylesheet, const xmlChar** transform_params) 
1.132     paf       707: {
1.149     paf       708:        xmlDoc& xmldoc=vdoc.get_xmldoc();
                    709: 
1.65      paf       710:        xsltTransformContext_auto_ptr transformContext(
1.149     paf       711:                xsltNewTransformContext(stylesheet, &xmldoc));
1.90      paf       712:        // make params literal
1.111     paf       713:        if (transformContext->globalVars == NULL) // strangly not initialized by xsltNewTransformContext
1.90      paf       714:                transformContext->globalVars = xmlHashCreate(20);
1.149     paf       715:        xsltQuoteUserParams(transformContext.get(), (const char**)transform_params);
1.90      paf       716:        // do transform
1.59      paf       717:        xmlDoc *transformed=xsltApplyStylesheetUser(
                    718:                stylesheet,
1.149     paf       719:                &xmldoc,
1.90      paf       720:                0/*already quoted-inserted  transform_params*/,
1.111     paf       721:                0/*const char* output*/,
1.59      paf       722:                0/*FILE *profile*/,
1.65      paf       723:                transformContext.get());
1.149     paf       724:        if(!transformed || xmlHaveGenericErrors())
                    725:                throw XmlException(stylesheet_source);
1.42      paf       726: 
1.61      paf       727:        //gdome_xml_doc_mkref dislikes XML_HTML_DOCUMENT_NODE  type, fixing
                    728:        transformed->type=XML_DOCUMENT_NODE;
1.59      paf       729:        // constructing result
1.149     paf       730:        VXdoc& result=*new VXdoc(r.charsets, *transformed);
1.61      paf       731:        /* grabbing options
                    732: 
                    733:                <xsl:output
                    734:                !method = "xml" | "html" | "text"
                    735:                        X| qname-but-not-ncname 
                    736:                !version = nmtoken 
                    737:                !encoding = string 
                    738:                !omit-xml-declaration = "yes" | "no"
                    739:                !standalone = "yes" | "no"
                    740:                !doctype-public = string 
                    741:                !doctype-system = string 
                    742:                Xcdata-section-elements = qnames 
                    743:                !indent = "yes" | "no"
                    744:                !media-type = string /> 
1.59      paf       745:        */
                    746:        VXdoc::Output_options& oo=result.output_options;
                    747: 
1.111     paf       748:        oo.method=stylesheet->method?&r.transcode(stylesheet->method):0;
                    749:        oo.encoding=stylesheet->encoding?&r.transcode(stylesheet->encoding):0;
                    750:        oo.mediaType=stylesheet->mediaType?&r.transcode(stylesheet->mediaType):0;
1.143     paf       751:        oo.indent=stylesheet->indent;
1.111     paf       752:        oo.version=stylesheet->version?&r.transcode(stylesheet->version):0;
1.143     paf       753:        oo.standalone=stylesheet->standalone;
                    754:        oo.omitXmlDeclaration=stylesheet->omitXmlDeclaration;
1.59      paf       755: 
1.93      paf       756:        // return
                    757:        return result;
                    758: }
1.111     paf       759: static void _transform(Request& r, MethodParams& params) {
                    760:        VXdoc& vdoc=GET_SELF(r, VXdoc);
1.93      paf       761: 
                    762:        // params
1.149     paf       763:        Array<const xmlChar*> transform_strings;
                    764:        const xmlChar** transform_params=0;
1.111     paf       765:        if(params.count()>1) {
                    766:                Value& vparams=params.as_no_junction(1, "transform parameters must be hash");
1.106     paf       767:                if(!vparams.is_string())
1.111     paf       768:                        if(HashStringValue* hash=vparams.get_hash()) {
1.149     paf       769:                                transform_params=new(UseGC) const xmlChar*[hash->count()*2+1];
1.111     paf       770:                                Add_xslt_param_info info={
                    771:                                        &r, 
                    772:                                        &transform_strings,
                    773:                                        transform_params
                    774:                                };
1.150     paf       775:                                hash->for_each<Add_xslt_param_info*>(add_xslt_param, &info);
1.111     paf       776:                                transform_params[hash->count()*2]=0;                            
1.93      paf       777:                        } else
1.154   ! misha     778:                                throw Exception(PARSER_RUNTIME,
1.111     paf       779:                                        0,
1.93      paf       780:                                        "transform parameters parameter must be hash");
                    781:        }
                    782: 
1.111     paf       783:        VXdoc* result;
                    784:        if(Value *vxdoc=params[0].as(VXDOC_TYPE, false)) { // stylesheet (xdoc)
1.149     paf       785:                xmlDoc& stylesheetdoc=static_cast<VXdoc *>(vxdoc)->get_xmldoc();
1.93      paf       786:                // compile xdoc stylesheet
1.149     paf       787:                xsltStylesheet_auto_ptr stylesheet_ptr(xsltParseStylesheetDoc(&stylesheetdoc)); 
                    788:                if(xmlHaveGenericErrors())
                    789:                        throw XmlException(0);
1.93      paf       790:                if(!stylesheet_ptr.get())
                    791:                        throw Exception("xml",
1.111     paf       792:                                0,
1.93      paf       793:                                "stylesheet failed to compile");
1.128     paf       794:                // strange thing - xsltParseStylesheetDoc records document and destroys it in stylesheet destructor
                    795:                // we don't need that
                    796:                stylesheet_ptr->doc=0;
1.93      paf       797: 
                    798:                // transform!
1.111     paf       799:                result=&_transform(r, 0,
1.93      paf       800:                        vdoc, stylesheet_ptr.get(),
                    801:                        transform_params);
                    802:        } else { // stylesheet (file name)
                    803:                // extablish stylesheet connection
                    804:                const String& stylesheet_filespec=
1.111     paf       805:                        r.absolute(params.as_string(0, "stylesheet must be file name (string) or DOM document (xdoc)"));
1.125     paf       806:                Stylesheet_connection_ptr connection=stylesheet_manager->get_connection(stylesheet_filespec);
1.93      paf       807: 
                    808:                // load and compile file to stylesheet [or get cached if any]
                    809:                // transform!
1.122     paf       810:                result=&_transform(r, &stylesheet_filespec, vdoc, connection->stylesheet(),
1.93      paf       811:                        transform_params);
                    812:        }
                    813: 
1.59      paf       814:        // write out result
1.93      paf       815:        r.write_no_lang(*result);
1.1       parser    816: }
                    817: 
1.16      parser    818: // constructor
1.2       parser    819: 
1.92      paf       820: /// @test how to create empty type html?
1.111     paf       821: MXdoc::MXdoc(): MXnode(XDOC_CLASS_NAME, xnode_class) {
1.16      parser    822:        /// DOM1
1.2       parser    823: 
1.16      parser    824:        // Element createElement(in DOMString tagName) raises(DOMException);
                    825:        add_native_method("createElement", Method::CT_DYNAMIC, _createElement, 1, 1);
                    826:        // DocumentFragment createDocumentFragment(); 
                    827:        add_native_method("createDocumentFragment", Method::CT_DYNAMIC, _createDocumentFragment, 0, 0);
                    828:        // Text createTextNode(in DOMString data);
                    829:        add_native_method("createTextNode", Method::CT_DYNAMIC, _createTextNode, 1, 1);
                    830:        // Comment createComment(in DOMString data);
                    831:        add_native_method("createComment", Method::CT_DYNAMIC, _createComment, 1, 1);
                    832:        // CDATASection createCDATASection(in DOMString data) raises(DOMException);
                    833:        add_native_method("createCDATASection", Method::CT_DYNAMIC, _createCDATASection, 1, 1);
                    834:        // ProcessingInstruction createProcessingInstruction(in DOMString target, in DOMString data) raises(DOMException);
                    835:        add_native_method("createProcessingInstruction", Method::CT_DYNAMIC, _createProcessingInstruction, 2, 2);
                    836:        // Attr createAttribute(in DOMString name) raises(DOMException);
                    837:        add_native_method("createAttribute", Method::CT_DYNAMIC, _createAttribute, 1, 1);
                    838:        // EntityReference createEntityReference(in DOMString name) raises(DOMException);
                    839:        add_native_method("createEntityReference", Method::CT_DYNAMIC, _createEntityReference, 1, 1);
1.2       parser    840: 
1.79      paf       841:        /// DOM2
1.2       parser    842: 
1.16      parser    843:        // ^xdoc.getElementById[elementId]
                    844:        add_native_method("getElementById", Method::CT_DYNAMIC, _getElementById, 1, 1);
1.79      paf       845: 
                    846:     // Node (in Node importedNode, in boolean deep) raises(DOMException)
                    847:        add_native_method("importNode", Method::CT_DYNAMIC, _importNode, 2, 2);
1.112     paf       848: 
1.113     paf       849:        // Attr createAttributeNS(in DOMString namespaceURI, in DOMString qualifiedName) raises(DOMException);
1.112     paf       850:        add_native_method("createAttributeNS", Method::CT_DYNAMIC, _createAttributeNS, 2, 2);
                    851: 
1.113     paf       852:        // Element createElementNS(in DOMString namespaceURI, in DOMString qualifiedName) raises(DOMException);
1.112     paf       853:        add_native_method("createElementNS", Method::CT_DYNAMIC, _createElementNS, 2, 2);
1.99      paf       854: 
1.16      parser    855:        /// parser
                    856:        
1.54      paf       857:        // ^xdoc::create{qualifiedName}
1.95      paf       858:        // ^xdoc::create[<some>xml</some>]
                    859:        // ^xdoc::create[URI][<some>xml</some>]
                    860:        add_native_method("create", Method::CT_DYNAMIC, _create, 1, 2); 
1.83      paf       861:        // for backward compatibility with <=v 1.82 2002/01/31 11:51:46 paf
                    862:        add_native_method("set", Method::CT_DYNAMIC, _create, 1, 1);
1.54      paf       863: 
                    864:        // ^xdoc::load[some.xml]
1.122     paf       865:        add_native_method("load", Method::CT_DYNAMIC, _load, 1, 1);
1.54      paf       866: 
1.2       parser    867:        // ^xdoc.save[some.xml]
                    868:        // ^xdoc.save[some.xml;options hash]
1.1       parser    869:        add_native_method("save", Method::CT_DYNAMIC, _save, 1, 2);
                    870: 
1.2       parser    871:        // ^xdoc.string[] <doc/>
                    872:        // ^xdoc.string[options hash] <doc/>
1.1       parser    873:        add_native_method("string", Method::CT_DYNAMIC, _string, 0, 1);
                    874: 
1.2       parser    875:        // ^xdoc.file[] file with "<doc/>"
                    876:        // ^xdoc.file[options hash] file with "<doc/>"
1.58      paf       877:        add_native_method("file", Method::CT_DYNAMIC, _file, 0, 1);
1.1       parser    878: 
1.98      paf       879:        // ^xdoc.transform[stylesheet file_name/xdoc]
                    880:        // ^xdoc.transform[stylesheet file_name/xdoc;params hash]
1.59      paf       881:        add_native_method("transform", Method::CT_DYNAMIC, _transform, 1, 2);
1.2       parser    882: 
1.5       parser    883: }
                    884: 
1.111     paf       885: # else
                    886: 
                    887: #include "classes.h"
                    888: 
1.1       parser    889: // global variable
                    890: 
1.111     paf       891: DECLARE_CLASS_VAR(xdoc, 0, 0); // fictive
1.1       parser    892: 
                    893: #endif

E-mail: