Files
libreoffice/forms/source/xforms/submission/serialization_urlencoded.cxx
Oliver Bolte b8509cfe5e INTEGRATION: CWS pchfix02 (1.4.26); FILE MERGED
2006/09/01 17:27:48 kaib 1.4.26.1: #i68856# Added header markers and pch files
2006-09-16 23:07:16 +00:00

221 lines
6.8 KiB
C++

/*************************************************************************
*
* OpenOffice.org - a multi-platform office productivity suite
*
* $RCSfile: serialization_urlencoded.cxx,v $
*
* $Revision: 1.5 $
*
* last change: $Author: obo $ $Date: 2006-09-17 00:07:16 $
*
* The Contents of this file are made available subject to
* the terms of GNU Lesser General Public License Version 2.1.
*
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2005 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_forms.hxx"
#include <com/sun/star/xml/xpath/XPathObjectType.hpp>
#include <com/sun/star/xml/dom/XNode.hpp>
#include <com/sun/star/xml/dom/XText.hpp>
#include <com/sun/star/xml/dom/XNodeList.hpp>
#include <com/sun/star/xml/dom/NodeType.hpp>
#include <rtl/ustrbuf.hxx>
#include <rtl/strbuf.hxx>
#include <unotools/processfactory.hxx>
#include <stdio.h>
#include "serialization_urlencoded.hxx"
using namespace rtl;
using namespace utl;
using namespace CSS::uno;
using namespace CSS::io;
using namespace CSS::xml::xpath;
using namespace CSS::xml::dom;
CSerializationURLEncoded::CSerializationURLEncoded()
: m_aFactory(getProcessServiceFactory())
, m_aPipe(Reference< XOutputStream > (m_aFactory->createInstance(
OUString::createFromAscii("com.sun.star.io.Pipe")), UNO_QUERY))
{
}
/*
rfc2396
reserved = ";" | "/" | "?" | ":" | "@" | "&" | "=" | "+" |
"$" | ","
mark = "-" | "_" | "." | "!" | "~" | "*" | "'" | "(" | ")"
unreserved = alphanum | mark
*/
sal_Bool CSerializationURLEncoded::is_unreserved(sal_Char c)
{
//digit?
if (c >= '0' && c <= '9') return sal_True;
if (c >= 'A' && c <= 'Z') return sal_True;
if (c >= 'a' && c <= 'z') return sal_True;
switch (c) {
case '-':
case '_':
case '.':
case '!':
case '~':
case '*':
case '\'':
case '(':
case ')':
return sal_True;
}
return sal_False;
}
void CSerializationURLEncoded::encode_and_append(const OUString& aString, OStringBuffer& aBuffer)
{
OString utf8String = OUStringToOString(aString, RTL_TEXTENCODING_UTF8);
const sal_uInt8 *pString = reinterpret_cast< const sal_uInt8 * >( utf8String.getStr() );
sal_Char tmpChar[4]; tmpChar[3] = 0;
while( *pString != 0)
{
if( *pString < 0x80 )
{
if ( is_unreserved(*pString) ) {
aBuffer.append(*pString);
} else if (*pString == 0x20) {
aBuffer.append('+');
} else if (*pString == 0x0d && *(pString+1) == 0x0a) {
aBuffer.append("%0D%0A");
pString++;
} else if (*pString == 0x0a) {
aBuffer.append("%0D%0A");
} else {
snprintf(tmpChar, 3, "%%%X", *pString % 0x100);
aBuffer.append(tmpChar);
}
} else {
snprintf(tmpChar, 3, "%%%X", *pString % 0x100);
aBuffer.append(tmpChar);
while (*pString >= 0x80) {
// continuation...
pString++;
snprintf(tmpChar, 3, "%%%X", *pString % 0x100);
aBuffer.append(tmpChar);
}
}
pString++;
}
}
void CSerializationURLEncoded::serialize_node(const Reference< XNode >& aNode)
{
// serialize recursive
// every element node E that has a text child T will be serialized in document order
// <E1>T1<E2>T2</E2></E1><E3>T3</E3> -> E1=T2&E2=T2&E3=T3 (En := local name)
// this node
Reference< XNodeList > aChildList = aNode->getChildNodes();
Reference< XNode > aChild;
// is this an element node?
if (aNode->getNodeType() == NodeType_ELEMENT_NODE)
{
OUString aName = aNode->getNodeName();
// find any text children
OUStringBuffer aValue;
Reference< XText > aText;
for(sal_Int32 i=0; i < aChildList->getLength(); i++)
{
aChild = aChildList->item(i);
if (aChild->getNodeType() == NodeType_TEXT_NODE)
{
aText = Reference< XText >(aChild, UNO_QUERY);
aValue.append(aText->getData());
}
}
// found anything?
if (aValue.getLength() > 0)
{
OUString aUnencValue = aValue.makeStringAndClear();
OStringBuffer aEncodedBuffer;
encode_and_append(aName, aEncodedBuffer);
aEncodedBuffer.append("=");
encode_and_append(aUnencValue, aEncodedBuffer);
aEncodedBuffer.append("&");
sal_Int8 *pData = (sal_Int8*)aEncodedBuffer.getStr();
Sequence< sal_Int8 > sData(pData, aEncodedBuffer.getLength());
m_aPipe->writeBytes(sData);
}
}
// element children...
for(sal_Int32 i=0; i < aChildList->getLength(); i++)
{
aChild = aChildList->item(i);
// if this is an element node, it might be a candidate for serialization
if (aChild.is() && aChild->getNodeType() == NodeType_ELEMENT_NODE)
serialize_node(aChild);
}
// siblings...
// Reference< XNode > aSibling = aNode->getNextSibling();
// if (aSibling.is())
// serialize_node(aSibling);
}
/*
void CSerializationURLEncoded::serialize_nodeset()
{
Reference< XNodeList > aNodeList = m_aXPathObject->getNodeList();
for (sal_Int32 i=0; i<aNodeList->getLength(); i++)
serialize_node(aNodeList->item(i));
m_aPipe->closeOutput();
}
*/
void CSerializationURLEncoded::serialize()
{
// output stream to the pipe buffer
Reference< XOutputStream > out(m_aPipe, UNO_QUERY);
CSS::uno::Reference< CSS::xml::dom::XNode > cur = m_aFragment->getFirstChild();
while (cur.is())
{
serialize_node(cur);
cur = cur->getNextSibling();
}
m_aPipe->closeOutput();
}
Reference< XInputStream > CSerializationURLEncoded::getInputStream()
{
return Reference< XInputStream >(m_aPipe, UNO_QUERY);
}