//%LICENSE//////////////////////////////////////////////////////////////// // // Licensed to The Open Group (TOG) under one or more contributor license // agreements. Refer to the OpenPegasusNOTICE.txt file distributed with // this work for additional information regarding copyright ownership. // Each contributor licenses this file to you under the OpenPegasus Open // Source License; you may not use this file except in compliance with the // License. // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. // IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY // CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, // TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE // SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // ////////////////////////////////////////////////////////////////////////// // //%///////////////////////////////////////////////////////////////////////////// #include #include #include #include "Constants.h" #include "CIMClass.h" #include "CIMClassRep.h" #include "CIMInstance.h" #include "CIMInstanceRep.h" #include "CIMProperty.h" #include "CIMPropertyRep.h" #include "CIMMethod.h" #include "CIMMethodRep.h" #include "CIMParameter.h" #include "CIMParameterRep.h" #include "CIMParamValue.h" #include "CIMParamValueRep.h" #include "CIMQualifier.h" #include "CIMQualifierRep.h" #include "CIMQualifierDecl.h" #include "CIMQualifierDeclRep.h" #include "CIMValue.h" #include "XmlWriter.h" #include "Tracer.h" #include #include "CommonUTF.h" #include "Buffer.h" #include "StrLit.h" #include "IDFactory.h" #include "StringConversion.h" PEGASUS_NAMESPACE_BEGIN //------------------------------------------------------------------------------ // // appendLocalNameSpacePathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendLocalNameSpacePathElement( Buffer& out, const CIMNamespaceName& nameSpace) { out << STRLIT("\n"); char* nameSpaceCopy = strdup(nameSpace.getString().getCString()); #if !defined(PEGASUS_COMPILER_MSVC) && !defined(PEGASUS_OS_ZOS) char *last; for (const char* p = strtok_r(nameSpaceCopy, "/", &last); p; p = strtok_r(NULL, "/", &last)) #else for (const char* p = strtok(nameSpaceCopy, "/"); p; p = strtok(NULL, "/")) #endif { out << STRLIT("\n"); } free(nameSpaceCopy); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendNameSpacePathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendNameSpacePathElement( Buffer& out, const String& host, const CIMNamespaceName& nameSpace) { out << STRLIT("\n" "") << host << STRLIT("\n"); appendLocalNameSpacePathElement(out, nameSpace); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendClassNameElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendClassNameElement( Buffer& out, const CIMName& className) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendInstanceNameElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendInstanceNameElement( Buffer& out, const CIMObjectPath& instanceName) { out << STRLIT("\n"); const Array& keyBindings = instanceName.getKeyBindings(); for (Uint32 i = 0, n = keyBindings.size(); i < n; i++) { out << STRLIT("\n"); if (keyBindings[i].getType() == CIMKeyBinding::REFERENCE) { CIMObjectPath ref = keyBindings[i].getValue(); // create an instancePath (i.e. isClassPath = false) appendValueReferenceElement(out, ref, false, true); } else { out << STRLIT(""); // fixed the special characters appendSpecial(out, keyBindings[i].getValue()); out << STRLIT("\n"); } out << STRLIT("\n"); } out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendClassPathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendClassPathElement( Buffer& out, const CIMObjectPath& classPath) { out << STRLIT("\n"); appendNameSpacePathElement(out, classPath.getHost(), classPath.getNameSpace()); appendClassNameElement(out, classPath.getClassName()); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendInstancePathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendInstancePathElement( Buffer& out, const CIMObjectPath& instancePath) { out << STRLIT("\n"); appendNameSpacePathElement(out, instancePath.getHost(), instancePath.getNameSpace()); appendInstanceNameElement(out, instancePath); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendLocalClassPathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendLocalClassPathElement( Buffer& out, const CIMObjectPath& classPath) { out << STRLIT("\n"); appendLocalNameSpacePathElement(out, classPath.getNameSpace()); appendClassNameElement(out, classPath.getClassName()); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendLocalInstancePathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendLocalInstancePathElement( Buffer& out, const CIMObjectPath& instancePath) { out << STRLIT("\n"); appendLocalNameSpacePathElement(out, instancePath.getNameSpace()); appendInstanceNameElement(out, instancePath); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendLocalObjectPathElement() // // If the reference refers to an instance, write a LOCALINSTANCEPATH; // otherwise write a LOCALCLASSPATH. // //------------------------------------------------------------------------------ // appendValueReferenceElement does this correctly with isClassPath flag // Called only but formatSimple void XmlWriter::appendLocalObjectPathElement( Buffer& out, const CIMObjectPath& objectPath) { // // ATTN-CAKG-P2-20020726: The following condition does not correctly // distinguish instanceNames from classNames in every case // The instanceName of a singleton instance of a keyless class has no // key bindings // See bBUG_3302. // if (objectPath.getKeyBindings ().size () != 0) { appendLocalInstancePathElement(out, objectPath); } else { appendLocalClassPathElement(out, objectPath); } } //------------------------------------------------------------------------------ // // Helper functions for appendValueElement() // //------------------------------------------------------------------------------ inline void _xmlWritter_appendValue(Buffer& out, Boolean x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, Uint8 x) { XmlWriter::append(out, Uint32(x)); } inline void _xmlWritter_appendValue(Buffer& out, Sint8 x) { XmlWriter::append(out, Sint32(x)); } inline void _xmlWritter_appendValue(Buffer& out, Uint16 x) { XmlWriter::append(out, Uint32(x)); } inline void _xmlWritter_appendValue(Buffer& out, Sint16 x) { XmlWriter::append(out, Sint32(x)); } inline void _xmlWritter_appendValue(Buffer& out, Uint32 x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, Sint32 x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, Uint64 x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, Sint64 x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, Real32 x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, Real64 x) { XmlWriter::append(out, x); } inline void _xmlWritter_appendValue(Buffer& out, const Char16& x) { XmlWriter::appendSpecial(out, x); } inline void _xmlWritter_appendValue(Buffer& out, const String& x) { XmlWriter::appendSpecial(out, x); } inline void _xmlWritter_appendValue(Buffer& out, const CIMDateTime& x) { // It is not necessary to use XmlWriter::appendSpecial(), because // CIMDateTime values do not contain special characters. out << x.toString(); } inline void _xmlWritter_appendValue(Buffer& out, const CIMObjectPath& x) { // Emit Instance Path with value wrapper XmlWriter::appendValueReferenceElement(out, x, false, true); } inline void _xmlWritter_appendValue(Buffer& out, const CIMObject& x) { String myStr = x.toString(); _xmlWritter_appendValue(out, myStr); } void _xmlWritter_appendValueArray( Buffer& out, const CIMObjectPath* p, Uint32 size) { out << STRLIT("\n"); while (size--) { _xmlWritter_appendValue(out, *p++); } out << STRLIT("\n"); } template void _xmlWritter_appendValueArray(Buffer& out, const T* p, Uint32 size) { out << STRLIT("\n"); while (size--) { out << STRLIT(""); _xmlWritter_appendValue(out, *p++); out << STRLIT("\n"); } out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendValueElement() // // // // // // //------------------------------------------------------------------------------ void XmlWriter::appendValueElement( Buffer& out, const CIMValue& value) { if (value.isNull()) { return; } if (value.isArray()) { switch (value.getType()) { case CIMTYPE_BOOLEAN: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_UINT8: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_SINT8: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_UINT16: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_SINT16: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_UINT32: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_SINT32: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_UINT64: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_SINT64: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_REAL32: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_REAL64: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_CHAR16: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_STRING: { const String* data; Uint32 size; value._get(data, size); _xmlWritter_appendValueArray(out, data, size); break; } case CIMTYPE_DATETIME: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_REFERENCE: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_OBJECT: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } case CIMTYPE_INSTANCE: { Array a; value.get(a); _xmlWritter_appendValueArray(out, a.getData(), a.size()); break; } default: PEGASUS_UNREACHABLE(PEGASUS_ASSERT(false);) } } else if (value.getType() == CIMTYPE_REFERENCE) { // Has to be separate because it uses VALUE.REFERENCE tag CIMObjectPath v; value.get(v); _xmlWritter_appendValue(out, v); } else { out << STRLIT(""); switch (value.getType()) { case CIMTYPE_BOOLEAN: { Boolean v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_UINT8: { Uint8 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_SINT8: { Sint8 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_UINT16: { Uint16 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_SINT16: { Sint16 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_UINT32: { Uint32 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_SINT32: { Sint32 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_UINT64: { Uint64 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_SINT64: { Sint64 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_REAL32: { Real32 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_REAL64: { Real64 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_CHAR16: { Char16 v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_STRING: { String v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_DATETIME: { CIMDateTime v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_OBJECT: { CIMObject v; value.get(v); _xmlWritter_appendValue(out, v); break; } case CIMTYPE_INSTANCE: { CIMInstance v; value.get(v); _xmlWritter_appendValue(out, v); break; } default: PEGASUS_UNREACHABLE(PEGASUS_ASSERT(false);) } out << STRLIT("\n"); } } void XmlWriter::printValueElement( const CIMValue& value, PEGASUS_STD(ostream)& os) { Buffer tmp; appendValueElement(tmp, value); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendValueObjectWithPathElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendValueObjectWithPathElement( Buffer& out, const CIMObject& objectWithPath, Boolean includeQualifiers, Boolean includeClassOrigin, Boolean isClassObject, const CIMPropertyList& propertyList) { out << STRLIT("\n"); appendValueReferenceElement(out, objectWithPath.getPath (), isClassObject , false); appendObjectElement( out, objectWithPath, includeQualifiers, includeClassOrigin, propertyList); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendValueReferenceElement() // // // //------------------------------------------------------------------------------ // appends INSTANCEPATH | LOCALINSTANCEPATH | INSTANCENAME void XmlWriter::appendValueInstancePathElement( Buffer& out, const CIMObjectPath& reference) { if (reference.getHost().size()) { appendInstancePathElement(out, reference); } else if (!reference.getNameSpace().isNull()) { appendLocalInstancePathElement(out, reference); } else { appendInstanceNameElement(out, reference); } } // appends CLASSPATH | LOCALCLASSPATH | CLASSNAME void XmlWriter::appendValueClassPathElement( Buffer& out, const CIMObjectPath& reference) { if (reference.getHost().size()) { appendClassPathElement(out, reference); } else if (!reference.getNameSpace().isNull()) { appendLocalClassPathElement(out, reference); } else { appendClassNameElement(out, reference.getClassName()); } } // Builds either a classPath or InstancePath based on isClassPath // parameter which was carried forward from, for example, the // request. If wrapper true, wrap output with void XmlWriter::appendValueReferenceElement( Buffer& out, const CIMObjectPath& reference, Boolean isClassPath, Boolean addValueWrapper) { if (addValueWrapper) { out << STRLIT("\n"); } if (isClassPath) { appendValueClassPathElement(out,reference); } else { appendValueInstancePathElement(out,reference); } if (addValueWrapper) { out << STRLIT("\n"); } } void XmlWriter::printValueReferenceElement( const CIMObjectPath& reference, Boolean isClassPath, PEGASUS_STD(ostream)& os) { Buffer tmp; appendValueReferenceElement(tmp, reference, isClassPath, true); indentedPrint(os, tmp.getData()); } //------------------------------------------------------------------------------ // // appendValueNamedInstanceElement() // // // //------------------------------------------------------------------------------ void XmlWriter::appendValueNamedInstanceElement( Buffer& out, const CIMInstance& namedInstance, Boolean includeQualifiers, Boolean includeClassOrigin, const CIMPropertyList& propertyList) { out << STRLIT("\n"); appendInstanceNameElement(out, namedInstance.getPath ()); appendInstanceElement( out, namedInstance, includeQualifiers, includeClassOrigin, propertyList); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendClassElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendClassElement( Buffer& out, const CIMConstClass& cimClass) { CheckRep(cimClass._rep); const CIMClassRep* rep = cimClass._rep; // Class opening element: out << STRLIT("getClassName() << STRLIT("\" "); if (!rep->getSuperClassName().isNull()) { out << STRLIT(" SUPERCLASS=\"") << rep->getSuperClassName() << STRLIT("\" "); } out << STRLIT(">\n"); // Append Class Qualifiers: for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); // Append Property definitions: for (Uint32 i = 0, n = rep->getPropertyCount(); i < n; i++) XmlWriter::appendPropertyElement(out, rep->getProperty(i)); // Append Method definitions: for (Uint32 i = 0, n = rep->getMethodCount(); i < n; i++) XmlWriter::appendMethodElement(out, rep->getMethod(i)); // Class closing element: out << STRLIT("\n"); } void XmlWriter::printClassElement( const CIMConstClass& cimclass, PEGASUS_STD(ostream)& os) { Buffer tmp; appendClassElement(tmp, cimclass); indentedPrint(os, tmp.getData(), 4); } //------------------------------------------------------------------------------ // // appendInstanceElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendInstanceElement( Buffer& out, const CIMConstInstance& instance, Boolean includeQualifiers, Boolean includeClassOrigin, const CIMPropertyList& propertyList) { CheckRep(instance._rep); const CIMInstanceRep* rep = instance._rep; // Class opening element: out << STRLIT("getClassName() << STRLIT("\" >\n"); // Append Instance Qualifiers: if(includeQualifiers) { for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); } if(propertyList.isNull()) { for (Uint32 i = 0, n = rep->getPropertyCount(); i < n; i++) { XmlWriter::appendPropertyElement( out, rep->getProperty(i), includeQualifiers,includeClassOrigin); } } else { for (Uint32 i = 0, n = propertyList.size(); i < n; i++) { CIMName name = propertyList[i]; Uint32 pos = rep->_properties.find( propertyList[i], propertyList.getCIMNameTag(i)); if(pos != PEG_NOT_FOUND) { PEG_TRACE((TRC_XML,Tracer::LEVEL4, "XmlWriter::appendInstanceElement" " Filtering the property name:%s for the className:%s" "since it was not filtered by the provider.", (const char *)name.getString().getCString(), (const char *)instance.getClassName(). getString().getCString())); XmlWriter::appendPropertyElement( out, rep->getProperty(pos), includeQualifiers,includeClassOrigin); } } } // Instance closing element: out << STRLIT("\n"); } void XmlWriter::printInstanceElement( const CIMConstInstance& instance, PEGASUS_STD(ostream)& os) { Buffer tmp; appendInstanceElement(tmp, instance); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendObjectElement() // // May refer to a CLASS or an INSTANCE // //------------------------------------------------------------------------------ void XmlWriter::appendObjectElement( Buffer& out, const CIMConstObject& object, Boolean includeQualifiers, Boolean includeClassOrigin, const CIMPropertyList& propertyList) { if (object.isClass()) { CIMConstClass c(object); appendClassElement(out, c); } else if (object.isInstance()) { CIMConstInstance i(object); appendInstanceElement( out, i, includeQualifiers, includeClassOrigin, propertyList); } // else PEGASUS_ASSERT(0); } //------------------------------------------------------------------------------ // // appendPropertyElement() // // // // // // // // // // //------------------------------------------------------------------------------ void XmlWriter::appendPropertyElement( Buffer& out, const CIMConstProperty& property, Boolean includeQualifiers, Boolean includeClassOrigin) { CheckRep(property._rep); const CIMPropertyRep* rep = property._rep; if (rep->getValue().isArray()) { out << STRLIT("getName() << STRLIT("\" "); if (rep->getValue().getType() == CIMTYPE_OBJECT) { // If the property array type is CIMObject, then // encode the property in CIM-XML as a string array with the // EmbeddedObject attribute (there is not currently a CIM-XML // "object" datatype) Array a; rep->getValue().get(a); out << STRLIT(" TYPE=\"string\""); // If the Embedded Object is an instance, always add the // EmbeddedObject attribute. if (a.size() > 0 && a[0].isInstance()) { out << STRLIT(" EmbeddedObject=\"object\"" " EMBEDDEDOBJECT=\"object\""); } #ifndef PEGASUS_SNIA_INTEROP_COMPATIBILITY else #endif { // Else the Embedded Object is a class, always add the // EmbeddedObject qualifier. Note that if the macro // PEGASUS_SNIA_INTEROP_COMPATIBILITY is defined, then // the EmbeddedObject qualifier will always be added, // whether it's a class or an instance. if (rep->findQualifier(PEGASUS_QUALIFIERNAME_EMBEDDEDOBJECT) == PEG_NOT_FOUND) { // Note that addQualifiers() cannot be called on a const // CIMQualifierRep. In this case we really do want to add // the EmbeddedObject qualifier, so we cast away the // constness. CIMPropertyRep* tmpRep = const_cast(rep); tmpRep->addQualifier( CIMQualifier(PEGASUS_QUALIFIERNAME_EMBEDDEDOBJECT, true)); } } } else if (rep->getValue().getType() == CIMTYPE_INSTANCE) { // If the property array type is CIMInstance, then // encode the property in CIM-XML as a string array with the // EmbeddedObject attribute (there is not currently a CIM-XML // "instance" datatype) Array a; rep->getValue().get(a); out << STRLIT(" TYPE=\"string\""); // add the EmbeddedObject attribute if (a.size() > 0) { out << STRLIT(" EmbeddedObject=\"instance\"" " EMBEDDEDOBJECT=\"instance\""); // Note that if the macro PEGASUS_SNIA_INTEROP_COMPATIBILITY is // defined, then the EmbeddedInstance qualifier will be added # ifdef PEGASUS_SNIA_INTEROP_COMPATIBILITY if (rep->findQualifier(PEGASUS_QUALIFIERNAME_EMBEDDEDINSTANCE) == PEG_NOT_FOUND) { // Note that addQualifiers() cannot be called on a const // CIMQualifierRep. In this case we really do want to add // the EmbeddedInstance qualifier, so we cast away the // constness. // For now, we assume that all the embedded instances in // the array are of the same type CIMPropertyRep* tmpRep = const_cast(rep); tmpRep->addQualifier(CIMQualifier( PEGASUS_QUALIFIERNAME_EMBEDDEDINSTANCE, a[0].getClassName().getString())); } # endif } } else { out.append(' '); out << xmlWriterTypeStrings(rep->getValue().getType()); } if (rep->getArraySize()) { char buffer[32]; sprintf(buffer, "%u", rep->getArraySize()); out << STRLIT(" ARRAYSIZE=\"") << buffer; out.append('"'); } if(includeClassOrigin && !rep->getClassOrigin().isNull()) { out << STRLIT(" CLASSORIGIN=\"") << rep->getClassOrigin(); out.append('"'); } if (rep->getPropagated()) { out << STRLIT(" PROPAGATED=\"true\""); } out << STRLIT(">\n"); if(includeQualifiers) { for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); } XmlWriter::appendValueElement(out, rep->getValue()); out << STRLIT("\n"); } else if (rep->getValue().getType() == CIMTYPE_REFERENCE) { out << STRLIT("getName() << STRLIT("\" "); if (!rep->getReferenceClassName().isNull()) { out << STRLIT(" REFERENCECLASS=\"") << rep->getReferenceClassName(); out.append('"'); } if(includeClassOrigin && !rep->getClassOrigin().isNull()) { out << STRLIT(" CLASSORIGIN=\"") << rep->getClassOrigin(); out.append('"'); } if (rep->getPropagated()) { out << STRLIT(" PROPAGATED=\"true\""); } out << STRLIT(">\n"); if(includeQualifiers) { for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); } XmlWriter::appendValueElement(out, rep->getValue()); out << STRLIT("\n"); } else { out << STRLIT("getName() << STRLIT("\" "); if(includeClassOrigin && !rep->getClassOrigin().isNull()) { out << STRLIT(" CLASSORIGIN=\"") << rep->getClassOrigin(); out.append('"'); } if (rep->getPropagated()) { out << STRLIT(" PROPAGATED=\"true\""); } if (rep->getValue().getType() == CIMTYPE_OBJECT) { // If the property type is CIMObject, then // encode the property in CIM-XML as a string with the // EmbeddedObject attribute (there is not currently a CIM-XML // "object" datatype) CIMObject a; rep->getValue().get(a); out << STRLIT(" TYPE=\"string\""); // If the Embedded Object is an instance, always add the // EmbeddedObject attribute. if (a.isInstance()) { out << STRLIT(" EmbeddedObject=\"object\"" " EMBEDDEDOBJECT=\"object\""); } // Else the Embedded Object is a class, always add the // EmbeddedObject qualifier. #ifndef PEGASUS_SNIA_INTEROP_COMPATIBILITY else #endif { // Note that if the macro PEGASUS_SNIA_INTEROP_COMPATIBILITY // is defined, then the EmbeddedObject qualifier will always // be added, whether it's a class or an instance. if (rep->findQualifier(PEGASUS_QUALIFIERNAME_EMBEDDEDOBJECT) == PEG_NOT_FOUND) { // Note that addQualifiers() cannot be called on a const // CIMQualifierRep. In this case we really do want to add // the EmbeddedObject qualifier, so we cast away the // constness. CIMPropertyRep* tmpRep = const_cast(rep); tmpRep->addQualifier( CIMQualifier(PEGASUS_QUALIFIERNAME_EMBEDDEDOBJECT, true)); } } } else if (rep->getValue().getType() == CIMTYPE_INSTANCE) { CIMInstance a; rep->getValue().get(a); out << STRLIT(" TYPE=\"string\"" " EmbeddedObject=\"instance\"" " EMBEDDEDOBJECT=\"instance\""); # ifdef PEGASUS_SNIA_INTEROP_COMPATIBILITY if (rep->findQualifier(PEGASUS_QUALIFIERNAME_EMBEDDEDOBJECT) == PEG_NOT_FOUND) { // Note that addQualifiers() cannot be called on a const // CIMQualifierRep. In this case we really do want to add // the EmbeddedInstance qualifier, so we cast away the // constness. CIMPropertyRep* tmpRep = const_cast(rep); tmpRep->addQualifier(CIMQualifier( PEGASUS_QUALIFIERNAME_EMBEDDEDINSTANCE, a.getClassName().getString())); } # endif } else { out.append(' '); out << xmlWriterTypeStrings(rep->getValue().getType()); } out << STRLIT(">\n"); if(includeQualifiers) { for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); } XmlWriter::appendValueElement(out, rep->getValue()); out << STRLIT("\n"); } } void XmlWriter::printPropertyElement( const CIMConstProperty& property, PEGASUS_STD(ostream)& os) { Buffer tmp; appendPropertyElement(tmp, property); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendMethodElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendMethodElement( Buffer& out, const CIMConstMethod& method) { CheckRep(method._rep); const CIMMethodRep* rep = method._rep; out << STRLIT("getName(); out << STRLIT("\" "); out << xmlWriterTypeStrings(rep->getType()); if (!rep->getClassOrigin().isNull()) { out << STRLIT(" CLASSORIGIN=\"") << rep->getClassOrigin(); out.append('"'); } if (rep->getPropagated()) { out << STRLIT(" PROPAGATED=\"true\""); } out << STRLIT(">\n"); for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); for (Uint32 i = 0, n = rep->getParameterCount(); i < n; i++) XmlWriter::appendParameterElement(out, rep->getParameter(i)); out << STRLIT("\n"); } void XmlWriter::printMethodElement( const CIMConstMethod& method, PEGASUS_STD(ostream)& os) { Buffer tmp; appendMethodElement(tmp, method); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendParameterElement() // // // // // // // // // // // // // //------------------------------------------------------------------------------ void XmlWriter::appendParameterElement( Buffer& out, const CIMConstParameter& parameter) { CheckRep(parameter._rep); const CIMParameterRep* rep = parameter._rep; if (rep->isArray()) { if (rep->getType() == CIMTYPE_REFERENCE) { out << STRLIT("getName(); out.append('"'); if (!rep->getReferenceClassName().isNull()) { out << STRLIT(" REFERENCECLASS=\""); out << rep->getReferenceClassName().getString(); out.append('"'); } if (rep->getArraySize()) { char buffer[32]; int n = sprintf(buffer, "%u", rep->getArraySize()); out << STRLIT(" ARRAYSIZE=\""); out.append(buffer, n); out.append('"'); } out << STRLIT(">\n"); for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); out << STRLIT("\n"); } else { out << STRLIT("getName(); out << STRLIT("\" ") << xmlWriterTypeStrings(rep->getType()); if (rep->getArraySize()) { char buffer[32]; sprintf(buffer, "%u", rep->getArraySize()); out << STRLIT(" ARRAYSIZE=\"") << buffer; out.append('"'); } out << STRLIT(">\n"); for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); out << STRLIT("\n"); } } else if (rep->getType() == CIMTYPE_REFERENCE) { out << STRLIT("getName(); out.append('"'); if (!rep->getReferenceClassName().isNull()) { out << STRLIT(" REFERENCECLASS=\""); out << rep->getReferenceClassName().getString(); out.append('"'); } out << STRLIT(">\n"); for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); out << STRLIT("\n"); } else { out << STRLIT("getName(); out << STRLIT("\" ") << xmlWriterTypeStrings(rep->getType()); out << STRLIT(">\n"); for (Uint32 i = 0, n = rep->getQualifierCount(); i < n; i++) XmlWriter::appendQualifierElement(out, rep->getQualifier(i)); out << STRLIT("\n"); } } void XmlWriter::printParameterElement( const CIMConstParameter& parameter, PEGASUS_STD(ostream)& os) { Buffer tmp; appendParameterElement(tmp, parameter); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendParamValueElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendParamValueElement( Buffer& out, const CIMParamValue& paramValue) { CheckRep(paramValue._rep); const CIMParamValueRep* rep = paramValue._rep; out << STRLIT("getParameterName(); out.append('"'); CIMType type = rep->getValue().getType(); if (rep->isTyped()) { XmlWriter::appendParamTypeAndEmbeddedObjAttrib(out, type); } out << STRLIT(">\n"); XmlWriter::appendValueElement(out, rep->getValue()); out << STRLIT("\n"); } void XmlWriter::printParamValueElement( const CIMParamValue& paramValue, PEGASUS_STD(ostream)& os) { Buffer tmp; appendParamValueElement(tmp, paramValue); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendQualifierElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendQualifierElement( Buffer& out, const CIMConstQualifier& qualifier) { CheckRep(qualifier._rep); const CIMQualifierRep* rep = qualifier._rep; out << STRLIT("getName(); out << STRLIT("\" ") << xmlWriterTypeStrings(rep->getValue().getType()); if (rep->getPropagated()) { out << STRLIT(" PROPAGATED=\"true\""); } XmlWriter::appendQualifierFlavorEntity(out, rep->getFlavor()); out << STRLIT(">\n"); XmlWriter::appendValueElement(out, rep->getValue()); out << STRLIT("\n"); } void XmlWriter::printQualifierElement( const CIMConstQualifier& qualifier, PEGASUS_STD(ostream)& os) { Buffer tmp; appendQualifierElement(tmp, qualifier); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendQualifierDeclElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendQualifierDeclElement( Buffer& out, const CIMConstQualifierDecl& qualifierDecl) { CheckRep(qualifierDecl._rep); const CIMQualifierDeclRep* rep = qualifierDecl._rep; out << STRLIT("getName(); out << STRLIT("\" ") << xmlWriterTypeStrings(rep->getValue().getType()); if (rep->getValue().isArray()) { out << STRLIT(" ISARRAY=\"true\""); if (rep->getArraySize()) { char buffer[64]; int n = sprintf(buffer, " ARRAYSIZE=\"%u\"", rep->getArraySize()); out.append(buffer, n); } } XmlWriter::appendQualifierFlavorEntity(out, rep->getFlavor()); out << STRLIT(">\n"); XmlWriter::appendScopeElement(out, rep->getScope()); XmlWriter::appendValueElement(out, rep->getValue()); out << STRLIT("\n"); } void XmlWriter::printQualifierDeclElement( const CIMConstQualifierDecl& qualifierDecl, PEGASUS_STD(ostream)& os) { Buffer tmp; appendQualifierDeclElement(tmp, qualifierDecl); os << tmp.getData() << PEGASUS_STD(endl); } //------------------------------------------------------------------------------ // // appendQualifierFlavorEntity() // // // // DEPRECATION NOTE: The attribute TOINSTANCE is DEPRECATED and MAY be // removed from the QualifierFlavor entity in a future version of this // document. Use of this qualifier is discouraged. // //------------------------------------------------------------------------------ void XmlWriter::appendQualifierFlavorEntity( Buffer& out, const CIMFlavor & flavor) { if (!(flavor.hasFlavor (CIMFlavor::OVERRIDABLE))) out << STRLIT(" OVERRIDABLE=\"false\""); if (!(flavor.hasFlavor (CIMFlavor::TOSUBCLASS))) out << STRLIT(" TOSUBCLASS=\"false\""); //if (flavor.hasFlavor (CIMFlavor::TOINSTANCE)) // out << STRLIT(" TOINSTANCE=\"true\""); if (flavor.hasFlavor (CIMFlavor::TRANSLATABLE)) out << STRLIT(" TRANSLATABLE=\"true\""); } //------------------------------------------------------------------------------ // // appendScopeElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendScopeElement( Buffer& out, const CIMScope & scope) { if (!(scope.equal (CIMScope ()))) { out << STRLIT(""); } } // l10n - added content language and accept language support to // the header methods below //------------------------------------------------------------------------------ // // appendMethodCallHeader() // // Build HTTP method call request header. // //------------------------------------------------------------------------------ void XmlWriter::appendMethodCallHeader( Buffer& out, const char* host, const CIMName& cimMethod, const String& cimObject, const String& authenticationHeader, HttpMethod httpMethod, const AcceptLanguageList& acceptLanguages, const ContentLanguageList& contentLanguages, Uint32 contentLength, bool binaryRequest, bool binaryResponse) { char nn[] = { char('0' + (rand() % 10)), char('0' + (rand() % 10)),'\0'}; // ATTN: KS 20020926 - Temporary change to issue only POST. This may // be changed in the DMTF CIM Operations standard in the future. // If we kept M-Post we would have to retry with Post. Does not // do that in client today. Permanent change is to retry until spec // updated. This change is temp to finish tests or until the retry // installed. Required because of change to wbemservices cimom if (httpMethod == HTTP_METHOD_M_POST) { out << STRLIT("M-POST /cimom HTTP/1.1\r\n"); } else { out << STRLIT("POST /cimom HTTP/1.1\r\n"); } out << STRLIT("HOST: ") << host << STRLIT("\r\n"); if (binaryRequest) { // Tell the server that the payload is encoded in the OpenPegasus // binary protocol. out << STRLIT("Content-Type: application/x-openpegasus\r\n"); } else { out << STRLIT("Content-Type: application/xml; charset=utf-8\r\n"); } if (binaryResponse) { // Tell the server that this client accepts the OpenPegasus binary // protocol. out << STRLIT("Accept: application/x-openpegasus\r\n"); } OUTPUT_CONTENTLENGTH(out, contentLength); if (acceptLanguages.size() > 0) { out << STRLIT("Accept-Language: ") << acceptLanguages << STRLIT("\r\n"); } if (contentLanguages.size() > 0) { out << STRLIT("Content-Language: ") << contentLanguages << STRLIT("\r\n"); } #ifdef PEGASUS_DEBUG // backdoor environment variable to turn OFF client requesting transfer // encoding. The default is on. to turn off, set this variable to zero. // This should be removed when stable. This should only be turned off in // a debugging/testing environment. static const char *clientTransferEncodingOff = getenv("PEGASUS_HTTP_TRANSFER_ENCODING_REQUEST"); if (!clientTransferEncodingOff || *clientTransferEncodingOff != '0') #endif if (!binaryResponse) { // The binary protocol does not allow chunking. out << STRLIT("TE: chunked, trailers\r\n"); } if (httpMethod == HTTP_METHOD_M_POST) { out << STRLIT("Man: http://www.dmtf.org/cim/mapping/http/v1.0; ns="); out << nn << STRLIT("\r\n"); out << nn << STRLIT("-CIMOperation: MethodCall\r\n"); out << nn << STRLIT("-CIMMethod: ") << encodeURICharacters(cimMethod.getString()) << STRLIT("\r\n"); out << nn << STRLIT("-CIMObject: ") << encodeURICharacters(cimObject) << STRLIT("\r\n"); } else { out << STRLIT("CIMOperation: MethodCall\r\n"); out << STRLIT("CIMMethod: ") << encodeURICharacters(cimMethod.getString()) << STRLIT("\r\n"); out << STRLIT("CIMObject: ") << encodeURICharacters(cimObject) << STRLIT("\r\n"); } if (authenticationHeader.size()) { out << authenticationHeader << STRLIT("\r\n"); } out << STRLIT("\r\n"); } void XmlWriter::appendMethodResponseHeader( Buffer& out, HttpMethod httpMethod, const ContentLanguageList& contentLanguages, Uint32 contentLength, Uint64 serverResponseTime, bool binaryResponse) { // Optimize the typical case for binary messages, circumventing the // more expensive logic below. if (binaryResponse && contentLength == 0 && httpMethod != HTTP_METHOD_M_POST && contentLanguages.size() == 0) { static const char HEADERS[] = "HTTP/1.1 200 OK\r\n" "Content-Type: application/x-openpegasus\r\n" "content-length: 0000000000\r\n" "CIMOperation: MethodResponse\r\n" "\r\n"; // The HTTP processor fills in the content-length value later. // It searches for a field matching "content-length" (so the first // character must be lower case). out.append(HEADERS, sizeof(HEADERS) - 1); return; } out << STRLIT("HTTP/1.1 " HTTP_STATUS_OK "\r\n"); #ifndef PEGASUS_DISABLE_PERFINST if (StatisticalData::current()->copyGSD) { out << STRLIT("WBEMServerResponseTime: ") << CIMValue(serverResponseTime).toString() << STRLIT("\r\n"); } #endif if (binaryResponse) { // According to MIME RFC, the "x-" prefix should be used for all // non-registered values. out << STRLIT("Content-Type: application/x-openpegasus\r\n"); } else { out << STRLIT("Content-Type: application/xml; charset=utf-8\r\n"); } OUTPUT_CONTENTLENGTH(out, contentLength); if (contentLanguages.size() > 0) { out << STRLIT("Content-Language: ") << contentLanguages << STRLIT("\r\n"); } if (httpMethod == HTTP_METHOD_M_POST) { char nn[] = {char('0'+(rand() % 10)),char('0' + (rand() % 10)),'\0'}; out << STRLIT("Ext:\r\n" "Cache-Control: no-cache\r\n" "Man: http://www.dmtf.org/cim/mapping/http/v1.0; ns="); out << nn << STRLIT("\r\n"); out << nn << STRLIT("-CIMOperation: MethodResponse\r\n\r\n"); } else { out << STRLIT("CIMOperation: MethodResponse\r\n\r\n"); } } //------------------------------------------------------------------------------ // // appendHttpErrorResponseHeader() // // Build HTTP error response header. // // Returns error response message in the following format: // // HTTP/1.1 400 Bad Request (using specified status code) // CIMError: (if specified by caller) // PGErrorDetail: (if specified by caller) // //------------------------------------------------------------------------------ void XmlWriter::appendHttpErrorResponseHeader( Buffer& out, const String& status, const String& cimError, const String& errorDetail) { out << STRLIT("HTTP/1.1 ") << status << STRLIT("\r\n"); if (cimError != String::EMPTY) { out << STRLIT("CIMError: ") << cimError << STRLIT("\r\n"); } if (errorDetail != String::EMPTY) { out << STRLIT(PEGASUS_HTTPHEADERTAG_ERRORDETAIL ": ") << encodeURICharacters(errorDetail) << STRLIT("\r\n"); } out << STRLIT("\r\n"); } //------------------------------------------------------------------------------ // // appendUnauthorizedResponseHeader() // // Build HTTP authentication response header for unauthorized requests. // // Returns unauthorized message in the following format: // // HTTP/1.1 401 Unauthorized // PGErrorDetail: (if specified by caller) // WWW-Authenticate: Basic realm="HostName" // // 401 Unauthorized // //

