diff options
author | Gordon Sim <gsim@apache.org> | 2008-10-24 11:33:40 +0000 |
---|---|---|
committer | Gordon Sim <gsim@apache.org> | 2008-10-24 11:33:40 +0000 |
commit | 93b372e4d8d51a3db161a95a8b1884ce6af117fd (patch) | |
tree | 4a48e33e5042ea1f3a7be46100d3d6a9c73d5c09 /cpp | |
parent | 55c976dbede7ada5dbcc581945f7d5b1a038344c (diff) | |
download | qpid-python-93b372e4d8d51a3db161a95a8b1884ce6af117fd.tar.gz |
Fix for bug in encoding/decoding of floats and doubles.
Added test to verify double/float headers set by c++ are correctly read by python
[TODO: add wider range of header types, add bidirectional tests, add tests against other languages]
[Also fixed one or two SSL related files missing from distribution list]
git-svn-id: https://svn.apache.org/repos/asf/incubator/qpid/trunk/qpid@707604 13f79535-47bb-0310-9956-ffa450edef68
Diffstat (limited to 'cpp')
-rw-r--r-- | cpp/Makefile.am | 2 | ||||
-rw-r--r-- | cpp/src/Makefile.am | 2 | ||||
-rw-r--r-- | cpp/src/qpid/framing/Endian.cpp | 52 | ||||
-rw-r--r-- | cpp/src/qpid/framing/Endian.h | 46 | ||||
-rw-r--r-- | cpp/src/qpid/framing/FieldTable.cpp | 5 | ||||
-rw-r--r-- | cpp/src/qpid/framing/FieldValue.cpp | 5 | ||||
-rw-r--r-- | cpp/src/qpid/framing/FieldValue.h | 2 | ||||
-rw-r--r-- | cpp/src/tests/Makefile.am | 9 | ||||
-rw-r--r-- | cpp/src/tests/header_test.cpp | 59 | ||||
-rwxr-xr-x | cpp/src/tests/header_test.py | 86 | ||||
-rwxr-xr-x | cpp/src/tests/run_header_test | 13 |
11 files changed, 275 insertions, 6 deletions
diff --git a/cpp/Makefile.am b/cpp/Makefile.am index 67300e370f..f4f424861c 100644 --- a/cpp/Makefile.am +++ b/cpp/Makefile.am @@ -2,7 +2,7 @@ AUTOMAKE_OPTIONS = 1.9.2 foreign ACLOCAL_AMFLAGS = -I m4 EXTRA_DIST = \ - LICENSE NOTICE README RELEASE_NOTES DESIGN DISCLAIMER\ + LICENSE NOTICE README SSL RELEASE_NOTES DESIGN DISCLAIMER\ xml/cluster.xml SUBDIRS = managementgen etc src docs/api docs/man examples diff --git a/cpp/src/Makefile.am b/cpp/src/Makefile.am index ef0af1b4d3..5d742877f5 100644 --- a/cpp/src/Makefile.am +++ b/cpp/src/Makefile.am @@ -257,6 +257,7 @@ libqpidcommon_la_SOURCES = \ qpid/framing/BodyHolder.cpp \ qpid/framing/BodyHandler.cpp \ qpid/framing/Buffer.cpp \ + qpid/framing/Endian.cpp \ qpid/framing/FieldTable.cpp \ qpid/framing/FieldValue.cpp \ qpid/framing/FrameSet.cpp \ @@ -544,6 +545,7 @@ nobase_include_HEADERS = \ qpid/framing/BodyHandler.h \ qpid/framing/Buffer.h \ qpid/framing/ChannelHandler.h \ + qpid/framing/Endian.h \ qpid/framing/FieldTable.h \ qpid/framing/FieldValue.h \ qpid/framing/FrameDefaultVisitor.h \ diff --git a/cpp/src/qpid/framing/Endian.cpp b/cpp/src/qpid/framing/Endian.cpp new file mode 100644 index 0000000000..a2d4566e5e --- /dev/null +++ b/cpp/src/qpid/framing/Endian.cpp @@ -0,0 +1,52 @@ +/* + * + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * + */ +#include "Endian.h" + +namespace qpid { +namespace framing { + +Endian::Endian() : littleEndian(!testBigEndian()) {} + +bool Endian::testBigEndian() +{ + uint16_t a = 1; + uint16_t b; + uint8_t* p = (uint8_t*) &b; + p[0] = 0xFF & (a >> 8); + p[1] = 0xFF & (a); + return a == b; +} + +uint8_t* const Endian::convertIfRequired(uint8_t* const octets, int width) +{ + if (instance.littleEndian) { + for (int i = 0; i < (width/2); i++) { + uint8_t temp = octets[i]; + octets[i] = octets[width - (1 + i)]; + octets[width - (1 + i)] = temp; + } + } + return octets; +} + +const Endian Endian::instance; + +}} // namespace qpid::framing diff --git a/cpp/src/qpid/framing/Endian.h b/cpp/src/qpid/framing/Endian.h new file mode 100644 index 0000000000..14d3f9e66d --- /dev/null +++ b/cpp/src/qpid/framing/Endian.h @@ -0,0 +1,46 @@ +#ifndef QPID_FRAMING_ENDIAN_H +#define QPID_FRAMING_ENDIAN_H + +/* + * + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * + */ + +#include "qpid/sys/IntegerTypes.h" + +namespace qpid { +namespace framing { + +/** + * Conversion utility for little-endian platforms that need to convert + * to and from network ordered octet sequences + */ +class Endian +{ + public: + static uint8_t* const convertIfRequired(uint8_t* const octets, int width); + private: + const bool littleEndian; + Endian(); + static const Endian instance; + static bool testBigEndian(); +}; +}} // namespace qpid::framing + +#endif /*!QPID_FRAMING_ENDIAN_H*/ diff --git a/cpp/src/qpid/framing/FieldTable.cpp b/cpp/src/qpid/framing/FieldTable.cpp index cf0e03180c..90acbaf6bc 100644 --- a/cpp/src/qpid/framing/FieldTable.cpp +++ b/cpp/src/qpid/framing/FieldTable.cpp @@ -21,6 +21,7 @@ #include "FieldTable.h" #include "Array.h" #include "Buffer.h" +#include "Endian.h" #include "FieldValue.h" #include "qpid/Exception.h" #include "qpid/framing/reply_exceptions.h" @@ -157,7 +158,9 @@ bool getRawFixedWidthValue(FieldTable::ValuePtr vptr, T& value) if (vptr && vptr->getType() == typecode) { FixedWidthValue<width>* fwv = dynamic_cast< FixedWidthValue<width>* >(&vptr->getData()); if (fwv) { - fwv->copyInto(reinterpret_cast<uint8_t*>(&value)); + uint8_t* const octets = Endian::convertIfRequired(fwv->rawOctets(), width); + uint8_t* const target = reinterpret_cast<uint8_t*>(&value); + for (uint i = 0; i < width; ++i) target[i] = octets[i]; return true; } } diff --git a/cpp/src/qpid/framing/FieldValue.cpp b/cpp/src/qpid/framing/FieldValue.cpp index 5fbfe7d0c1..9107ceeeea 100644 --- a/cpp/src/qpid/framing/FieldValue.cpp +++ b/cpp/src/qpid/framing/FieldValue.cpp @@ -21,6 +21,7 @@ #include "FieldValue.h" #include "Array.h" #include "Buffer.h" +#include "Endian.h" #include "qpid/framing/reply_exceptions.h" namespace qpid { @@ -138,11 +139,11 @@ IntegerValue::IntegerValue(int v) : {} FloatValue::FloatValue(float v) : - FieldValue(0x23, new FixedWidthValue<4>(reinterpret_cast<uint8_t*>(&v))) + FieldValue(0x23, new FixedWidthValue<4>(Endian::convertIfRequired(reinterpret_cast<uint8_t*>(&v), 4))) {} DoubleValue::DoubleValue(double v) : - FieldValue(0x33, new FixedWidthValue<8>(reinterpret_cast<uint8_t*>(&v))) + FieldValue(0x33, new FixedWidthValue<8>(Endian::convertIfRequired(reinterpret_cast<uint8_t*>(&v), 8))) {} Integer64Value::Integer64Value(int64_t v) : diff --git a/cpp/src/qpid/framing/FieldValue.h b/cpp/src/qpid/framing/FieldValue.h index 0a70360cbd..18c45f3ff8 100644 --- a/cpp/src/qpid/framing/FieldValue.h +++ b/cpp/src/qpid/framing/FieldValue.h @@ -166,6 +166,7 @@ class FixedWidthValue : public FieldValue::Data { { for (uint i = 0; i < width; ++i) data[i] = octets[i]; } + uint8_t* const rawOctets() { return octets; } void print(std::ostream& o) const { o << "F" << width << ":"; }; }; @@ -304,7 +305,6 @@ class ArrayValue : public FieldValue { ArrayValue(const Array&); }; - template <class T> bool getEncodedValue(FieldTable::ValuePtr vptr, T& value) { diff --git a/cpp/src/tests/Makefile.am b/cpp/src/tests/Makefile.am index 35728a3549..91c3995e77 100644 --- a/cpp/src/tests/Makefile.am +++ b/cpp/src/tests/Makefile.am @@ -128,9 +128,13 @@ check_PROGRAMS+=consume consume_SOURCES=consume.cpp TestOptions.h ConnectionOptions.h consume_LDADD=$(lib_client) +check_PROGRAMS+=header_test +header_test_SOURCES=header_test.cpp TestOptions.h ConnectionOptions.h +header_test_LDADD=$(lib_client) + TESTS_ENVIRONMENT = VALGRIND=$(VALGRIND) srcdir=$(srcdir) QPID_DATA_DIR= $(srcdir)/run_test -system_tests = client_test quick_perftest quick_topictest +system_tests = client_test quick_perftest quick_topictest run_header_test TESTS += start_broker $(system_tests) python_tests stop_broker run_federation_tests run_acl_tests EXTRA_DIST += \ @@ -139,6 +143,9 @@ EXTRA_DIST += \ quick_topictest \ quick_perftest \ topictest \ + run_header_tests \ + header_test.py \ + run_ssl_tests \ run_federation_tests \ run_acl_tests \ .valgrind.supp \ diff --git a/cpp/src/tests/header_test.cpp b/cpp/src/tests/header_test.cpp new file mode 100644 index 0000000000..ba9ffacc9b --- /dev/null +++ b/cpp/src/tests/header_test.cpp @@ -0,0 +1,59 @@ +/* + * + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * + */ + +#include <iostream> + +#include "TestOptions.h" +#include "qpid/client/Connection.h" +#include "qpid/client/Message.h" +#include "qpid/client/Session.h" +#include "qpid/client/SubscriptionManager.h" + +using namespace qpid; +using namespace qpid::client; +using namespace std; + +int main(int argc, char** argv) +{ + TestOptions opts; + try { + opts.parse(argc, argv); + Connection connection; + connection.open(opts.con); + Session session = connection.newSession(); + std::string q("header_interop_test_queue"); + session.queueDeclare(arg::queue=q); + double pi = 3.14159265; + float e = 2.71828; + Message msg("", q); + msg.getMessageProperties().getApplicationHeaders().setDouble("pi", pi); + msg.getMessageProperties().getApplicationHeaders().setFloat("e", e); + session.messageTransfer(arg::content=msg); + + session.close(); + connection.close(); + + return 0; + } catch(const exception& e) { + cout << e.what() << endl; + } + return 1; +} diff --git a/cpp/src/tests/header_test.py b/cpp/src/tests/header_test.py new file mode 100755 index 0000000000..d5a2c16c01 --- /dev/null +++ b/cpp/src/tests/header_test.py @@ -0,0 +1,86 @@ +#!/usr/bin/env python +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +import qpid +import sys +import os +from qpid.util import connect +from qpid.connection import Connection +from qpid.datatypes import Message, RangedSet, uuid4 +from qpid.queue import Empty +from math import fabs + +def getApplicationHeaders(msg): + for h in msg.headers: + if hasattr(h, 'application_headers'): return getattr(h, 'application_headers') + return None + +# Set parameters for login + +host="127.0.0.1" +port=5672 +user="guest" +password="guest" + +if len(sys.argv) > 1 : + host=sys.argv[1] +if len(sys.argv) > 2 : + port=int(sys.argv[2]) + +# Create a connection. +socket = connect(host, port) +connection = Connection (sock=socket) +connection.start() +session = connection.session(str(uuid4())) + +q = "header_interop_test_queue" +session.queue_declare(queue=q) + +session.message_subscribe(queue=q, destination="received") +queue = session.incoming("received") +queue.start() + +msg = queue.get(timeout=10) +pi = 3.14159265 +e = 2.71828 + +headers = getApplicationHeaders(msg) +pi_ = headers["pi"] +e_ = headers["e"] +session.close(timeout=10) + +failed = False + +if pi != pi_: + print "got incorrect value for pi: ", pi_, " expected:", pi + failed = True + +if fabs(e - e_) > 0.0001: + print "got incorrect value for e: ", e_, " expected:", e + failed = True + +if failed: + sys.exit(1) +else: + print "Correct header values received." + sys.exit(0) + + + diff --git a/cpp/src/tests/run_header_test b/cpp/src/tests/run_header_test new file mode 100755 index 0000000000..aedd2619a6 --- /dev/null +++ b/cpp/src/tests/run_header_test @@ -0,0 +1,13 @@ +#!/bin/sh +# Simple test of encode/decode of a double in application headers +# TODO: this should be expanded to cover a wider set of types and go +# in both directions + +srcdir=`dirname $0` +PYTHON_DIR=$srcdir/../../../python +test -f qpidd.port && QPID_PORT=`cat qpidd.port` + +./header_test -p $QPID_PORT +export PYTHONPATH=$PYTHON_DIR:$PYTHONPATH +$srcdir/header_test.py "localhost" $QPID_PORT + |