TEST401 Unauthorized

//
// // //------------------------------------------------------------------------------ void XmlWriter::appendUnauthorizedResponseHeader( Buffer& out, const String& errorDetail, const String& content) { out << STRLIT("HTTP/1.1 " HTTP_STATUS_UNAUTHORIZED "\r\n"); if (errorDetail.size() > 0) { out << STRLIT(PEGASUS_HTTPHEADERTAG_ERRORDETAIL ": ") << encodeURICharacters(errorDetail) << STRLIT("\r\n"); } OUTPUT_CONTENTLENGTH(out, 0); out << content << STRLIT("\r\n\r\n"); //ATTN: We may need to include the following line, so that the browsers // can display the error message. // out << "\r\n"; // out << "" << "401 Unauthorized" << "\r\n"; // out << "\r\n"; // out << "

TEST" << "401 Unauthorized" << "

\r\n"; // out << "
\r\n"; // out << "\r\n"; } #ifdef PEGASUS_KERBEROS_AUTHENTICATION //------------------------------------------------------------------------------ // // appendOKResponseHeader() // // Build HTTP authentication response header for unauthorized requests. // // Returns OK message in the following format: // // HTTP/1.1 200 OK // Content-Length: 0 // WWW-Authenticate: Negotiate "token" // // 200 OK // //

TEST200 OK

//
// // //------------------------------------------------------------------------------ void XmlWriter::appendOKResponseHeader( Buffer& out, const String& content) { out << STRLIT("HTTP/1.1 " HTTP_STATUS_OK "\r\n"); // Content-Length header needs to be added because 200 OK record // is usually intended to have content. But, for Kerberos this // may not always be the case so we need to indicate that there // is no content OUTPUT_CONTENTLENGTH(out, 0); out << content << STRLIT("\r\n\r\n"); //ATTN: We may need to include the following line, so that the browsers // can display the error message. // out << "\r\n"; // out << "" << "200 OK" << "\r\n"; // out << "\r\n"; // out << "

TEST" << "200 OK" << "

\r\n"; // out << "
\r\n"; // out << "\r\n"; } #endif //------------------------------------------------------------------------------ // // _appendMessageElementBegin() // _appendMessageElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendMessageElementBegin( Buffer& out, const String& messageId) { out << STRLIT("\n" "\n" "\n"); } void XmlWriter::_appendMessageElementEnd( Buffer& out) { out << STRLIT("\n\n"); } //------------------------------------------------------------------------------ // // _appendSimpleReqElementBegin() // _appendSimpleReqElementEnd() // // // //------------------------------------------------------------------------------ void XmlWriter::_appendSimpleReqElementBegin( Buffer& out) { out << STRLIT("\n"); } void XmlWriter::_appendSimpleReqElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendMethodCallElementBegin() // _appendMethodCallElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendMethodCallElementBegin( Buffer& out, const CIMName& name) { out << STRLIT("\n"); } void XmlWriter::_appendMethodCallElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendIMethodCallElementBegin() // _appendIMethodCallElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendIMethodCallElementBegin( Buffer& out, const CIMName& name) { out << STRLIT("\n"); } void XmlWriter::_appendIMethodCallElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendIParamValueElementBegin() // _appendIParamValueElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendIParamValueElementBegin( Buffer& out, const char* name) { out << STRLIT("\n"); } void XmlWriter::_appendIParamValueElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendSimpleRspElementBegin() // _appendSimpleRspElementEnd() // // // //------------------------------------------------------------------------------ void XmlWriter::_appendSimpleRspElementBegin( Buffer& out) { out << STRLIT("\n"); } void XmlWriter::_appendSimpleRspElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendMethodResponseElementBegin() // _appendMethodResponseElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendMethodResponseElementBegin( Buffer& out, const CIMName& name) { out << STRLIT("\n"); } void XmlWriter::_appendMethodResponseElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendIMethodResponseElementBegin() // _appendIMethodResponseElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendIMethodResponseElementBegin( Buffer& out, const CIMName& name) { out << STRLIT("\n"); } void XmlWriter::_appendIMethodResponseElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendErrorElement() // //------------------------------------------------------------------------------ void XmlWriter::_appendErrorElement( Buffer& out, const CIMException& cimException) { Tracer::traceCIMException(TRC_XML, Tracer::LEVEL2, cimException); out << STRLIT(""); for (Uint32 i = 0, n = cimException.getErrorCount(); i < n; i++) appendInstanceElement(out, cimException.getError(i)); out << STRLIT(""); } else out << STRLIT("/>"); } //---------------------------------------------------------------------- // // appendParamTypeAndEmbeddedObjAttrib // Appends the Param type and EmbeddedObject Info to the buffer // %EmbeddedObject; #IMPLIED // %ParamType;> // //--------------------------------------------------------------------- void XmlWriter::appendParamTypeAndEmbeddedObjAttrib( Buffer& out, const CIMType& type) { // If the property type is CIMObject, then // encode the property in CIM-XML as a string with the EmbeddedObject // attribute (there is not currently a CIM-XML "object" // datatype). // Because of an error in Pegasus we were earlier outputting // upper case "EMBEDDEDOBJECT" as the attribute name. The // spec calls for mixed case "EmbeddedObject. Fixed in // bug 7131 to output EmbeddedObject attribute in upper // case and mixed case. Receiver will ignore one or the // other. //else // output the real type if (type == CIMTYPE_OBJECT) { out << STRLIT(" PARAMTYPE=\"string\"" " EmbeddedObject=\"object\"" " EMBEDDEDOBJECT=\"object\""); } else if (type == CIMTYPE_INSTANCE) { out << STRLIT(" PARAMTYPE=\"string\"" " EmbeddedObject=\"instance\"" " EMBEDDEDOBJECT=\"instance\""); } else { out << STRLIT(" PARAM") << xmlWriterTypeStrings(type); } } //------------------------------------------------------------------------------ // // appendReturnValueElement() // // // // //------------------------------------------------------------------------------ void XmlWriter::appendReturnValueElement( Buffer& out, const CIMValue& value) { out << STRLIT("\n"); // Add value. appendValueElement(out, value); out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendIReturnValueElementBegin() // _appendIReturnValueElementEnd() // // // //------------------------------------------------------------------------------ void XmlWriter::_appendIReturnValueElementBegin( Buffer& out) { out << STRLIT("\n"); } void XmlWriter::_appendIReturnValueElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendBooleanIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendBooleanIParameter( Buffer& out, const char* name, Boolean flag) { _appendIParamValueElementBegin(out, name); out << STRLIT(""); append(out, flag); out << STRLIT("\n"); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendStringIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendStringIParameter( Buffer& out, const char* name, const String& str) { _appendIParamValueElementBegin(out, name); out << STRLIT(""); appendSpecial(out, str); out << STRLIT("\n"); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendClassNameIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendClassNameIParameter( Buffer& out, const char* name, const CIMName& className) { _appendIParamValueElementBegin(out, name); // // A NULL (unassigned) value for a parameter is specified by an // element with no subelement // if (!className.isNull ()) { appendClassNameElement(out, className); } _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendInstanceNameIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendInstanceNameIParameter( Buffer& out, const char* name, const CIMObjectPath& instanceName) { _appendIParamValueElementBegin(out, name); appendInstanceNameElement(out, instanceName); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendClassIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendClassIParameter( Buffer& out, const char* name, const CIMConstClass& cimClass) { _appendIParamValueElementBegin(out, name); appendClassElement(out, cimClass); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendInstanceIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendInstanceIParameter( Buffer& out, const char* name, const CIMConstInstance& instance) { _appendIParamValueElementBegin(out, name); appendInstanceElement(out, instance); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendNamedInstanceIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendNamedInstanceIParameter( Buffer& out, const char* name, const CIMInstance& namedInstance) { _appendIParamValueElementBegin(out, name); appendValueNamedInstanceElement(out, namedInstance); _appendIParamValueElementEnd(out); } //---------------------------------------------------------- // // appendPropertyNameIParameter() // // // FreeSpace // // USE: Create parameter for getProperty operation //========================================================== void XmlWriter::appendPropertyNameIParameter( Buffer& out, const CIMName& propertyName) { _appendIParamValueElementBegin(out, "PropertyName"); out << STRLIT("") << propertyName << STRLIT("\n"); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendPropertyValueIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendPropertyValueIParameter( Buffer& out, const char* name, const CIMValue& value) { _appendIParamValueElementBegin(out, name); appendValueElement(out, value); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendPropertyListIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendPropertyListIParameter( Buffer& out, const CIMPropertyList& propertyList) { _appendIParamValueElementBegin(out, "PropertyList"); // // A NULL (unassigned) value for a parameter is specified by an // element with no subelement // if (!propertyList.isNull ()) { out << STRLIT("\n"); for (Uint32 i = 0; i < propertyList.size(); i++) { out << STRLIT("") << propertyList[i] << STRLIT("\n"); } out << STRLIT("\n"); } _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // appendQualifierDeclarationIParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendQualifierDeclarationIParameter( Buffer& out, const char* name, const CIMConstQualifierDecl& qualifierDecl) { _appendIParamValueElementBegin(out, name); appendQualifierDeclElement(out, qualifierDecl); _appendIParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // XmlWriter::formatHttpErrorRspMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatHttpErrorRspMessage( const String& status, const String& cimError, const String& errorDetail) { Buffer out; appendHttpErrorResponseHeader(out, status, cimError, errorDetail); return out; } // l10n - add content language support to the format methods below //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleMethodReqMessage() // //------------------------------------------------------------------------------ // ATTN-RK-P1-20020228: Need to complete copy elimination optimization Buffer XmlWriter::formatSimpleMethodReqMessage( const char* host, const CIMNamespaceName& nameSpace, const CIMObjectPath& path, const CIMName& methodName, const Array& parameters, const String& messageId, HttpMethod httpMethod, const String& authenticationHeader, const AcceptLanguageList& httpAcceptLanguages, const ContentLanguageList& httpContentLanguages, bool binaryResponse) { Buffer out; Buffer tmp; CIMObjectPath localObjectPath = path; localObjectPath.setNameSpace(nameSpace.getString()); localObjectPath.setHost(String::EMPTY); _appendMessageElementBegin(out, messageId); _appendSimpleReqElementBegin(out); _appendMethodCallElementBegin(out, methodName); appendLocalObjectPathElement(out, localObjectPath); for (Uint32 i=0; i < parameters.size(); i++) { appendParamValueElement(out, parameters[i]); } _appendMethodCallElementEnd(out); _appendSimpleReqElementEnd(out); _appendMessageElementEnd(out); appendMethodCallHeader( tmp, host, methodName, localObjectPath.toString(), authenticationHeader, httpMethod, httpAcceptLanguages, httpContentLanguages, out.size(), false, binaryResponse); tmp << out; return tmp; } Buffer XmlWriter::formatSimpleMethodRspMessage( const CIMName& methodName, const String& messageId, HttpMethod httpMethod, const ContentLanguageList& httpContentLanguages, const Buffer& body, Uint64 serverResponseTime, Boolean isFirst, Boolean isLast) { Buffer out; if (isFirst == true) { // NOTE: temporarily put zero for content length. the http code // will later decide to fill in the length or remove it altogether appendMethodResponseHeader( out, httpMethod, httpContentLanguages, 0, serverResponseTime); _appendMessageElementBegin(out, messageId); _appendSimpleRspElementBegin(out); _appendMethodResponseElementBegin(out, methodName); } if (body.size() != 0) { out << body; } if (isLast == true) { _appendMethodResponseElementEnd(out); _appendSimpleRspElementEnd(out); _appendMessageElementEnd(out); } return out; } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleMethodErrorRspMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleMethodErrorRspMessage( const CIMName& methodName, const String& messageId, HttpMethod httpMethod, const CIMException& cimException) { Buffer out; Buffer tmp; _appendMessageElementBegin(out, messageId); _appendSimpleRspElementBegin(out); _appendMethodResponseElementBegin(out, methodName); _appendErrorElement(out, cimException); _appendMethodResponseElementEnd(out); _appendSimpleRspElementEnd(out); _appendMessageElementEnd(out); appendMethodResponseHeader( tmp, httpMethod, cimException.getContentLanguages(), out.size(), false); tmp << out; return tmp; } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleIMethodReqMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleIMethodReqMessage( const char* host, const CIMNamespaceName& nameSpace, const CIMName& iMethodName, const String& messageId, HttpMethod httpMethod, const String& authenticationHeader, const AcceptLanguageList& httpAcceptLanguages, const ContentLanguageList& httpContentLanguages, const Buffer& body, bool binaryResponse) { Buffer out; Buffer tmp; _appendMessageElementBegin(out, messageId); _appendSimpleReqElementBegin(out); _appendIMethodCallElementBegin(out, iMethodName); appendLocalNameSpacePathElement(out, nameSpace.getString()); out << body; _appendIMethodCallElementEnd(out); _appendSimpleReqElementEnd(out); _appendMessageElementEnd(out); appendMethodCallHeader( tmp, host, iMethodName, nameSpace.getString(), authenticationHeader, httpMethod, httpAcceptLanguages, httpContentLanguages, out.size(), false, binaryResponse); tmp << out; return tmp; } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleIMethodRspMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleIMethodRspMessage( const CIMName& iMethodName, const String& messageId, HttpMethod httpMethod, const ContentLanguageList& httpContentLanguages, const Buffer& body, Uint64 serverResponseTime, Boolean isFirst, Boolean isLast) { Buffer out; if (isFirst == true) { // NOTE: temporarily put zero for content length. the http code // will later decide to fill in the length or remove it altogether appendMethodResponseHeader( out, httpMethod, httpContentLanguages, 0, serverResponseTime); _appendMessageElementBegin(out, messageId); _appendSimpleRspElementBegin(out); _appendIMethodResponseElementBegin(out, iMethodName); // output the start of the return tag. Test if there is response data // by: // 1. there is data on the first chunk OR // 2. there is no data on the first chunk but isLast is false implying // there is more non-empty data to come. If all subsequent chunks // are empty, then this generates and empty response. if (body.size() != 0 || isLast == false) _appendIReturnValueElementBegin(out); } if (body.size() != 0) { out << body; } if (isLast == true) { if (body.size() != 0 || isFirst == false) _appendIReturnValueElementEnd(out); _appendIMethodResponseElementEnd(out); _appendSimpleRspElementEnd(out); _appendMessageElementEnd(out); } return out; } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleIMethodErrorRspMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleIMethodErrorRspMessage( const CIMName& iMethodName, const String& messageId, HttpMethod httpMethod, const CIMException& cimException) { Buffer out; Buffer tmp; _appendMessageElementBegin(out, messageId); _appendSimpleRspElementBegin(out); _appendIMethodResponseElementBegin(out, iMethodName); _appendErrorElement(out, cimException); _appendIMethodResponseElementEnd(out); _appendSimpleRspElementEnd(out); _appendMessageElementEnd(out); appendMethodResponseHeader(tmp, httpMethod, cimException.getContentLanguages(), out.size(), false); tmp << out; return tmp; } //****************************************************************************** // // Export Messages (used for indications) // //****************************************************************************** //------------------------------------------------------------------------------ // // appendEMethodRequestHeader() // // Build HTTP request header for export operation. // //------------------------------------------------------------------------------ void XmlWriter::appendEMethodRequestHeader( Buffer& out, const char* requestUri, const char* host, const CIMName& cimMethod, HttpMethod httpMethod, const String& authenticationHeader, const AcceptLanguageList& acceptLanguages, const ContentLanguageList& contentLanguages, Uint32 contentLength) { char nn[] = { char('0' + (rand() % 10)), char('0' + (rand() % 10)), '\0' }; if (httpMethod == HTTP_METHOD_M_POST) { out << STRLIT("M-POST ") << requestUri << STRLIT(" HTTP/1.1\r\n"); } else { out << STRLIT("POST ") << requestUri << STRLIT(" HTTP/1.1\r\n"); } out << STRLIT("HOST: ") << host << STRLIT("\r\n" "Content-Type: application/xml; charset=utf-8\r\n"); OUTPUT_CONTENTLENGTH(out, contentLength); if (acceptLanguages.size() > 0) { out << STRLIT("Accept-Language: ") << acceptLanguages << STRLIT("\r\n"); } if (contentLanguages.size() > 0) { out << STRLIT("Content-Language: ") << contentLanguages << STRLIT("\r\n"); } #ifdef PEGASUS_DEBUG // backdoor environment variable to turn OFF client requesting transfer // encoding. The default is on. to turn off, set this variable to zero. // This should be removed when stable. This should only be turned off in // a debugging/testing environment. static const char *clientTransferEncodingOff = getenv("PEGASUS_HTTP_TRANSFER_ENCODING_REQUEST"); if (!clientTransferEncodingOff || *clientTransferEncodingOff != '0') #endif out << STRLIT("TE: chunked, trailers\r\n"); if (httpMethod == HTTP_METHOD_M_POST) { out << STRLIT("Man: http://www.dmtf.org/cim/mapping/http/v1.0; ns="); out << nn << STRLIT("\r\n"); out << nn << STRLIT("-CIMExport: MethodRequest\r\n"); out << nn << STRLIT("-CIMExportMethod: ") << cimMethod << STRLIT("\r\n"); } else { out << STRLIT("CIMExport: MethodRequest\r\n" "CIMExportMethod: ") << cimMethod << STRLIT("\r\n"); } if (authenticationHeader.size()) { out << authenticationHeader << STRLIT("\r\n"); } out << STRLIT("\r\n"); } //------------------------------------------------------------------------------ // // appendEMethodResponseHeader() // // Build HTTP response header for export operation. // //------------------------------------------------------------------------------ void XmlWriter::appendEMethodResponseHeader( Buffer& out, HttpMethod httpMethod, const ContentLanguageList& contentLanguages, Uint32 contentLength) { char nn[] = { char('0' + (rand() % 10)), char('0' + (rand() % 10)), '\0' }; out << STRLIT("HTTP/1.1 " HTTP_STATUS_OK "\r\n" "Content-Type: application/xml; charset=utf-8\r\n"); OUTPUT_CONTENTLENGTH(out, contentLength); if (contentLanguages.size() > 0) { out << STRLIT("Content-Language: ") << contentLanguages << STRLIT("\r\n"); } if (httpMethod == HTTP_METHOD_M_POST) { out << STRLIT("Ext:\r\n" "Cache-Control: no-cache\r\n" "Man: http://www.dmtf.org/cim/mapping/http/v1.0; ns="); out << nn << STRLIT("\r\n"); out << nn << STRLIT("-CIMExport: MethodResponse\r\n\r\n"); } else { out << STRLIT("CIMExport: MethodResponse\r\n\r\n"); } } //------------------------------------------------------------------------------ // // _appendSimpleExportReqElementBegin() // _appendSimpleExportReqElementEnd() // // // //------------------------------------------------------------------------------ void XmlWriter::_appendSimpleExportReqElementBegin( Buffer& out) { out << STRLIT("\n"); } void XmlWriter::_appendSimpleExportReqElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendEMethodCallElementBegin() // _appendEMethodCallElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendEMethodCallElementBegin( Buffer& out, const CIMName& name) { out << STRLIT("\n"); } void XmlWriter::_appendEMethodCallElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendEParamValueElementBegin() // _appendEParamValueElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendEParamValueElementBegin( Buffer& out, const char* name) { out << STRLIT("\n"); } void XmlWriter::_appendEParamValueElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // appendInstanceEParameter() // //------------------------------------------------------------------------------ void XmlWriter::appendInstanceEParameter( Buffer& out, const char* name, const CIMInstance& instance) { _appendEParamValueElementBegin(out, name); appendInstanceElement(out, instance); _appendEParamValueElementEnd(out); } //------------------------------------------------------------------------------ // // _appendSimpleExportRspElementBegin() // _appendSimpleExportRspElementEnd() // // // //------------------------------------------------------------------------------ void XmlWriter::_appendSimpleExportRspElementBegin( Buffer& out) { out << STRLIT("\n"); } void XmlWriter::_appendSimpleExportRspElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // _appendEMethodResponseElementBegin() // _appendEMethodResponseElementEnd() // // // // //------------------------------------------------------------------------------ void XmlWriter::_appendEMethodResponseElementBegin( Buffer& out, const CIMName& name) { out << STRLIT("\n"); } void XmlWriter::_appendEMethodResponseElementEnd( Buffer& out) { out << STRLIT("\n"); } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleEMethodReqMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleEMethodReqMessage( const char* requestUri, const char* host, const CIMName& eMethodName, const String& messageId, HttpMethod httpMethod, const String& authenticationHeader, const AcceptLanguageList& httpAcceptLanguages, const ContentLanguageList& httpContentLanguages, const Buffer& body) { Buffer out; Buffer tmp; _appendMessageElementBegin(out, messageId); _appendSimpleExportReqElementBegin(out); _appendEMethodCallElementBegin(out, eMethodName); out << body; _appendEMethodCallElementEnd(out); _appendSimpleExportReqElementEnd(out); _appendMessageElementEnd(out); appendEMethodRequestHeader( tmp, requestUri, host, eMethodName, httpMethod, authenticationHeader, httpAcceptLanguages, httpContentLanguages, out.size()); tmp << out; return tmp; } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleEMethodRspMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleEMethodRspMessage( const CIMName& eMethodName, const String& messageId, HttpMethod httpMethod, const ContentLanguageList& httpContentLanguages, const Buffer& body) { Buffer out; Buffer tmp; _appendMessageElementBegin(out, messageId); _appendSimpleExportRspElementBegin(out); _appendEMethodResponseElementBegin(out, eMethodName); out << body; _appendEMethodResponseElementEnd(out); _appendSimpleExportRspElementEnd(out); _appendMessageElementEnd(out); appendEMethodResponseHeader(tmp, httpMethod, httpContentLanguages, out.size()); tmp << out; return tmp; } //------------------------------------------------------------------------------ // // XmlWriter::formatSimpleEMethodErrorRspMessage() // //------------------------------------------------------------------------------ Buffer XmlWriter::formatSimpleEMethodErrorRspMessage( const CIMName& eMethodName, const String& messageId, HttpMethod httpMethod, const CIMException& cimException) { Buffer out; Buffer tmp; _appendMessageElementBegin(out, messageId); _appendSimpleExportRspElementBegin(out); _appendEMethodResponseElementBegin(out, eMethodName); _appendErrorElement(out, cimException); _appendEMethodResponseElementEnd(out); _appendSimpleExportRspElementEnd(out); _appendMessageElementEnd(out); appendEMethodResponseHeader( tmp, httpMethod, cimException.getContentLanguages(), out.size()); tmp << out; return tmp; } //------------------------------------------------------------------------------ // // XmlWriter::getNextMessageId() // //------------------------------------------------------------------------------ static IDFactory _messageIDFactory(1000); String XmlWriter::getNextMessageId() { char scratchBuffer[22]; Uint32 n; const char * startP = Uint32ToString(scratchBuffer, _messageIDFactory.getID(), n); return String(startP, n); } //------------------------------------------------------------------------------ // // XmlWriter::keyBindingTypeToString // //------------------------------------------------------------------------------ const StrLit XmlWriter::keyBindingTypeToString (CIMKeyBinding::Type type) { switch (type) { case CIMKeyBinding::BOOLEAN: return STRLIT("boolean"); case CIMKeyBinding::STRING: return STRLIT("string"); case CIMKeyBinding::NUMERIC: return STRLIT("numeric"); case CIMKeyBinding::REFERENCE: default: PEGASUS_UNREACHABLE(PEGASUS_ASSERT(false);) } return STRLIT("unknown"); } PEGASUS_NAMESPACE_END