mirror of
				https://github.com/msgpack/msgpack-c.git
				synced 2025-10-23 00:08:01 +02:00 
			
		
		
		
	Merge branch 'master' of http://github.com/msgpack/msgpack
This commit is contained in:
		
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| *.o | ||||
| *.so | ||||
| ruby/Makefile | ||||
| @@ -1,4 +1,31 @@ | ||||
|  | ||||
| 2010-08-29 version 0.5.4: | ||||
|  | ||||
|   * includes msgpack_vc2008.vcproj file in source package | ||||
|   * fixes type::fix_int types | ||||
|  | ||||
| 2010-08-27 version 0.5.3: | ||||
|  | ||||
|   * adds type::fix_{u,}int{8,16,32,64} types | ||||
|   * adds msgpack_pack_fix_{u,}int{8,16,32,64} functions | ||||
|   * adds packer<Stream>::pack_fix_{u,}int{8,16,32,64} functions | ||||
|   * fixes include paths | ||||
|  | ||||
| 2010-07-14 version 0.5.2: | ||||
|  | ||||
|   * type::raw::str(), operator==, operator!=, operator< and operator> are now const | ||||
|   * generates version.h using AC_OUTPUT macro in ./configure | ||||
|  | ||||
| 2010-07-06 version 0.5.1: | ||||
|  | ||||
|   * Add msgpack_vrefbuffer_new and msgpack_vrefbuffer_free | ||||
|   * Add msgpack_sbuffer_new and msgpack_sbuffer_free | ||||
|   * Add msgpack_unpacker_next and msgpack_unpack_next | ||||
|   * msgpack::unpack returns void | ||||
|   * Add MSGPACK_VERSION{,_MAJOR,_MINOR} macros to check header version | ||||
|   * Add msgpack_version{,_major,_minor} functions to check library version | ||||
|   * ./configure supports --disable-cxx option not to build C++ API | ||||
|  | ||||
| 2010-04-29 version 0.5.0: | ||||
|  | ||||
|   * msgpack_object_type is changed. MSGPACK_OBJECT_NIL is now 0x00. | ||||
|   | ||||
| @@ -6,13 +6,14 @@ DOC_FILES = \ | ||||
| 		NOTICE \ | ||||
| 		msgpack_vc8.vcproj \ | ||||
| 		msgpack_vc8.sln \ | ||||
| 		msgpack_vc8.postbuild.bat | ||||
| 		msgpack_vc2008.vcproj \ | ||||
| 		msgpack_vc2008.sln \ | ||||
| 		msgpack_vc.postbuild.bat | ||||
|  | ||||
| EXTRA_DIST = \ | ||||
| 		$(DOC_FILES) | ||||
|  | ||||
| doxygen: | ||||
| 	./preprocess | ||||
| 	./preprocess clean | ||||
| 	cd src && $(MAKE) doxygen | ||||
| 	./preprocess | ||||
|   | ||||
| @@ -13,9 +13,10 @@ On UNIX-like platform, run ./configure && make && sudo make install: | ||||
|     $ make | ||||
|     $ sudo make install | ||||
|  | ||||
| On Windows, open msgpack_vc8.vcproj file and build it using batch build. DLLs are built on lib folder, and the headers are built on include folder. | ||||
| On Windows, open msgpack_vc8.vcproj or msgpack_vc2008 file and build it using batch build. DLLs are built on lib folder, | ||||
| and the headers are built on include folder. | ||||
|  | ||||
| To use the library in your program, include msgpack.hpp header and link msgpack and msgpackc library. | ||||
| To use the library in your program, include msgpack.hpp header and link "msgpack" library. | ||||
|  | ||||
|  | ||||
| ## Example | ||||
| @@ -34,15 +35,9 @@ To use the library in your program, include msgpack.hpp header and link msgpack | ||||
|         msgpack::pack(&buffer, target); | ||||
|      | ||||
|         // Deserialize the serialized data. | ||||
|         msgpack::zone mempool;    // this manages the life of deserialized object | ||||
|         msgpack::object obj; | ||||
|         msgpack::unpack_return ret = | ||||
|                 msgpack::unpack(buffer.data, buffer.size, NULL, &mempool, &obj); | ||||
|      | ||||
|         if(ret != msgapck::UNPACK_SUCCESS) { | ||||
|             // error check | ||||
|             exit(1); | ||||
|         } | ||||
|         msgpack::unpacked msg;    // includes memory pool and deserialized object | ||||
|         msgpack::unpack(&msg, sbuf.data(), sbuf.size()); | ||||
|         msgpack::object obj = msg.get(); | ||||
|      | ||||
|         // Print the deserialized object to stdout. | ||||
|         std::cout << obj << std::endl;    // ["Hello," "World!"] | ||||
| @@ -55,24 +50,24 @@ To use the library in your program, include msgpack.hpp header and link msgpack | ||||
|         obj.as<int>();  // type is mismatched, msgpack::type_error is thrown | ||||
|     } | ||||
|  | ||||
| API document and other example codes are available at the [wiki.](http://msgpack.sourceforge.net/start) | ||||
| API documents and other example codes are available at the [wiki.](http://redmine.msgpack.org/projects/msgpack/wiki) | ||||
|  | ||||
|  | ||||
| ## License | ||||
|  | ||||
| Copyright (C) 2008-2010 FURUHASHI Sadayuki | ||||
|  | ||||
|    Licensed 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. | ||||
|     Copyright (C) 2008-2010 FURUHASHI Sadayuki | ||||
|      | ||||
|        Licensed 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. | ||||
|  | ||||
| See also NOTICE file. | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| AC_INIT(src/object.cpp) | ||||
| AC_CONFIG_AUX_DIR(ac) | ||||
| AM_INIT_AUTOMAKE(msgpack, 0.5.0) | ||||
| AM_INIT_AUTOMAKE(msgpack, 0.5.4) | ||||
| AC_CONFIG_HEADER(config.h) | ||||
|  | ||||
| AC_SUBST(CFLAGS) | ||||
| @@ -9,23 +9,31 @@ CFLAGS="-O4 -Wall $CFLAGS" | ||||
| AC_SUBST(CXXFLAGS) | ||||
| CXXFLAGS="-O4 -Wall $CXXFLAGS" | ||||
|  | ||||
|  | ||||
| AC_PROG_CC | ||||
| AC_PROG_CXX | ||||
|  | ||||
|  | ||||
| AC_MSG_CHECKING([if C++ API is enabled]) | ||||
| AC_ARG_ENABLE(cxx, | ||||
| 	AS_HELP_STRING([--disable-cxx], | ||||
| 				   [don't build C++ API]) ) | ||||
| AC_MSG_RESULT([$enable_cxx]) | ||||
| if test "$enable_cxx" != "no"; then | ||||
| 	AC_PROG_CXX | ||||
| 	AM_PROG_CC_C_O | ||||
| fi | ||||
| AM_CONDITIONAL(ENABLE_CXX, test "$enable_cxx" != "no") | ||||
|  | ||||
|  | ||||
| AC_PROG_LIBTOOL | ||||
| AM_PROG_AS | ||||
| AM_PROG_CC_C_O | ||||
|  | ||||
| AC_LANG_PUSH([C++]) | ||||
| AC_CHECK_HEADERS(tr1/unordered_map) | ||||
| AC_CHECK_HEADERS(tr1/unordered_set) | ||||
| AC_LANG_POP([C++]) | ||||
|  | ||||
|  | ||||
| AC_MSG_CHECKING([if debug option is enabled]) | ||||
| AC_ARG_ENABLE(debug, | ||||
| 	AS_HELP_STRING([--disable-debug], | ||||
| 				   [disable assert macros and omit -g option.]) ) | ||||
| 				   [disable assert macros and omit -g option]) ) | ||||
| AC_MSG_RESULT([$enable_debug]) | ||||
| if test "$enable_debug" != "no"; then | ||||
| 	CXXFLAGS="$CXXFLAGS -g" | ||||
| 	CFLAGS="$CFLAGS -g" | ||||
| @@ -33,7 +41,6 @@ else | ||||
| 	CXXFLAGS="$CXXFLAGS -DNDEBUG" | ||||
| 	CFLAGS="$CFLAGS -DNDEBUG" | ||||
| fi | ||||
| AC_MSG_RESULT($enable_debug) | ||||
|  | ||||
|  | ||||
| AC_CACHE_CHECK([for __sync_* atomic operations], msgpack_cv_atomic_ops, [ | ||||
| @@ -54,5 +61,15 @@ add CFLAGS="--march=i686" and CXXFLAGS="-march=i686" options to ./configure as f | ||||
| ]) | ||||
| fi | ||||
|  | ||||
| AC_OUTPUT([Makefile src/Makefile test/Makefile]) | ||||
|  | ||||
| major=`echo $VERSION | sed 's/\([[0-9]]*\)\.\([[0-9]]*\).*/\1/'` | ||||
| minor=`echo $VERSION | sed 's/\([[0-9]]*\)\.\([[0-9]]*\).*/\2/'` | ||||
| AC_SUBST(VERSION_MAJOR, $major) | ||||
| AC_SUBST(VERSION_MINOR, $minor) | ||||
|  | ||||
|  | ||||
| AC_OUTPUT([Makefile | ||||
| 		   src/Makefile | ||||
| 		   src/msgpack/version.h | ||||
| 		   test/Makefile]) | ||||
|  | ||||
|   | ||||
							
								
								
									
										45
									
								
								cpp/msgpack_vc.postbuild.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								cpp/msgpack_vc.postbuild.bat
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| IF NOT EXIST include                  MKDIR include | ||||
| IF NOT EXIST include\msgpack          MKDIR include\msgpack | ||||
| IF NOT EXIST include\msgpack\type     MKDIR include\msgpack\type | ||||
| IF NOT EXIST include\msgpack\type\tr1 MKDIR include\msgpack\type\tr1 | ||||
| copy src\msgpack\pack_define.h      include\msgpack\ | ||||
| copy src\msgpack\pack_template.h    include\msgpack\ | ||||
| copy src\msgpack\unpack_define.h    include\msgpack\ | ||||
| copy src\msgpack\unpack_template.h  include\msgpack\ | ||||
| copy src\msgpack\sysdep.h           include\msgpack\ | ||||
| copy src\msgpack.h                     include\ | ||||
| copy src\msgpack\sbuffer.h             include\msgpack\ | ||||
| copy src\msgpack\version.h             include\msgpack\ | ||||
| copy src\msgpack\vrefbuffer.h          include\msgpack\ | ||||
| copy src\msgpack\zbuffer.h             include\msgpack\ | ||||
| copy src\msgpack\pack.h                include\msgpack\ | ||||
| copy src\msgpack\unpack.h              include\msgpack\ | ||||
| copy src\msgpack\object.h              include\msgpack\ | ||||
| copy src\msgpack\zone.h                include\msgpack\ | ||||
| copy src\msgpack.hpp                   include\ | ||||
| copy src\msgpack\sbuffer.hpp           include\msgpack\ | ||||
| copy src\msgpack\vrefbuffer.hpp        include\msgpack\ | ||||
| copy src\msgpack\zbuffer.hpp           include\msgpack\ | ||||
| copy src\msgpack\pack.hpp              include\msgpack\ | ||||
| copy src\msgpack\unpack.hpp            include\msgpack\ | ||||
| copy src\msgpack\object.hpp            include\msgpack\ | ||||
| copy src\msgpack\zone.hpp              include\msgpack\ | ||||
| copy src\msgpack\type.hpp              include\msgpack\type\ | ||||
| copy src\msgpack\type\bool.hpp         include\msgpack\type\ | ||||
| copy src\msgpack\type\deque.hpp        include\msgpack\type\ | ||||
| copy src\msgpack\type\fixint.hpp       include\msgpack\type\ | ||||
| copy src\msgpack\type\float.hpp        include\msgpack\type\ | ||||
| copy src\msgpack\type\int.hpp          include\msgpack\type\ | ||||
| copy src\msgpack\type\list.hpp         include\msgpack\type\ | ||||
| copy src\msgpack\type\map.hpp          include\msgpack\type\ | ||||
| copy src\msgpack\type\nil.hpp          include\msgpack\type\ | ||||
| copy src\msgpack\type\pair.hpp         include\msgpack\type\ | ||||
| copy src\msgpack\type\raw.hpp          include\msgpack\type\ | ||||
| copy src\msgpack\type\set.hpp          include\msgpack\type\ | ||||
| copy src\msgpack\type\string.hpp       include\msgpack\type\ | ||||
| copy src\msgpack\type\vector.hpp       include\msgpack\type\ | ||||
| copy src\msgpack\type\tuple.hpp        include\msgpack\type\ | ||||
| copy src\msgpack\type\define.hpp       include\msgpack\type\ | ||||
| copy src\msgpack\type\tr1\unordered_map.hpp  include\msgpack\type\ | ||||
| copy src\msgpack\type\tr1\unordered_set.hpp  include\msgpack\type\ | ||||
|  | ||||
| @@ -1,43 +0,0 @@ | ||||
| IF NOT EXIST include                  MKDIR include | ||||
| IF NOT EXIST include\msgpack          MKDIR include\msgpack | ||||
| IF NOT EXIST include\msgpack\type     MKDIR include\msgpack\type | ||||
| IF NOT EXIST include\msgpack\type\tr1 MKDIR include\msgpack\type\tr1 | ||||
| copy msgpack\pack_define.h      include\msgpack\ | ||||
| copy msgpack\pack_template.h    include\msgpack\ | ||||
| copy msgpack\unpack_define.h    include\msgpack\ | ||||
| copy msgpack\unpack_template.h  include\msgpack\ | ||||
| copy msgpack\sysdep.h           include\msgpack\ | ||||
| copy msgpack.h                     include\ | ||||
| copy msgpack\sbuffer.h             include\msgpack\ | ||||
| copy msgpack\vrefbuffer.h          include\msgpack\ | ||||
| copy msgpack\zbuffer.h             include\msgpack\ | ||||
| copy msgpack\pack.h                include\msgpack\ | ||||
| copy msgpack\unpack.h              include\msgpack\ | ||||
| copy msgpack\object.h              include\msgpack\ | ||||
| copy msgpack\zone.h                include\msgpack\ | ||||
| copy msgpack.hpp                   include\ | ||||
| copy msgpack\sbuffer.hpp           include\msgpack\ | ||||
| copy msgpack\vrefbuffer.hpp        include\msgpack\ | ||||
| copy msgpack\zbuffer.hpp           include\msgpack\ | ||||
| copy msgpack\pack.hpp              include\msgpack\ | ||||
| copy msgpack\unpack.hpp            include\msgpack\ | ||||
| copy msgpack\object.hpp            include\msgpack\ | ||||
| copy msgpack\zone.hpp              include\msgpack\ | ||||
| copy msgpack\type.hpp              include\msgpack\type\ | ||||
| copy msgpack\type\bool.hpp         include\msgpack\type\ | ||||
| copy msgpack\type\float.hpp        include\msgpack\type\ | ||||
| copy msgpack\type\int.hpp          include\msgpack\type\ | ||||
| copy msgpack\type\list.hpp         include\msgpack\type\ | ||||
| copy msgpack\type\deque.hpp        include\msgpack\type\ | ||||
| copy msgpack\type\map.hpp          include\msgpack\type\ | ||||
| copy msgpack\type\nil.hpp          include\msgpack\type\ | ||||
| copy msgpack\type\pair.hpp         include\msgpack\type\ | ||||
| copy msgpack\type\raw.hpp          include\msgpack\type\ | ||||
| copy msgpack\type\set.hpp          include\msgpack\type\ | ||||
| copy msgpack\type\string.hpp       include\msgpack\type\ | ||||
| copy msgpack\type\vector.hpp       include\msgpack\type\ | ||||
| copy msgpack\type\tuple.hpp        include\msgpack\type\ | ||||
| copy msgpack\type\define.hpp       include\msgpack\type\ | ||||
| copy msgpack\type\tr1\unordered_map.hpp  include\msgpack\type\ | ||||
| copy msgpack\type\tr1\unordered_set.hpp  include\msgpack\type\ | ||||
|  | ||||
| @@ -28,7 +28,7 @@ | ||||
| 			<Tool | ||||
| 				Name="VCCustomBuildTool" | ||||
| 				Description="Gathering header files" | ||||
| 				CommandLine="msgpack_vc8.postbuild.bat" | ||||
| 				CommandLine="msgpack_vc.postbuild.bat" | ||||
| 				Outputs="include" | ||||
| 			/> | ||||
| 			<Tool | ||||
| @@ -96,7 +96,7 @@ | ||||
| 			<Tool | ||||
| 				Name="VCCustomBuildTool" | ||||
| 				Description="Gathering header files" | ||||
| 				CommandLine="msgpack_vc8.postbuild.bat" | ||||
| 				CommandLine="msgpack_vc.postbuild.bat" | ||||
| 				Outputs="include" | ||||
| 			/> | ||||
| 			<Tool | ||||
| @@ -157,7 +157,7 @@ | ||||
| 			UniqueIdentifier="{4FC737F1-C7A5-4376-A066-2A32D752A2FF}" | ||||
| 			> | ||||
| 			<File | ||||
| 				RelativePath=".\objectc.c" | ||||
| 				RelativePath=".\src\objectc.c" | ||||
| 				> | ||||
| 				<FileConfiguration | ||||
| 					Name="Debug|Win32" | ||||
| @@ -177,7 +177,7 @@ | ||||
| 				</FileConfiguration> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\unpack.c" | ||||
| 				RelativePath=".\src\unpack.c" | ||||
| 				> | ||||
| 				<FileConfiguration | ||||
| 					Name="Debug|Win32" | ||||
| @@ -197,7 +197,7 @@ | ||||
| 				</FileConfiguration> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\vrefbuffer.c" | ||||
| 				RelativePath=".\src\version.c" | ||||
| 				> | ||||
| 				<FileConfiguration | ||||
| 					Name="Debug|Win32" | ||||
| @@ -217,7 +217,7 @@ | ||||
| 				</FileConfiguration> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\zone.c" | ||||
| 				RelativePath=".\src\vrefbuffer.c" | ||||
| 				> | ||||
| 				<FileConfiguration | ||||
| 					Name="Debug|Win32" | ||||
| @@ -237,7 +237,27 @@ | ||||
| 				</FileConfiguration> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\object.cpp" | ||||
| 				RelativePath=".\src\zone.c" | ||||
| 				> | ||||
| 				<FileConfiguration | ||||
| 					Name="Debug|Win32" | ||||
| 					> | ||||
| 					<Tool | ||||
| 						Name="VCCLCompilerTool" | ||||
| 						CompileAs="2" | ||||
| 					/> | ||||
| 				</FileConfiguration> | ||||
| 				<FileConfiguration | ||||
| 					Name="Release|Win32" | ||||
| 					> | ||||
| 					<Tool | ||||
| 						Name="VCCLCompilerTool" | ||||
| 						CompileAs="2" | ||||
| 					/> | ||||
| 				</FileConfiguration> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\src\object.cpp" | ||||
| 				> | ||||
| 			</File> | ||||
| 		</Filter> | ||||
| @@ -247,23 +267,23 @@ | ||||
| 			UniqueIdentifier="{93995380-89BD-4b04-88EB-625FBE52EBFB}" | ||||
| 			> | ||||
| 			<File | ||||
| 				RelativePath=".\msgpack\pack_define.h" | ||||
| 				RelativePath=".\src\msgpack\pack_define.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\msgpack\pack_template.h" | ||||
| 				RelativePath=".\src\msgpack\pack_template.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\msgpack\sysdep.h" | ||||
| 				RelativePath=".\src\msgpack\sysdep.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\msgpack\unpack_define.h" | ||||
| 				RelativePath=".\src\msgpack\unpack_define.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath=".\msgpack\unpack_template.h" | ||||
| 				RelativePath=".\src\msgpack\unpack_template.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 		</Filter> | ||||
|   | ||||
| @@ -29,3 +29,6 @@ cp -f ../msgpack/unpack_template.h src/msgpack/ | ||||
| cp -f ../test/cases.mpac           test/ | ||||
| cp -f ../test/cases_compact.mpac   test/ | ||||
|  | ||||
| sed -e 's/8\.00/9.00/' < msgpack_vc8.vcproj > msgpack_vc2008.vcproj | ||||
| sed -e 's/9\.00/10.00/' -e 's/msgpack_vc8/msgpack_vc2008/' < msgpack_vc8.sln > msgpack_vc2008.sln | ||||
|  | ||||
|   | ||||
| @@ -4,9 +4,14 @@ lib_LTLIBRARIES = libmsgpack.la | ||||
| libmsgpack_la_SOURCES = \ | ||||
| 		unpack.c \ | ||||
| 		objectc.c \ | ||||
| 		version.c \ | ||||
| 		vrefbuffer.c \ | ||||
| 		zone.c \ | ||||
| 		zone.c | ||||
|  | ||||
| if ENABLE_CXX | ||||
| libmsgpack_la_SOURCES += \ | ||||
| 		object.cpp | ||||
| endif | ||||
|  | ||||
| # -version-info CURRENT:REVISION:AGE | ||||
| libmsgpack_la_LDFLAGS = -version-info 3:0:0 | ||||
| @@ -18,15 +23,12 @@ lib_LTLIBRARIES += libmsgpackc.la | ||||
| libmsgpackc_la_SOURCES = \ | ||||
| 		unpack.c \ | ||||
| 		objectc.c \ | ||||
| 		version.c \ | ||||
| 		vrefbuffer.c \ | ||||
| 		zone.c | ||||
|  | ||||
| libmsgpackc_la_LDFLAGS = -version-info 2:0:0 | ||||
|  | ||||
| # work around for duplicated file name | ||||
| kumo_manager_CFLAGS = $(AM_CFLAGS) | ||||
| kumo_manager_CXXFLAGS = $(AM_CXXFLAGS) | ||||
|  | ||||
|  | ||||
| nobase_include_HEADERS = \ | ||||
| 		msgpack/pack_define.h \ | ||||
| @@ -36,12 +38,16 @@ nobase_include_HEADERS = \ | ||||
| 		msgpack/sysdep.h \ | ||||
| 		msgpack.h \ | ||||
| 		msgpack/sbuffer.h \ | ||||
| 		msgpack/version.h \ | ||||
| 		msgpack/vrefbuffer.h \ | ||||
| 		msgpack/zbuffer.h \ | ||||
| 		msgpack/pack.h \ | ||||
| 		msgpack/unpack.h \ | ||||
| 		msgpack/object.h \ | ||||
| 		msgpack/zone.h \ | ||||
| 		msgpack/zone.h | ||||
|  | ||||
| if ENABLE_CXX | ||||
| nobase_include_HEADERS += \ | ||||
| 		msgpack.hpp \ | ||||
| 		msgpack/sbuffer.hpp \ | ||||
| 		msgpack/vrefbuffer.hpp \ | ||||
| @@ -52,10 +58,11 @@ nobase_include_HEADERS = \ | ||||
| 		msgpack/zone.hpp \ | ||||
| 		msgpack/type.hpp \ | ||||
| 		msgpack/type/bool.hpp \ | ||||
| 		msgpack/type/deque.hpp \ | ||||
| 		msgpack/type/float.hpp \ | ||||
| 		msgpack/type/fixint.hpp \ | ||||
| 		msgpack/type/int.hpp \ | ||||
| 		msgpack/type/list.hpp \ | ||||
| 		msgpack/type/deque.hpp \ | ||||
| 		msgpack/type/map.hpp \ | ||||
| 		msgpack/type/nil.hpp \ | ||||
| 		msgpack/type/pair.hpp \ | ||||
| @@ -67,8 +74,10 @@ nobase_include_HEADERS = \ | ||||
| 		msgpack/type/define.hpp \ | ||||
| 		msgpack/type/tr1/unordered_map.hpp \ | ||||
| 		msgpack/type/tr1/unordered_set.hpp | ||||
| endif | ||||
|  | ||||
| EXTRA_DIST = \ | ||||
| 		msgpack/version.h.in \ | ||||
| 		msgpack/zone.hpp.erb \ | ||||
| 		msgpack/type/define.hpp.erb \ | ||||
| 		msgpack/type/tuple.hpp.erb | ||||
|   | ||||
| @@ -26,3 +26,5 @@ | ||||
| #include "msgpack/unpack.h" | ||||
| #include "msgpack/sbuffer.h" | ||||
| #include "msgpack/vrefbuffer.h" | ||||
| #include "msgpack/version.h" | ||||
|  | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_OBJECT_H__ | ||||
| #define MSGPACK_OBJECT_H__ | ||||
|  | ||||
| #include "msgpack/zone.h" | ||||
| #include "zone.h" | ||||
| #include <stdio.h> | ||||
|  | ||||
| #ifdef __cplusplus | ||||
|   | ||||
| @@ -18,9 +18,9 @@ | ||||
| #ifndef MSGPACK_OBJECT_HPP__ | ||||
| #define MSGPACK_OBJECT_HPP__ | ||||
|  | ||||
| #include "msgpack/object.h" | ||||
| #include "msgpack/pack.hpp" | ||||
| #include "msgpack/zone.hpp" | ||||
| #include "object.h" | ||||
| #include "pack.hpp" | ||||
| #include "zone.hpp" | ||||
| #include <string.h> | ||||
| #include <stdexcept> | ||||
| #include <typeinfo> | ||||
|   | ||||
| @@ -18,8 +18,8 @@ | ||||
| #ifndef MSGPACK_PACK_H__ | ||||
| #define MSGPACK_PACK_H__ | ||||
|  | ||||
| #include "msgpack/pack_define.h" | ||||
| #include "msgpack/object.h" | ||||
| #include "pack_define.h" | ||||
| #include "object.h" | ||||
| #include <stdlib.h> | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| @@ -70,6 +70,15 @@ static int msgpack_pack_int16(msgpack_packer* pk, int16_t d); | ||||
| static int msgpack_pack_int32(msgpack_packer* pk, int32_t d); | ||||
| static int msgpack_pack_int64(msgpack_packer* pk, int64_t d); | ||||
|  | ||||
| static int msgpack_pack_fix_uint8(msgpack_packer* pk, uint8_t d); | ||||
| static int msgpack_pack_fix_uint16(msgpack_packer* pk, uint16_t d); | ||||
| static int msgpack_pack_fix_uint32(msgpack_packer* pk, uint32_t d); | ||||
| static int msgpack_pack_fix_uint64(msgpack_packer* pk, uint64_t d); | ||||
| static int msgpack_pack_fix_int8(msgpack_packer* pk, int8_t d); | ||||
| static int msgpack_pack_fix_int16(msgpack_packer* pk, int16_t d); | ||||
| static int msgpack_pack_fix_int32(msgpack_packer* pk, int32_t d); | ||||
| static int msgpack_pack_fix_int64(msgpack_packer* pk, int64_t d); | ||||
|  | ||||
| static int msgpack_pack_float(msgpack_packer* pk, float d); | ||||
| static int msgpack_pack_double(msgpack_packer* pk, double d); | ||||
|  | ||||
| @@ -96,12 +105,18 @@ int msgpack_pack_object(msgpack_packer* pk, msgpack_object d); | ||||
| #define msgpack_pack_inline_func_cint(name) \ | ||||
| 	inline int msgpack_pack ## name | ||||
|  | ||||
| #define msgpack_pack_inline_func_cint(name) \ | ||||
| 	inline int msgpack_pack ## name | ||||
|  | ||||
| #define msgpack_pack_inline_func_fixint(name) \ | ||||
| 	inline int msgpack_pack_fix ## name | ||||
|  | ||||
| #define msgpack_pack_user msgpack_packer* | ||||
|  | ||||
| #define msgpack_pack_append_buffer(user, buf, len) \ | ||||
| 	return (*(user)->callback)((user)->data, (const char*)buf, len) | ||||
|  | ||||
| #include "msgpack/pack_template.h" | ||||
| #include "pack_template.h" | ||||
|  | ||||
| inline void msgpack_packer_init(msgpack_packer* pk, void* data, msgpack_packer_write callback) | ||||
| { | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_PACK_HPP__ | ||||
| #define MSGPACK_PACK_HPP__ | ||||
|  | ||||
| #include "msgpack/pack_define.h" | ||||
| #include "pack_define.h" | ||||
| #include <stdexcept> | ||||
| #include <limits.h> | ||||
|  | ||||
| @@ -45,6 +45,15 @@ public: | ||||
| 	packer<Stream>& pack_int32(int32_t d); | ||||
| 	packer<Stream>& pack_int64(int64_t d); | ||||
|  | ||||
| 	packer<Stream>& pack_fix_uint8(uint8_t d); | ||||
| 	packer<Stream>& pack_fix_uint16(uint16_t d); | ||||
| 	packer<Stream>& pack_fix_uint32(uint32_t d); | ||||
| 	packer<Stream>& pack_fix_uint64(uint64_t d); | ||||
| 	packer<Stream>& pack_fix_int8(int8_t d); | ||||
| 	packer<Stream>& pack_fix_int16(int16_t d); | ||||
| 	packer<Stream>& pack_fix_int32(int32_t d); | ||||
| 	packer<Stream>& pack_fix_int64(int64_t d); | ||||
|  | ||||
| 	packer<Stream>& pack_short(short d); | ||||
| 	packer<Stream>& pack_int(int d); | ||||
| 	packer<Stream>& pack_long(long d); | ||||
| @@ -78,6 +87,15 @@ private: | ||||
| 	static void _pack_int32(Stream& x, int32_t d); | ||||
| 	static void _pack_int64(Stream& x, int64_t d); | ||||
|  | ||||
| 	static void _pack_fix_uint8(Stream& x, uint8_t d); | ||||
| 	static void _pack_fix_uint16(Stream& x, uint16_t d); | ||||
| 	static void _pack_fix_uint32(Stream& x, uint32_t d); | ||||
| 	static void _pack_fix_uint64(Stream& x, uint64_t d); | ||||
| 	static void _pack_fix_int8(Stream& x, int8_t d); | ||||
| 	static void _pack_fix_int16(Stream& x, int16_t d); | ||||
| 	static void _pack_fix_int32(Stream& x, int32_t d); | ||||
| 	static void _pack_fix_int64(Stream& x, int64_t d); | ||||
|  | ||||
| 	static void _pack_short(Stream& x, short d); | ||||
| 	static void _pack_int(Stream& x, int d); | ||||
| 	static void _pack_long(Stream& x, long d); | ||||
| @@ -133,11 +151,15 @@ inline void pack(Stream& s, const T& v) | ||||
| 	template <typename Stream> \ | ||||
| 	inline void packer<Stream>::_pack ## name | ||||
|  | ||||
| #define msgpack_pack_inline_func_fixint(name) \ | ||||
| 	template <typename Stream> \ | ||||
| 	inline void packer<Stream>::_pack_fix ## name | ||||
|  | ||||
| #define msgpack_pack_user Stream& | ||||
|  | ||||
| #define msgpack_pack_append_buffer append_buffer | ||||
|  | ||||
| #include "msgpack/pack_template.h" | ||||
| #include "pack_template.h" | ||||
|  | ||||
|  | ||||
| template <typename Stream> | ||||
| @@ -149,6 +171,7 @@ packer<Stream>::packer(Stream& s) : m_stream(s) { } | ||||
| template <typename Stream> | ||||
| packer<Stream>::~packer() { } | ||||
|  | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_uint8(uint8_t d) | ||||
| { _pack_uint8(m_stream, d); return *this; } | ||||
| @@ -182,6 +205,39 @@ inline packer<Stream>& packer<Stream>::pack_int64(int64_t d) | ||||
| { _pack_int64(m_stream, d); return *this;} | ||||
|  | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_uint8(uint8_t d) | ||||
| { _pack_fix_uint8(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_uint16(uint16_t d) | ||||
| { _pack_fix_uint16(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_uint32(uint32_t d) | ||||
| { _pack_fix_uint32(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_uint64(uint64_t d) | ||||
| { _pack_fix_uint64(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_int8(int8_t d) | ||||
| { _pack_fix_int8(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_int16(int16_t d) | ||||
| { _pack_fix_int16(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_int32(int32_t d) | ||||
| { _pack_fix_int32(m_stream, d); return *this; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_fix_int64(int64_t d) | ||||
| { _pack_fix_int64(m_stream, d); return *this;} | ||||
|  | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& packer<Stream>::pack_short(short d) | ||||
| { _pack_short(m_stream, d); return *this; } | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_SBUFFER_HPP__ | ||||
| #define MSGPACK_SBUFFER_HPP__ | ||||
|  | ||||
| #include "msgpack/sbuffer.h" | ||||
| #include "sbuffer.h" | ||||
| #include <stdexcept> | ||||
|  | ||||
| namespace msgpack { | ||||
|   | ||||
| @@ -1,15 +1,16 @@ | ||||
| #include "msgpack/type/bool.hpp" | ||||
| #include "msgpack/type/float.hpp" | ||||
| #include "msgpack/type/int.hpp" | ||||
| #include "msgpack/type/list.hpp" | ||||
| #include "msgpack/type/deque.hpp" | ||||
| #include "msgpack/type/map.hpp" | ||||
| #include "msgpack/type/nil.hpp" | ||||
| #include "msgpack/type/pair.hpp" | ||||
| #include "msgpack/type/raw.hpp" | ||||
| #include "msgpack/type/set.hpp" | ||||
| #include "msgpack/type/string.hpp" | ||||
| #include "msgpack/type/vector.hpp" | ||||
| #include "msgpack/type/tuple.hpp" | ||||
| #include "msgpack/type/define.hpp" | ||||
| #include "type/bool.hpp" | ||||
| #include "type/deque.hpp" | ||||
| #include "type/fixint.hpp" | ||||
| #include "type/float.hpp" | ||||
| #include "type/int.hpp" | ||||
| #include "type/list.hpp" | ||||
| #include "type/map.hpp" | ||||
| #include "type/nil.hpp" | ||||
| #include "type/pair.hpp" | ||||
| #include "type/raw.hpp" | ||||
| #include "type/set.hpp" | ||||
| #include "type/string.hpp" | ||||
| #include "type/vector.hpp" | ||||
| #include "type/tuple.hpp" | ||||
| #include "type/define.hpp" | ||||
|  | ||||
|   | ||||
							
								
								
									
										172
									
								
								cpp/src/msgpack/type/fixint.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								cpp/src/msgpack/type/fixint.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | ||||
| // | ||||
| // MessagePack for C++ static resolution routine | ||||
| // | ||||
| // Copyright (C) 2020 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| #ifndef MSGPACK_TYPE_FIXINT_HPP__ | ||||
| #define MSGPACK_TYPE_FIXINT_HPP__ | ||||
|  | ||||
| #include "msgpack/object.hpp" | ||||
| #include "msgpack/type/int.hpp" | ||||
|  | ||||
| namespace msgpack { | ||||
|  | ||||
| namespace type { | ||||
|  | ||||
|  | ||||
| template <typename T> | ||||
| struct fix_int { | ||||
| 	fix_int() : value(0) { } | ||||
| 	fix_int(T value) : value(value) { } | ||||
|  | ||||
| 	operator T() const { return value; } | ||||
|  | ||||
| 	T get() const { return value; } | ||||
|  | ||||
| private: | ||||
| 	T value; | ||||
| }; | ||||
|  | ||||
|  | ||||
| typedef fix_int<uint8_t>  fix_uint8; | ||||
| typedef fix_int<uint16_t> fix_uint16; | ||||
| typedef fix_int<uint32_t> fix_uint32; | ||||
| typedef fix_int<uint64_t> fix_uint64; | ||||
|  | ||||
| typedef fix_int<int8_t>  fix_int8; | ||||
| typedef fix_int<int16_t> fix_int16; | ||||
| typedef fix_int<int32_t> fix_int32; | ||||
| typedef fix_int<int64_t> fix_int64; | ||||
|  | ||||
|  | ||||
| }  // namespace type | ||||
|  | ||||
|  | ||||
| inline type::fix_int8& operator>> (object o, type::fix_int8& v) | ||||
| 	{ v = type::detail::convert_integer<int8_t>(o); return v; } | ||||
|  | ||||
| inline type::fix_int16& operator>> (object o, type::fix_int16& v) | ||||
| 	{ v = type::detail::convert_integer<int16_t>(o); return v; } | ||||
|  | ||||
| inline type::fix_int32& operator>> (object o, type::fix_int32& v) | ||||
| 	{ v = type::detail::convert_integer<int32_t>(o); return v; } | ||||
|  | ||||
| inline type::fix_int64& operator>> (object o, type::fix_int64& v) | ||||
| 	{ v = type::detail::convert_integer<int64_t>(o); return v; } | ||||
|  | ||||
|  | ||||
| inline type::fix_uint8& operator>> (object o, type::fix_uint8& v) | ||||
| 	{ v = type::detail::convert_integer<uint8_t>(o); return v; } | ||||
|  | ||||
| inline type::fix_uint16& operator>> (object o, type::fix_uint16& v) | ||||
| 	{ v = type::detail::convert_integer<uint16_t>(o); return v; } | ||||
|  | ||||
| inline type::fix_uint32& operator>> (object o, type::fix_uint32& v) | ||||
| 	{ v = type::detail::convert_integer<uint32_t>(o); return v; } | ||||
|  | ||||
| inline type::fix_uint64& operator>> (object o, type::fix_uint64& v) | ||||
| 	{ v = type::detail::convert_integer<uint64_t>(o); return v; } | ||||
|  | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int8& v) | ||||
| 	{ o.pack_fix_int8(v); return o; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int16& v) | ||||
| 	{ o.pack_fix_int16(v); return o; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int32& v) | ||||
| 	{ o.pack_fix_int32(v); return o; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int64& v) | ||||
| 	{ o.pack_fix_int64(v); return o; } | ||||
|  | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint8& v) | ||||
| 	{ o.pack_fix_uint8(v); return o; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint16& v) | ||||
| 	{ o.pack_fix_uint16(v); return o; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint32& v) | ||||
| 	{ o.pack_fix_uint32(v); return o; } | ||||
|  | ||||
| template <typename Stream> | ||||
| inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint64& v) | ||||
| 	{ o.pack_fix_uint64(v); return o; } | ||||
|  | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_int8 v) | ||||
| 	{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_int16 v) | ||||
| 	{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_int32 v) | ||||
| 	{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_int64 v) | ||||
| 	{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_uint8 v) | ||||
| 	{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_uint16 v) | ||||
| 	{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_uint32 v) | ||||
| 	{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
| inline void operator<< (object& o, type::fix_uint64 v) | ||||
| 	{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); } | ||||
|  | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_int8 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_int16 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_int32 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_int64 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_uint8 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_uint16 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_uint32 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
| inline void operator<< (object::with_zone& o, type::fix_uint64 v) | ||||
| 	{ static_cast<object&>(o) << v; } | ||||
|  | ||||
|  | ||||
| }  // namespace msgpack | ||||
|  | ||||
| #endif /* msgpack/type/fixint.hpp */ | ||||
|  | ||||
| @@ -33,25 +33,25 @@ struct raw_ref { | ||||
| 	uint32_t size; | ||||
| 	const char* ptr; | ||||
|  | ||||
| 	std::string str() { return std::string(ptr, size); } | ||||
| 	std::string str() const { return std::string(ptr, size); } | ||||
|  | ||||
| 	bool operator== (const raw_ref& x) | ||||
| 	bool operator== (const raw_ref& x) const | ||||
| 	{ | ||||
| 		return size == x.size && memcmp(ptr, x.ptr, size) == 0; | ||||
| 	} | ||||
|  | ||||
| 	bool operator!= (const raw_ref& x) | ||||
| 	bool operator!= (const raw_ref& x) const | ||||
| 	{ | ||||
| 		return !(*this != x); | ||||
| 	} | ||||
|  | ||||
| 	bool operator< (const raw_ref& x) | ||||
| 	bool operator< (const raw_ref& x) const | ||||
| 	{ | ||||
| 		if(size == x.size) { return memcmp(ptr, x.ptr, size) < 0; } | ||||
| 		else { return size < x.size; } | ||||
| 	} | ||||
|  | ||||
| 	bool operator> (const raw_ref& x) | ||||
| 	bool operator> (const raw_ref& x) const | ||||
| 	{ | ||||
| 		if(size == x.size) { return memcmp(ptr, x.ptr, size) > 0; } | ||||
| 		else { return size > x.size; } | ||||
|   | ||||
| @@ -187,5 +187,20 @@ inline void operator<< ( | ||||
|  | ||||
| }  // namespace msgpack | ||||
|  | ||||
|  | ||||
| //inline std::ostream& operator<< (std::ostream& o, const msgpack::type::tuple<>& v) { | ||||
| //  return o << "[]"; | ||||
| //} | ||||
| //<%0.upto(GENERATION_LIMIT) {|i|%> | ||||
| //template <typename A0<%1.upto(i) {|j|%>, typename A<%=j%><%}%>> | ||||
| //inline std::ostream& operator<< (std::ostream& o, | ||||
| //		const msgpack::type::tuple<A0<%1.upto(i) {|j|%>, A<%=j%><%}%>>& v) { | ||||
| //	return o << "[" | ||||
| //	<%0.upto(i) {|j|%> | ||||
| //	<<<%if j != 0 then%> ", " <<<%end%> v.template get<<%=j%>>()<%}%> | ||||
| //	<< "]"; | ||||
| //} | ||||
| //<%}%> | ||||
|  | ||||
| #endif /* msgpack/type/tuple.hpp */ | ||||
|  | ||||
|   | ||||
| @@ -18,8 +18,8 @@ | ||||
| #ifndef MSGPACK_UNPACKER_H__ | ||||
| #define MSGPACK_UNPACKER_H__ | ||||
|  | ||||
| #include "msgpack/zone.h" | ||||
| #include "msgpack/object.h" | ||||
| #include "zone.h" | ||||
| #include "object.h" | ||||
| #include <string.h> | ||||
|  | ||||
| #ifdef __cplusplus | ||||
|   | ||||
| @@ -18,9 +18,9 @@ | ||||
| #ifndef MSGPACK_UNPACK_HPP__ | ||||
| #define MSGPACK_UNPACK_HPP__ | ||||
|  | ||||
| #include "msgpack/unpack.h" | ||||
| #include "msgpack/object.hpp" | ||||
| #include "msgpack/zone.hpp" | ||||
| #include "unpack.h" | ||||
| #include "object.hpp" | ||||
| #include "zone.hpp" | ||||
| #include <memory> | ||||
| #include <stdexcept> | ||||
|  | ||||
|   | ||||
							
								
								
									
										40
									
								
								cpp/src/msgpack/version.h.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								cpp/src/msgpack/version.h.in
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| /* | ||||
|  * MessagePack for C version information | ||||
|  * | ||||
|  * Copyright (C) 2008-2009 FURUHASHI Sadayuki | ||||
|  * | ||||
|  *    Licensed 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. | ||||
|  */ | ||||
| #ifndef MSGPACK_VERSION_H__ | ||||
| #define MSGPACK_VERSION_H__ | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
|  | ||||
| const char* msgpack_version(void); | ||||
| int msgpack_version_major(void); | ||||
| int msgpack_version_minor(void); | ||||
|  | ||||
| #define MSGPACK_VERSION "@VERSION@" | ||||
| #define MSGPACK_VERSION_MAJOR @VERSION_MAJOR@ | ||||
| #define MSGPACK_VERSION_MINOR @VERSION_MINOR@ | ||||
|  | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #endif /* msgpack/version.h */ | ||||
|  | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_VREFBUFFER_H__ | ||||
| #define MSGPACK_VREFBUFFER_H__ | ||||
|  | ||||
| #include "msgpack/zone.h" | ||||
| #include "zone.h" | ||||
| #include <stdlib.h> | ||||
|  | ||||
| #ifndef _WIN32 | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_VREFBUFFER_HPP__ | ||||
| #define MSGPACK_VREFBUFFER_HPP__ | ||||
|  | ||||
| #include "msgpack/vrefbuffer.h" | ||||
| #include "vrefbuffer.h" | ||||
| #include <stdexcept> | ||||
|  | ||||
| namespace msgpack { | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_ZBUFFER_H__ | ||||
| #define MSGPACK_ZBUFFER_H__ | ||||
|  | ||||
| #include "msgpack/sysdep.h" | ||||
| #include "sysdep.h" | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <zlib.h> | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_ZBUFFER_HPP__ | ||||
| #define MSGPACK_ZBUFFER_HPP__ | ||||
|  | ||||
| #include "msgpack/zbuffer.h" | ||||
| #include "zbuffer.h" | ||||
| #include <stdexcept> | ||||
|  | ||||
| namespace msgpack { | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_ZONE_H__ | ||||
| #define MSGPACK_ZONE_H__ | ||||
|  | ||||
| #include "msgpack/sysdep.h" | ||||
| #include "sysdep.h" | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
| #ifndef MSGPACK_ZONE_HPP__ | ||||
| #define MSGPACK_ZONE_HPP__ | ||||
|  | ||||
| #include "msgpack/zone.h" | ||||
| #include "zone.h" | ||||
| #include <cstdlib> | ||||
| #include <memory> | ||||
| #include <vector> | ||||
|   | ||||
							
								
								
									
										17
									
								
								cpp/src/version.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								cpp/src/version.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| #include "msgpack.h" | ||||
|  | ||||
| const char* msgpack_version(void) | ||||
| { | ||||
| 	return MSGPACK_VERSION; | ||||
| } | ||||
|  | ||||
| int msgpack_version_major(void) | ||||
| { | ||||
| 	return MSGPACK_VERSION_MAJOR; | ||||
| } | ||||
|  | ||||
| int msgpack_version_minor(void) | ||||
| { | ||||
| 	return MSGPACK_VERSION_MINOR; | ||||
| } | ||||
|  | ||||
| @@ -13,6 +13,9 @@ check_PROGRAMS = \ | ||||
| 		convert \ | ||||
| 		buffer \ | ||||
| 		cases \ | ||||
| 		fixint \ | ||||
| 		fixint_c \ | ||||
| 		version \ | ||||
| 		msgpackc_test \ | ||||
| 		msgpack_test | ||||
|  | ||||
| @@ -37,6 +40,12 @@ buffer_LDADD = -lz | ||||
|  | ||||
| cases_SOURCES = cases.cc | ||||
|  | ||||
| fixint_SOURCES = fixint.cc | ||||
|  | ||||
| fixint_c_SOURCES = fixint_c.cc | ||||
|  | ||||
| version_SOURCES = version.cc | ||||
|  | ||||
| msgpackc_test_SOURCES = msgpackc_test.cpp | ||||
|  | ||||
| msgpack_test_SOURCES = msgpack_test.cpp | ||||
|   | ||||
| @@ -32,5 +32,7 @@ TEST(cases, format) | ||||
| 		EXPECT_TRUE( pac_compact.next(&result_compact) ); | ||||
| 		EXPECT_EQ(result_compact.get(), result.get()); | ||||
| 	} | ||||
|  | ||||
| 	EXPECT_FALSE( pac_compact.next(&result) ); | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										55
									
								
								cpp/test/fixint.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								cpp/test/fixint.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| #include <msgpack.hpp> | ||||
| #include <gtest/gtest.h> | ||||
|  | ||||
| template <typename T> | ||||
| void check_size(size_t size) { | ||||
| 	T v(0); | ||||
| 	msgpack::sbuffer sbuf; | ||||
| 	msgpack::pack(sbuf, v); | ||||
| 	EXPECT_EQ(size, sbuf.size()); | ||||
| } | ||||
|  | ||||
| TEST(fixint, size) | ||||
| { | ||||
| 	check_size<msgpack::type::fix_int8>(2); | ||||
| 	check_size<msgpack::type::fix_int16>(3); | ||||
| 	check_size<msgpack::type::fix_int32>(5); | ||||
| 	check_size<msgpack::type::fix_int64>(9); | ||||
|  | ||||
| 	check_size<msgpack::type::fix_uint8>(2); | ||||
| 	check_size<msgpack::type::fix_uint16>(3); | ||||
| 	check_size<msgpack::type::fix_uint32>(5); | ||||
| 	check_size<msgpack::type::fix_uint64>(9); | ||||
| } | ||||
|  | ||||
|  | ||||
| template <typename T> | ||||
| void check_convert() { | ||||
| 	T v1(-11); | ||||
| 	msgpack::sbuffer sbuf; | ||||
| 	msgpack::pack(sbuf, v1); | ||||
|  | ||||
| 	msgpack::unpacked msg; | ||||
| 	msgpack::unpack(&msg, sbuf.data(), sbuf.size()); | ||||
|  | ||||
| 	T v2; | ||||
| 	msg.get().convert(&v2); | ||||
|  | ||||
| 	EXPECT_EQ(v1.get(), v2.get()); | ||||
|  | ||||
| 	EXPECT_EQ(msg.get(), msgpack::object(T(v1.get()))); | ||||
| } | ||||
|  | ||||
| TEST(fixint, convert) | ||||
| { | ||||
| 	check_convert<msgpack::type::fix_int8>(); | ||||
| 	check_convert<msgpack::type::fix_int16>(); | ||||
| 	check_convert<msgpack::type::fix_int32>(); | ||||
| 	check_convert<msgpack::type::fix_int64>(); | ||||
|  | ||||
| 	check_convert<msgpack::type::fix_uint8>(); | ||||
| 	check_convert<msgpack::type::fix_uint16>(); | ||||
| 	check_convert<msgpack::type::fix_uint32>(); | ||||
| 	check_convert<msgpack::type::fix_uint64>(); | ||||
| } | ||||
|  | ||||
							
								
								
									
										32
									
								
								cpp/test/fixint_c.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								cpp/test/fixint_c.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| #include <msgpack.hpp> | ||||
| #include <gtest/gtest.h> | ||||
|  | ||||
| TEST(fixint, size) | ||||
| { | ||||
| 	msgpack_sbuffer* sbuf = msgpack_sbuffer_new(); | ||||
| 	msgpack_packer* pk = msgpack_packer_new(sbuf, msgpack_sbuffer_write); | ||||
|  | ||||
| 	size_t sum = 0; | ||||
|  | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_int8(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=2, sbuf->size); | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_int16(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=3, sbuf->size); | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_int32(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=5, sbuf->size); | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_int64(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=9, sbuf->size); | ||||
|  | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_uint8(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=2, sbuf->size); | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_uint16(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=3, sbuf->size); | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_uint32(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=5, sbuf->size); | ||||
| 	EXPECT_EQ(0, msgpack_pack_fix_uint64(pk, 0)); | ||||
| 	EXPECT_EQ(sum+=9, sbuf->size); | ||||
|  | ||||
| 	msgpack_sbuffer_free(sbuf); | ||||
| 	msgpack_packer_free(pk); | ||||
| } | ||||
|  | ||||
							
								
								
									
										13
									
								
								cpp/test/version.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								cpp/test/version.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| #include <msgpack.hpp> | ||||
| #include <gtest/gtest.h> | ||||
|  | ||||
| TEST(version, print) | ||||
| { | ||||
| 	printf("MSGPACK_VERSION          : %s\n", MSGPACK_VERSION); | ||||
| 	printf("MSGPACK_VERSION_MAJOR    : %d\n", MSGPACK_VERSION_MAJOR); | ||||
| 	printf("MSGPACK_VERSION_MINOR    : %d\n", MSGPACK_VERSION_MINOR); | ||||
| 	printf("msgpack_version()        : %s\n", msgpack_version()); | ||||
| 	printf("msgpack_version_major()  : %d\n", msgpack_version_major()); | ||||
| 	printf("msgpack_version_minor()  : %d\n", msgpack_version_minor()); | ||||
| } | ||||
|  | ||||
							
								
								
									
										1
									
								
								erlang/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								erlang/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,3 +2,4 @@ MANIFEST | ||||
| *.beam | ||||
| .omakedb* | ||||
| *.omc | ||||
| *~ | ||||
| @@ -22,7 +22,7 @@ | ||||
| # Phony targets are scoped, so you probably want to declare them first. | ||||
| # | ||||
|  | ||||
| .PHONY: all clean test #install | ||||
| .PHONY: all clean test edoc dialyzer #install | ||||
|  | ||||
| ######################################################################## | ||||
| # Subdirectories. | ||||
| @@ -33,10 +33,19 @@ | ||||
| .DEFAULT: msgpack.beam | ||||
|  | ||||
| msgpack.beam: msgpack.erl | ||||
| 	erlc $< | ||||
| 	erlc -Wall +debug_info $< | ||||
|  | ||||
| msgpack.html: msgpack.erl | ||||
| 	erl -noshell -run edoc_run file $< | ||||
|  | ||||
| test: msgpack.beam | ||||
| 	erl -s msgpack test -s init stop | ||||
| 	erl -noshell -s msgpack test -s init stop | ||||
|  | ||||
| edoc: msgpack.erl | ||||
| 	erl -noshell -eval 'ok=edoc:files(["msgpack.erl"], [{dir, "edoc"}]).' -s init stop | ||||
|  | ||||
| dialyzer: msgpack.erl | ||||
| 	dialyzer --src $< | ||||
|  | ||||
| clean: | ||||
| 	-rm *.beam | ||||
| 	-rm -f *.beam *.html | ||||
|   | ||||
| @@ -2,18 +2,8 @@ MessagePack for Erlang | ||||
| ====================== | ||||
| Binary-based efficient object serialization library. | ||||
|  | ||||
| ## Status  | ||||
| see wiki ( http://redmine.msgpack.org/projects/msgpack/wiki/QuickStartErlang ) for details | ||||
|  | ||||
| still in development. | ||||
| # Status | ||||
|  | ||||
| TODOs: | ||||
|  - decide string specification. | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| ## Example | ||||
|  | ||||
| ## License | ||||
|  | ||||
|  | ||||
|  -  | ||||
| 0.1.0 released. | ||||
|   | ||||
							
								
								
									
										4
									
								
								erlang/edoc/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								erlang/edoc/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| *.html | ||||
| *.css | ||||
| *.png | ||||
| edoc-info | ||||
| @@ -15,318 +15,314 @@ | ||||
| %%    See the License for the specific language governing permissions and | ||||
| %%    limitations under the License. | ||||
|  | ||||
|  | ||||
| %% @doc <a href="http://msgpack.org/">MessagePack</a> codec for Erlang. | ||||
| %% | ||||
| %%      APIs are almost compatible with <a href="http://redmine.msgpack.org/projects/msgpack/wiki/QuickStartC">C API</a> | ||||
| %%      except for buffering functions (both copying and zero-copying), which are unavailable. | ||||
| %% | ||||
| %%   <table border="1"> | ||||
| %%     <caption>Equivalence between Erlang and <a href="http://msgpack.sourceforge.jp/spec">Msgpack type</a> :</caption> | ||||
| %%     <tr><th>    erlang    </th><th>                            msgpack                                      </th></tr> | ||||
| %%     <tr><td> integer()    </td><td> pos_fixnum/neg_fixnum/uint8/uint16/uint32/uint64/int8/int16/int32/int64 </td></tr> | ||||
| %%     <tr><td> float()      </td><td> float/double                                                            </td></tr> | ||||
| %%     <tr><td> nil          </td><td> nil                                                                     </td></tr> | ||||
| %%     <tr><td> boolean()    </td><td> boolean                                                                 </td></tr> | ||||
| %%     <tr><td> binary()     </td><td> fix_raw/raw16/raw32                                                     </td></tr> | ||||
| %%     <tr><td> list()       </td><td> fix_array/array16/array32                                               </td></tr> | ||||
| %%     <tr><td> {proplist()} </td><td> fix_map/map16/map32                                                     </td></tr> | ||||
| %%   </table> | ||||
| %% @end | ||||
|  | ||||
| -module(msgpack). | ||||
| -author('kuenishi+msgpack@gmail.com'). | ||||
|  | ||||
| %% tuples, atoms are not supported.  lists, integers, double, and so on. | ||||
| %% see http://msgpack.sourceforge.jp/spec for | ||||
| %% supported formats. APIs are almost compatible | ||||
| %% for C API (http://msgpack.sourceforge.jp/c:doc) | ||||
| %% except buffering functions (both copying and zero-copying). | ||||
| -export([pack/1, unpack/1, unpack_all/1, test/0]). | ||||
| -export([pack/1, unpack/1, unpack_all/1]). | ||||
|  | ||||
| -include_lib("eunit/include/eunit.hrl"). | ||||
|  | ||||
| % compile: | ||||
| % erl> c(msgpack). | ||||
| % erl> S = <some term>. | ||||
| % erl> {S, <<>>} = msgpack:unpack( msgpack:pack(S) ). | ||||
| % @type msgpack_term() = [msgpack_term()] | ||||
| %                      | {[{msgpack_term(),msgpack_term()}]} | ||||
| %                      | integer() | float() | binary(). | ||||
| % Erlang representation of msgpack data. | ||||
| -type msgpack_term() :: [msgpack_term()] | ||||
| 		      | {[{msgpack_term(),msgpack_term()}]} | ||||
| 		      | integer() | float() | binary(). | ||||
|  | ||||
|  | ||||
| -type reason() ::  enomem. | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| % external APIs | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
|  | ||||
| % @doc Encode an erlang term into an msgpack binary. | ||||
| %      Returns {error, {badarg, term()}} if the input is illegal. | ||||
| % @spec pack(Term::msgpack_term()) -> binary() | {error, {badarg, term()}} | ||||
| -spec pack(Term::msgpack_term()) -> binary() | {error, {badarg, term()}}. | ||||
| pack(Term)-> | ||||
|     try | ||||
| 	pack_(Term) | ||||
|     catch | ||||
| 	throw:Exception -> | ||||
| 	    {error, Exception} | ||||
|     end. | ||||
|  | ||||
| % @doc Decode an msgpack binary into an erlang term. | ||||
| %      It only decodes the first msgpack packet contained in the binary; the rest is returned as is. | ||||
| %      Returns {error, {badarg, term()}} if the input is corrupted. | ||||
| %      Returns {error, incomplete} if the input is not a full msgpack packet (caller should gather more data and try again). | ||||
| % @spec unpack(Bin::binary()) -> {msgpack_term(), binary()} | {error, incomplete} | {error, {badarg, term()}} | ||||
| -spec unpack(Bin::binary()) -> {msgpack_term(), binary()} | {error, incomplete} | {error, {badarg, term()}}. | ||||
| unpack(Bin) when is_binary(Bin) -> | ||||
|     try | ||||
| 	unpack_(Bin) | ||||
|     catch | ||||
| 	throw:Exception -> | ||||
| 	    {error, Exception} | ||||
|     end; | ||||
| unpack(Other) -> | ||||
|     {error, {badarg, Other}}. | ||||
|  | ||||
| % @doc Decode an msgpack binary into an erlang terms. | ||||
| %      It only decodes ALL msgpack packets contained in the binary. No packets should not remain. | ||||
| %      Returns {error, {badarg, term()}} if the input is corrupted. | ||||
| %      Returns {error, incomplete} if the input is not a full msgpack packet (caller should gather more data and try again). | ||||
| % @spec unpack_all(binary()) -> [msgpack_term()] | {error, incomplete} | {error, {badarg, term()}} | ||||
| -spec unpack_all(binary()) -> [msgpack_term()] | {error, incomplete} | {error, {badarg, term()}}. | ||||
| unpack_all(Data)-> | ||||
|     try | ||||
| 	unpack_all_(Data) | ||||
|     catch | ||||
| 	throw:Exception -> | ||||
| 	    {error, Exception} | ||||
|     end. | ||||
| unpack_all_(Data)-> | ||||
|     case unpack_(Data) of | ||||
| 	{ Term, <<>> } -> | ||||
| 	    [Term]; | ||||
| 	{ Term, Binary } when is_binary(Binary) -> | ||||
| 	    [Term|unpack_all_(Binary)] | ||||
|     end. | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| % internal APIs | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
|  | ||||
| % pack them all | ||||
| -spec pack_(msgpack_term()) -> binary() | no_return(). | ||||
| pack_(I) when is_integer(I) andalso I < 0 -> | ||||
|     pack_int_(I); | ||||
| pack_(I) when is_integer(I) -> | ||||
|     pack_uint_(I); | ||||
| pack_(F) when is_float(F) -> | ||||
|     pack_double(F); | ||||
| pack_(nil) -> | ||||
|     << 16#C0:8 >>; | ||||
| pack_(true) -> | ||||
|     << 16#C3:8 >>; | ||||
| pack_(false) -> | ||||
|     << 16#C2:8 >>; | ||||
| pack_(Bin) when is_binary(Bin) -> | ||||
|     pack_raw(Bin); | ||||
| pack_(List)  when is_list(List) -> | ||||
|     pack_array(List); | ||||
| pack_({Map}) when is_list(Map) -> | ||||
|     pack_map(Map); | ||||
| pack_(Other) -> | ||||
|     throw({badarg, Other}). | ||||
|  | ||||
|  | ||||
| -spec pack_uint_(non_neg_integer()) -> binary(). | ||||
| % positive fixnum | ||||
| pack_uint_(N) when is_integer( N ) , N < 128 ->   | ||||
| pack_uint_(N) when N < 128 -> | ||||
|     << 2#0:1, N:7 >>; | ||||
| % uint 8 | ||||
| pack_uint_( N ) when is_integer( N ) andalso N < 256 -> | ||||
| pack_uint_(N) when N < 256 -> | ||||
|     << 16#CC:8, N:8 >>; | ||||
|  | ||||
| % uint 16 | ||||
| pack_uint_( N ) when is_integer( N ) andalso N < 65536 -> | ||||
| pack_uint_(N) when N < 65536 -> | ||||
|     << 16#CD:8, N:16/big-unsigned-integer-unit:1 >>; | ||||
|  | ||||
| % uint 32 | ||||
| pack_uint_( N ) when is_integer( N ) andalso N < 16#FFFFFFFF-> | ||||
| pack_uint_(N) when N < 16#FFFFFFFF-> | ||||
|     << 16#CE:8, N:32/big-unsigned-integer-unit:1 >>; | ||||
|  | ||||
| % uint 64 | ||||
| pack_uint_( N ) when is_integer( N )-> | ||||
| pack_uint_(N) -> | ||||
|     << 16#CF:8, N:64/big-unsigned-integer-unit:1 >>. | ||||
|  | ||||
| -spec pack_int_(integer()) -> binary(). | ||||
| % negative fixnum | ||||
| pack_int_( N ) when is_integer( N ) , N >= -32-> | ||||
| pack_int_(N) when N >= -32-> | ||||
|     << 2#111:3, N:5 >>; | ||||
| % int 8 | ||||
| pack_int_( N ) when is_integer( N ) , N >= -256 -> | ||||
|     << 16#D0:8, N:8 >>; | ||||
| pack_int_(N) when N > -128 -> | ||||
|     << 16#D0:8, N:8/big-signed-integer-unit:1 >>; | ||||
| % int 16 | ||||
| pack_int_( N ) when is_integer( N ), N >= -65536 -> | ||||
| pack_int_(N) when N > -32768 -> | ||||
|     << 16#D1:8, N:16/big-signed-integer-unit:1 >>; | ||||
| % int 32 | ||||
| pack_int_( N ) when is_integer( N ), N >= -16#FFFFFFFF -> | ||||
| pack_int_(N) when N > -16#FFFFFFFF -> | ||||
|     << 16#D2:8, N:32/big-signed-integer-unit:1 >>; | ||||
| % int 64 | ||||
| pack_int_( N ) when is_integer( N )-> | ||||
| pack_int_(N) -> | ||||
|     << 16#D3:8, N:64/big-signed-integer-unit:1 >>. | ||||
|  | ||||
| % nil | ||||
| pack_nil()-> | ||||
|     << 16#C0:8 >>. | ||||
| % pack_true / pack_false | ||||
| pack_bool(true)->    << 16#C3:8 >>; | ||||
| pack_bool(false)->   << 16#C2:8 >>. | ||||
|  | ||||
| -spec pack_double(float()) -> binary(). | ||||
| % float : erlang's float is always IEEE 754 64bit format. | ||||
| %pack_float(F) when is_float(F)-> | ||||
| % pack_float(F) when is_float(F)-> | ||||
| %    << 16#CA:8, F:32/big-float-unit:1 >>. | ||||
| %    pack_double(F). | ||||
| % double | ||||
| pack_double(F) when is_float(F)-> | ||||
| pack_double(F) -> | ||||
|     << 16#CB:8, F:64/big-float-unit:1 >>. | ||||
|  | ||||
| power(N,0) when is_integer(N) -> 1; | ||||
| power(N,D) when is_integer(N) and is_integer(D) -> N * power(N, D-1). | ||||
|  | ||||
| -spec pack_raw(binary()) -> binary(). | ||||
| % raw bytes | ||||
| pack_raw(Bin) when is_binary(Bin)-> | ||||
|     MaxLen = power(2,16), | ||||
| pack_raw(Bin) -> | ||||
|     case byte_size(Bin) of | ||||
| 	Len when Len < 6-> | ||||
| 	    << 2#101:3, Len:5, Bin/binary >>; | ||||
| 	Len when Len < MaxLen -> | ||||
| 	Len when Len < 16#10000 -> % 65536 | ||||
| 	    << 16#DA:8, Len:16/big-unsigned-integer-unit:1, Bin/binary >>; | ||||
| 	Len -> | ||||
| 	    << 16#DB:8, Len:32/big-unsigned-integer-unit:1, Bin/binary >> | ||||
|     end. | ||||
|  | ||||
| % list / tuple | ||||
| pack_array(L) when is_list(L)-> | ||||
|     MaxLen = power(2,16), | ||||
|  | ||||
| -spec pack_array([msgpack_term()]) -> binary() | no_return(). | ||||
| % list | ||||
| pack_array(L) -> | ||||
|     case length(L) of | ||||
|  	Len when Len < 16 -> | ||||
|  	    << 2#1001:4, Len:4/integer-unit:1, (pack_array_(L))/binary >>; | ||||
| 	Len when Len < MaxLen -> | ||||
| 	    << 16#DC:8, Len:16/big-unsigned-integer-unit:1,(pack_array_(L))/binary >>; | ||||
|  	    << 2#1001:4, Len:4/integer-unit:1, (pack_array_(L, <<>>))/binary >>; | ||||
| 	Len when Len < 16#10000 -> % 65536 | ||||
| 	    << 16#DC:8, Len:16/big-unsigned-integer-unit:1, (pack_array_(L, <<>>))/binary >>; | ||||
| 	Len -> | ||||
| 	    << 16#DD:8, Len:32/big-unsigned-integer-unit:1,(pack_array_(L))/binary >> | ||||
| 	    << 16#DD:8, Len:32/big-unsigned-integer-unit:1, (pack_array_(L, <<>>))/binary >> | ||||
|     end. | ||||
| pack_array_([])-> <<>>; | ||||
| pack_array_([Head|Tail])-> | ||||
|     << (pack_object(Head))/binary, (pack_array_(Tail))/binary >>. | ||||
|  | ||||
| unpack_array_(<<>>, 0)-> []; | ||||
| unpack_array_(Remain, 0) when is_binary(Remain)-> [Remain]; | ||||
| unpack_array_(Bin, RestLen) when is_binary(Bin)-> | ||||
|     {Term, Rest} = unpack(Bin), | ||||
|     [Term|unpack_array_(Rest, RestLen-1)]. | ||||
|      | ||||
| pack_map({dict,M})-> | ||||
|     MaxLen = power(2,16), | ||||
|     case dict:size(M) of | ||||
| pack_array_([], Acc) -> Acc; | ||||
| pack_array_([Head|Tail], Acc) -> | ||||
|     pack_array_(Tail, <<Acc/binary,  (pack_(Head))/binary>>). | ||||
|  | ||||
| % Users SHOULD NOT send too long list: this uses lists:reverse/1 | ||||
| -spec unpack_array_(binary(), non_neg_integer(), [msgpack_term()]) -> {[msgpack_term()], binary()} | no_return(). | ||||
| unpack_array_(Bin, 0,   Acc) -> {lists:reverse(Acc), Bin}; | ||||
| unpack_array_(Bin, Len, Acc) -> | ||||
|     {Term, Rest} = unpack_(Bin), | ||||
|     unpack_array_(Rest, Len-1, [Term|Acc]). | ||||
|  | ||||
|  | ||||
| -spec pack_map(M::[{msgpack_term(),msgpack_term()}]) -> binary() | no_return(). | ||||
| pack_map(M)-> | ||||
|     case length(M) of | ||||
| 	Len when Len < 16 -> | ||||
|  	    << 2#1001:4, Len:4/integer-unit:1, (pack_map_(dict:to_list(M))) >>; | ||||
| 	Len when Len < MaxLen -> | ||||
| 	    << 16#DE:8, Len:16/big-unsigned-integer-unit:1, (pack_map_(dict:to_list(M))) >>; | ||||
| 	    << 2#1000:4, Len:4/integer-unit:1, (pack_map_(M, <<>>))/binary >>; | ||||
| 	Len when Len < 16#10000 -> % 65536 | ||||
| 	    << 16#DE:8, Len:16/big-unsigned-integer-unit:1, (pack_map_(M, <<>>))/binary >>; | ||||
| 	Len -> | ||||
| 	    << 16#DF:8, Len:32/big-unsigned-integer-unit:1, (pack_map_(dict:to_list(M))) >> | ||||
| 	    << 16#DF:8, Len:32/big-unsigned-integer-unit:1, (pack_map_(M, <<>>))/binary >> | ||||
|     end. | ||||
|  | ||||
| pack_map_([])-> <<>>; | ||||
| pack_map_([{Key,Value}|Tail]) -> | ||||
|     << (pack_object(Key)),(pack_object(Value)),(pack_map_(Tail)) >>. | ||||
| pack_map_([], Acc) -> Acc; | ||||
| pack_map_([{Key,Value}|Tail], Acc) -> | ||||
|     pack_map_(Tail, << Acc/binary, (pack_(Key))/binary, (pack_(Value))/binary>>). | ||||
|  | ||||
| unpack_map_(<<>>, 0)-> []; | ||||
| unpack_map_(Bin,  0) when is_binary(Bin)-> [Bin]; | ||||
| unpack_map_(Bin, Len) when is_binary(Bin) and is_integer(Len) -> | ||||
|     { Key, Rest } = unpack(Bin), | ||||
|     { Value, Rest2 } = unpack(Rest), | ||||
|     [{Key,Value}|unpack_map_(Rest2,Len-1)]. | ||||
| % Users SHOULD NOT send too long list: this uses lists:reverse/1 | ||||
| -spec unpack_map_(binary(), non_neg_integer(), [{msgpack_term(), msgpack_term()}]) -> | ||||
| 			 {{[{msgpack_term(), msgpack_term()}]}, binary()} | no_return(). | ||||
| unpack_map_(Bin, 0,   Acc) -> {{lists:reverse(Acc)}, Bin}; | ||||
| unpack_map_(Bin, Len, Acc) -> | ||||
|     {Key, Rest} = unpack_(Bin), | ||||
|     {Value, Rest2} = unpack_(Rest), | ||||
|     unpack_map_(Rest2, Len-1, [{Key,Value}|Acc]). | ||||
|  | ||||
| pack_object(O) when is_integer(O) andalso O < 0 -> | ||||
|     pack_int_(O); | ||||
| pack_object(O) when is_integer(O) -> | ||||
|     pack_uint_(O); | ||||
| pack_object(O) when is_float(O)-> | ||||
|     pack_double(O); | ||||
| pack_object(nil) -> | ||||
|     pack_nil(); | ||||
| pack_object(Bool) when is_atom(Bool) -> | ||||
|     pack_bool(Bool); | ||||
| pack_object(Bin) when is_binary(Bin)-> | ||||
|     pack_raw(Bin); | ||||
| pack_object(List)  when is_list(List)-> | ||||
|     pack_array(List); | ||||
| pack_object({dict, Map})-> | ||||
|     pack_map({dict, Map}); | ||||
| pack_object(_) -> | ||||
|     undefined. | ||||
| % unpack them all | ||||
| -spec unpack_(Bin::binary()) -> {msgpack_term(), binary()} | no_return(). | ||||
| unpack_(Bin) -> | ||||
|     case Bin of | ||||
| % ATOMS | ||||
| 	<<16#C0, Rest/binary>> -> {nil, Rest}; | ||||
| 	<<16#C2, Rest/binary>> -> {false, Rest}; | ||||
| 	<<16#C3, Rest/binary>> -> {true, Rest}; | ||||
| % Floats | ||||
| 	<<16#CA, V:32/float-unit:1, Rest/binary>> ->   {V, Rest}; | ||||
| 	<<16#CB, V:64/float-unit:1, Rest/binary>> ->   {V, Rest}; | ||||
| % Unsigned integers | ||||
| 	<<16#CC, V:8/unsigned-integer, Rest/binary>> ->             {V, Rest}; | ||||
| 	<<16#CD, V:16/big-unsigned-integer-unit:1, Rest/binary>> -> {V, Rest}; | ||||
| 	<<16#CE, V:32/big-unsigned-integer-unit:1, Rest/binary>> -> {V, Rest}; | ||||
| 	<<16#CF, V:64/big-unsigned-integer-unit:1, Rest/binary>> -> {V, Rest}; | ||||
| % Signed integers | ||||
| 	<<16#D0, V:8/signed-integer, Rest/binary>> ->             {V, Rest}; | ||||
| 	<<16#D1, V:16/big-signed-integer-unit:1, Rest/binary>> -> {V, Rest}; | ||||
| 	<<16#D2, V:32/big-signed-integer-unit:1, Rest/binary>> -> {V, Rest}; | ||||
| 	<<16#D3, V:64/big-signed-integer-unit:1, Rest/binary>> -> {V, Rest}; | ||||
| % Raw bytes | ||||
| 	<<16#DA, L:16/unsigned-integer-unit:1, V:L/binary, Rest/binary>> -> {V, Rest}; | ||||
| 	<<16#DB, L:32/unsigned-integer-unit:1, V:L/binary, Rest/binary>> -> {V, Rest}; | ||||
| % Arrays | ||||
| 	<<16#DC, L:16/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_array_(Rest, L, []); | ||||
| 	<<16#DD, L:32/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_array_(Rest, L, []); | ||||
| % Maps | ||||
| 	<<16#DE, L:16/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_map_(Rest, L, []); | ||||
| 	<<16#DF, L:32/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_map_(Rest, L, []); | ||||
|  | ||||
| pack(Obj)-> | ||||
|     pack_object(Obj). | ||||
| % Tag-encoded lengths (kept last, for speed) | ||||
| 	<<0:1, V:7, Rest/binary>> ->                 {V, Rest};                  % positive int | ||||
| 	<<2#111:3, V:5, Rest/binary>> ->             {V - 2#100000, Rest};       % negative int | ||||
| 	<<2#101:3, L:5, V:L/binary, Rest/binary>> -> {V, Rest};                  % raw bytes | ||||
| 	<<2#1001:4, L:4, Rest/binary>> ->            unpack_array_(Rest, L, []); % array | ||||
| 	<<2#1000:4, L:4, Rest/binary>> ->            unpack_map_(Rest, L, []);   % map | ||||
|  | ||||
|  | ||||
| % unpacking. | ||||
| % if failed in decoding and not end, get more data  | ||||
| % and feed more Bin into this function. | ||||
| % TODO: error case for imcomplete format when short for any type formats. | ||||
| -spec unpack( binary() )-> {term(), binary()} | {more, non_neg_integer()} | {error, reason()}. | ||||
| unpack(Bin) when not is_binary(Bin)-> | ||||
|     {error, badard}; | ||||
| unpack(Bin) when bit_size(Bin) >= 8 -> | ||||
|     << Flag:8/unsigned-integer, Payload/binary >> = Bin, | ||||
|     case Flag of  | ||||
| 	16#C0 -> | ||||
| 	    {nil, Payload}; | ||||
| 	16#C2 -> | ||||
| 	    {false, Payload}; | ||||
| 	16#C3 -> | ||||
| 	    {true, Payload}; | ||||
|  | ||||
| 	16#CA -> % 32bit float | ||||
| 	    << Return:32/float-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Return, Rest}; | ||||
| 	16#CB -> % 64bit float | ||||
| 	    << Return:64/float-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Return, Rest}; | ||||
|  | ||||
| 	16#CC -> % uint 8 | ||||
| 	    << Int:8/unsigned-integer, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#CD -> % uint 16 | ||||
| 	    << Int:16/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#CE -> | ||||
| 	    << Int:32/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#CF -> | ||||
| 	    << Int:64/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
|  | ||||
| 	16#D0 -> % int 8 | ||||
| 	    << Int:8/big-signed-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#D1 -> % int 16 | ||||
| 	    << Int:16/big-signed-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#D2 -> % int 32 | ||||
| 	    << Int:32/big-signed-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#D3 -> % int 64 | ||||
| 	    << Int:64/big-signed-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    {Int, Rest}; | ||||
| 	16#DA -> % raw 16 | ||||
| 	    << Len:16/unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    << Return:Len/binary, Remain/binary >> = Rest, | ||||
| 	    {Return, Remain}; | ||||
| 	16#DB -> % raw 32 | ||||
| 	    << Len:32/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    << Return:Len/binary, Remain/binary >> = Rest, | ||||
| 	    {Return, Remain}; | ||||
| 	16#DC -> % array 16 | ||||
| 	    << Len:16/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    Array=unpack_array_(Rest, Len), | ||||
| 	    case length(Array) of | ||||
| 		Len -> {Array, <<>>}; | ||||
| 		_ ->  | ||||
| 		    {Return, RemainRest} = lists:split(Len, Array), | ||||
| 		    [Remain] = RemainRest, | ||||
| 		    {Return, Remain} | ||||
| 	    end; | ||||
| 	16#DD -> % array 32 | ||||
| 	    << Len:32/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    Array=unpack_array_(Rest, Len), | ||||
| 	    case length(Array) of | ||||
| 		Len -> {Array, <<>>}; | ||||
| 		_ ->  | ||||
| 		    {Return, RemainRest} = lists:split(Len, Array), | ||||
| 		    [Remain] = RemainRest, | ||||
| 		    {Return, Remain} | ||||
| 	    end; | ||||
| 	16#DE -> % map 16 | ||||
| 	    << Len:16/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    Array=unpack_map_(Rest, Len), | ||||
| 	    case length(Array) of | ||||
| 		Len -> { dict:from_list(Array), <<>>}; | ||||
| 		_ ->  | ||||
| 		    {Return, RemainRest} = lists:split(Len, Array), | ||||
| 		    [Remain] = RemainRest, | ||||
| 		    {dict:from_list(Return), Remain} | ||||
| 	    end; | ||||
| 	16#DF -> % map 32 | ||||
| 	    << Len:32/big-unsigned-integer-unit:1, Rest/binary >> = Payload, | ||||
| 	    Array=unpack_map_(Rest, Len), | ||||
| 	    case length(Array) of | ||||
| 		Len -> { dict:from_list(Array), <<>>}; | ||||
| 		_ ->  | ||||
| 		    {Return, RemainRest} = lists:split(Len, Array), | ||||
| 		    [Remain] = RemainRest, | ||||
| 		    {dict:from_list(Return), Remain} | ||||
| 	    end; | ||||
|  | ||||
| 	% positive fixnum | ||||
| 	Code when Code >= 2#00000000, Code < 2#10000000-> | ||||
| 	    {Code, Payload}; | ||||
|  | ||||
| 	% negative fixnum | ||||
| 	Code when Code >= 2#11100000 -> | ||||
| 	    {(Code - 16#100), Payload}; | ||||
|  | ||||
| 	Code when Code >= 2#10100000 , Code < 2#11000000 -> | ||||
| %	 101XXXXX for FixRaw  | ||||
| 	    Len = Code rem 2#10100000, | ||||
| 	    << Return:Len/binary, Remain/binary >> = Payload, | ||||
| 	    {Return, Remain}; | ||||
|  | ||||
| 	Code when Code >= 2#10010000 , Code < 2#10100000 -> | ||||
| %        1001XXXX for FixArray | ||||
| 	    Len = Code rem 2#10010000, | ||||
| 	    Array=unpack_array_(Payload, Len), | ||||
| 	    case length(Array) of | ||||
| 		Len -> { Array, <<>>}; | ||||
| 		_ ->  | ||||
| 		    {Return, RemainRest} = lists:split(Len, Array), | ||||
| 		    [Remain] = RemainRest, | ||||
| 		    {Return, Remain} | ||||
| 	    end; | ||||
|  | ||||
| 	Code when Code >= 2#10000000 , Code < 2#10010000 -> | ||||
| %        1000XXXX for FixMap | ||||
| 	    Len = Code rem 2#10000000, | ||||
| 	    Array=unpack_map_(Payload, Len), | ||||
| 	    case length(Array) of | ||||
| 		Len -> { dict:from_list(Array), <<>>}; | ||||
| 		_ ->  | ||||
| 		    {Return, RemainRest} = lists:split(Len, Array), | ||||
| 		    [Remain] = RemainRest, | ||||
| 		    {dict:from_list(Return), Remain} | ||||
| 	    end; | ||||
|  | ||||
| 	_Other -> | ||||
| 	    erlang:display(_Other), | ||||
| 	    {error, no_code_matches} | ||||
|     end; | ||||
| unpack(_)-> % when bit_size(Bin) < 8 -> | ||||
|     {more, 8}. | ||||
|      | ||||
| unpack_all(Data)-> | ||||
|     case unpack(Data) of | ||||
| 	{ Term, Binary } when bit_size(Binary) =:= 0 ->  | ||||
| 	    [Term]; | ||||
| 	{ Term, Binary } when is_binary(Binary) -> | ||||
| 	    [Term|unpack_all(Binary)] | ||||
| % Invalid data | ||||
| 	<<F, R/binary>> when F==16#C1; | ||||
| 	                     F==16#C4; F==16#C5; F==16#C6; F==16#C7; F==16#C8; F==16#C9; | ||||
| 	                     F==16#D4; F==16#D5; F==16#D6; F==16#D7; F==16#D8; F==16#D9 -> | ||||
| 	    throw({badarg, <<F, R/binary>>}); | ||||
| % Incomplete data (we've covered every complete/invalid case; anything left is incomplete) | ||||
| 	_ -> | ||||
| 	    throw(incomplete) | ||||
|     end. | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| % unit tests | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| -include_lib("eunit/include/eunit.hrl"). | ||||
| -ifdef(EUNIT). | ||||
|  | ||||
| compare_all([], [])-> ok; | ||||
| compare_all([],  R)-> {toomuchrhs, R}; | ||||
| compare_all(L,  [])-> {toomuchlhs, L}; | ||||
| compare_all([LH|LTL], [RH|RTL]) -> | ||||
|     ?assertEqual(LH, RH), | ||||
|     compare_all(LTL, RTL). | ||||
|  | ||||
| port_receive(Port) -> | ||||
|     port_receive(Port, <<>>). | ||||
| port_receive(Port, Acc) -> | ||||
|     receive | ||||
|         {Port, {data, Data}} -> port_receive(Port, <<Acc/binary, Data/binary>>); | ||||
|         {Port, eof} -> Acc | ||||
|     after 1000 -> Acc | ||||
|     end. | ||||
|  | ||||
| test_([]) -> 0; | ||||
| test_([Term|Rest])-> | ||||
|     Pack = msgpack:pack(Term), | ||||
|     ?assertEqual({Term, <<>>}, msgpack:unpack( Pack )), | ||||
|     1+test_(Rest). | ||||
|  | ||||
| test_data()-> | ||||
|     [0, 1, 2, 123, 512, 1230, 678908, 16#FFFFFFFFFF, | ||||
|     [true, false, nil, | ||||
|      0, 1, 2, 123, 512, 1230, 678908, 16#FFFFFFFFFF, | ||||
|      -1, -23, -512, -1230, -567898, -16#FFFFFFFFFF, | ||||
|      123.123, -234.4355, 1.0e-34, 1.0e64, | ||||
|      [23, 234, 0.23], | ||||
|      "hogehoge", "243546rf7g68h798j", | ||||
|      <<"hogehoge">>, <<"243546rf7g68h798j", 0, 23, 255>>, | ||||
|      <<"hoasfdafdas][">>, | ||||
|      [0,42,"sum", [1,2]], [1,42, nil, [3]] | ||||
|      [0,42, <<"sum">>, [1,2]], [1,42, nil, [3]], | ||||
|      -234, -40000, -16#10000000, -16#100000000, | ||||
|      42 | ||||
|     ]. | ||||
|  | ||||
| basic_test()-> | ||||
| @@ -336,37 +332,64 @@ basic_test()-> | ||||
|  | ||||
| port_test()-> | ||||
|     Tests = test_data(), | ||||
|     {[Tests],<<>>} = msgpack:unpack(msgpack:pack([Tests])), | ||||
|     Port = open_port({spawn, "ruby ../test/crosslang.rb"}, [binary]), | ||||
|     true = port_command(Port, msgpack:pack(Tests) ), | ||||
|     %Port ! {self, {command, msgpack:pack(Tests)}}, ... not owner | ||||
|     receive | ||||
| 	{Port, {data, Data}}->  {Tests, <<>>}=msgpack:unpack(Data) | ||||
|     after 1024-> ?assert(false)   end, | ||||
|     ?assertEqual({[Tests],<<>>}, msgpack:unpack(msgpack:pack([Tests]))), | ||||
|  | ||||
|     Port = open_port({spawn, "ruby ../test/crosslang.rb"}, [binary, eof]), | ||||
|     true = port_command(Port, msgpack:pack(Tests)), | ||||
|     ?assertEqual({Tests, <<>>}, msgpack:unpack(port_receive(Port))), | ||||
|     port_close(Port). | ||||
|  | ||||
| test_p(Len,Term,OrigBin,Len) -> | ||||
|     {Term, <<>>}=msgpack:unpack(OrigBin); | ||||
| test_p(I,_,OrigBin,Len) when I < Len-> | ||||
|     <<Bin:I/binary, _/binary>> = OrigBin, | ||||
|     ?assertEqual({error,incomplete}, msgpack:unpack(Bin)). | ||||
|  | ||||
| partial_test()-> % error handling test. | ||||
|     Term = lists:seq(0, 45), | ||||
|     Bin=msgpack:pack(Term), | ||||
|     BinLen = byte_size(Bin), | ||||
|     [test_p(X, Term, Bin, BinLen) || X <- lists:seq(0,BinLen)]. | ||||
|  | ||||
| long_test()-> | ||||
|     Longer = lists:seq(0, 655), | ||||
|     {Longer, <<>>} = msgpack:unpack(msgpack:pack(Longer)). | ||||
|  | ||||
| map_test()-> | ||||
|     Ints = lists:seq(0, 65), | ||||
|     Map = {[ {X, X*2} || X <- Ints ] ++ [{<<"hage">>, 324}, {43542, [nil, true, false]}]}, | ||||
|     {Map2, <<>>} = msgpack:unpack(msgpack:pack(Map)), | ||||
|     ?assertEqual(Map, Map2), | ||||
|     ok. | ||||
|  | ||||
| unknown_test()-> | ||||
|     Port = open_port({spawn, "ruby testcase_generator.rb"}, [binary, eof]), | ||||
|     Tests = [0, 1, 2, 123, 512, 1230, 678908, | ||||
| 	     -1, -23, -512, -1230, -567898, | ||||
| %	     "hogehoge", "243546rf7g68h798j", | ||||
| 	     123.123 %-234.4355, 1.0e-34, 1.0e64, | ||||
| %	     [23, 234, 0.23] | ||||
| %	     [0,42,"sum", [1,2]], [1,42, nil, [3]] | ||||
| 	     <<"hogehoge">>, <<"243546rf7g68h798j">>, | ||||
| 	     123.123, | ||||
| 	     -234.4355, 1.0e-34, 1.0e64, | ||||
| 	     [23, 234, 0.23], | ||||
| 	     [0,42,<<"sum">>, [1,2]], [1,42, nil, [3]], | ||||
| 	     {[{1,2},{<<"hoge">>,nil}]}, % map | ||||
| 	     -234, -50000, | ||||
| 	     42 | ||||
| 	    ], | ||||
|     Port = open_port({spawn, "ruby testcase_generator.rb"}, [binary]), | ||||
|     %Port ! {self, {command, msgpack:pack(Tests)}}, ... not owner | ||||
|     receive | ||||
| 	{Port, {data, Data}}-> | ||||
| 	    Tests=msgpack:unpack_all(Data) | ||||
| %	    io:format("~p~n", [Tests]) | ||||
|     after 1024-> ?assert(false)   end, | ||||
|     ?assertEqual(ok, compare_all(Tests, msgpack:unpack_all(port_receive(Port)))), | ||||
|     port_close(Port). | ||||
|  | ||||
| test_([]) -> 0; | ||||
| test_([S|Rest])-> | ||||
|     Pack = msgpack:pack(S), | ||||
| %    io:format("testing: ~p => ~p~n", [S, Pack]), | ||||
|     {S, <<>>} = msgpack:unpack( Pack ), | ||||
|     1+test_(Rest). | ||||
| other_test()-> | ||||
|     ?assertEqual({error,incomplete},msgpack:unpack(<<>>)). | ||||
|  | ||||
| benchmark_test()-> | ||||
|     Data=[test_data() || _ <- lists:seq(0, 10000)], | ||||
|     S=?debugTime("  serialize", msgpack:pack(Data)), | ||||
|     {Data,<<>>}=?debugTime("deserialize", msgpack:unpack(S)), | ||||
|     ?debugFmt("for ~p KB test data.", [byte_size(S) div 1024]). | ||||
|  | ||||
| error_test()-> | ||||
|     ?assertEqual({error,{badarg, atom}}, msgpack:pack(atom)), | ||||
|     Term = {"hoge", "hage", atom}, | ||||
|     ?assertEqual({error,{badarg, Term}}, msgpack:pack(Term)). | ||||
|  | ||||
| -endif. | ||||
|   | ||||
| @@ -39,10 +39,14 @@ end | ||||
|  | ||||
| objs = [0, 1, 2, 123, 512, 1230, 678908, | ||||
|         -1, -23, -512, -1230, -567898, | ||||
| #        "hogehoge", "243546rf7g68h798j", | ||||
|         123.123, #-234.4355, 1.0e-34, 1.0e64, | ||||
| #        [23, 234, 0.23] | ||||
| #        [0,42,"sum", [1,2]], [1,42, nil, [3]] | ||||
|         "hogehoge", "243546rf7g68h798j", | ||||
|         123.123, | ||||
|        -234.4355, 1.0e-34, 1.0e64, | ||||
|         [23, 234, 0.23], | ||||
|         [0,42,"sum", [1,2]], [1,42, nil, [3]], | ||||
|         { 1 => 2, "hoge" => nil }, | ||||
|         -234, -50000, | ||||
|         42 | ||||
|        ] | ||||
| begin | ||||
|   objs.each do |obj| | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| Copyright (c) 2009, Hideyuki Tanaka | ||||
| Copyright (c) 2009-2010, Hideyuki Tanaka | ||||
| All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
|   | ||||
| @@ -1,137 +0,0 @@ | ||||
| #include <msgpack.h> | ||||
|  | ||||
| void msgpack_sbuffer_init_wrap(msgpack_sbuffer* sbuf) | ||||
| { | ||||
|   msgpack_sbuffer_init(sbuf); | ||||
| } | ||||
|  | ||||
| void msgpack_sbuffer_destroy_wrap(msgpack_sbuffer* sbuf) | ||||
| { | ||||
|   msgpack_sbuffer_destroy(sbuf); | ||||
| } | ||||
|  | ||||
| int msgpack_sbuffer_write_wrap(void* data, const char* buf, unsigned int len) | ||||
| { | ||||
|   return msgpack_sbuffer_write(data, buf, len); | ||||
| } | ||||
|  | ||||
| msgpack_packer* msgpack_packer_new_wrap(void *data, msgpack_packer_write callback) | ||||
| { | ||||
|   return msgpack_packer_new(data, callback); | ||||
| } | ||||
|  | ||||
| void msgpack_packer_free_wrap(msgpack_packer* pk) | ||||
| { | ||||
|   msgpack_packer_free(pk); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_uint8_wrap(msgpack_packer* pk, uint8_t d) | ||||
| { | ||||
|   return msgpack_pack_uint8(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_uint16_wrap(msgpack_packer* pk, uint16_t d) | ||||
| { | ||||
|   return msgpack_pack_uint16(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_uint32_wrap(msgpack_packer* pk, uint32_t d) | ||||
| { | ||||
|   return msgpack_pack_uint32(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_uint64_wrap(msgpack_packer* pk, uint64_t d) | ||||
| { | ||||
|   return msgpack_pack_uint64(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_int8_wrap(msgpack_packer* pk, int8_t d) | ||||
| { | ||||
|   return msgpack_pack_int8(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_int16_wrap(msgpack_packer* pk, int16_t d) | ||||
| { | ||||
|   return msgpack_pack_int16(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_int32_wrap(msgpack_packer* pk, int32_t d) | ||||
| { | ||||
|   return msgpack_pack_int32(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_int64_wrap(msgpack_packer* pk, int64_t d) | ||||
| { | ||||
|   return msgpack_pack_int64(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_double_wrap(msgpack_packer* pk, double d) | ||||
| { | ||||
|   return msgpack_pack_double(pk, d); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_nil_wrap(msgpack_packer* pk) | ||||
| { | ||||
|   return msgpack_pack_nil(pk); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_true_wrap(msgpack_packer* pk) | ||||
| { | ||||
|   return msgpack_pack_true(pk); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_false_wrap(msgpack_packer* pk) | ||||
| { | ||||
|   return msgpack_pack_false(pk); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_array_wrap(msgpack_packer* pk, unsigned int n) | ||||
| { | ||||
|   return msgpack_pack_array(pk, n); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_map_wrap(msgpack_packer* pk, unsigned int n) | ||||
| { | ||||
|   return msgpack_pack_map(pk, n); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_raw_wrap(msgpack_packer* pk, size_t l) | ||||
| { | ||||
|   return msgpack_pack_raw(pk, l); | ||||
| } | ||||
|  | ||||
| int msgpack_pack_raw_body_wrap(msgpack_packer* pk, const void *b, size_t l) | ||||
| { | ||||
|   return msgpack_pack_raw_body(pk, b, l); | ||||
| } | ||||
|  | ||||
| bool msgpack_unpacker_reserve_buffer_wrap(msgpack_unpacker *mpac, size_t size) | ||||
| { | ||||
|   return msgpack_unpacker_reserve_buffer(mpac, size); | ||||
| } | ||||
|  | ||||
| char *msgpack_unpacker_buffer_wrap(msgpack_unpacker *mpac) | ||||
| { | ||||
|   return msgpack_unpacker_buffer(mpac); | ||||
| } | ||||
|  | ||||
| size_t msgpack_unpacker_buffer_capacity_wrap(const msgpack_unpacker *mpac) | ||||
| { | ||||
|   return msgpack_unpacker_buffer_capacity(mpac); | ||||
| } | ||||
|  | ||||
| void msgpack_unpacker_buffer_consumed_wrap(msgpack_unpacker *mpac, size_t size) | ||||
| { | ||||
|   msgpack_unpacker_buffer_consumed(mpac, size); | ||||
| } | ||||
|  | ||||
| void msgpack_unpacker_data_wrap(msgpack_unpacker *mpac, msgpack_object *obj) | ||||
| { | ||||
|   *obj=msgpack_unpacker_data(mpac); | ||||
| } | ||||
|  | ||||
| size_t msgpack_unpacker_message_size_wrap(const msgpack_unpacker *mpac) | ||||
| { | ||||
|   return msgpack_unpacker_message_size(mpac); | ||||
| } | ||||
|  | ||||
| @@ -1,32 +1,48 @@ | ||||
| Name:                msgpack | ||||
| Version:             0.2.2 | ||||
| License:             BSD3 | ||||
| License-File:        LICENSE | ||||
| Author:              Hideyuki Tanaka | ||||
| Maintainer:          Hideyuki Tanaka <tanaka.hideyuki@gmail.com> | ||||
| Category:            Data | ||||
| Synopsis:            A Haskell binding to MessagePack | ||||
| Name:               msgpack | ||||
| Version:            0.4.0.1 | ||||
| Synopsis:           A Haskell binding to MessagePack | ||||
| Description: | ||||
|   A Haskell binding to MessagePack <http://msgpack.sourceforge.jp/> | ||||
| Homepage:	     http://github.com/tanakh/hsmsgpack | ||||
| Stability:	     Experimental | ||||
| Tested-with:	     GHC==6.10.4 | ||||
| Cabal-Version:	     >=1.2 | ||||
| Build-Type:          Simple | ||||
|   A Haskell binding to MessagePack <http://msgpack.org/> | ||||
|  | ||||
| library | ||||
|   build-depends:	base>=4 && <5, mtl, bytestring | ||||
|   ghc-options:		-O2 -Wall | ||||
|   hs-source-dirs:	src | ||||
|   extra-libraries:	msgpackc | ||||
| License:            BSD3 | ||||
| License-File:       LICENSE | ||||
| Copyright:          Copyright (c) 2009-2010, Hideyuki Tanaka | ||||
| Category:           Data | ||||
| Author:             Hideyuki Tanaka | ||||
| Maintainer:         Hideyuki Tanaka <tanaka.hideyuki@gmail.com> | ||||
| Homepage:           http://github.com/msgpack/msgpack | ||||
| Stability:          Experimental | ||||
| Cabal-Version:      >= 1.6 | ||||
| Build-Type:         Simple | ||||
|  | ||||
| Extra-source-files: | ||||
|   test/Test.hs | ||||
|   test/UserData.hs | ||||
|  | ||||
| Library | ||||
|   Build-depends:    base >=4 && <5, | ||||
|                     transformers >= 0.2.1 && < 0.2.2, | ||||
|                     MonadCatchIO-transformers >= 0.2.2 && < 0.2.3, | ||||
|                     bytestring >= 0.9 && < 0.10, | ||||
|                     vector >= 0.6.0 && < 0.6.1, | ||||
|                     iteratee >= 0.4 && < 0.5, | ||||
|                     attoparsec >= 0.8.1 && < 0.8.2, | ||||
|                     binary >= 0.5.0 && < 0.5.1, | ||||
|                     data-binary-ieee754 >= 0.4 && < 0.5, | ||||
|                     deepseq >= 1.1 && <1.2, | ||||
|                     template-haskell >= 2.4 && < 2.5 | ||||
|  | ||||
|   Ghc-options:      -Wall | ||||
|   Hs-source-dirs:   src | ||||
|  | ||||
|   Exposed-modules: | ||||
|     Data.MessagePack | ||||
|     Data.MessagePack.Base | ||||
|     Data.MessagePack.Class | ||||
|     Data.MessagePack.Feed | ||||
|     Data.MessagePack.Monad | ||||
|     Data.MessagePack.Stream | ||||
|     Data.MessagePack.Pack | ||||
|     Data.MessagePack.Unpack | ||||
|     Data.MessagePack.Object | ||||
|     Data.MessagePack.Iteratee | ||||
|     Data.MessagePack.Derive | ||||
|  | ||||
|   C-Sources: | ||||
|     cbits/msgpack.c | ||||
| Source-repository head | ||||
|   Type:     git | ||||
|   Location: git://github.com/msgpack/msgpack.git | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009 | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009-2010 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| @@ -13,51 +13,91 @@ | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack( | ||||
|   module Data.MessagePack.Base, | ||||
|   module Data.MessagePack.Class, | ||||
|   module Data.MessagePack.Feed, | ||||
|   module Data.MessagePack.Monad, | ||||
|   module Data.MessagePack.Stream, | ||||
|   module Data.MessagePack.Pack, | ||||
|   module Data.MessagePack.Unpack, | ||||
|   module Data.MessagePack.Object, | ||||
|   module Data.MessagePack.Iteratee, | ||||
|   module Data.MessagePack.Derive, | ||||
|    | ||||
|   -- * Pack and Unpack | ||||
|   packb, | ||||
|   unpackb, | ||||
|   -- * Pack functions | ||||
|   packToString, | ||||
|   packToHandle, | ||||
|   packToHandle', | ||||
|   packToFile, | ||||
|    | ||||
|   -- * Unpack functions | ||||
|   unpackFromString, | ||||
|   unpackFromHandle, | ||||
|   unpackFromFile, | ||||
|   unpackFromStringI, | ||||
|   unpackFromHandleI, | ||||
|   unpackFromFileI, | ||||
|    | ||||
|   -- * Pure version of Pack and Unpack | ||||
|   packb', | ||||
|   unpackb', | ||||
|   ) where | ||||
|  | ||||
| import Data.ByteString (ByteString) | ||||
| import System.IO.Unsafe | ||||
| import qualified Control.Monad.CatchIO as CIO | ||||
| import Control.Monad.IO.Class | ||||
| import qualified Data.Attoparsec as A | ||||
| import Data.Binary.Put | ||||
| import qualified Data.ByteString as B | ||||
| import qualified Data.ByteString.Lazy as L | ||||
| import qualified Data.Iteratee as I | ||||
| import System.IO | ||||
|  | ||||
| import Data.MessagePack.Base | ||||
| import Data.MessagePack.Class | ||||
| import Data.MessagePack.Feed | ||||
| import Data.MessagePack.Monad | ||||
| import Data.MessagePack.Stream | ||||
| import Data.MessagePack.Pack | ||||
| import Data.MessagePack.Unpack | ||||
| import Data.MessagePack.Object | ||||
| import Data.MessagePack.Iteratee | ||||
| import Data.MessagePack.Derive | ||||
|  | ||||
| -- | Pack Haskell data to MessagePack string. | ||||
| packb :: OBJECT a => a -> IO ByteString | ||||
| packb dat = do | ||||
|   sb <- newSimpleBuffer | ||||
|   pc <- newPacker sb | ||||
|   pack pc dat | ||||
|   simpleBufferData sb | ||||
| bufferSize :: Int | ||||
| bufferSize = 4 * 1024 | ||||
|  | ||||
| -- | Unpack MessagePack string to Haskell data. | ||||
| unpackb :: OBJECT a => ByteString -> IO (Result a) | ||||
| unpackb bs = do | ||||
|   withZone $ \z -> do | ||||
|     r <- unpackObject z bs | ||||
|     return $ case r of | ||||
|       Left err -> Left (show err) | ||||
|       Right (_, dat) -> fromObject dat | ||||
| -- | Pack to ByteString. | ||||
| packToString :: Put -> L.ByteString | ||||
| packToString = runPut | ||||
|  | ||||
| -- | Pure version of 'packb'. | ||||
| packb' :: OBJECT a => a -> ByteString | ||||
| packb' dat = unsafePerformIO $ packb dat | ||||
| -- | Pack to Handle | ||||
| packToHandle :: Handle -> Put -> IO () | ||||
| packToHandle h = L.hPutStr h . packToString | ||||
|  | ||||
| -- | Pure version of 'unpackb'. | ||||
| unpackb' :: OBJECT a => ByteString -> Result a | ||||
| unpackb' bs = unsafePerformIO $ unpackb bs | ||||
| -- | Pack to Handle and Flush Handle | ||||
| packToHandle' :: Handle -> Put -> IO () | ||||
| packToHandle' h p = packToHandle h p >> hFlush h | ||||
|  | ||||
| -- | Pack to File | ||||
| packToFile :: FilePath -> Put -> IO () | ||||
| packToFile path = L.writeFile path . packToString | ||||
|  | ||||
| -- | Unpack from ByteString | ||||
| unpackFromString :: (Monad m, IsByteString s) => s -> A.Parser a -> m a | ||||
| unpackFromString bs = | ||||
|   unpackFromStringI bs . parserToIteratee | ||||
|  | ||||
| -- | Unpack from Handle | ||||
| unpackFromHandle :: CIO.MonadCatchIO m => Handle -> A.Parser a -> m a | ||||
| unpackFromHandle h = | ||||
|   unpackFromHandleI h .parserToIteratee | ||||
|  | ||||
| -- | Unpack from File | ||||
| unpackFromFile :: CIO.MonadCatchIO m => FilePath -> A.Parser a -> m a | ||||
| unpackFromFile path = | ||||
|   unpackFromFileI path . parserToIteratee | ||||
|  | ||||
| -- | Iteratee interface to unpack from ByteString | ||||
| unpackFromStringI :: (Monad m, IsByteString s) => s -> I.Iteratee B.ByteString m a -> m a | ||||
| unpackFromStringI bs = | ||||
|   I.run . I.joinIM . I.enumPure1Chunk (toBS bs) | ||||
|  | ||||
| -- | Iteratee interface to unpack from Handle | ||||
| unpackFromHandleI :: CIO.MonadCatchIO m => Handle -> I.Iteratee B.ByteString m a -> m a | ||||
| unpackFromHandleI h = | ||||
|   I.run . I.joinIM . enumHandleNonBlocking bufferSize h | ||||
|  | ||||
| -- | Iteratee interface to unpack from File | ||||
| unpackFromFileI :: CIO.MonadCatchIO m => FilePath -> I.Iteratee B.ByteString m a -> m a | ||||
| unpackFromFileI path p = | ||||
|   CIO.bracket | ||||
|   (liftIO $ openBinaryFile path ReadMode) | ||||
|   (liftIO . hClose) | ||||
|   (flip unpackFromHandleI p) | ||||
|   | ||||
| @@ -1,584 +0,0 @@ | ||||
| {-# LANGUAGE CPP #-} | ||||
| {-# LANGUAGE ForeignFunctionInterface #-} | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Base | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- Low Level Interface to MessagePack C API | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Base( | ||||
|   -- * Simple Buffer | ||||
|   SimpleBuffer, | ||||
|   newSimpleBuffer, | ||||
|   simpleBufferData, | ||||
|    | ||||
|   -- * Serializer | ||||
|   Packer, | ||||
|   newPacker, | ||||
|    | ||||
|   packU8, | ||||
|   packU16, | ||||
|   packU32, | ||||
|   packU64,   | ||||
|   packS8, | ||||
|   packS16, | ||||
|   packS32, | ||||
|   packS64, | ||||
|    | ||||
|   packTrue, | ||||
|   packFalse, | ||||
|    | ||||
|   packInt, | ||||
|   packDouble, | ||||
|   packNil, | ||||
|   packBool, | ||||
|    | ||||
|   packArray, | ||||
|   packMap, | ||||
|   packRAW, | ||||
|   packRAWBody, | ||||
|   packRAW', | ||||
|    | ||||
|   -- * Stream Deserializer | ||||
|   Unpacker, | ||||
|   defaultInitialBufferSize, | ||||
|   newUnpacker, | ||||
|   unpackerReserveBuffer, | ||||
|   unpackerBuffer, | ||||
|   unpackerBufferCapacity, | ||||
|   unpackerBufferConsumed, | ||||
|   unpackerFeed, | ||||
|   unpackerExecute, | ||||
|   unpackerData, | ||||
|   unpackerReleaseZone, | ||||
|   unpackerResetZone, | ||||
|   unpackerReset, | ||||
|   unpackerMessageSize, | ||||
|    | ||||
|   -- * MessagePack Object | ||||
|   Object(..), | ||||
|   packObject, | ||||
|    | ||||
|   UnpackReturn(..), | ||||
|   unpackObject, | ||||
|    | ||||
|   -- * Memory Zone | ||||
|   Zone, | ||||
|   newZone, | ||||
|   freeZone, | ||||
|   withZone, | ||||
|   ) where | ||||
|  | ||||
| import Control.Exception | ||||
| import Control.Monad | ||||
| import Data.ByteString (ByteString) | ||||
| import qualified Data.ByteString as BS hiding (pack, unpack) | ||||
| import Data.Int | ||||
| import Data.Word | ||||
| import Foreign.C | ||||
| import Foreign.Concurrent | ||||
| import Foreign.ForeignPtr hiding (newForeignPtr) | ||||
| import Foreign.Marshal.Alloc | ||||
| import Foreign.Marshal.Array | ||||
| import Foreign.Ptr | ||||
| import Foreign.Storable | ||||
|  | ||||
| #include <msgpack.h> | ||||
|  | ||||
| type SimpleBuffer = ForeignPtr () | ||||
|  | ||||
| type WriteCallback = Ptr () -> CString -> CUInt -> IO CInt | ||||
|  | ||||
| -- | Create a new Simple Buffer. It will be deleted automatically. | ||||
| newSimpleBuffer :: IO SimpleBuffer | ||||
| newSimpleBuffer = do | ||||
|   ptr <- mallocBytes (#size msgpack_sbuffer) | ||||
|   fptr <- newForeignPtr ptr $ do | ||||
|     msgpack_sbuffer_destroy ptr | ||||
|     free ptr | ||||
|   withForeignPtr fptr $ \p -> | ||||
|     msgpack_sbuffer_init p | ||||
|   return fptr | ||||
|  | ||||
| -- | Get data of Simple Buffer. | ||||
| simpleBufferData :: SimpleBuffer -> IO ByteString | ||||
| simpleBufferData sb = | ||||
|   withForeignPtr sb $ \ptr -> do | ||||
|     size <- (#peek msgpack_sbuffer, size) ptr | ||||
|     dat  <- (#peek msgpack_sbuffer, data) ptr | ||||
|     BS.packCStringLen (dat, fromIntegral (size :: CSize)) | ||||
|  | ||||
| foreign import ccall "msgpack_sbuffer_init_wrap" msgpack_sbuffer_init :: | ||||
|   Ptr () -> IO () | ||||
|  | ||||
| foreign import ccall "msgpack_sbuffer_destroy_wrap" msgpack_sbuffer_destroy :: | ||||
|   Ptr () -> IO () | ||||
|  | ||||
| foreign import ccall "msgpack_sbuffer_write_wrap" msgpack_sbuffer_write :: | ||||
|   WriteCallback | ||||
|  | ||||
| type Packer = ForeignPtr () | ||||
|  | ||||
| -- | Create new Packer. It will be deleted automatically. | ||||
| newPacker :: SimpleBuffer -> IO Packer | ||||
| newPacker sbuf = do | ||||
|   cb <- wrap_callback msgpack_sbuffer_write | ||||
|   ptr <- withForeignPtr sbuf $ \ptr -> | ||||
|     msgpack_packer_new ptr cb | ||||
|   fptr <- newForeignPtr ptr $ do | ||||
|     msgpack_packer_free ptr | ||||
|   return fptr | ||||
|  | ||||
| foreign import ccall "msgpack_packer_new_wrap" msgpack_packer_new :: | ||||
|   Ptr () -> FunPtr WriteCallback -> IO (Ptr ()) | ||||
|  | ||||
| foreign import ccall "msgpack_packer_free_wrap" msgpack_packer_free :: | ||||
|   Ptr () -> IO () | ||||
|  | ||||
| foreign import ccall "wrapper" wrap_callback :: | ||||
|   WriteCallback -> IO (FunPtr WriteCallback) | ||||
|  | ||||
| packU8 :: Packer -> Word8 -> IO Int | ||||
| packU8 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_uint8 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_uint8_wrap" msgpack_pack_uint8 :: | ||||
|   Ptr () -> Word8 -> IO CInt | ||||
|  | ||||
| packU16 :: Packer -> Word16 -> IO Int | ||||
| packU16 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_uint16 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_uint16_wrap" msgpack_pack_uint16 :: | ||||
|   Ptr () -> Word16 -> IO CInt | ||||
|  | ||||
| packU32 :: Packer -> Word32 -> IO Int | ||||
| packU32 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_uint32 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_uint32_wrap" msgpack_pack_uint32 :: | ||||
|   Ptr () -> Word32 -> IO CInt | ||||
|  | ||||
| packU64 :: Packer -> Word64 -> IO Int | ||||
| packU64 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_uint64 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_uint64_wrap" msgpack_pack_uint64 :: | ||||
|   Ptr () -> Word64 -> IO CInt | ||||
|  | ||||
| packS8 :: Packer -> Int8 -> IO Int | ||||
| packS8 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_int8 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_int8_wrap" msgpack_pack_int8 :: | ||||
|   Ptr () -> Int8 -> IO CInt | ||||
|  | ||||
| packS16 :: Packer -> Int16 -> IO Int | ||||
| packS16 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_int16 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_int16_wrap" msgpack_pack_int16 :: | ||||
|   Ptr () -> Int16 -> IO CInt | ||||
|  | ||||
| packS32 :: Packer -> Int32 -> IO Int | ||||
| packS32 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_int32 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_int32_wrap" msgpack_pack_int32 :: | ||||
|   Ptr () -> Int32 -> IO CInt | ||||
|  | ||||
| packS64 :: Packer -> Int64 -> IO Int | ||||
| packS64 pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_int64 ptr n | ||||
|  | ||||
| foreign import ccall "msgpack_pack_int64_wrap" msgpack_pack_int64 :: | ||||
|   Ptr () -> Int64 -> IO CInt | ||||
|  | ||||
| -- | Pack an integral data. | ||||
| packInt :: Integral a => Packer -> a -> IO Int | ||||
| packInt pc n = packS64 pc $ fromIntegral n | ||||
|  | ||||
| -- | Pack a double data. | ||||
| packDouble :: Packer -> Double -> IO Int | ||||
| packDouble pc d = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_double ptr (realToFrac d) | ||||
|  | ||||
| foreign import ccall "msgpack_pack_double_wrap" msgpack_pack_double :: | ||||
|   Ptr () -> CDouble -> IO CInt | ||||
|  | ||||
| -- | Pack a nil. | ||||
| packNil :: Packer -> IO Int | ||||
| packNil pc = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_nil ptr | ||||
|  | ||||
| foreign import ccall "msgpack_pack_nil_wrap" msgpack_pack_nil :: | ||||
|   Ptr () -> IO CInt | ||||
|  | ||||
| packTrue :: Packer -> IO Int | ||||
| packTrue pc = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_true ptr | ||||
|  | ||||
| foreign import ccall "msgpack_pack_true_wrap" msgpack_pack_true :: | ||||
|   Ptr () -> IO CInt | ||||
|  | ||||
| packFalse :: Packer -> IO Int | ||||
| packFalse pc = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_false ptr | ||||
|  | ||||
| foreign import ccall "msgpack_pack_false_wrap" msgpack_pack_false :: | ||||
|   Ptr () -> IO CInt | ||||
|  | ||||
| -- | Pack a bool data. | ||||
| packBool :: Packer -> Bool -> IO Int | ||||
| packBool pc True  = packTrue pc | ||||
| packBool pc False = packFalse pc | ||||
|  | ||||
| -- | 'packArray' @p n@ starts packing an array.  | ||||
| -- Next @n@ data will consist this array. | ||||
| packArray :: Packer -> Int -> IO Int | ||||
| packArray pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_array ptr (fromIntegral n) | ||||
|  | ||||
| foreign import ccall "msgpack_pack_array_wrap" msgpack_pack_array :: | ||||
|   Ptr () -> CUInt -> IO CInt | ||||
|  | ||||
| -- | 'packMap' @p n@ starts packing a map.  | ||||
| -- Next @n@ pairs of data (2*n data) will consist this map. | ||||
| packMap :: Packer -> Int -> IO Int | ||||
| packMap pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_map ptr (fromIntegral n) | ||||
|  | ||||
| foreign import ccall "msgpack_pack_map_wrap" msgpack_pack_map :: | ||||
|   Ptr () -> CUInt -> IO CInt | ||||
|  | ||||
| -- | 'packRAW' @p n@ starts packing a byte sequence.  | ||||
| -- Next total @n@ bytes of 'packRAWBody' call will consist this sequence. | ||||
| packRAW :: Packer -> Int -> IO Int | ||||
| packRAW pc n = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|     msgpack_pack_raw ptr (fromIntegral n) | ||||
|  | ||||
| foreign import ccall "msgpack_pack_raw_wrap" msgpack_pack_raw :: | ||||
|   Ptr () -> CSize -> IO CInt | ||||
|  | ||||
| -- | Pack a byte sequence. | ||||
| packRAWBody :: Packer -> ByteString -> IO Int | ||||
| packRAWBody pc bs = | ||||
|   liftM fromIntegral $ withForeignPtr pc $ \ptr -> | ||||
|   BS.useAsCStringLen bs $ \(str, len) -> | ||||
|     msgpack_pack_raw_body ptr (castPtr str) (fromIntegral len) | ||||
|  | ||||
| foreign import ccall "msgpack_pack_raw_body_wrap" msgpack_pack_raw_body :: | ||||
|   Ptr () -> Ptr () -> CSize -> IO CInt | ||||
|  | ||||
| -- | Pack a single byte stream. It calls 'packRAW' and 'packRAWBody'. | ||||
| packRAW' :: Packer -> ByteString -> IO Int | ||||
| packRAW' pc bs = do | ||||
|   _ <- packRAW pc (BS.length bs) | ||||
|   packRAWBody pc bs | ||||
|  | ||||
| type Unpacker = ForeignPtr () | ||||
|  | ||||
| defaultInitialBufferSize :: Int | ||||
| defaultInitialBufferSize = 32 * 1024 -- #const MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE | ||||
|  | ||||
| -- | 'newUnpacker' @initialBufferSize@ creates a new Unpacker. It will be deleted automatically. | ||||
| newUnpacker :: Int -> IO Unpacker | ||||
| newUnpacker initialBufferSize = do | ||||
|   ptr <- msgpack_unpacker_new (fromIntegral initialBufferSize) | ||||
|   fptr <- newForeignPtr ptr $ do | ||||
|     msgpack_unpacker_free ptr | ||||
|   return fptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_new" msgpack_unpacker_new :: | ||||
|   CSize -> IO (Ptr ()) | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_free" msgpack_unpacker_free :: | ||||
|   Ptr() -> IO () | ||||
|  | ||||
| -- | 'unpackerReserveBuffer' @up size@ reserves at least @size@ bytes of buffer. | ||||
| unpackerReserveBuffer :: Unpacker -> Int -> IO Bool | ||||
| unpackerReserveBuffer up size = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   liftM (/=0) $ msgpack_unpacker_reserve_buffer ptr (fromIntegral size) | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_reserve_buffer_wrap" msgpack_unpacker_reserve_buffer :: | ||||
|   Ptr () -> CSize -> IO CChar | ||||
|  | ||||
| -- | Get a pointer of unpacker buffer. | ||||
| unpackerBuffer :: Unpacker -> IO (Ptr CChar) | ||||
| unpackerBuffer up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   msgpack_unpacker_buffer ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_buffer_wrap" msgpack_unpacker_buffer :: | ||||
|   Ptr () -> IO (Ptr CChar) | ||||
|  | ||||
| -- | Get size of allocated buffer. | ||||
| unpackerBufferCapacity :: Unpacker -> IO Int | ||||
| unpackerBufferCapacity up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   liftM fromIntegral $ msgpack_unpacker_buffer_capacity ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_buffer_capacity_wrap" msgpack_unpacker_buffer_capacity :: | ||||
|   Ptr () -> IO CSize | ||||
|  | ||||
| -- | 'unpackerBufferConsumed' @up size@ notices that writed @size@ bytes to buffer. | ||||
| unpackerBufferConsumed :: Unpacker -> Int -> IO () | ||||
| unpackerBufferConsumed up size = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   msgpack_unpacker_buffer_consumed ptr (fromIntegral size) | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_buffer_consumed_wrap" msgpack_unpacker_buffer_consumed :: | ||||
|   Ptr () -> CSize -> IO () | ||||
|  | ||||
| -- | Write byte sequence to Unpacker. It is utility funciton, calls 'unpackerReserveBuffer', 'unpackerBuffer' and 'unpackerBufferConsumed'. | ||||
| unpackerFeed :: Unpacker -> ByteString -> IO () | ||||
| unpackerFeed up bs = | ||||
|   BS.useAsCStringLen bs $ \(str, len) -> do | ||||
|     True <- unpackerReserveBuffer up len | ||||
|     ptr <- unpackerBuffer up | ||||
|     copyArray ptr str len | ||||
|     unpackerBufferConsumed up len | ||||
|  | ||||
| -- | Execute deserializing. It returns 0 when buffer contains not enough bytes, returns 1 when succeeded, returns negative value when it failed. | ||||
| unpackerExecute :: Unpacker -> IO Int | ||||
| unpackerExecute up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   liftM fromIntegral $ msgpack_unpacker_execute ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_execute" msgpack_unpacker_execute :: | ||||
|   Ptr () -> IO CInt | ||||
|  | ||||
| -- | Returns a deserialized object when 'unpackerExecute' returned 1. | ||||
| unpackerData :: Unpacker -> IO Object | ||||
| unpackerData up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   allocaBytes (#size msgpack_object) $ \pobj -> do | ||||
|     msgpack_unpacker_data ptr pobj | ||||
|     peekObject pobj | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_data_wrap" msgpack_unpacker_data :: | ||||
|   Ptr () -> Ptr () -> IO () | ||||
|  | ||||
| -- | Release memory zone. The returned zone must be freed by calling 'freeZone'. | ||||
| unpackerReleaseZone :: Unpacker -> IO Zone | ||||
| unpackerReleaseZone up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   msgpack_unpacker_release_zone ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_release_zone" msgpack_unpacker_release_zone :: | ||||
|   Ptr () -> IO (Ptr ()) | ||||
|  | ||||
| -- | Free memory zone used by Unapcker. | ||||
| unpackerResetZone :: Unpacker -> IO () | ||||
| unpackerResetZone up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   msgpack_unpacker_reset_zone ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_reset_zone" msgpack_unpacker_reset_zone :: | ||||
|   Ptr () -> IO () | ||||
|  | ||||
| -- | Reset Unpacker state except memory zone. | ||||
| unpackerReset :: Unpacker -> IO () | ||||
| unpackerReset up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   msgpack_unpacker_reset ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_reset" msgpack_unpacker_reset :: | ||||
|   Ptr () -> IO () | ||||
|  | ||||
| -- | Returns number of bytes of sequence of deserializing object. | ||||
| unpackerMessageSize :: Unpacker -> IO Int | ||||
| unpackerMessageSize up = | ||||
|   withForeignPtr up $ \ptr -> | ||||
|   liftM fromIntegral $ msgpack_unpacker_message_size ptr | ||||
|  | ||||
| foreign import ccall "msgpack_unpacker_message_size_wrap" msgpack_unpacker_message_size :: | ||||
|   Ptr () -> IO CSize | ||||
|  | ||||
| type Zone = Ptr () | ||||
|  | ||||
| -- | Create a new memory zone. It must be freed manually. | ||||
| newZone :: IO Zone | ||||
| newZone = | ||||
|   msgpack_zone_new (#const MSGPACK_ZONE_CHUNK_SIZE) | ||||
|  | ||||
| -- | Free a memory zone. | ||||
| freeZone :: Zone -> IO () | ||||
| freeZone z = | ||||
|   msgpack_zone_free z | ||||
|  | ||||
| -- | Create a memory zone, then execute argument, then free memory zone. | ||||
| withZone :: (Zone -> IO a) -> IO a | ||||
| withZone z = | ||||
|   bracket newZone freeZone z | ||||
|  | ||||
| foreign import ccall "msgpack_zone_new" msgpack_zone_new :: | ||||
|   CSize -> IO Zone | ||||
|  | ||||
| foreign import ccall "msgpack_zone_free" msgpack_zone_free :: | ||||
|   Zone -> IO () | ||||
|  | ||||
| -- | Object Representation of MessagePack data. | ||||
| data Object = | ||||
|   ObjectNil | ||||
|   | ObjectBool Bool | ||||
|   | ObjectInteger Int | ||||
|   | ObjectDouble Double | ||||
|   | ObjectRAW ByteString | ||||
|   | ObjectArray [Object] | ||||
|   | ObjectMap [(Object, Object)] | ||||
|   deriving (Show) | ||||
|  | ||||
| peekObject :: Ptr a -> IO Object | ||||
| peekObject ptr = do | ||||
|   typ <- (#peek msgpack_object, type) ptr | ||||
|   case (typ :: CInt) of | ||||
|     (#const MSGPACK_OBJECT_NIL) -> | ||||
|       return ObjectNil | ||||
|     (#const MSGPACK_OBJECT_BOOLEAN) -> | ||||
|       peekObjectBool ptr | ||||
|     (#const MSGPACK_OBJECT_POSITIVE_INTEGER) -> | ||||
|       peekObjectPositiveInteger ptr | ||||
|     (#const MSGPACK_OBJECT_NEGATIVE_INTEGER) -> | ||||
|       peekObjectNegativeInteger ptr | ||||
|     (#const MSGPACK_OBJECT_DOUBLE) -> | ||||
|       peekObjectDouble ptr | ||||
|     (#const MSGPACK_OBJECT_RAW) -> | ||||
|       peekObjectRAW ptr | ||||
|     (#const MSGPACK_OBJECT_ARRAY) -> | ||||
|       peekObjectArray ptr | ||||
|     (#const MSGPACK_OBJECT_MAP) -> | ||||
|       peekObjectMap ptr | ||||
|     _ -> | ||||
|       fail $ "peekObject: unknown object type (" ++ show typ ++ ")" | ||||
|  | ||||
| peekObjectBool :: Ptr a -> IO Object | ||||
| peekObjectBool ptr = do | ||||
|   b <- (#peek msgpack_object, via.boolean) ptr | ||||
|   return $ ObjectBool $ (b :: CUChar) /= 0 | ||||
|  | ||||
| peekObjectPositiveInteger :: Ptr a -> IO Object | ||||
| peekObjectPositiveInteger ptr = do | ||||
|   n <- (#peek msgpack_object, via.u64) ptr | ||||
|   return $ ObjectInteger $ fromIntegral (n :: Word64) | ||||
|  | ||||
| peekObjectNegativeInteger :: Ptr a -> IO Object | ||||
| peekObjectNegativeInteger ptr = do | ||||
|   n <- (#peek msgpack_object, via.i64) ptr | ||||
|   return $ ObjectInteger $ fromIntegral (n :: Int64) | ||||
|  | ||||
| peekObjectDouble :: Ptr a -> IO Object | ||||
| peekObjectDouble ptr = do | ||||
|   d <- (#peek msgpack_object, via.dec) ptr | ||||
|   return $ ObjectDouble $ realToFrac (d :: CDouble) | ||||
|  | ||||
| peekObjectRAW :: Ptr a -> IO Object | ||||
| peekObjectRAW ptr = do | ||||
|   size <- (#peek msgpack_object, via.raw.size) ptr | ||||
|   p    <- (#peek msgpack_object, via.raw.ptr) ptr | ||||
|   bs   <- BS.packCStringLen (p, fromIntegral (size :: Word32)) | ||||
|   return $ ObjectRAW bs | ||||
|  | ||||
| peekObjectArray :: Ptr a -> IO Object | ||||
| peekObjectArray ptr = do | ||||
|   csize <- (#peek msgpack_object, via.array.size) ptr | ||||
|   let size = fromIntegral (csize :: Word32) | ||||
|   p     <- (#peek msgpack_object, via.array.ptr) ptr | ||||
|   objs  <- mapM (\i -> peekObject $ p `plusPtr` | ||||
|                       ((#size msgpack_object) * i)) | ||||
|            [0..size-1] | ||||
|   return $ ObjectArray objs | ||||
|  | ||||
| peekObjectMap :: Ptr a -> IO Object | ||||
| peekObjectMap ptr = do | ||||
|   csize <- (#peek msgpack_object, via.map.size) ptr | ||||
|   let size = fromIntegral (csize :: Word32) | ||||
|   p     <- (#peek msgpack_object, via.map.ptr) ptr | ||||
|   dat   <- mapM (\i -> peekObjectKV $ p `plusPtr` | ||||
|                       ((#size msgpack_object_kv) * i)) | ||||
|            [0..size-1] | ||||
|   return $ ObjectMap dat | ||||
|  | ||||
| peekObjectKV :: Ptr a -> IO (Object, Object) | ||||
| peekObjectKV ptr = do | ||||
|   k <- peekObject $ ptr `plusPtr` (#offset msgpack_object_kv, key) | ||||
|   v <- peekObject $ ptr `plusPtr` (#offset msgpack_object_kv, val) | ||||
|   return (k, v) | ||||
|  | ||||
| -- | Pack a Object. | ||||
| packObject :: Packer -> Object -> IO () | ||||
| packObject pc ObjectNil = packNil pc >> return () | ||||
|  | ||||
| packObject pc (ObjectBool b) = packBool pc b >> return () | ||||
|  | ||||
| packObject pc (ObjectInteger n) = packInt pc n >> return () | ||||
|  | ||||
| packObject pc (ObjectDouble d) = packDouble pc d >> return () | ||||
|  | ||||
| packObject pc (ObjectRAW bs) = packRAW' pc bs >> return () | ||||
|  | ||||
| packObject pc (ObjectArray ls) = do | ||||
|   _ <- packArray pc (length ls) | ||||
|   mapM_ (packObject pc) ls | ||||
|  | ||||
| packObject pc (ObjectMap ls) = do | ||||
|   _ <- packMap pc (length ls) | ||||
|   mapM_ (\(a, b) -> packObject pc a >> packObject pc b) ls | ||||
|  | ||||
| data UnpackReturn = | ||||
|   UnpackContinue     -- ^ not enough bytes to unpack object | ||||
|   | UnpackParseError -- ^ got invalid bytes | ||||
|   | UnpackError      -- ^ other error | ||||
|   deriving (Eq, Show) | ||||
|  | ||||
| -- | Unpack a single MessagePack object from byte sequence. | ||||
| unpackObject :: Zone -> ByteString -> IO (Either UnpackReturn (Int, Object)) | ||||
| unpackObject z dat = | ||||
|   allocaBytes (#size msgpack_object) $ \ptr -> | ||||
|   BS.useAsCStringLen dat $ \(str, len) -> | ||||
|   alloca $ \poff -> do | ||||
|     poke poff 0 | ||||
|     ret <- msgpack_unpack str (fromIntegral len) poff z ptr | ||||
|     case ret of | ||||
|       (#const MSGPACK_UNPACK_SUCCESS) -> do | ||||
|         off <- peek poff | ||||
|         obj <- peekObject ptr | ||||
|         return $ Right (fromIntegral off, obj) | ||||
|       (#const MSGPACK_UNPACK_EXTRA_BYTES) -> do | ||||
|         off <- peek poff | ||||
|         obj <- peekObject ptr | ||||
|         return $ Right (fromIntegral off, obj) | ||||
|       (#const MSGPACK_UNPACK_CONTINUE) -> | ||||
|         return $ Left UnpackContinue | ||||
|       (#const MSGPACK_UNPACK_PARSE_ERROR) -> | ||||
|         return $ Left UnpackParseError | ||||
|       _ -> | ||||
|         return $ Left UnpackError | ||||
|  | ||||
| foreign import ccall "msgpack_unpack" msgpack_unpack :: | ||||
|   Ptr CChar -> CSize -> Ptr CSize -> Zone -> Ptr () -> IO CInt | ||||
| @@ -1,101 +0,0 @@ | ||||
| {-# LANGUAGE TypeSynonymInstances #-} | ||||
| {-# LANGUAGE FlexibleInstances #-} | ||||
| {-# LANGUAGE OverlappingInstances #-} | ||||
| {-# LANGUAGE IncoherentInstances #-} | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Class | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- Serializing Haskell values to and from MessagePack Objects. | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Class( | ||||
|   -- * Serialization to and from Object | ||||
|   OBJECT(..), | ||||
|   Result, | ||||
|   pack, | ||||
|   ) where | ||||
|  | ||||
| import Control.Monad.Error | ||||
| import Data.ByteString.Char8 (ByteString) | ||||
| import qualified Data.ByteString.Char8 as C8 | ||||
|  | ||||
| import Data.MessagePack.Base | ||||
|  | ||||
| -- | The class of types serializable to and from MessagePack object | ||||
| class OBJECT a where | ||||
|   toObject :: a -> Object | ||||
|   fromObject :: Object -> Result a | ||||
|  | ||||
| -- | A type for parser results | ||||
| type Result a = Either String a | ||||
|  | ||||
| instance OBJECT Object where | ||||
|   toObject = id | ||||
|   fromObject = Right | ||||
|  | ||||
| fromObjectError :: String | ||||
| fromObjectError = "fromObject: cannot cast" | ||||
|  | ||||
| instance OBJECT () where | ||||
|   toObject = const ObjectNil | ||||
|   fromObject ObjectNil = Right () | ||||
|   fromObject _ = Left fromObjectError | ||||
|  | ||||
| instance OBJECT Int where | ||||
|   toObject = ObjectInteger | ||||
|   fromObject (ObjectInteger n) = Right n | ||||
|   fromObject _ = Left fromObjectError | ||||
|  | ||||
| instance OBJECT Bool where | ||||
|   toObject = ObjectBool | ||||
|   fromObject (ObjectBool b) = Right b | ||||
|   fromObject _ = Left fromObjectError | ||||
|  | ||||
| instance OBJECT Double where | ||||
|   toObject = ObjectDouble | ||||
|   fromObject (ObjectDouble d) = Right d | ||||
|   fromObject _ = Left fromObjectError | ||||
|  | ||||
| instance OBJECT ByteString where | ||||
|   toObject = ObjectRAW | ||||
|   fromObject (ObjectRAW bs) = Right bs | ||||
|   fromObject _ = Left fromObjectError | ||||
|  | ||||
| instance OBJECT String where | ||||
|   toObject = toObject . C8.pack | ||||
|   fromObject obj = liftM C8.unpack $ fromObject obj | ||||
|  | ||||
| instance OBJECT a => OBJECT [a] where | ||||
|   toObject = ObjectArray . map toObject | ||||
|   fromObject (ObjectArray arr) = | ||||
|     mapM fromObject arr | ||||
|   fromObject _ = | ||||
|     Left fromObjectError | ||||
|  | ||||
| instance (OBJECT a, OBJECT b) => OBJECT [(a, b)] where | ||||
|   toObject = | ||||
|     ObjectMap . map (\(a, b) -> (toObject a, toObject b)) | ||||
|   fromObject (ObjectMap mem) = do | ||||
|     mapM (\(a, b) -> liftM2 (,) (fromObject a) (fromObject b)) mem | ||||
|   fromObject _ = | ||||
|     Left fromObjectError | ||||
|  | ||||
| instance OBJECT a => OBJECT (Maybe a) where | ||||
|   toObject (Just a) = toObject a | ||||
|   toObject Nothing = ObjectNil | ||||
|    | ||||
|   fromObject ObjectNil = return Nothing | ||||
|   fromObject obj = liftM Just $ fromObject obj | ||||
|  | ||||
| -- | Pack a serializable Haskell value. | ||||
| pack :: OBJECT a => Packer -> a -> IO () | ||||
| pack pc = packObject pc . toObject | ||||
							
								
								
									
										106
									
								
								haskell/src/Data/MessagePack/Derive.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								haskell/src/Data/MessagePack/Derive.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,106 @@ | ||||
| {-# Language TemplateHaskell #-} | ||||
|  | ||||
| module Data.MessagePack.Derive ( | ||||
|   derivePack, | ||||
|   deriveUnpack, | ||||
|   deriveObject, | ||||
|   ) where | ||||
|  | ||||
| import Control.Applicative | ||||
| import Control.Monad | ||||
| import Language.Haskell.TH | ||||
|  | ||||
| import Data.MessagePack.Pack | ||||
| import Data.MessagePack.Unpack | ||||
| import Data.MessagePack.Object | ||||
|  | ||||
| deriveUnpack :: Name -> Q [Dec] | ||||
| deriveUnpack typName = do | ||||
|   TyConI (DataD _ name _ cons _) <- reify typName | ||||
|    | ||||
|   return | ||||
|     [ InstanceD [] (AppT (ConT ''Unpackable) (ConT name)) | ||||
|       [ FunD 'get [Clause [] (NormalB $ ch $ map body cons) []] | ||||
|         ]] | ||||
|          | ||||
|   where | ||||
|     body (NormalC conName elms) = | ||||
|       DoE $ | ||||
|       tupOrListP (map VarP names) (VarE 'get) ++ | ||||
|       [ NoBindS $ AppE (VarE 'return) $ foldl AppE (ConE conName) $ map VarE names ] | ||||
|         where | ||||
|           names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms | ||||
|  | ||||
|     body (RecC conName elms) = | ||||
|       body (NormalC conName $ map (\(_, b, c) -> (b, c)) elms) | ||||
|  | ||||
|     ch = foldl1 (\e f -> AppE (AppE (VarE '(<|>)) e) f) | ||||
|  | ||||
| derivePack :: Name -> Q [Dec] | ||||
| derivePack typName = do | ||||
|   TyConI (DataD _ name _ cons _) <- reify typName | ||||
|    | ||||
|   return | ||||
|     [ InstanceD [] (AppT (ConT ''Packable) (ConT name)) | ||||
|       [ FunD 'put (map body cons) | ||||
|         ]] | ||||
|          | ||||
|   where | ||||
|     body (NormalC conName elms) = | ||||
|       Clause | ||||
|         [ ConP conName $ map VarP names ] | ||||
|         (NormalB $ AppE (VarE 'put) $ tupOrListE $ map VarE names) [] | ||||
|       where | ||||
|         names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms | ||||
|  | ||||
|     body (RecC conName elms) = | ||||
|       body (NormalC conName $ map (\(_, b, c) -> (b, c)) elms) | ||||
|  | ||||
| deriveObject :: Name -> Q [Dec] | ||||
| deriveObject typName = do | ||||
|   g <- derivePack typName | ||||
|   p <- deriveUnpack typName | ||||
|  | ||||
|   TyConI (DataD _ name _ cons _) <- reify typName | ||||
|   let o = InstanceD [] (AppT (ConT ''OBJECT) (ConT name)) | ||||
|           [ FunD 'toObject (map toObjectBody cons), | ||||
|             FunD 'tryFromObject [Clause [ VarP oname ] | ||||
|                                  (NormalB $ ch $ map tryFromObjectBody cons) []]] | ||||
|  | ||||
|   return $ g ++ p ++ [o] | ||||
|   where | ||||
|     toObjectBody (NormalC conName elms) = | ||||
|       Clause | ||||
|         [ ConP conName $ map VarP names ] | ||||
|         (NormalB $ AppE (VarE 'toObject) $ tupOrListE $ map VarE names) [] | ||||
|       where | ||||
|         names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms | ||||
|     toObjectBody (RecC conName elms) = | ||||
|       toObjectBody (NormalC conName $ map (\(_, b, c) -> (b, c)) elms) | ||||
|  | ||||
|     tryFromObjectBody (NormalC conName elms) = | ||||
|       DoE $ | ||||
|       tupOrListP (map VarP names) (AppE (VarE 'tryFromObject) (VarE oname)) ++ | ||||
|       [ NoBindS $ AppE (VarE 'return) $ foldl AppE (ConE conName) $ map VarE names ] | ||||
|         where | ||||
|           names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms | ||||
|     tryFromObjectBody (RecC conName elms) = | ||||
|       tryFromObjectBody (NormalC conName $ map (\(_, b, c) -> (b, c)) elms) | ||||
|  | ||||
|     oname = mkName "o" | ||||
|     ch = foldl1 (\e f -> AppE (AppE (VarE '(<|>)) e) f) | ||||
|  | ||||
| tupOrListP :: [Pat] -> Exp -> [Stmt] | ||||
| tupOrListP ls e | ||||
|   | length ls == 0 = | ||||
|     let lsname = mkName "ls" in | ||||
|     [ BindS (VarP lsname) e | ||||
|     , NoBindS $ AppE (VarE 'guard) $ AppE (VarE 'null) $ SigE (VarE lsname) (AppT ListT (ConT ''())) ] | ||||
|   | length ls == 1 = [ BindS (ListP ls) e ] | ||||
|   | otherwise = [ BindS (TupP ls) e ] | ||||
|  | ||||
| tupOrListE :: [Exp] -> Exp | ||||
| tupOrListE ls | ||||
|   | length ls == 0 = SigE (ListE []) (AppT ListT (ConT ''())) | ||||
|   | length ls == 1 = ListE ls | ||||
|   | otherwise = TupE ls | ||||
| @@ -1,62 +0,0 @@ | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Feed | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- Feeders for Stream Deserializers | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Feed( | ||||
|   -- * Feeder type | ||||
|   Feeder, | ||||
|   -- * Feeders | ||||
|   feederFromHandle, | ||||
|   feederFromFile, | ||||
|   feederFromString, | ||||
|   ) where | ||||
|  | ||||
| import Data.ByteString (ByteString) | ||||
| import qualified Data.ByteString as BS | ||||
| import Data.IORef | ||||
| import System.IO | ||||
|  | ||||
| -- | Feeder returns Just ByteString when bytes remains, otherwise Nothing. | ||||
| type Feeder = IO (Maybe ByteString) | ||||
|  | ||||
| -- | Feeder from Handle | ||||
| feederFromHandle :: Handle -> IO Feeder | ||||
| feederFromHandle h = return $ do | ||||
|   bs <- BS.hGetNonBlocking h bufSize | ||||
|   if BS.length bs > 0 | ||||
|     then do return $ Just bs | ||||
|     else do | ||||
|     c <- BS.hGet h 1 | ||||
|     if BS.length c > 0 | ||||
|       then do return $ Just c | ||||
|       else do | ||||
|       hClose h | ||||
|       return Nothing | ||||
|   where | ||||
|     bufSize = 4096 | ||||
|  | ||||
| -- | Feeder from File | ||||
| feederFromFile :: FilePath -> IO Feeder | ||||
| feederFromFile path = | ||||
|   openFile path ReadMode >>= feederFromHandle | ||||
|  | ||||
| -- | Feeder from ByteString | ||||
| feederFromString :: ByteString -> IO Feeder | ||||
| feederFromString bs = do | ||||
|   r <- newIORef (Just bs) | ||||
|   return $ f r | ||||
|   where | ||||
|     f r = do | ||||
|       mb <- readIORef r | ||||
|       writeIORef r Nothing | ||||
|       return mb | ||||
							
								
								
									
										82
									
								
								haskell/src/Data/MessagePack/Iteratee.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								haskell/src/Data/MessagePack/Iteratee.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Iteratee | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009-2010 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- MessagePack Deserializer interface to @Data.Iteratee@ | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Iteratee( | ||||
|   -- * Iteratee version of deserializer | ||||
|   getI, | ||||
|   -- * Non Blocking Enumerator | ||||
|   enumHandleNonBlocking, | ||||
|   -- * Convert Parser to Iteratee | ||||
|   parserToIteratee, | ||||
|   ) where | ||||
|  | ||||
| import Control.Exception | ||||
| import Control.Monad.IO.Class | ||||
| import qualified Data.Attoparsec as A | ||||
| import qualified Data.ByteString as B | ||||
| import qualified Data.Iteratee as I | ||||
| import System.IO | ||||
|  | ||||
| import Data.MessagePack.Unpack | ||||
|  | ||||
| -- | Deserialize a value | ||||
| getI :: (Monad m, Unpackable a) => I.Iteratee B.ByteString m a | ||||
| getI = parserToIteratee get | ||||
|  | ||||
| -- | Enumerator | ||||
| enumHandleNonBlocking :: MonadIO m => Int -> Handle -> I.Enumerator B.ByteString m a | ||||
| enumHandleNonBlocking bufSize h = | ||||
|   I.enumFromCallback $ readSome bufSize h | ||||
|  | ||||
| readSome :: MonadIO m => Int -> Handle -> m (Either SomeException (Bool, B.ByteString)) | ||||
| readSome bufSize h = liftIO $ do | ||||
|   ebs <- try $ hGetSome bufSize h | ||||
|   case ebs of | ||||
|     Left exc -> | ||||
|       return $ Left (exc :: SomeException) | ||||
|     Right bs | B.null bs -> | ||||
|       return $ Right (False, B.empty) | ||||
|     Right bs -> | ||||
|       return $ Right (True, bs) | ||||
|  | ||||
| hGetSome :: Int -> Handle -> IO B.ByteString | ||||
| hGetSome bufSize h = do | ||||
|   bs <- B.hGetNonBlocking h bufSize | ||||
|   if B.null bs | ||||
|     then do | ||||
|     hd <- B.hGet h 1 | ||||
|     if B.null hd | ||||
|       then do | ||||
|       return B.empty | ||||
|       else do | ||||
|       rest <- B.hGetNonBlocking h (bufSize - 1) | ||||
|       return $ B.cons (B.head hd) rest | ||||
|     else do | ||||
|     return bs | ||||
|  | ||||
| -- | Convert Parser to Iteratee | ||||
| parserToIteratee :: Monad m => A.Parser a -> I.Iteratee B.ByteString m a | ||||
| parserToIteratee p = I.icont (itr (A.parse p)) Nothing | ||||
|   where | ||||
|     itr pcont s = case s of | ||||
|       I.EOF _ -> | ||||
|         I.throwErr (I.setEOF s) | ||||
|       I.Chunk bs -> | ||||
|         case pcont bs of | ||||
|           A.Fail _ _ msg -> | ||||
|             I.throwErr (I.iterStrExc msg) | ||||
|           A.Partial cont -> | ||||
|             I.icont (itr cont) Nothing | ||||
|           A.Done remain ret -> | ||||
|             I.idone ret (I.Chunk remain) | ||||
| @@ -1,156 +0,0 @@ | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Monad | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- Monadic Stream Serializers and Deserializers | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Monad( | ||||
|   -- * Classes | ||||
|   MonadPacker(..), | ||||
|   MonadUnpacker(..), | ||||
|    | ||||
|   -- * Packer and Unpacker type | ||||
|   PackerT(..), | ||||
|   UnpackerT(..), | ||||
|    | ||||
|   -- * Packers | ||||
|   packToString, | ||||
|   packToHandle, | ||||
|   packToFile, | ||||
|    | ||||
|   -- * Unpackers | ||||
|   unpackFrom, | ||||
|   unpackFromString, | ||||
|   unpackFromHandle, | ||||
|   unpackFromFile, | ||||
|   ) where | ||||
|  | ||||
| import Control.Monad | ||||
| import Control.Monad.Trans | ||||
| import Data.ByteString (ByteString) | ||||
| import qualified Data.ByteString as BS | ||||
| import System.IO | ||||
|  | ||||
| import Data.MessagePack.Base hiding (Unpacker) | ||||
| import qualified Data.MessagePack.Base as Base | ||||
| import Data.MessagePack.Class | ||||
| import Data.MessagePack.Feed | ||||
|  | ||||
| class Monad m => MonadPacker m where | ||||
|   -- | Serialize a object | ||||
|   put :: OBJECT a => a -> m () | ||||
|  | ||||
| class Monad m => MonadUnpacker m where | ||||
|   -- | Deserialize a object | ||||
|   get :: OBJECT a => m a | ||||
|  | ||||
| -- | Serializer Type | ||||
| newtype PackerT m r = PackerT { runPackerT :: Base.Packer -> m r } | ||||
|  | ||||
| instance Monad m => Monad (PackerT m) where | ||||
|   a >>= b = | ||||
|     PackerT $ \pc -> do | ||||
|       r <- runPackerT a pc | ||||
|       runPackerT (b r) pc | ||||
|    | ||||
|   return r = | ||||
|     PackerT $ \_ -> return r | ||||
|  | ||||
| instance MonadTrans PackerT where | ||||
|   lift m = PackerT $ \_ -> m | ||||
|  | ||||
| instance MonadIO m => MonadIO (PackerT m) where | ||||
|   liftIO = lift . liftIO | ||||
|  | ||||
| instance MonadIO m => MonadPacker (PackerT m) where | ||||
|   put v = PackerT $ \pc -> liftIO $ do | ||||
|     pack pc v | ||||
|  | ||||
| -- | Execute given serializer and returns byte sequence. | ||||
| packToString :: MonadIO m => PackerT m r -> m ByteString | ||||
| packToString m = do | ||||
|   sb <- liftIO $ newSimpleBuffer | ||||
|   pc <- liftIO $ newPacker sb | ||||
|   _ <- runPackerT m pc | ||||
|   liftIO $ simpleBufferData sb | ||||
|  | ||||
| -- | Execute given serializer and write byte sequence to Handle. | ||||
| packToHandle :: MonadIO m => Handle -> PackerT m r -> m () | ||||
| packToHandle h m = do | ||||
|   sb <- packToString m | ||||
|   liftIO $ BS.hPut h sb | ||||
|   liftIO $ hFlush h | ||||
|  | ||||
| -- | Execute given serializer and write byte sequence to file. | ||||
| packToFile :: MonadIO m => FilePath -> PackerT m r -> m () | ||||
| packToFile p m = do | ||||
|   sb <- packToString m | ||||
|   liftIO $ BS.writeFile p sb | ||||
|  | ||||
| -- | Deserializer type | ||||
| newtype UnpackerT m r = UnpackerT { runUnpackerT :: Base.Unpacker -> Feeder -> m r } | ||||
|  | ||||
| instance Monad m => Monad (UnpackerT m) where | ||||
|   a >>= b = | ||||
|     UnpackerT $ \up feed -> do | ||||
|       r <- runUnpackerT a up feed | ||||
|       runUnpackerT (b r) up feed | ||||
|    | ||||
|   return r = | ||||
|     UnpackerT $ \_ _ -> return r | ||||
|  | ||||
| instance MonadTrans UnpackerT where | ||||
|   lift m = UnpackerT $ \_ _ -> m | ||||
|  | ||||
| instance MonadIO m => MonadIO (UnpackerT m) where | ||||
|   liftIO = lift . liftIO | ||||
|  | ||||
| instance MonadIO m => MonadUnpacker (UnpackerT m) where | ||||
|   get = UnpackerT $ \up feed -> liftIO $ do | ||||
|     executeOne up feed | ||||
|     obj <- unpackerData up | ||||
|     freeZone =<< unpackerReleaseZone up | ||||
|     unpackerReset up | ||||
|     let Right r = fromObject obj | ||||
|     return r | ||||
|      | ||||
|     where | ||||
|       executeOne up feed = do | ||||
|         resp <- unpackerExecute up | ||||
|         guard $ resp>=0 | ||||
|         when (resp==0) $ do | ||||
|           Just bs <- feed | ||||
|           unpackerFeed up bs | ||||
|           executeOne up feed | ||||
|  | ||||
| -- | Execute deserializer using given feeder. | ||||
| unpackFrom :: MonadIO m => Feeder -> UnpackerT m r -> m r | ||||
| unpackFrom f m = do | ||||
|   up <- liftIO $ newUnpacker defaultInitialBufferSize | ||||
|   runUnpackerT m up f | ||||
|  | ||||
| -- | Execute deserializer using given handle. | ||||
| unpackFromHandle :: MonadIO m => Handle -> UnpackerT m r -> m r | ||||
| unpackFromHandle h m = | ||||
|   flip unpackFrom m =<< liftIO (feederFromHandle h) | ||||
|  | ||||
| -- | Execute deserializer using given file content. | ||||
| unpackFromFile :: MonadIO m => FilePath -> UnpackerT m r -> m r | ||||
| unpackFromFile p m = do | ||||
|   h <- liftIO $ openFile p ReadMode | ||||
|   r <- flip unpackFrom m =<< liftIO (feederFromHandle h) | ||||
|   liftIO $ hClose h | ||||
|   return r | ||||
|  | ||||
| -- | Execute deserializer from given byte sequence. | ||||
| unpackFromString :: MonadIO m => ByteString -> UnpackerT m r -> m r | ||||
| unpackFromString bs m = do | ||||
|   flip unpackFrom m =<< liftIO (feederFromString bs) | ||||
							
								
								
									
										301
									
								
								haskell/src/Data/MessagePack/Object.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										301
									
								
								haskell/src/Data/MessagePack/Object.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,301 @@ | ||||
| {-# Language TypeSynonymInstances #-} | ||||
| {-# Language FlexibleInstances #-} | ||||
| {-# Language OverlappingInstances #-} | ||||
| {-# Language IncoherentInstances #-} | ||||
| {-# Language DeriveDataTypeable #-} | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Object | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009-2010 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- MessagePack object definition | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Object( | ||||
|   -- * MessagePack Object | ||||
|   Object(..), | ||||
|    | ||||
|   -- * Serialization to and from Object | ||||
|   OBJECT(..), | ||||
|   -- Result, | ||||
|   ) where | ||||
|  | ||||
| import Control.DeepSeq | ||||
| import Control.Exception | ||||
| import Control.Monad | ||||
| import Control.Monad.Trans.Error () | ||||
| import qualified Data.Attoparsec as A | ||||
| import qualified Data.ByteString as B | ||||
| import qualified Data.ByteString.Char8 as C8 | ||||
| import Data.Typeable | ||||
|  | ||||
| import Data.MessagePack.Pack | ||||
| import Data.MessagePack.Unpack | ||||
|  | ||||
| -- | Object Representation of MessagePack data. | ||||
| data Object = | ||||
|   ObjectNil | ||||
|   | ObjectBool Bool | ||||
|   | ObjectInteger Int | ||||
|   | ObjectDouble Double | ||||
|   | ObjectRAW B.ByteString | ||||
|   | ObjectArray [Object] | ||||
|   | ObjectMap [(Object, Object)] | ||||
|   deriving (Show, Eq, Ord, Typeable) | ||||
|  | ||||
| instance NFData Object where | ||||
|   rnf obj = | ||||
|     case obj of | ||||
|       ObjectNil -> () | ||||
|       ObjectBool b -> rnf b | ||||
|       ObjectInteger n -> rnf n | ||||
|       ObjectDouble d -> rnf d | ||||
|       ObjectRAW bs -> bs `seq` () | ||||
|       ObjectArray a -> rnf a | ||||
|       ObjectMap m -> rnf m | ||||
|  | ||||
| instance Unpackable Object where | ||||
|   get = | ||||
|     A.choice | ||||
|     [ liftM ObjectInteger get | ||||
|     , liftM (\() -> ObjectNil) get | ||||
|     , liftM ObjectBool get | ||||
|     , liftM ObjectDouble get | ||||
|     , liftM ObjectRAW get | ||||
|     , liftM ObjectArray get | ||||
|     , liftM ObjectMap get | ||||
|     ] | ||||
|  | ||||
| instance Packable Object where | ||||
|   put obj = | ||||
|     case obj of | ||||
|       ObjectInteger n -> | ||||
|         put n | ||||
|       ObjectNil -> | ||||
|         put () | ||||
|       ObjectBool b -> | ||||
|         put b | ||||
|       ObjectDouble d -> | ||||
|         put d | ||||
|       ObjectRAW raw -> | ||||
|         put raw | ||||
|       ObjectArray arr -> | ||||
|         put arr | ||||
|       ObjectMap m -> | ||||
|         put m | ||||
|  | ||||
| -- | The class of types serializable to and from MessagePack object | ||||
| class (Unpackable a, Packable a) => OBJECT a where | ||||
|   -- | Encode a value to MessagePack object | ||||
|   toObject :: a -> Object | ||||
|   toObject = unpack . pack | ||||
|    | ||||
|   -- | Decode a value from MessagePack object | ||||
|   fromObject :: Object -> a | ||||
|   fromObject a = | ||||
|     case tryFromObject a of | ||||
|       Left err -> | ||||
|         throw $ UnpackError err | ||||
|       Right ret -> | ||||
|         ret | ||||
|  | ||||
|   -- | Decode a value from MessagePack object | ||||
|   tryFromObject :: Object -> Either String a | ||||
|   tryFromObject = tryUnpack . pack | ||||
|  | ||||
| instance OBJECT Object where | ||||
|   toObject = id | ||||
|   tryFromObject = Right | ||||
|  | ||||
| tryFromObjectError :: Either String a | ||||
| tryFromObjectError = Left "tryFromObject: cannot cast" | ||||
|  | ||||
| instance OBJECT () where | ||||
|   toObject = const ObjectNil | ||||
|   tryFromObject ObjectNil = Right () | ||||
|   tryFromObject _ = tryFromObjectError | ||||
|  | ||||
| instance OBJECT Int where | ||||
|   toObject = ObjectInteger | ||||
|   tryFromObject (ObjectInteger n) = Right n | ||||
|   tryFromObject _ = tryFromObjectError | ||||
|  | ||||
| instance OBJECT Bool where | ||||
|   toObject = ObjectBool | ||||
|   tryFromObject (ObjectBool b) = Right b | ||||
|   tryFromObject _ = tryFromObjectError | ||||
|  | ||||
| instance OBJECT Double where | ||||
|   toObject = ObjectDouble | ||||
|   tryFromObject (ObjectDouble d) = Right d | ||||
|   tryFromObject _ = tryFromObjectError | ||||
|  | ||||
| instance OBJECT B.ByteString where | ||||
|   toObject = ObjectRAW | ||||
|   tryFromObject (ObjectRAW bs) = Right bs | ||||
|   tryFromObject _ = tryFromObjectError | ||||
|  | ||||
| instance OBJECT String where | ||||
|   toObject = toObject . C8.pack | ||||
|   tryFromObject obj = liftM C8.unpack $ tryFromObject obj | ||||
|  | ||||
| instance OBJECT a => OBJECT [a] where | ||||
|   toObject = ObjectArray . map toObject | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     mapM tryFromObject arr | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2) => OBJECT (a1, a2) where | ||||
|   toObject (a1, a2) = ObjectArray [toObject a1, toObject a2] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         return (v1, v2) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3) => OBJECT (a1, a2, a3) where | ||||
|   toObject (a1, a2, a3) = ObjectArray [toObject a1, toObject a2, toObject a3] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         return (v1, v2, v3) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4) => OBJECT (a1, a2, a3, a4) where | ||||
|   toObject (a1, a2, a3, a4) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3, o4] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         v4 <- tryFromObject o4 | ||||
|         return (v1, v2, v3, v4) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5) => OBJECT (a1, a2, a3, a4, a5) where | ||||
|   toObject (a1, a2, a3, a4, a5) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3, o4, o5] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         v4 <- tryFromObject o4 | ||||
|         v5 <- tryFromObject o5 | ||||
|         return (v1, v2, v3, v4, v5) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6) => OBJECT (a1, a2, a3, a4, a5, a6) where | ||||
|   toObject (a1, a2, a3, a4, a5, a6) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3, o4, o5, o6] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         v4 <- tryFromObject o4 | ||||
|         v5 <- tryFromObject o5 | ||||
|         v6 <- tryFromObject o6 | ||||
|         return (v1, v2, v3, v4, v5, v6) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6, OBJECT a7) => OBJECT (a1, a2, a3, a4, a5, a6, a7) where | ||||
|   toObject (a1, a2, a3, a4, a5, a6, a7) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3, o4, o5, o6, o7] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         v4 <- tryFromObject o4 | ||||
|         v5 <- tryFromObject o5 | ||||
|         v6 <- tryFromObject o6 | ||||
|         v7 <- tryFromObject o7 | ||||
|         return (v1, v2, v3, v4, v5, v6, v7) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6, OBJECT a7, OBJECT a8) => OBJECT (a1, a2, a3, a4, a5, a6, a7, a8) where | ||||
|   toObject (a1, a2, a3, a4, a5, a6, a7, a8) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7, toObject a8] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3, o4, o5, o6, o7, o8] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         v4 <- tryFromObject o4 | ||||
|         v5 <- tryFromObject o5 | ||||
|         v6 <- tryFromObject o6 | ||||
|         v7 <- tryFromObject o7 | ||||
|         v8 <- tryFromObject o8 | ||||
|         return (v1, v2, v3, v4, v5, v6, v7, v8) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6, OBJECT a7, OBJECT a8, OBJECT a9) => OBJECT (a1, a2, a3, a4, a5, a6, a7, a8, a9) where | ||||
|   toObject (a1, a2, a3, a4, a5, a6, a7, a8, a9) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7, toObject a8, toObject a9] | ||||
|   tryFromObject (ObjectArray arr) = | ||||
|     case arr of | ||||
|       [o1, o2, o3, o4, o5, o6, o7, o8, o9] -> do | ||||
|         v1 <- tryFromObject o1 | ||||
|         v2 <- tryFromObject o2 | ||||
|         v3 <- tryFromObject o3 | ||||
|         v4 <- tryFromObject o4 | ||||
|         v5 <- tryFromObject o5 | ||||
|         v6 <- tryFromObject o6 | ||||
|         v7 <- tryFromObject o7 | ||||
|         v8 <- tryFromObject o8 | ||||
|         v9 <- tryFromObject o9 | ||||
|         return (v1, v2, v3, v4, v5, v6, v7, v8, v9) | ||||
|       _ -> | ||||
|         tryFromObjectError | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance (OBJECT a, OBJECT b) => OBJECT [(a, b)] where | ||||
|   toObject = | ||||
|     ObjectMap . map (\(a, b) -> (toObject a, toObject b)) | ||||
|   tryFromObject (ObjectMap mem) = do | ||||
|     mapM (\(a, b) -> liftM2 (,) (tryFromObject a) (tryFromObject b)) mem | ||||
|   tryFromObject _ = | ||||
|     tryFromObjectError | ||||
|  | ||||
| instance OBJECT a => OBJECT (Maybe a) where | ||||
|   toObject (Just a) = toObject a | ||||
|   toObject Nothing = ObjectNil | ||||
|    | ||||
|   tryFromObject ObjectNil = return Nothing | ||||
|   tryFromObject obj = liftM Just $ tryFromObject obj | ||||
							
								
								
									
										186
									
								
								haskell/src/Data/MessagePack/Pack.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										186
									
								
								haskell/src/Data/MessagePack/Pack.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,186 @@ | ||||
| {-# Language FlexibleInstances #-} | ||||
| {-# Language IncoherentInstances #-} | ||||
| {-# Language OverlappingInstances #-} | ||||
| {-# Language TypeSynonymInstances #-} | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Pack | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009-2010 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- MessagePack Serializer using @Data.Binary.Pack@ | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Pack ( | ||||
|   -- * Serializable class | ||||
|   Packable(..), | ||||
|   -- * Simple function to pack a Haskell value | ||||
|   pack, | ||||
|   ) where | ||||
|  | ||||
| import Data.Binary.Put | ||||
| import Data.Binary.IEEE754 | ||||
| import Data.Bits | ||||
| import qualified Data.ByteString as B | ||||
| import qualified Data.ByteString.Char8 as B8 | ||||
| import qualified Data.ByteString.Lazy as L | ||||
| import qualified Data.Vector as V | ||||
|  | ||||
| -- | Serializable class | ||||
| class Packable a where | ||||
|   -- | Serialize a value | ||||
|   put :: a -> Put | ||||
|  | ||||
| -- | Pack Haskell data to MessagePack string. | ||||
| pack :: Packable a => a -> L.ByteString | ||||
| pack = runPut . put | ||||
|  | ||||
| instance Packable Int where | ||||
|   put n = | ||||
|     case n of | ||||
|       _ | n >= 0 && n <= 127 -> | ||||
|         putWord8 $ fromIntegral n | ||||
|       _ | n >= -32 && n <= -1 -> | ||||
|         putWord8 $ fromIntegral n | ||||
|       _ | n >= 0 && n < 0x100 -> do | ||||
|         putWord8 0xCC | ||||
|         putWord8 $ fromIntegral n | ||||
|       _ | n >= 0 && n < 0x10000 -> do | ||||
|         putWord8 0xCD | ||||
|         putWord16be $ fromIntegral n | ||||
|       _ | n >= 0 && n < 0x100000000 -> do | ||||
|         putWord8 0xCE | ||||
|         putWord32be $ fromIntegral n | ||||
|       _ | n >= 0 -> do | ||||
|         putWord8 0xCF | ||||
|         putWord64be $ fromIntegral n | ||||
|       _ | n >= -0x80 -> do | ||||
|         putWord8 0xD0 | ||||
|         putWord8 $ fromIntegral n | ||||
|       _ | n >= -0x8000 -> do | ||||
|         putWord8 0xD1 | ||||
|         putWord16be $ fromIntegral n | ||||
|       _ | n >= -0x80000000 -> do | ||||
|         putWord8 0xD2 | ||||
|         putWord32be $ fromIntegral n | ||||
|       _ -> do | ||||
|         putWord8 0xD3 | ||||
|         putWord64be $ fromIntegral n | ||||
|  | ||||
| instance Packable () where | ||||
|   put _ =  | ||||
|     putWord8 0xC0 | ||||
|  | ||||
| instance Packable Bool where | ||||
|   put True = putWord8 0xC3 | ||||
|   put False = putWord8 0xC2 | ||||
|  | ||||
| instance Packable Double where | ||||
|   put d = do | ||||
|     putWord8 0xCB | ||||
|     putFloat64be d | ||||
|  | ||||
| instance Packable String where | ||||
|   put = putString length (putByteString . B8.pack) | ||||
|  | ||||
| instance Packable B.ByteString where | ||||
|   put = putString B.length putByteString | ||||
|  | ||||
| instance Packable L.ByteString where | ||||
|   put = putString (fromIntegral . L.length) putLazyByteString | ||||
|  | ||||
| putString :: (s -> Int) -> (s -> Put) -> s -> Put | ||||
| putString lf pf str = do | ||||
|   case lf str of | ||||
|     len | len <= 31 -> do | ||||
|       putWord8 $ 0xA0 .|. fromIntegral len | ||||
|     len | len < 0x10000 -> do | ||||
|       putWord8 0xDA | ||||
|       putWord16be $ fromIntegral len | ||||
|     len -> do | ||||
|       putWord8 0xDB | ||||
|       putWord32be $ fromIntegral len | ||||
|   pf str | ||||
|  | ||||
| instance Packable a => Packable [a] where | ||||
|   put = putArray length (mapM_ put) | ||||
|  | ||||
| instance Packable a => Packable (V.Vector a) where | ||||
|   put = putArray V.length (V.mapM_ put) | ||||
|  | ||||
| instance (Packable a1, Packable a2) => Packable (a1, a2) where | ||||
|   put = putArray (const 2) f where | ||||
|     f (a1, a2) = put a1 >> put a2 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3) => Packable (a1, a2, a3) where | ||||
|   put = putArray (const 3) f where | ||||
|     f (a1, a2, a3) = put a1 >> put a2 >> put a3 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3, Packable a4) => Packable (a1, a2, a3, a4) where | ||||
|   put = putArray (const 4) f where | ||||
|     f (a1, a2, a3, a4) = put a1 >> put a2 >> put a3 >> put a4 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5) => Packable (a1, a2, a3, a4, a5) where | ||||
|   put = putArray (const 5) f where | ||||
|     f (a1, a2, a3, a4, a5) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6) => Packable (a1, a2, a3, a4, a5, a6) where | ||||
|   put = putArray (const 6) f where | ||||
|     f (a1, a2, a3, a4, a5, a6) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6, Packable a7) => Packable (a1, a2, a3, a4, a5, a6, a7) where | ||||
|   put = putArray (const 7) f where | ||||
|     f (a1, a2, a3, a4, a5, a6, a7) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 >> put a7 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6, Packable a7, Packable a8) => Packable (a1, a2, a3, a4, a5, a6, a7, a8) where | ||||
|   put = putArray (const 8) f where | ||||
|     f (a1, a2, a3, a4, a5, a6, a7, a8) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 >> put a7 >> put a8 | ||||
|  | ||||
| instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6, Packable a7, Packable a8, Packable a9) => Packable (a1, a2, a3, a4, a5, a6, a7, a8, a9) where | ||||
|   put = putArray (const 9) f where | ||||
|     f (a1, a2, a3, a4, a5, a6, a7, a8, a9) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 >> put a7 >> put a8 >> put a9 | ||||
|  | ||||
| putArray :: (a -> Int) -> (a -> Put) -> a -> Put | ||||
| putArray lf pf arr = do | ||||
|   case lf arr of | ||||
|     len | len <= 15 -> | ||||
|       putWord8 $ 0x90 .|. fromIntegral len | ||||
|     len | len < 0x10000 -> do | ||||
|       putWord8 0xDC | ||||
|       putWord16be $ fromIntegral len | ||||
|     len -> do | ||||
|       putWord8 0xDD | ||||
|       putWord32be $ fromIntegral len | ||||
|   pf arr | ||||
|  | ||||
| instance (Packable k, Packable v) => Packable [(k, v)] where | ||||
|   put = putMap length (mapM_ putPair) | ||||
|  | ||||
| instance (Packable k, Packable v) => Packable (V.Vector (k, v)) where | ||||
|   put = putMap V.length (V.mapM_ putPair) | ||||
|  | ||||
| putPair :: (Packable a, Packable b) => (a, b) -> Put | ||||
| putPair (a, b) = put a >> put b | ||||
|  | ||||
| putMap :: (a -> Int) -> (a -> Put) -> a -> Put | ||||
| putMap lf pf m = do | ||||
|   case lf m of | ||||
|     len | len <= 15 -> | ||||
|       putWord8 $ 0x80 .|. fromIntegral len | ||||
|     len | len < 0x10000 -> do | ||||
|       putWord8 0xDE | ||||
|       putWord16be $ fromIntegral len | ||||
|     len -> do | ||||
|       putWord8 0xDF | ||||
|       putWord32be $ fromIntegral len | ||||
|   pf m | ||||
|  | ||||
| instance Packable a => Packable (Maybe a) where | ||||
|   put Nothing = put () | ||||
|   put (Just a) = put a | ||||
| @@ -1,82 +0,0 @@ | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Stream | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- Lazy Stream Serializers and Deserializers | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Stream( | ||||
|   unpackObjects, | ||||
|   unpackObjectsFromFile, | ||||
|   unpackObjectsFromHandle, | ||||
|   unpackObjectsFromString, | ||||
|   ) where | ||||
|  | ||||
| import Data.ByteString (ByteString) | ||||
| import System.IO | ||||
| import System.IO.Unsafe | ||||
|  | ||||
| import Data.MessagePack.Base | ||||
| import Data.MessagePack.Feed | ||||
|  | ||||
| -- | Unpack objects using given feeder. | ||||
| unpackObjects :: Feeder -> IO [Object] | ||||
| unpackObjects feeder = do | ||||
|   up <- newUnpacker defaultInitialBufferSize | ||||
|   f up | ||||
|   where | ||||
|   f up = unsafeInterleaveIO $ do | ||||
|     mbo <- unpackOnce up | ||||
|     case mbo of | ||||
|       Just o -> do | ||||
|         os <- f up | ||||
|         return $ o:os | ||||
|       Nothing -> | ||||
|         return [] | ||||
|  | ||||
|   unpackOnce up = do | ||||
|     resp <- unpackerExecute up | ||||
|     case resp of | ||||
|       0 -> do | ||||
|         r <- feedOnce up | ||||
|         if r | ||||
|           then unpackOnce up | ||||
|           else return Nothing | ||||
|       1 -> do | ||||
|         obj <- unpackerData up | ||||
|         freeZone =<< unpackerReleaseZone up | ||||
|         unpackerReset up | ||||
|         return $ Just obj | ||||
|       _ -> | ||||
|         error $ "unpackerExecute fails: " ++ show resp | ||||
|  | ||||
|   feedOnce up = do | ||||
|     dat <- feeder | ||||
|     case dat of | ||||
|       Nothing -> | ||||
|         return False | ||||
|       Just bs -> do | ||||
|         unpackerFeed up bs | ||||
|         return True | ||||
|  | ||||
| -- | Unpack objects from file. | ||||
| unpackObjectsFromFile :: FilePath -> IO [Object] | ||||
| unpackObjectsFromFile fname = | ||||
|   unpackObjects =<< feederFromFile fname | ||||
|  | ||||
| -- | Unpack objects from handle. | ||||
| unpackObjectsFromHandle :: Handle -> IO [Object] | ||||
| unpackObjectsFromHandle h = | ||||
|   unpackObjects =<< feederFromHandle h | ||||
|    | ||||
| -- | Unpack oobjects from given byte sequence. | ||||
| unpackObjectsFromString :: ByteString -> IO [Object] | ||||
| unpackObjectsFromString bs = | ||||
|   unpackObjects =<< feederFromString bs | ||||
							
								
								
									
										308
									
								
								haskell/src/Data/MessagePack/Unpack.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										308
									
								
								haskell/src/Data/MessagePack/Unpack.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,308 @@ | ||||
| {-# Language FlexibleInstances #-} | ||||
| {-# Language IncoherentInstances #-} | ||||
| {-# Language OverlappingInstances #-} | ||||
| {-# Language TypeSynonymInstances #-} | ||||
| {-# Language DeriveDataTypeable #-} | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
| -- | | ||||
| -- Module    : Data.MessagePack.Unpack | ||||
| -- Copyright : (c) Hideyuki Tanaka, 2009-2010 | ||||
| -- License   : BSD3 | ||||
| -- | ||||
| -- Maintainer:  tanaka.hideyuki@gmail.com | ||||
| -- Stability :  experimental | ||||
| -- Portability: portable | ||||
| -- | ||||
| -- MessagePack Deserializer using @Data.Attoparsec@ | ||||
| -- | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| module Data.MessagePack.Unpack( | ||||
|   -- * MessagePack deserializer | ||||
|   Unpackable(..), | ||||
|   -- * Simple function to unpack a Haskell value | ||||
|   unpack, | ||||
|   tryUnpack, | ||||
|   -- * Unpack exception | ||||
|   UnpackError(..), | ||||
|   -- * ByteString utils | ||||
|   IsByteString(..), | ||||
|   ) where | ||||
|  | ||||
| import Control.Exception | ||||
| import Control.Monad | ||||
| import qualified Data.Attoparsec as A | ||||
| import Data.Binary.Get | ||||
| import Data.Binary.IEEE754 | ||||
| import Data.Bits | ||||
| import qualified Data.ByteString as B | ||||
| import qualified Data.ByteString.Char8 as B8 | ||||
| import qualified Data.ByteString.Lazy as L | ||||
| import Data.Int | ||||
| import Data.Typeable | ||||
| import qualified Data.Vector as V | ||||
| import Data.Word | ||||
| import Text.Printf | ||||
|  | ||||
| -- | Deserializable class | ||||
| class Unpackable a where | ||||
|   -- | Deserialize a value | ||||
|   get :: A.Parser a | ||||
|  | ||||
| class IsByteString s where | ||||
|   toBS :: s -> B.ByteString | ||||
|  | ||||
| instance IsByteString B.ByteString where | ||||
|   toBS = id | ||||
|  | ||||
| instance IsByteString L.ByteString where | ||||
|   toBS = B.concat . L.toChunks | ||||
|  | ||||
| -- | The exception of unpack | ||||
| data UnpackError = | ||||
|   UnpackError String | ||||
|   deriving (Show, Typeable) | ||||
|  | ||||
| instance Exception UnpackError | ||||
|  | ||||
| -- | Unpack MessagePack string to Haskell data. | ||||
| unpack :: (Unpackable a, IsByteString s) => s -> a | ||||
| unpack bs = | ||||
|   case tryUnpack bs of | ||||
|     Left err -> | ||||
|       throw $ UnpackError err | ||||
|     Right ret -> | ||||
|       ret | ||||
|  | ||||
| -- | Unpack MessagePack string to Haskell data. | ||||
| tryUnpack :: (Unpackable a, IsByteString s) => s -> Either String a | ||||
| tryUnpack bs = | ||||
|   case A.parse get (toBS bs) of | ||||
|     A.Fail _ _ err -> | ||||
|       Left err | ||||
|     A.Partial _ -> | ||||
|       Left "not enough input" | ||||
|     A.Done _ ret -> | ||||
|       Right ret | ||||
|  | ||||
| instance Unpackable Int where | ||||
|   get = do | ||||
|     c <- A.anyWord8 | ||||
|     case c of | ||||
|       _ | c .&. 0x80 == 0x00 -> | ||||
|         return $ fromIntegral c | ||||
|       _ | c .&. 0xE0 == 0xE0 -> | ||||
|         return $ fromIntegral (fromIntegral c :: Int8) | ||||
|       0xCC -> | ||||
|         return . fromIntegral =<< A.anyWord8 | ||||
|       0xCD -> | ||||
|         return . fromIntegral =<< parseUint16 | ||||
|       0xCE -> | ||||
|         return . fromIntegral =<< parseUint32 | ||||
|       0xCF -> | ||||
|         return . fromIntegral =<< parseUint64 | ||||
|       0xD0 -> | ||||
|         return . fromIntegral =<< parseInt8 | ||||
|       0xD1 -> | ||||
|         return . fromIntegral =<< parseInt16 | ||||
|       0xD2 -> | ||||
|         return . fromIntegral =<< parseInt32 | ||||
|       0xD3 -> | ||||
|         return . fromIntegral =<< parseInt64 | ||||
|       _ -> | ||||
|         fail $ printf "invlid integer tag: 0x%02X" c | ||||
|  | ||||
| instance Unpackable () where | ||||
|   get = do | ||||
|     c <- A.anyWord8 | ||||
|     case c of | ||||
|       0xC0 -> | ||||
|         return () | ||||
|       _ -> | ||||
|         fail $ printf "invlid nil tag: 0x%02X" c | ||||
|  | ||||
| instance Unpackable Bool where | ||||
|   get = do | ||||
|     c <- A.anyWord8 | ||||
|     case c of | ||||
|       0xC3 -> | ||||
|         return True | ||||
|       0xC2 -> | ||||
|         return False | ||||
|       _ -> | ||||
|         fail $ printf "invlid bool tag: 0x%02X" c | ||||
|  | ||||
| instance Unpackable Double where | ||||
|   get = do | ||||
|     c <- A.anyWord8 | ||||
|     case c of | ||||
|       0xCA -> | ||||
|         return . realToFrac . runGet getFloat32be . toLBS =<< A.take 4 | ||||
|       0xCB -> | ||||
|         return . runGet getFloat64be . toLBS =<< A.take 8 | ||||
|       _ -> | ||||
|         fail $ printf "invlid double tag: 0x%02X" c | ||||
|  | ||||
| instance Unpackable String where | ||||
|   get = parseString (\n -> return . B8.unpack =<< A.take n) | ||||
|  | ||||
| instance Unpackable B.ByteString where | ||||
|   get = parseString A.take | ||||
|  | ||||
| instance Unpackable L.ByteString where | ||||
|   get = parseString (\n -> do bs <- A.take n; return $ L.fromChunks [bs]) | ||||
|  | ||||
| parseString :: (Int -> A.Parser a) -> A.Parser a | ||||
| parseString aget = do | ||||
|   c <- A.anyWord8 | ||||
|   case c of | ||||
|     _ | c .&. 0xE0 == 0xA0 -> | ||||
|       aget . fromIntegral $ c .&. 0x1F | ||||
|     0xDA -> | ||||
|       aget . fromIntegral =<< parseUint16 | ||||
|     0xDB -> | ||||
|       aget . fromIntegral =<< parseUint32 | ||||
|     _ -> | ||||
|       fail $ printf "invlid raw tag: 0x%02X" c | ||||
|  | ||||
| instance Unpackable a => Unpackable [a] where | ||||
|   get = parseArray (flip replicateM get) | ||||
|  | ||||
| instance Unpackable a => Unpackable (V.Vector a) where | ||||
|   get = parseArray (flip V.replicateM get) | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2) => Unpackable (a1, a2) where | ||||
|   get = parseArray f where | ||||
|     f 2 = get >>= \a1 -> get >>= \a2 -> return (a1, a2) | ||||
|     f n = fail $ printf "wrong tupple size: expected 2 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3) => Unpackable (a1, a2, a3) where | ||||
|   get = parseArray f where | ||||
|     f 3 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> return (a1, a2, a3) | ||||
|     f n = fail $ printf "wrong tupple size: expected 3 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4) => Unpackable (a1, a2, a3, a4) where | ||||
|   get = parseArray f where | ||||
|     f 4 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> return (a1, a2, a3, a4) | ||||
|     f n = fail $ printf "wrong tupple size: expected 4 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5) => Unpackable (a1, a2, a3, a4, a5) where | ||||
|   get = parseArray f where | ||||
|     f 5 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> return (a1, a2, a3, a4, a5) | ||||
|     f n = fail $ printf "wrong tupple size: expected 5 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6) => Unpackable (a1, a2, a3, a4, a5, a6) where | ||||
|   get = parseArray f where | ||||
|     f 6 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> return (a1, a2, a3, a4, a5, a6) | ||||
|     f n = fail $ printf "wrong tupple size: expected 6 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6, Unpackable a7) => Unpackable (a1, a2, a3, a4, a5, a6, a7) where | ||||
|   get = parseArray f where | ||||
|     f 7 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> get >>= \a7 -> return (a1, a2, a3, a4, a5, a6, a7) | ||||
|     f n = fail $ printf "wrong tupple size: expected 7 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6, Unpackable a7, Unpackable a8) => Unpackable (a1, a2, a3, a4, a5, a6, a7, a8) where | ||||
|   get = parseArray f where | ||||
|     f 8 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> get >>= \a7 -> get >>= \a8 -> return (a1, a2, a3, a4, a5, a6, a7, a8) | ||||
|     f n = fail $ printf "wrong tupple size: expected 8 but got " n | ||||
|  | ||||
| instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6, Unpackable a7, Unpackable a8, Unpackable a9) => Unpackable (a1, a2, a3, a4, a5, a6, a7, a8, a9) where | ||||
|   get = parseArray f where | ||||
|     f 9 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> get >>= \a7 -> get >>= \a8 -> get >>= \a9 -> return (a1, a2, a3, a4, a5, a6, a7, a8, a9) | ||||
|     f n = fail $ printf "wrong tupple size: expected 9 but got " n | ||||
|  | ||||
| parseArray :: (Int -> A.Parser a) -> A.Parser a | ||||
| parseArray aget = do | ||||
|   c <- A.anyWord8 | ||||
|   case c of | ||||
|     _ | c .&. 0xF0 == 0x90 -> | ||||
|       aget . fromIntegral $ c .&. 0x0F | ||||
|     0xDC -> | ||||
|       aget . fromIntegral =<< parseUint16 | ||||
|     0xDD -> | ||||
|       aget . fromIntegral =<< parseUint32 | ||||
|     _ -> | ||||
|       fail $ printf "invlid array tag: 0x%02X" c | ||||
|  | ||||
| instance (Unpackable k, Unpackable v) => Unpackable [(k, v)] where | ||||
|   get = parseMap (flip replicateM parsePair) | ||||
|  | ||||
| instance (Unpackable k, Unpackable v) => Unpackable (V.Vector (k, v)) where | ||||
|   get = parseMap (flip V.replicateM parsePair) | ||||
|  | ||||
| parsePair :: (Unpackable k, Unpackable v) => A.Parser (k, v) | ||||
| parsePair = do | ||||
|   a <- get | ||||
|   b <- get | ||||
|   return (a, b) | ||||
|  | ||||
| parseMap :: (Int -> A.Parser a) -> A.Parser a | ||||
| parseMap aget = do | ||||
|   c <- A.anyWord8 | ||||
|   case c of | ||||
|     _ | c .&. 0xF0 == 0x80 -> | ||||
|       aget . fromIntegral $ c .&. 0x0F | ||||
|     0xDE -> | ||||
|       aget . fromIntegral =<< parseUint16 | ||||
|     0xDF -> | ||||
|       aget . fromIntegral =<< parseUint32 | ||||
|     _ -> | ||||
|       fail $ printf "invlid map tag: 0x%02X" c | ||||
|  | ||||
| instance Unpackable a => Unpackable (Maybe a) where | ||||
|   get =  | ||||
|     A.choice | ||||
|     [ liftM Just get | ||||
|     , liftM (\() -> Nothing) get ] | ||||
|  | ||||
| parseUint16 :: A.Parser Word16 | ||||
| parseUint16 = do | ||||
|   b0 <- A.anyWord8 | ||||
|   b1 <- A.anyWord8 | ||||
|   return $ (fromIntegral b0 `shiftL` 8) .|. fromIntegral b1 | ||||
|  | ||||
| parseUint32 :: A.Parser Word32 | ||||
| parseUint32 = do | ||||
|   b0 <- A.anyWord8 | ||||
|   b1 <- A.anyWord8 | ||||
|   b2 <- A.anyWord8 | ||||
|   b3 <- A.anyWord8 | ||||
|   return $ (fromIntegral b0 `shiftL` 24) .|. | ||||
|            (fromIntegral b1 `shiftL` 16) .|. | ||||
|            (fromIntegral b2 `shiftL` 8) .|. | ||||
|            fromIntegral b3 | ||||
|  | ||||
| parseUint64 :: A.Parser Word64 | ||||
| parseUint64 = do | ||||
|   b0 <- A.anyWord8 | ||||
|   b1 <- A.anyWord8 | ||||
|   b2 <- A.anyWord8 | ||||
|   b3 <- A.anyWord8 | ||||
|   b4 <- A.anyWord8 | ||||
|   b5 <- A.anyWord8 | ||||
|   b6 <- A.anyWord8 | ||||
|   b7 <- A.anyWord8 | ||||
|   return $ (fromIntegral b0 `shiftL` 56) .|. | ||||
|            (fromIntegral b1 `shiftL` 48) .|. | ||||
|            (fromIntegral b2 `shiftL` 40) .|. | ||||
|            (fromIntegral b3 `shiftL` 32) .|. | ||||
|            (fromIntegral b4 `shiftL` 24) .|. | ||||
|            (fromIntegral b5 `shiftL` 16) .|. | ||||
|            (fromIntegral b6 `shiftL` 8) .|. | ||||
|            fromIntegral b7 | ||||
|  | ||||
| parseInt8 :: A.Parser Int8 | ||||
| parseInt8 = return . fromIntegral =<< A.anyWord8 | ||||
|  | ||||
| parseInt16 :: A.Parser Int16 | ||||
| parseInt16 = return . fromIntegral =<< parseUint16 | ||||
|  | ||||
| parseInt32 :: A.Parser Int32 | ||||
| parseInt32 = return . fromIntegral =<< parseUint32 | ||||
|  | ||||
| parseInt64 :: A.Parser Int64 | ||||
| parseInt64 = return . fromIntegral =<< parseUint64 | ||||
|  | ||||
| toLBS :: B.ByteString -> L.ByteString | ||||
| toLBS bs = L.fromChunks [bs] | ||||
| @@ -1,16 +1,21 @@ | ||||
| import Control.Monad.Trans | ||||
| {-# Language OverloadedStrings #-} | ||||
|  | ||||
| import Control.Monad.IO.Class | ||||
| import qualified Data.ByteString as B | ||||
| import Data.MessagePack | ||||
|  | ||||
| main = do | ||||
|   sb <- packToString $ do | ||||
|   sb <- return $ packToString $ do | ||||
|     put [1,2,3::Int] | ||||
|     put (3.14 :: Double) | ||||
|     put "Hoge" | ||||
|     put ("Hoge" :: B.ByteString) | ||||
|    | ||||
|   print sb | ||||
|    | ||||
|   unpackFromString sb $ do | ||||
|   r <- unpackFromString sb $ do | ||||
|     arr <- get | ||||
|     dbl <- get | ||||
|     str <- get | ||||
|     liftIO $ print (arr :: [Int], dbl :: Double, str :: String) | ||||
|     return (arr :: [Int], dbl :: Double, str :: B.ByteString) | ||||
|    | ||||
|   print r | ||||
|   | ||||
| @@ -1,14 +0,0 @@ | ||||
| import Control.Applicative | ||||
| import qualified Data.ByteString as BS | ||||
| import Data.MessagePack | ||||
|  | ||||
| main = do | ||||
|   sb <- newSimpleBuffer | ||||
|   pc <- newPacker sb | ||||
|   pack pc [1,2,3::Int] | ||||
|   pack pc True | ||||
|   pack pc "hoge" | ||||
|   bs <- simpleBufferData sb | ||||
|    | ||||
|   os <- unpackObjectsFromString bs | ||||
|   mapM_ print os | ||||
| @@ -1,36 +1,64 @@ | ||||
| import Test.Framework | ||||
| import Test.Framework.Providers.QuickCheck2 | ||||
| import Test.QuickCheck | ||||
|  | ||||
| import Control.Monad | ||||
| import qualified Data.ByteString.Char8 as B | ||||
| import qualified Data.ByteString.Lazy.Char8 as L | ||||
| import Data.MessagePack | ||||
|  | ||||
| {- | ||||
| main = do | ||||
|   sb <- newSimpleBuffer | ||||
|   pc <- newPacker sb | ||||
|    | ||||
|   pack pc [(1,2),(2,3),(3::Int,4::Int)] | ||||
|   pack pc [4,5,6::Int] | ||||
|   pack pc "hoge" | ||||
|    | ||||
|   bs <- simpleBufferData sb | ||||
|   print bs | ||||
|    | ||||
|   up <- newUnpacker defaultInitialBufferSize | ||||
|    | ||||
|   unpackerFeed up bs | ||||
| mid :: (Packable a, Unpackable a) => a -> a | ||||
| mid = unpack . pack | ||||
|  | ||||
|   let f = do | ||||
|         res <- unpackerExecute up | ||||
|         when (res==1) $ do | ||||
|           obj <- unpackerData up | ||||
|           print obj | ||||
|           f | ||||
|    | ||||
|   f | ||||
| prop_mid_int a = a == mid a | ||||
|   where types = a :: Int | ||||
| prop_mid_nil a = a == mid a | ||||
|   where types = a :: () | ||||
| prop_mid_bool a = a == mid a | ||||
|   where types = a :: Bool | ||||
| prop_mid_double a = a == mid a | ||||
|   where types = a :: Double | ||||
| prop_mid_string a = a == mid a | ||||
|   where types = a :: String | ||||
| prop_mid_bytestring a = B.pack a == mid (B.pack a) | ||||
|   where types = a :: String | ||||
| prop_mid_lazy_bytestring a = (L.pack a) == mid (L.pack a) | ||||
|   where types = a :: String | ||||
| prop_mid_array_int a = a == mid a | ||||
|   where types = a :: [Int] | ||||
| prop_mid_array_string a = a == mid a | ||||
|   where types = a :: [String] | ||||
| prop_mid_pair2 a = a == mid a | ||||
|   where types = a :: (Int, Int) | ||||
| prop_mid_pair3 a = a == mid a | ||||
|   where types = a :: (Int, Int, Int) | ||||
| prop_mid_pair4 a = a == mid a | ||||
|   where types = a :: (Int, Int, Int, Int) | ||||
| prop_mid_pair5 a = a == mid a | ||||
|   where types = a :: (Int, Int, Int, Int, Int) | ||||
| prop_mid_map_int_double a = a == mid a | ||||
|   where types = a :: [(Int, Double)] | ||||
| prop_mid_map_string_string a = a == mid a | ||||
|   where types = a :: [(String, String)] | ||||
|  | ||||
|   return () | ||||
| -} | ||||
| tests = | ||||
|   [ testGroup "simple" | ||||
|     [ testProperty "int" prop_mid_int | ||||
|     , testProperty "nil" prop_mid_nil | ||||
|     , testProperty "bool" prop_mid_bool | ||||
|     , testProperty "double" prop_mid_double | ||||
|     , testProperty "string" prop_mid_string | ||||
|     , testProperty "bytestring" prop_mid_bytestring | ||||
|     , testProperty "lazy-bytestring" prop_mid_lazy_bytestring | ||||
|     , testProperty "[int]" prop_mid_array_int | ||||
|     , testProperty "[string]" prop_mid_array_string | ||||
|     , testProperty "(int, int)" prop_mid_pair2 | ||||
|     , testProperty "(int, int, int)" prop_mid_pair3 | ||||
|     , testProperty "(int, int, int, int)" prop_mid_pair4 | ||||
|     , testProperty "(int, int, int, int, int)" prop_mid_pair5 | ||||
|     , testProperty "[(int, double)]" prop_mid_map_int_double | ||||
|     , testProperty "[(string, string)]" prop_mid_map_string_string | ||||
|     ] | ||||
|   ] | ||||
|  | ||||
| main = do | ||||
|   bs <- packb [(1,2),(2,3),(3::Int,4::Int)] | ||||
|   print bs | ||||
|   dat <- unpackb bs | ||||
|   print (dat :: Result [(Int, Int)]) | ||||
| main = defaultMain tests | ||||
|   | ||||
							
								
								
									
										43
									
								
								haskell/test/UserData.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								haskell/test/UserData.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| {-# Language TemplateHaskell #-} | ||||
|  | ||||
| import Data.MessagePack | ||||
| import Data.MessagePack.Derive | ||||
|  | ||||
| data T | ||||
|   = A Int String | ||||
|   | B Double | ||||
|   deriving (Show, Eq) | ||||
|  | ||||
| $(deriveObject ''T) | ||||
|  | ||||
| data U | ||||
|   = C { c1 :: Int, c2 :: String } | ||||
|   | D { d1 :: Double } | ||||
|   deriving (Show, Eq) | ||||
|  | ||||
| $(deriveObject ''U) | ||||
|  | ||||
| data V | ||||
|   = E String | F | ||||
|   deriving (Show, Eq) | ||||
|  | ||||
| $(deriveObject ''V) | ||||
|  | ||||
| test :: (OBJECT a, Show a, Eq a) => a -> IO () | ||||
| test v = do | ||||
|   let bs = pack v | ||||
|   print bs | ||||
|   print (unpack bs == v) | ||||
|  | ||||
|   let oa = toObject v | ||||
|   print oa | ||||
|   print (fromObject oa == v) | ||||
|  | ||||
| main = do | ||||
|   test $ A 123 "hoge" | ||||
|   test $ B 3.14 | ||||
|   test $ C 123 "hoge" | ||||
|   test $ D 3.14 | ||||
|   test $ E "hello" | ||||
|   test $ F | ||||
|   return () | ||||
							
								
								
									
										32
									
								
								java/pom.xml
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								java/pom.xml
									
									
									
									
									
								
							| @@ -7,7 +7,7 @@ | ||||
|   <description>MessagePack for Java</description> | ||||
|  | ||||
|   <name>MessagePack for Java</name> | ||||
|   <url>http://msgpack.sourceforge.net/</url> | ||||
|   <url>http://msgpack.org/</url> | ||||
|  | ||||
|   <licenses> | ||||
|     <license> | ||||
| @@ -29,6 +29,12 @@ | ||||
|       <version>4.8.1</version> | ||||
|       <scope>test</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>javassist</groupId> | ||||
|       <artifactId>javassist</artifactId> | ||||
|       <version>3.12.1.GA</version> | ||||
|       <scope>compile</scope> | ||||
|     </dependency> | ||||
|   </dependencies> | ||||
|  | ||||
|   <build> | ||||
| @@ -97,29 +103,31 @@ | ||||
|  | ||||
|   <repositories> | ||||
|     <repository> | ||||
|       <id>msgpack.sourceforge.net</id> | ||||
|       <id>msgpack.org</id> | ||||
|       <name>MessagePack Maven2 Repository</name> | ||||
|       <url>http://msgpack.sourceforge.net/maven2</url> | ||||
|       <url>http://msgpack.org/maven2</url> | ||||
|     </repository> | ||||
|     <repository> | ||||
|       <id>msgpack.sourceforge.net</id> | ||||
|       <name>MessagePack Maven2 Snapshot Repository</name> | ||||
|       <url>http://msgpack.sourceforge.net/maven2-snapshot</url> | ||||
|       <id>repository.jboss.org</id> | ||||
|       <url>https://repository.jboss.org/nexus/content/groups/public/</url> | ||||
|       <snapshots> | ||||
|         <enabled>false</enabled> | ||||
|       </snapshots> | ||||
|     </repository> | ||||
|   </repositories> | ||||
|  | ||||
|   <distributionManagement> | ||||
|     <repository> | ||||
|       <uniqueVersion>false</uniqueVersion> | ||||
|       <id>shell.sourceforge.net</id> | ||||
|       <name>Repository at sourceforge.net</name> | ||||
|       <url>scp://shell.sourceforge.net/home/groups/m/ms/msgpack/htdocs/maven2/</url> | ||||
|       <id>msgpack.org</id> | ||||
|       <name>Repository at msgpack.org</name> | ||||
| 	  <url>file://${project.build.directory}/website/maven2/</url> | ||||
|     </repository> | ||||
|     <snapshotRepository> | ||||
|        <uniqueVersion>true</uniqueVersion> | ||||
|        <id>shell.sourceforge.net</id> | ||||
|        <name>Repository Name</name> | ||||
|        <url>scp://shell.sourceforge.net/home/groups/m/ms/msgpack/htdocs/maven2-snapshot/</url> | ||||
|        <id>msgpack.org</id> | ||||
|        <name>Repository at msgpack.org</name> | ||||
| 	   <url>file://${project.build.directory}/website/maven2/</url> | ||||
|     </snapshotRepository> | ||||
|   </distributionManagement> | ||||
|  | ||||
|   | ||||
| @@ -19,7 +19,7 @@ package org.msgpack; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.nio.ByteBuffer; | ||||
| //import java.math.BigInteger; | ||||
| import java.math.BigInteger; | ||||
|  | ||||
| abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 	int offset = 0; | ||||
| @@ -47,7 +47,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 			offset = noffset; | ||||
| 		} while(!super.isFinished()); | ||||
|  | ||||
| 		Object obj = super.getData(); | ||||
| 		MessagePackObject obj = super.getData(); | ||||
| 		super.reset(); | ||||
| 		result.done(obj); | ||||
|  | ||||
| @@ -103,7 +103,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 		case 0xcc:  // unsigned int  8 | ||||
| 			more(2); | ||||
| 			advance(2); | ||||
| 			return (int)((short)buffer[offset+1] & 0xff); | ||||
| 			return (int)((short)(buffer[offset-1]) & 0xff); | ||||
| 		case 0xcd:  // unsigned int 16 | ||||
| 			more(3); | ||||
| 			castBuffer.rewind(); | ||||
| @@ -137,7 +137,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 		case 0xd0:  // signed int  8 | ||||
| 			more(2); | ||||
| 			advance(2); | ||||
| 			return (int)buffer[offset+1]; | ||||
| 			return (int)buffer[offset-1]; | ||||
| 		case 0xd1:  // signed int 16 | ||||
| 			more(3); | ||||
| 			castBuffer.rewind(); | ||||
| @@ -178,7 +178,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 		case 0xcc:  // unsigned int  8 | ||||
| 			more(2); | ||||
| 			advance(2); | ||||
| 			return (long)((short)buffer[offset+1] & 0xff); | ||||
| 			return (long)((short)(buffer[offset-1]) & 0xff); | ||||
| 		case 0xcd:  // unsigned int 16 | ||||
| 			more(3); | ||||
| 			castBuffer.rewind(); | ||||
| @@ -198,8 +198,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 			{ | ||||
| 				long o = castBuffer.getLong(0); | ||||
| 				if(o < 0) { | ||||
| 					// FIXME | ||||
| 					throw new MessageTypeException("uint 64 bigger than 0x7fffffff is not supported"); | ||||
| 					throw new MessageTypeException(); | ||||
| 				} | ||||
| 				advance(9); | ||||
| 				return o; | ||||
| @@ -207,7 +206,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 		case 0xd0:  // signed int  8 | ||||
| 			more(2); | ||||
| 			advance(2); | ||||
| 			return (long)buffer[offset+1]; | ||||
| 			return (long)buffer[offset-1]; | ||||
| 		case 0xd1:  // signed int 16 | ||||
| 			more(3); | ||||
| 			castBuffer.rewind(); | ||||
| @@ -231,6 +230,26 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	final BigInteger unpackBigInteger() throws IOException, MessageTypeException { | ||||
| 		more(1); | ||||
| 		int b = buffer[offset]; | ||||
| 		if((b & 0xff) != 0xcf) { | ||||
| 			return BigInteger.valueOf(unpackLong()); | ||||
| 		} | ||||
|  | ||||
| 		// unsigned int 64 | ||||
| 		more(9); | ||||
| 		castBuffer.rewind(); | ||||
| 		castBuffer.put(buffer, offset+1, 8); | ||||
| 		advance(9); | ||||
| 		long o = castBuffer.getLong(0); | ||||
| 		if(o < 0) { | ||||
| 			return new BigInteger(1, castBuffer.array()); | ||||
| 		} else { | ||||
| 			return BigInteger.valueOf(o); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	final float unpackFloat() throws IOException, MessageTypeException { | ||||
| 		more(1); | ||||
| 		int b = buffer[offset]; | ||||
| @@ -414,7 +433,7 @@ abstract class BufferedUnpackerImpl extends UnpackerImpl { | ||||
| 		return s; | ||||
| 	} | ||||
|  | ||||
| 	final Object unpackObject() throws IOException { | ||||
| 	final MessagePackObject unpackObject() throws IOException { | ||||
| 		UnpackResult result = new UnpackResult(); | ||||
| 		if(!next(result)) { | ||||
| 			super.reset(); | ||||
|   | ||||
| @@ -18,6 +18,6 @@ | ||||
| package org.msgpack; | ||||
|  | ||||
| public interface MessageConvertable { | ||||
| 	public void messageConvert(Object obj) throws MessageTypeException; | ||||
| 	public void messageConvert(MessagePackObject obj) throws MessageTypeException; | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										136
									
								
								java/src/main/java/org/msgpack/MessagePackObject.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								java/src/main/java/org/msgpack/MessagePackObject.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,136 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Set; | ||||
| import java.util.Map; | ||||
| import java.math.BigInteger; | ||||
|  | ||||
| public abstract class MessagePackObject implements Cloneable, MessagePackable { | ||||
| 	public boolean isNil() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean isBooleanType() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean isIntegerType() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean isFloatType() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean isArrayType() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean isMapType() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean isRawType() { | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public boolean asBoolean() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public byte asByte() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public short asShort() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public int asInt() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public long asLong() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public BigInteger asBigInteger() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public float asFloat() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public double asDouble() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public byte[] asByteArray() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public String asString() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public MessagePackObject[] asArray() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public List<MessagePackObject> asList() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Map<MessagePackObject, MessagePackObject> asMap() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public byte byteValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public short shortValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public int intValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public long longValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public BigInteger bigIntegerValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public float floatValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public double doubleValue() { | ||||
| 		throw new MessageTypeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	abstract public Object clone(); | ||||
| } | ||||
|  | ||||
| @@ -23,15 +23,5 @@ public class MessageTypeException extends RuntimeException { | ||||
| 	public MessageTypeException(String s) { | ||||
| 		super(s); | ||||
| 	} | ||||
|  | ||||
| 	public static MessageTypeException invalidConvert(Object from, Schema to) { | ||||
| 		return new MessageTypeException(from.getClass().getName()+" cannot be convert to "+to.getExpression()); | ||||
| 	} | ||||
|  | ||||
| 	/* FIXME | ||||
| 	public static MessageTypeException schemaMismatch(Schema to) { | ||||
| 		return new MessageTypeException("schema mismatch "+to.getExpression()); | ||||
| 	} | ||||
| 	*/ | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -22,6 +22,7 @@ import java.io.IOException; | ||||
| import java.nio.ByteBuffer; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.math.BigInteger; | ||||
|  | ||||
| /** | ||||
|  * Packer enables you to serialize objects into OutputStream. | ||||
| @@ -194,6 +195,27 @@ public class Packer { | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	public Packer packBigInteger(BigInteger d) throws IOException { | ||||
| 		if(d.bitLength() <= 63) { | ||||
| 			return packLong(d.longValue()); | ||||
| 		} else if(d.bitLength() <= 64 && d.signum() >= 0) { | ||||
| 			castBytes[0] = (byte)0xcf; | ||||
| 			byte[] barray = d.toByteArray(); | ||||
| 			castBytes[1] = barray[barray.length-8]; | ||||
| 			castBytes[2] = barray[barray.length-7]; | ||||
| 			castBytes[3] = barray[barray.length-6]; | ||||
| 			castBytes[4] = barray[barray.length-5]; | ||||
| 			castBytes[5] = barray[barray.length-4]; | ||||
| 			castBytes[6] = barray[barray.length-3]; | ||||
| 			castBytes[7] = barray[barray.length-2]; | ||||
| 			castBytes[8] = barray[barray.length-1]; | ||||
| 			out.write(castBytes); | ||||
| 			return this; | ||||
| 		} else { | ||||
| 			throw new MessageTypeException("can't pack BigInteger larger than 0xffffffffffffffff"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public Packer packFloat(float d) throws IOException { | ||||
| 		castBytes[0] = (byte)0xca; | ||||
| 		castBuffer.putFloat(1, d); | ||||
| @@ -223,6 +245,10 @@ public class Packer { | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	public Packer packBoolean(boolean d) throws IOException { | ||||
| 		return d ? packTrue() : packFalse(); | ||||
| 	} | ||||
|  | ||||
| 	public Packer packArray(int n) throws IOException { | ||||
| 		if(n < 16) { | ||||
| 			final int d = 0x90 | n; | ||||
| @@ -282,12 +308,6 @@ public class Packer { | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	public Packer packWithSchema(Object o, Schema s) throws IOException { | ||||
| 		s.pack(this, o); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	public Packer packString(String s) throws IOException { | ||||
| 		byte[] b = ((String)s).getBytes("UTF-8"); | ||||
| 		packRaw(b.length); | ||||
| @@ -295,39 +315,36 @@ public class Packer { | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	public Packer pack(String o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		return packString(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(MessagePackable o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		o.messagePack(this); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(byte[] o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		packRaw(o.length); | ||||
| 		return packRawBody(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(List o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		packArray(o.size()); | ||||
| 		for(Object i : o) { pack(i); } | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public Packer pack(Map o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		packMap(o.size()); | ||||
| 		for(Map.Entry e : ((Map<Object,Object>)o).entrySet()) { | ||||
| 			pack(e.getKey()); | ||||
| 			pack(e.getValue()); | ||||
| 	public Packer pack(boolean o) throws IOException { | ||||
| 		if(o) { | ||||
| 			return packTrue(); | ||||
| 		} else { | ||||
| 			return packFalse(); | ||||
| 		} | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(byte o) throws IOException { | ||||
| 		return packByte(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(short o) throws IOException { | ||||
| 		return packShort(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(int o) throws IOException { | ||||
| 		return packInt(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(long o) throws IOException { | ||||
| 		return packLong(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(float o) throws IOException { | ||||
| 		return packFloat(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(double o) throws IOException { | ||||
| 		return packDouble(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(Boolean o) throws IOException { | ||||
| @@ -359,6 +376,11 @@ public class Packer { | ||||
| 		return packLong(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(BigInteger o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		return packBigInteger(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(Float o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		return packFloat(o); | ||||
| @@ -369,8 +391,41 @@ public class Packer { | ||||
| 		return packDouble(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(String o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		return packString(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(MessagePackable o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		o.messagePack(this); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(byte[] o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		packRaw(o.length); | ||||
| 		return packRawBody(o); | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(List o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		packArray(o.size()); | ||||
| 		for(Object i : o) { pack(i); } | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	public Packer pack(Map o) throws IOException { | ||||
| 		if(o == null) { return packNil(); } | ||||
| 		packMap(o.size()); | ||||
| 		for(Map.Entry<Object,Object> e : ((Map<Object,Object>)o).entrySet()) { | ||||
| 			pack(e.getKey()); | ||||
| 			pack(e.getValue()); | ||||
| 		} | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public Packer pack(Object o) throws IOException { | ||||
| 		if(o == null) { | ||||
| 			return packNil(); | ||||
| @@ -393,7 +448,7 @@ public class Packer { | ||||
| 		} else if(o instanceof Map) { | ||||
| 			Map<Object,Object> m = (Map<Object,Object>)o; | ||||
| 			packMap(m.size()); | ||||
| 			for(Map.Entry e : m.entrySet()) { | ||||
| 			for(Map.Entry<Object,Object> e : m.entrySet()) { | ||||
| 				pack(e.getKey()); | ||||
| 				pack(e.getValue()); | ||||
| 			} | ||||
| @@ -416,6 +471,8 @@ public class Packer { | ||||
| 			return packFloat((Float)o); | ||||
| 		} else if(o instanceof Double) { | ||||
| 			return packDouble((Double)o); | ||||
| 		} else if(o instanceof BigInteger) { | ||||
| 			return packBigInteger((BigInteger)o); | ||||
| 		} else { | ||||
| 			throw new MessageTypeException("unknown object "+o+" ("+o.getClass()+")"); | ||||
| 		} | ||||
|   | ||||
| @@ -1,133 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack; | ||||
|  | ||||
| import java.io.Writer; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.schema.SSchemaParser; | ||||
| import org.msgpack.schema.ClassGenerator; | ||||
|  | ||||
| public abstract class Schema { | ||||
| 	private String expression; | ||||
| 	private String name; | ||||
|  | ||||
| 	public Schema(String name) { | ||||
| 		this.expression = expression; | ||||
| 		this.name = name; | ||||
| 	} | ||||
|  | ||||
| 	public String getName() { | ||||
| 		return name; | ||||
| 	} | ||||
|  | ||||
| 	public String getFullName() { | ||||
| 		return name; | ||||
| 	} | ||||
|  | ||||
| 	public String getExpression() { | ||||
| 		return name; | ||||
| 	} | ||||
|  | ||||
| 	public static Schema parse(String source) { | ||||
| 		return SSchemaParser.parse(source); | ||||
| 	} | ||||
|  | ||||
| 	public static Schema load(String source) { | ||||
| 		return SSchemaParser.load(source); | ||||
| 	} | ||||
|  | ||||
| 	public void write(Writer output) throws IOException { | ||||
| 		ClassGenerator.write(this, output); | ||||
| 	} | ||||
|  | ||||
| 	public abstract void pack(Packer pk, Object obj) throws IOException; | ||||
|  | ||||
| 	public abstract Object convert(Object obj) throws MessageTypeException; | ||||
|  | ||||
|  | ||||
| 	public Object createFromNil() { | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromBoolean(boolean v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromShort(short v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromInt(int v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromLong(long v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromRaw(byte[] b, int offset, int length) { | ||||
| 		throw new RuntimeException("type error"); | ||||
| 	} | ||||
|  | ||||
| 	/* FIXME | ||||
| 	public Object createFromBoolean(boolean v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromShort(short v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromInt(int v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromLong(long v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromRaw(byte[] b, int offset, int length) { | ||||
| 		throw MessageTypeException.schemaMismatch(this); | ||||
| 	} | ||||
| 	*/ | ||||
| } | ||||
|  | ||||
| @@ -21,7 +21,7 @@ import java.io.IOException; | ||||
| import java.util.Iterator; | ||||
| import java.util.NoSuchElementException; | ||||
|  | ||||
| public class UnpackIterator extends UnpackResult implements Iterator<Object> { | ||||
| public class UnpackIterator extends UnpackResult implements Iterator<MessagePackObject> { | ||||
| 	private Unpacker pac; | ||||
|  | ||||
| 	UnpackIterator(Unpacker pac) { | ||||
| @@ -38,7 +38,7 @@ public class UnpackIterator extends UnpackResult implements Iterator<Object> { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public Object next() { | ||||
| 	public MessagePackObject next() { | ||||
| 		if(!finished && !hasNext()) { | ||||
| 			throw new NoSuchElementException(); | ||||
| 		} | ||||
|   | ||||
| @@ -19,13 +19,13 @@ package org.msgpack; | ||||
|  | ||||
| public class UnpackResult { | ||||
| 	protected boolean finished = false; | ||||
| 	protected Object data = null; | ||||
| 	protected MessagePackObject data = null; | ||||
|  | ||||
| 	public boolean isFinished() { | ||||
| 		return finished; | ||||
| 	} | ||||
|  | ||||
| 	public Object getData() { | ||||
| 	public MessagePackObject getData() { | ||||
| 		return data; | ||||
| 	} | ||||
|  | ||||
| @@ -34,7 +34,7 @@ public class UnpackResult { | ||||
| 		data = null; | ||||
| 	} | ||||
|  | ||||
| 	void done(Object obj) { | ||||
| 	void done(MessagePackObject obj) { | ||||
| 		finished = true; | ||||
| 		data = obj; | ||||
| 	} | ||||
|   | ||||
| @@ -22,12 +22,13 @@ import java.io.InputStream; | ||||
| import java.io.IOException; | ||||
| import java.util.Iterator; | ||||
| import java.nio.ByteBuffer; | ||||
| import java.math.BigInteger; | ||||
|  | ||||
| /** | ||||
|  * Unpacker enables you to deserialize objects from stream. | ||||
|  * | ||||
|  * Unpacker provides Buffered API, Unbuffered API, Schema API | ||||
|  * and Direct Conversion API. | ||||
|  * Unpacker provides Buffered API, Unbuffered API and | ||||
|  * Direct Conversion API. | ||||
|  * | ||||
|  * Buffered API uses the internal buffer of the Unpacker. | ||||
|  * Following code uses Buffered API with an InputStream: | ||||
| @@ -39,7 +40,7 @@ import java.nio.ByteBuffer; | ||||
|  * UnpackResult result = pac.next(); | ||||
|  * | ||||
|  * // use an iterator. | ||||
|  * for(Object obj : pac) { | ||||
|  * for(MessagePackObject obj : pac) { | ||||
|  *   // use MessageConvertable interface to convert the | ||||
|  *   // the generic object to the specific type. | ||||
|  * } | ||||
| @@ -56,7 +57,7 @@ import java.nio.ByteBuffer; | ||||
|  * pac.feed(input_bytes); | ||||
|  * | ||||
|  * // use next() method or iterators. | ||||
|  * for(Object obj : pac) { | ||||
|  * for(MessagePackObject obj : pac) { | ||||
|  *   // ... | ||||
|  * } | ||||
|  * </pre> | ||||
| @@ -75,7 +76,7 @@ import java.nio.ByteBuffer; | ||||
|  * System.in.read(pac.getBuffer(), pac.getBufferOffset(), pac.getBufferCapacity()); | ||||
|  * | ||||
|  * // use next() method or iterators. | ||||
|  * for(Object obj : pac) { | ||||
|  * for(MessagePackObject obj : pac) { | ||||
|  *     // ... | ||||
|  * } | ||||
|  * </pre> | ||||
| @@ -96,12 +97,12 @@ import java.nio.ByteBuffer; | ||||
|  * | ||||
|  * // take out object if deserialized object is ready. | ||||
|  * if(pac.isFinished()) { | ||||
|  *     Object obj = pac.getData(); | ||||
|  *     MessagePackObject obj = pac.getData(); | ||||
|  *     // ... | ||||
|  * } | ||||
|  * </pre> | ||||
|  */ | ||||
| public class Unpacker implements Iterable<Object> { | ||||
| public class Unpacker implements Iterable<MessagePackObject> { | ||||
|  | ||||
| 	// buffer: | ||||
| 	// +---------------------------------------------+ | ||||
| @@ -170,16 +171,6 @@ public class Unpacker implements Iterable<Object> { | ||||
| 		this.stream = stream; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Sets schema to convert deserialized object into specific type. | ||||
| 	 * Default schema is {@link GenericSchema} that leaves objects for generic type. Use {@link MessageConvertable#messageConvert(Object)} method to convert the generic object. | ||||
| 	 * @param s schem to use | ||||
| 	 */ | ||||
| 	public Unpacker useSchema(Schema s) { | ||||
| 		impl.setSchema(s); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	/** | ||||
| 	 * Gets the input stream. | ||||
| @@ -255,7 +246,7 @@ public class Unpacker implements Iterable<Object> { | ||||
| 	/** | ||||
| 	 * Returns the iterator that calls {@link next()} method repeatedly. | ||||
| 	 */ | ||||
| 	public Iterator<Object> iterator() { | ||||
| 	public Iterator<MessagePackObject> iterator() { | ||||
| 		return new UnpackIterator(this); | ||||
| 	} | ||||
|  | ||||
| @@ -387,7 +378,7 @@ public class Unpacker implements Iterable<Object> { | ||||
| 	/** | ||||
| 	 * Gets the object deserialized by {@link execute(byte[])} method. | ||||
| 	 */ | ||||
| 	public Object getData() { | ||||
| 	public MessagePackObject getData() { | ||||
| 		return impl.getData(); | ||||
| 	} | ||||
|  | ||||
| @@ -462,6 +453,15 @@ public class Unpacker implements Iterable<Object> { | ||||
| 		return impl.unpackLong(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Gets one {@code BigInteger} value from the buffer. | ||||
| 	 * This method calls {@link fill()} method if needed. | ||||
| 	 * @throws MessageTypeException the first value of the buffer is not a {@code BigInteger}. | ||||
| 	 */ | ||||
| 	public BigInteger unpackBigInteger() throws IOException, MessageTypeException { | ||||
| 		return impl.unpackBigInteger(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Gets one {@code float} value from the buffer. | ||||
| 	 * This method calls {@link fill()} method if needed. | ||||
| @@ -557,15 +557,15 @@ public class Unpacker implements Iterable<Object> { | ||||
| 	 * Gets one {@code Object} value from the buffer. | ||||
| 	 * This method calls {@link fill()} method if needed. | ||||
| 	 */ | ||||
| 	final public Object unpackObject() throws IOException { | ||||
| 	final public MessagePackObject unpackObject() throws IOException { | ||||
| 		return impl.unpackObject(); | ||||
| 	} | ||||
|  | ||||
| 	final void unpack(MessageUnpackable obj) throws IOException, MessageTypeException { | ||||
| 	final public void unpack(MessageUnpackable obj) throws IOException, MessageTypeException { | ||||
| 		obj.messageUnpack(this); | ||||
| 	} | ||||
|  | ||||
| 	final boolean tryUnpackNull() throws IOException { | ||||
| 	final public boolean tryUnpackNull() throws IOException { | ||||
| 		return impl.tryUnpackNull(); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -18,11 +18,8 @@ | ||||
| package org.msgpack; | ||||
|  | ||||
| import java.nio.ByteBuffer; | ||||
| //import java.math.BigInteger; | ||||
| import org.msgpack.*; | ||||
| import org.msgpack.schema.GenericSchema; | ||||
| import org.msgpack.schema.IMapSchema; | ||||
| import org.msgpack.schema.IArraySchema; | ||||
| import java.math.BigInteger; | ||||
| import org.msgpack.object.*; | ||||
|  | ||||
| public class UnpackerImpl { | ||||
| 	static final int CS_HEADER      = 0x00; | ||||
| @@ -55,30 +52,19 @@ public class UnpackerImpl { | ||||
| 	private int[]    stack_ct       = new int[MAX_STACK_SIZE]; | ||||
| 	private int[]    stack_count    = new int[MAX_STACK_SIZE]; | ||||
| 	private Object[] stack_obj      = new Object[MAX_STACK_SIZE]; | ||||
| 	private Schema[] stack_schema   = new Schema[MAX_STACK_SIZE]; | ||||
| 	private int top_ct; | ||||
| 	private int top_count; | ||||
| 	private Object top_obj; | ||||
| 	private Schema top_schema; | ||||
| 	private ByteBuffer castBuffer   = ByteBuffer.allocate(8); | ||||
| 	private boolean finished = false; | ||||
| 	private Object data = null; | ||||
|  | ||||
| 	private static final Schema GENERIC_SCHEMA = new GenericSchema(); | ||||
| 	private Schema rootSchema; | ||||
| 	private MessagePackObject data = null; | ||||
|  | ||||
| 	public UnpackerImpl() | ||||
| 	{ | ||||
| 		setSchema(GENERIC_SCHEMA); | ||||
| 	} | ||||
|  | ||||
| 	public void setSchema(Schema schema) | ||||
| 	{ | ||||
| 		this.rootSchema = schema; | ||||
| 		reset(); | ||||
| 	} | ||||
|  | ||||
| 	public final Object getData() | ||||
| 	public final MessagePackObject getData() | ||||
| 	{ | ||||
| 		return data; | ||||
| 	} | ||||
| @@ -94,7 +80,6 @@ public class UnpackerImpl { | ||||
| 		top_ct = 0; | ||||
| 		top_count = 0; | ||||
| 		top_obj = null; | ||||
| 		top_schema = rootSchema; | ||||
| 	} | ||||
|  | ||||
| 	public final void reset() | ||||
| @@ -127,20 +112,20 @@ public class UnpackerImpl { | ||||
| 	 | ||||
| 					if((b & 0x80) == 0) {  // Positive Fixnum | ||||
| 						//System.out.println("positive fixnum "+b); | ||||
| 						obj = top_schema.createFromByte((byte)b); | ||||
| 						obj = IntegerType.create((byte)b); | ||||
| 						break _push; | ||||
| 					} | ||||
| 	 | ||||
| 					if((b & 0xe0) == 0xe0) {  // Negative Fixnum | ||||
| 						//System.out.println("negative fixnum "+b); | ||||
| 						obj = top_schema.createFromByte((byte)b); | ||||
| 						obj = IntegerType.create((byte)b); | ||||
| 						break _push; | ||||
| 					} | ||||
| 	 | ||||
| 					if((b & 0xe0) == 0xa0) {  // FixRaw | ||||
| 						trail = b & 0x1f; | ||||
| 						if(trail == 0) { | ||||
| 							obj = top_schema.createFromRaw(new byte[0], 0, 0); | ||||
| 							obj = RawType.create(new byte[0]); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						cs = ACS_RAW_VALUE; | ||||
| @@ -151,25 +136,20 @@ public class UnpackerImpl { | ||||
| 						if(top >= MAX_STACK_SIZE) { | ||||
| 							throw new UnpackException("parse error"); | ||||
| 						} | ||||
| 						if(!(top_schema instanceof IArraySchema)) { | ||||
| 							throw new RuntimeException("type error"); | ||||
| 						} | ||||
| 						count = b & 0x0f; | ||||
| 						//System.out.println("fixarray count:"+count); | ||||
| 						obj = new Object[count]; | ||||
| 						obj = new MessagePackObject[count]; | ||||
| 						if(count == 0) { | ||||
| 							obj = ((IArraySchema)top_schema).createFromArray((Object[])obj); | ||||
| 							obj = ArrayType.create((MessagePackObject[])obj); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						++top; | ||||
| 						stack_obj[top]    = top_obj; | ||||
| 						stack_ct[top]     = top_ct; | ||||
| 						stack_count[top]  = top_count; | ||||
| 						stack_schema[top] = top_schema; | ||||
| 						top_obj    = obj; | ||||
| 						top_ct     = CT_ARRAY_ITEM; | ||||
| 						top_count  = count; | ||||
| 						top_schema = ((IArraySchema)top_schema).getElementSchema(0); | ||||
| 						break _header_again; | ||||
| 					} | ||||
| 	 | ||||
| @@ -177,13 +157,10 @@ public class UnpackerImpl { | ||||
| 						if(top >= MAX_STACK_SIZE) { | ||||
| 							throw new UnpackException("parse error"); | ||||
| 						} | ||||
| 						if(!(top_schema instanceof IMapSchema)) { | ||||
| 							throw new RuntimeException("type error"); | ||||
| 						} | ||||
| 						count = b & 0x0f; | ||||
| 						obj = new Object[count*2]; | ||||
| 						obj = new MessagePackObject[count*2]; | ||||
| 						if(count == 0) { | ||||
| 							obj = ((IMapSchema)top_schema).createFromMap((Object[])obj); | ||||
| 							obj = MapType.create((MessagePackObject[])obj); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						//System.out.println("fixmap count:"+count); | ||||
| @@ -191,23 +168,21 @@ public class UnpackerImpl { | ||||
| 						stack_obj[top]    = top_obj; | ||||
| 						stack_ct[top]     = top_ct; | ||||
| 						stack_count[top]  = top_count; | ||||
| 						stack_schema[top] = top_schema; | ||||
| 						top_obj    = obj; | ||||
| 						top_ct     = CT_MAP_KEY; | ||||
| 						top_count  = count; | ||||
| 						top_schema = ((IMapSchema)top_schema).getKeySchema(); | ||||
| 						break _header_again; | ||||
| 					} | ||||
| 	 | ||||
| 					switch(b & 0xff) {    // FIXME | ||||
| 					case 0xc0:  // nil | ||||
| 						obj = top_schema.createFromNil(); | ||||
| 						obj = NilType.create(); | ||||
| 						break _push; | ||||
| 					case 0xc2:  // false | ||||
| 						obj = top_schema.createFromBoolean(false); | ||||
| 						obj = BooleanType.create(false); | ||||
| 						break _push; | ||||
| 					case 0xc3:  // true | ||||
| 						obj = top_schema.createFromBoolean(true); | ||||
| 						obj = BooleanType.create(true); | ||||
| 						break _push; | ||||
| 					case 0xca:  // float | ||||
| 					case 0xcb:  // double | ||||
| @@ -251,13 +226,13 @@ public class UnpackerImpl { | ||||
| 					case CS_FLOAT: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 4); | ||||
| 						obj = top_schema.createFromFloat( castBuffer.getFloat(0) ); | ||||
| 						obj = FloatType.create( castBuffer.getFloat(0) ); | ||||
| 						//System.out.println("float "+obj); | ||||
| 						break _push; | ||||
| 					case CS_DOUBLE: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 8); | ||||
| 						obj = top_schema.createFromDouble( castBuffer.getDouble(0) ); | ||||
| 						obj = FloatType.create( castBuffer.getDouble(0) ); | ||||
| 						//System.out.println("double "+obj); | ||||
| 						break _push; | ||||
| 					case CS_UINT_8: | ||||
| @@ -265,7 +240,7 @@ public class UnpackerImpl { | ||||
| 						//System.out.println(src[n]); | ||||
| 						//System.out.println(src[n+1]); | ||||
| 						//System.out.println(src[n-1]); | ||||
| 						obj = top_schema.createFromShort( (short)((src[n]) & 0xff) ); | ||||
| 						obj = IntegerType.create( (short)((src[n]) & 0xff) ); | ||||
| 						//System.out.println("uint8 "+obj); | ||||
| 						break _push; | ||||
| 					case CS_UINT_16: | ||||
| @@ -273,13 +248,13 @@ public class UnpackerImpl { | ||||
| 						//System.out.println(src[n+1]); | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 2); | ||||
| 						obj = top_schema.createFromInt( ((int)castBuffer.getShort(0)) & 0xffff ); | ||||
| 						obj = IntegerType.create( ((int)castBuffer.getShort(0)) & 0xffff ); | ||||
| 						//System.out.println("uint 16 "+obj); | ||||
| 						break _push; | ||||
| 					case CS_UINT_32: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 4); | ||||
| 						obj = top_schema.createFromLong( ((long)castBuffer.getInt(0)) & 0xffffffffL ); | ||||
| 						obj = IntegerType.create( ((long)castBuffer.getInt(0)) & 0xffffffffL ); | ||||
| 						//System.out.println("uint 32 "+obj); | ||||
| 						break _push; | ||||
| 					case CS_UINT_64: | ||||
| @@ -288,38 +263,36 @@ public class UnpackerImpl { | ||||
| 						{ | ||||
| 							long o = castBuffer.getLong(0); | ||||
| 							if(o < 0) { | ||||
| 								// FIXME | ||||
| 								//obj = GenericBigInteger.valueOf(o & 0x7fffffffL).setBit(31); | ||||
| 								throw new UnpackException("uint 64 bigger than 0x7fffffff is not supported"); | ||||
| 								obj = IntegerType.create(new BigInteger(1, castBuffer.array())); | ||||
| 							} else { | ||||
| 								obj = top_schema.createFromLong( o ); | ||||
| 								obj = IntegerType.create(o); | ||||
| 							} | ||||
| 						} | ||||
| 						break _push; | ||||
| 					case CS_INT_8: | ||||
| 						obj = top_schema.createFromByte(  src[n] ); | ||||
| 						obj = IntegerType.create( src[n] ); | ||||
| 						break _push; | ||||
| 					case CS_INT_16: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 2); | ||||
| 						obj = top_schema.createFromShort( castBuffer.getShort(0) ); | ||||
| 						obj = IntegerType.create( castBuffer.getShort(0) ); | ||||
| 						break _push; | ||||
| 					case CS_INT_32: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 4); | ||||
| 						obj = top_schema.createFromInt( castBuffer.getInt(0) ); | ||||
| 						obj = IntegerType.create( castBuffer.getInt(0) ); | ||||
| 						break _push; | ||||
| 					case CS_INT_64: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 8); | ||||
| 						obj = top_schema.createFromLong( castBuffer.getLong(0) ); | ||||
| 						obj = IntegerType.create( castBuffer.getLong(0) ); | ||||
| 						break _push; | ||||
| 					case CS_RAW_16: | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 2); | ||||
| 						trail = ((int)castBuffer.getShort(0)) & 0xffff; | ||||
| 						if(trail == 0) { | ||||
| 							obj = top_schema.createFromRaw(new byte[0], 0, 0); | ||||
| 							obj = RawType.create(new byte[0]); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						cs = ACS_RAW_VALUE; | ||||
| @@ -330,77 +303,68 @@ public class UnpackerImpl { | ||||
| 						// FIXME overflow check | ||||
| 						trail = castBuffer.getInt(0) & 0x7fffffff; | ||||
| 						if(trail == 0) { | ||||
| 							obj = top_schema.createFromRaw(new byte[0], 0, 0); | ||||
| 							obj = RawType.create(new byte[0]); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						cs = ACS_RAW_VALUE; | ||||
| 					case ACS_RAW_VALUE: | ||||
| 						obj = top_schema.createFromRaw(src, n, trail); | ||||
| 						break _fixed_trail_again; | ||||
| 					case ACS_RAW_VALUE: { | ||||
| 							byte[] raw = new byte[trail]; | ||||
| 							System.arraycopy(src, n, raw, 0, trail); | ||||
| 							obj = RawType.create(raw); | ||||
| 						} | ||||
| 						break _push; | ||||
| 					case CS_ARRAY_16: | ||||
| 						if(top >= MAX_STACK_SIZE) { | ||||
| 							throw new UnpackException("parse error"); | ||||
| 						} | ||||
| 						if(!(top_schema instanceof IArraySchema)) { | ||||
| 							throw new RuntimeException("type error"); | ||||
| 						} | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 2); | ||||
| 						count = ((int)castBuffer.getShort(0)) & 0xffff; | ||||
| 						obj = new Object[count]; | ||||
| 						obj = new MessagePackObject[count]; | ||||
| 						if(count == 0) { | ||||
| 							obj = ((IArraySchema)top_schema).createFromArray((Object[])obj); | ||||
| 							obj = ArrayType.create((MessagePackObject[])obj); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						++top; | ||||
| 						stack_obj[top]    = top_obj; | ||||
| 						stack_ct[top]     = top_ct; | ||||
| 						stack_count[top]  = top_count; | ||||
| 						stack_schema[top] = top_schema; | ||||
| 						top_obj    = obj; | ||||
| 						top_ct     = CT_ARRAY_ITEM; | ||||
| 						top_count  = count; | ||||
| 						top_schema = ((IArraySchema)top_schema).getElementSchema(0); | ||||
| 						break _header_again; | ||||
| 					case CS_ARRAY_32: | ||||
| 						if(top >= MAX_STACK_SIZE) { | ||||
| 							throw new UnpackException("parse error"); | ||||
| 						} | ||||
| 						if(!(top_schema instanceof IArraySchema)) { | ||||
| 							throw new RuntimeException("type error"); | ||||
| 						} | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 4); | ||||
| 						// FIXME overflow check | ||||
| 						count = castBuffer.getInt(0) & 0x7fffffff; | ||||
| 						obj = new Object[count]; | ||||
| 						obj = new MessagePackObject[count]; | ||||
| 						if(count == 0) { | ||||
| 							obj = ((IArraySchema)top_schema).createFromArray((Object[])obj); | ||||
| 							obj = ArrayType.create((MessagePackObject[])obj); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						++top; | ||||
| 						stack_obj[top]    = top_obj; | ||||
| 						stack_ct[top]     = top_ct; | ||||
| 						stack_count[top]  = top_count; | ||||
| 						stack_schema[top] = top_schema; | ||||
| 						top_obj    = obj; | ||||
| 						top_ct     = CT_ARRAY_ITEM; | ||||
| 						top_count  = count; | ||||
| 						top_schema = ((IArraySchema)top_schema).getElementSchema(0); | ||||
| 						break _header_again; | ||||
| 					case CS_MAP_16: | ||||
| 						if(top >= MAX_STACK_SIZE) { | ||||
| 							throw new UnpackException("parse error"); | ||||
| 						} | ||||
| 						if(!(top_schema instanceof IMapSchema)) { | ||||
| 							throw new RuntimeException("type error"); | ||||
| 						} | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 2); | ||||
| 						count = ((int)castBuffer.getShort(0)) & 0xffff; | ||||
| 						obj = new Object[count*2]; | ||||
| 						obj = new MessagePackObject[count*2]; | ||||
| 						if(count == 0) { | ||||
| 							obj = ((IMapSchema)top_schema).createFromMap((Object[])obj); | ||||
| 							obj = MapType.create((MessagePackObject[])obj); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						//System.out.println("fixmap count:"+count); | ||||
| @@ -408,26 +372,21 @@ public class UnpackerImpl { | ||||
| 						stack_obj[top]    = top_obj; | ||||
| 						stack_ct[top]     = top_ct; | ||||
| 						stack_count[top]  = top_count; | ||||
| 						stack_schema[top] = top_schema; | ||||
| 						top_obj    = obj; | ||||
| 						top_ct     = CT_MAP_KEY; | ||||
| 						top_count  = count; | ||||
| 						top_schema = ((IMapSchema)top_schema).getKeySchema(); | ||||
| 						break _header_again; | ||||
| 					case CS_MAP_32: | ||||
| 						if(top >= MAX_STACK_SIZE) { | ||||
| 							throw new UnpackException("parse error"); | ||||
| 						} | ||||
| 						if(!(top_schema instanceof IMapSchema)) { | ||||
| 							throw new RuntimeException("type error"); | ||||
| 						} | ||||
| 						castBuffer.rewind(); | ||||
| 						castBuffer.put(src, n, 4); | ||||
| 						// FIXME overflow check | ||||
| 						count = castBuffer.getInt(0) & 0x7fffffff; | ||||
| 						obj = new Object[count*2]; | ||||
| 						obj = new MessagePackObject[count*2]; | ||||
| 						if(count == 0) { | ||||
| 							obj = ((IMapSchema)top_schema).createFromMap((Object[])obj); | ||||
| 							obj = MapType.create((MessagePackObject[])obj); | ||||
| 							break _push; | ||||
| 						} | ||||
| 						//System.out.println("fixmap count:"+count); | ||||
| @@ -435,11 +394,9 @@ public class UnpackerImpl { | ||||
| 						stack_obj[top]    = top_obj; | ||||
| 						stack_ct[top]     = top_ct; | ||||
| 						stack_count[top]  = top_count; | ||||
| 						stack_schema[top] = top_schema; | ||||
| 						top_obj    = obj; | ||||
| 						top_ct     = CT_MAP_KEY; | ||||
| 						top_count  = count; | ||||
| 						top_schema = ((IMapSchema)top_schema).getKeySchema(); | ||||
| 						break _header_again; | ||||
| 					default: | ||||
| 						throw new UnpackException("parse error"); | ||||
| @@ -454,7 +411,7 @@ public class UnpackerImpl { | ||||
| 				//System.out.println("push top:"+top); | ||||
| 				if(top == -1) { | ||||
| 					++i; | ||||
| 					data = obj; | ||||
| 					data = (MessagePackObject)obj; | ||||
| 					finished = true; | ||||
| 					break _out; | ||||
| 				} | ||||
| @@ -468,14 +425,10 @@ public class UnpackerImpl { | ||||
| 							top_obj    = stack_obj[top]; | ||||
| 							top_ct     = stack_ct[top]; | ||||
| 							top_count  = stack_count[top]; | ||||
| 							top_schema = stack_schema[top]; | ||||
| 							obj = ((IArraySchema)top_schema).createFromArray(ar); | ||||
| 							obj = ArrayType.create((MessagePackObject[])ar); | ||||
| 							stack_obj[top] = null; | ||||
| 							stack_schema[top] = null; | ||||
| 							--top; | ||||
| 							break _push; | ||||
| 						} else { | ||||
| 							top_schema = ((IArraySchema)stack_schema[top]).getElementSchema(ar.length - top_count); | ||||
| 						} | ||||
| 						break _header_again; | ||||
| 					} | ||||
| @@ -484,7 +437,6 @@ public class UnpackerImpl { | ||||
| 						Object[] mp = (Object[])top_obj; | ||||
| 						mp[mp.length - top_count*2] = obj; | ||||
| 						top_ct = CT_MAP_VALUE; | ||||
| 						top_schema = ((IMapSchema)stack_schema[top]).getValueSchema(); | ||||
| 						break _header_again; | ||||
| 					} | ||||
| 				case CT_MAP_VALUE: { | ||||
| @@ -495,10 +447,8 @@ public class UnpackerImpl { | ||||
| 							top_obj    = stack_obj[top]; | ||||
| 							top_ct     = stack_ct[top]; | ||||
| 							top_count  = stack_count[top]; | ||||
| 							top_schema = stack_schema[top]; | ||||
| 							obj = ((IMapSchema)top_schema).createFromMap(mp); | ||||
| 							obj = MapType.create((MessagePackObject[])mp); | ||||
| 							stack_obj[top] = null; | ||||
| 							stack_schema[top] = null; | ||||
| 							--top; | ||||
| 							break _push; | ||||
| 						} | ||||
|   | ||||
							
								
								
									
										81
									
								
								java/src/main/java/org/msgpack/object/ArrayType.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								java/src/main/java/org/msgpack/object/ArrayType.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Arrays; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class ArrayType extends MessagePackObject { | ||||
| 	private MessagePackObject[] array; | ||||
|  | ||||
| 	ArrayType(MessagePackObject[] array) { | ||||
| 		this.array = array; | ||||
| 	} | ||||
|  | ||||
| 	public static ArrayType create(MessagePackObject[] array) { | ||||
| 		return new ArrayType(array); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean isArrayType() { | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public MessagePackObject[] asArray() { | ||||
| 		return array; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public List<MessagePackObject> asList() { | ||||
| 		return Arrays.asList(array); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packArray(array.length); | ||||
| 		for(int i=0; i < array.length; i++) { | ||||
| 			array[i].messagePack(pk); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return Arrays.equals(((ArrayType)obj).array, array); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return array.hashCode(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		MessagePackObject[] copy = new MessagePackObject[array.length]; | ||||
| 		for(int i=0; i < array.length; i++) { | ||||
| 			copy[i] = (MessagePackObject)array[i].clone(); | ||||
| 		} | ||||
| 		return copy; | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										131
									
								
								java/src/main/java/org/msgpack/object/BigIntegerTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								java/src/main/java/org/msgpack/object/BigIntegerTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,131 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.math.BigInteger; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| class BigIntegerTypeIMPL extends IntegerType { | ||||
| 	private BigInteger value; | ||||
|  | ||||
| 	BigIntegerTypeIMPL(BigInteger value) { | ||||
| 		this.value = value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte asByte() { | ||||
| 		if(value.compareTo(BigInteger.valueOf((long)Byte.MAX_VALUE)) > 0) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return value.byteValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short asShort() { | ||||
| 		if(value.compareTo(BigInteger.valueOf((long)Short.MAX_VALUE)) > 0) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return value.shortValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int asInt() { | ||||
| 		if(value.compareTo(BigInteger.valueOf((long)Integer.MAX_VALUE)) > 0) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return value.intValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long asLong() { | ||||
| 		if(value.compareTo(BigInteger.valueOf(Long.MAX_VALUE)) > 0) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return value.longValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger asBigInteger() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte byteValue() { | ||||
| 		return value.byteValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short shortValue() { | ||||
| 		return value.shortValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int intValue() { | ||||
| 		return value.intValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long longValue() { | ||||
| 		return value.longValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger bigIntegerValue() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float floatValue() { | ||||
| 		return value.floatValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double doubleValue() { | ||||
| 		return value.doubleValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packBigInteger(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			if(obj.getClass() == ShortIntegerTypeIMPL.class) { | ||||
| 				return BigInteger.valueOf(((ShortIntegerTypeIMPL)obj).longValue()).equals(value); | ||||
| 			} else if(obj.getClass() == LongIntegerTypeIMPL.class) { | ||||
| 				return BigInteger.valueOf(((LongIntegerTypeIMPL)obj).longValue()).equals(value); | ||||
| 			} | ||||
| 			return false; | ||||
| 		} | ||||
| 		return ((BigIntegerTypeIMPL)obj).value.equals(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return value.hashCode(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new BigIntegerTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										71
									
								
								java/src/main/java/org/msgpack/object/BooleanType.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								java/src/main/java/org/msgpack/object/BooleanType.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class BooleanType extends MessagePackObject { | ||||
| 	private boolean value; | ||||
|  | ||||
| 	BooleanType(boolean value) { | ||||
| 		this.value = value; | ||||
| 	} | ||||
|  | ||||
| 	public static BooleanType create(boolean value) { | ||||
| 		return new BooleanType(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean isBooleanType() { | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean asBoolean() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packBoolean(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return ((BooleanType)obj).value == value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		if(value) { | ||||
| 			return 1231; | ||||
| 		} else { | ||||
| 			return 1237; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new BooleanType(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										101
									
								
								java/src/main/java/org/msgpack/object/DoubleTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								java/src/main/java/org/msgpack/object/DoubleTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,101 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.math.BigInteger; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| class DoubleTypeIMPL extends FloatType { | ||||
| 	private double value; | ||||
|  | ||||
| 	public DoubleTypeIMPL(double value) { | ||||
| 		this.value = value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float asFloat() { | ||||
| 		// FIXME check overflow, underflow? | ||||
| 		return (float)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double asDouble() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte byteValue() { | ||||
| 		return (byte)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short shortValue() { | ||||
| 		return (short)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int intValue() { | ||||
| 		return (int)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long longValue() { | ||||
| 		return (long)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger bigIntegerValue() { | ||||
| 		return BigInteger.valueOf((long)value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float floatValue() { | ||||
| 		return (float)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double doubleValue() { | ||||
| 		return (double)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packDouble(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return ((DoubleTypeIMPL)obj).value == value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		long v = Double.doubleToLongBits(value); | ||||
| 		return (int)(v^(v>>>32)); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new DoubleTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -15,12 +15,22 @@ | ||||
| //    See the License for the specific language governing permissions and | ||||
| //    limitations under the License. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
| package org.msgpack.object; | ||||
| 
 | ||||
| import org.msgpack.Schema; | ||||
| import org.msgpack.*; | ||||
| 
 | ||||
| public interface IArraySchema { | ||||
| 	public Schema getElementSchema(int index); | ||||
| 	public Object createFromArray(Object[] obj); | ||||
| public abstract class FloatType extends MessagePackObject { | ||||
| 	@Override | ||||
| 	public boolean isFloatType() { | ||||
| 		return true; | ||||
| 	} | ||||
| 
 | ||||
| 	public static FloatType create(float value) { | ||||
| 		return new FloatTypeIMPL(value); | ||||
| 	} | ||||
| 
 | ||||
| 	public static FloatType create(double value) { | ||||
| 		return new DoubleTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										94
									
								
								java/src/main/java/org/msgpack/object/FloatTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								java/src/main/java/org/msgpack/object/FloatTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.math.BigInteger; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| class FloatTypeIMPL extends FloatType { | ||||
| 	private float value; | ||||
|  | ||||
| 	public FloatTypeIMPL(float value) { | ||||
| 		this.value = value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float asFloat() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double asDouble() { | ||||
| 		return (double)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte byteValue() { | ||||
| 		return (byte)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short shortValue() { | ||||
| 		return (short)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int intValue() { | ||||
| 		return (int)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long longValue() { | ||||
| 		return (long)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float floatValue() { | ||||
| 		return (float)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double doubleValue() { | ||||
| 		return (double)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packFloat(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return ((FloatTypeIMPL)obj).value == value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return Float.floatToIntBits(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new FloatTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										49
									
								
								java/src/main/java/org/msgpack/object/IntegerType.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								java/src/main/java/org/msgpack/object/IntegerType.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.math.BigInteger; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public abstract class IntegerType extends MessagePackObject { | ||||
| 	@Override | ||||
| 	public boolean isIntegerType() { | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| 	public static IntegerType create(byte value) { | ||||
| 		return new ShortIntegerTypeIMPL((int)value); | ||||
| 	} | ||||
|  | ||||
| 	public static IntegerType create(short value) { | ||||
| 		return new ShortIntegerTypeIMPL((int)value); | ||||
| 	} | ||||
|  | ||||
| 	public static IntegerType create(int value) { | ||||
| 		return new ShortIntegerTypeIMPL(value); | ||||
| 	} | ||||
|  | ||||
| 	public static IntegerType create(long value) { | ||||
| 		return new LongIntegerTypeIMPL(value); | ||||
| 	} | ||||
|  | ||||
| 	public static IntegerType create(BigInteger value) { | ||||
| 		return new BigIntegerTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										128
									
								
								java/src/main/java/org/msgpack/object/LongIntegerTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								java/src/main/java/org/msgpack/object/LongIntegerTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.math.BigInteger; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| class LongIntegerTypeIMPL extends IntegerType { | ||||
| 	private long value; | ||||
|  | ||||
| 	LongIntegerTypeIMPL(long value) { | ||||
| 		this.value = value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte asByte() { | ||||
| 		if(value > (long)Byte.MAX_VALUE) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return (byte)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short asShort() { | ||||
| 		if(value > (long)Short.MAX_VALUE) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return (short)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int asInt() { | ||||
| 		if(value > (long)Integer.MAX_VALUE) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return (int)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long asLong() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger asBigInteger() { | ||||
| 		return BigInteger.valueOf(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte byteValue() { | ||||
| 		return (byte)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short shortValue() { | ||||
| 		return (short)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int intValue() { | ||||
| 		return (int)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long longValue() { | ||||
| 		return (long)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger bigIntegerValue() { | ||||
| 		return BigInteger.valueOf(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float floatValue() { | ||||
| 		return (float)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double doubleValue() { | ||||
| 		return (double)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packLong(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			if(obj.getClass() == ShortIntegerTypeIMPL.class) { | ||||
| 				return value == ((ShortIntegerTypeIMPL)obj).longValue(); | ||||
| 			} else if(obj.getClass() == BigIntegerTypeIMPL.class) { | ||||
| 				return BigInteger.valueOf(value).equals(((BigIntegerTypeIMPL)obj).bigIntegerValue()); | ||||
| 			} | ||||
| 			return false; | ||||
| 		} | ||||
| 		return ((LongIntegerTypeIMPL)obj).value == value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return (int)(value^(value>>>32)); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new LongIntegerTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										84
									
								
								java/src/main/java/org/msgpack/object/MapType.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								java/src/main/java/org/msgpack/object/MapType.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
| import java.util.Arrays; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class MapType extends MessagePackObject { | ||||
| 	private MessagePackObject[] map; | ||||
|  | ||||
| 	MapType(MessagePackObject[] map) { | ||||
| 		this.map = map; | ||||
| 	} | ||||
|  | ||||
| 	public static MapType create(MessagePackObject[] map) { | ||||
| 		return new MapType(map); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean isMapType() { | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Map<MessagePackObject, MessagePackObject> asMap() { | ||||
| 		HashMap<MessagePackObject, MessagePackObject> m = new HashMap(map.length / 2); | ||||
| 		int i = 0; | ||||
| 		while(i < map.length) { | ||||
| 			MessagePackObject k = map[i++]; | ||||
| 			MessagePackObject v = map[i++]; | ||||
| 			m.put(k, v); | ||||
| 		} | ||||
| 		return m; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packMap(map.length / 2); | ||||
| 		for(int i=0; i < map.length; i++) { | ||||
| 			map[i].messagePack(pk); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return Arrays.equals(((MapType)obj).map, map); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return map.hashCode(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		MessagePackObject[] copy = new MessagePackObject[map.length]; | ||||
| 		for(int i=0; i < map.length; i++) { | ||||
| 			copy[i] = (MessagePackObject)map[i].clone(); | ||||
| 		} | ||||
| 		return copy; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -15,29 +15,46 @@ | ||||
| //    See the License for the specific language governing permissions and | ||||
| //    limitations under the License. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
| package org.msgpack.object; | ||||
| 
 | ||||
| import org.msgpack.Schema; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
| 
 | ||||
| public class FieldSchema { | ||||
| 	private String name; | ||||
| 	private Schema schema; | ||||
| public class NilType extends MessagePackObject { | ||||
| 	private final static NilType INSTANCE = new NilType(); | ||||
| 
 | ||||
| 	public FieldSchema(String name, Schema schema) { | ||||
| 		this.name = name; | ||||
| 		this.schema = schema; | ||||
| 	public static NilType create() { | ||||
| 		return INSTANCE; | ||||
| 	} | ||||
| 
 | ||||
| 	public final String getName() { | ||||
| 		return name; | ||||
| 	private NilType() { } | ||||
| 
 | ||||
| 	@Override | ||||
| 	public boolean isNil() { | ||||
| 		return true; | ||||
| 	} | ||||
| 
 | ||||
| 	public final Schema getSchema() { | ||||
| 		return schema; | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packNil(); | ||||
| 	} | ||||
| 
 | ||||
| 	public String getExpression() { | ||||
| 		return "(field "+name+" "+schema.getExpression()+")"; | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return true; | ||||
| 	} | ||||
| 
 | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return INSTANCE; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										90
									
								
								java/src/main/java/org/msgpack/object/RawType.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								java/src/main/java/org/msgpack/object/RawType.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class RawType extends MessagePackObject { | ||||
| 	private byte[] bytes; | ||||
|  | ||||
| 	RawType(byte[] bytes) { | ||||
| 		this.bytes = bytes; | ||||
| 	} | ||||
|  | ||||
| 	RawType(String str) { | ||||
| 		try { | ||||
| 			this.bytes = str.getBytes("UTF-8"); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public static RawType create(byte[] bytes) { | ||||
| 		return new RawType(bytes); | ||||
| 	} | ||||
|  | ||||
| 	public static RawType create(String str) { | ||||
| 		return new RawType(str); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean isRawType() { | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte[] asByteArray() { | ||||
| 		return bytes; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String asString() { | ||||
| 		try { | ||||
| 			return new String(bytes, "UTF-8"); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packRaw(bytes.length); | ||||
| 		pk.packRawBody(bytes); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return Arrays.equals(((RawType)obj).bytes, bytes); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return bytes.hashCode(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new RawType((byte[])bytes.clone()); | ||||
| 	} | ||||
| } | ||||
|  | ||||
							
								
								
									
										125
									
								
								java/src/main/java/org/msgpack/object/ShortIntegerTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								java/src/main/java/org/msgpack/object/ShortIntegerTypeIMPL.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.object; | ||||
|  | ||||
| import java.math.BigInteger; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| class ShortIntegerTypeIMPL extends IntegerType { | ||||
| 	private int value; | ||||
|  | ||||
| 	ShortIntegerTypeIMPL(int value) { | ||||
| 		this.value = value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte asByte() { | ||||
| 		if(value > (int)Byte.MAX_VALUE) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return (byte)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short asShort() { | ||||
| 		if(value > (int)Short.MAX_VALUE) { | ||||
| 			throw new MessageTypeException("type error"); | ||||
| 		} | ||||
| 		return (short)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int asInt() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long asLong() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger asBigInteger() { | ||||
| 		return BigInteger.valueOf((long)value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public byte byteValue() { | ||||
| 		return (byte)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public short shortValue() { | ||||
| 		return (short)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int intValue() { | ||||
| 		return (int)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public long longValue() { | ||||
| 		return (long)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public BigInteger bigIntegerValue() { | ||||
| 		return BigInteger.valueOf((long)value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public float floatValue() { | ||||
| 		return (float)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public double doubleValue() { | ||||
| 		return (double)value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void messagePack(Packer pk) throws IOException { | ||||
| 		pk.packInt(value); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean equals(Object obj) { | ||||
| 		if(obj.getClass() != getClass()) { | ||||
| 			if(obj.getClass() == LongIntegerTypeIMPL.class) { | ||||
| 				return (long)value == ((LongIntegerTypeIMPL)obj).longValue(); | ||||
| 			} else if(obj.getClass() == BigIntegerTypeIMPL.class) { | ||||
| 				return ((BigIntegerTypeIMPL)obj).bigIntegerValue().equals(BigInteger.valueOf((long)value)); | ||||
| 			} | ||||
| 			return false; | ||||
| 		} | ||||
| 		return ((ShortIntegerTypeIMPL)obj).value == value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int hashCode() { | ||||
| 		return value; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object clone() { | ||||
| 		return new ShortIntegerTypeIMPL(value); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,125 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.Collection; | ||||
| import java.util.Set; | ||||
| import java.util.List; | ||||
| import java.util.ArrayList; | ||||
| import java.util.RandomAccess; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class ArraySchema extends Schema implements IArraySchema { | ||||
| 	private Schema elementSchema; | ||||
|  | ||||
| 	public ArraySchema(Schema elementSchema) | ||||
| 	{ | ||||
| 		super("array"); | ||||
| 		this.elementSchema = elementSchema; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getFullName() | ||||
| 	{ | ||||
| 		return "List<"+elementSchema.getFullName()+">"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() | ||||
| 	{ | ||||
| 		return "(array "+elementSchema.getExpression()+")"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public void pack(Packer pk, Object obj) throws IOException | ||||
| 	{ | ||||
| 		if(obj instanceof List) { | ||||
| 			ArrayList<Object> d = (ArrayList<Object>)obj; | ||||
| 			pk.packArray(d.size()); | ||||
| 			if(obj instanceof RandomAccess) { | ||||
| 				for(int i=0; i < d.size(); ++i) { | ||||
| 					elementSchema.pack(pk, d.get(i)); | ||||
| 				} | ||||
| 			} else { | ||||
| 				for(Object e : d) { | ||||
| 					elementSchema.pack(pk, e); | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj instanceof Set) { | ||||
| 			Set<Object> d = (Set<Object>)obj; | ||||
| 			pk.packArray(d.size()); | ||||
| 			for(Object e : d) { | ||||
| 				elementSchema.pack(pk, e); | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public Object convert(Object obj) throws MessageTypeException | ||||
| 	{ | ||||
| 		if(obj instanceof List) { | ||||
| 			List d = (List)obj; | ||||
| 			ArrayList ar = new ArrayList(d.size()); | ||||
| 			if(obj instanceof RandomAccess) { | ||||
| 				for(int i=0; i < d.size(); ++i) { | ||||
| 					ar.add( elementSchema.convert(d.get(i)) ); | ||||
| 				} | ||||
| 			} else { | ||||
| 				for(Object e : d) { | ||||
| 					ar.add( elementSchema.convert(e) ); | ||||
| 				} | ||||
| 			} | ||||
| 			return ar; | ||||
|  | ||||
| 		} else if(obj instanceof Collection) { | ||||
| 			Collection d = (Collection)obj; | ||||
| 			ArrayList ar = new ArrayList(d.size()); | ||||
| 			for(Object e : (Collection)obj) { | ||||
| 				ar.add( elementSchema.convert(e) ); | ||||
| 			} | ||||
| 			return ar; | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Schema getElementSchema(int index) | ||||
| 	{ | ||||
| 		return elementSchema; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromArray(Object[] obj) | ||||
| 	{ | ||||
| 		return Arrays.asList(obj); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,89 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class ByteSchema extends Schema { | ||||
| 	public ByteSchema() { | ||||
| 		super("Byte"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "byte"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Number) { | ||||
| 			pk.packByte( ((Number)obj).byteValue() ); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Byte) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof Number) { | ||||
| 			return ((Number)obj).byteValue(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return (byte)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return (byte)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return (byte)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromLong(long v) { | ||||
| 		return (byte)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return (byte)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return (byte)v; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,241 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.io.IOException; | ||||
| import java.io.File; | ||||
| import java.io.Writer; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class ClassGenerator { | ||||
| 	private ClassSchema schema; | ||||
| 	private Writer writer; | ||||
| 	private int indent; | ||||
|  | ||||
| 	private ClassGenerator(Writer writer) { | ||||
| 		this.writer = writer; | ||||
| 		this.indent = 0; | ||||
| 	} | ||||
|  | ||||
| 	public static void write(Schema schema, Writer dest) throws IOException { | ||||
| 		if(!(schema instanceof ClassSchema)) { | ||||
| 			throw new RuntimeException("schema is not class schema"); | ||||
| 		} | ||||
| 		ClassSchema cs = (ClassSchema)schema; | ||||
| 		new ClassGenerator(dest).run(cs); | ||||
| 	} | ||||
|  | ||||
| 	private void run(ClassSchema cs) throws IOException { | ||||
| 		List<ClassSchema> subclasses = new ArrayList<ClassSchema>(); | ||||
| 		for(FieldSchema f : cs.getFields()) { | ||||
| 			findSubclassSchema(subclasses, f.getSchema()); | ||||
| 		} | ||||
|  | ||||
| 		for(ClassSchema sub : subclasses) { | ||||
| 			sub.setNamespace(cs.getNamespace()); | ||||
| 			sub.setImports(cs.getImports()); | ||||
| 		} | ||||
|  | ||||
| 		this.schema = cs; | ||||
|  | ||||
| 		writeHeader(); | ||||
|  | ||||
| 		writeClass(); | ||||
|  | ||||
| 		for(ClassSchema sub : subclasses) { | ||||
| 			this.schema = sub; | ||||
| 			writeSubclass(); | ||||
| 		} | ||||
|  | ||||
| 		writeFooter(); | ||||
|  | ||||
| 		this.schema = null; | ||||
| 		writer.flush(); | ||||
| 	} | ||||
|  | ||||
| 	private void findSubclassSchema(List<ClassSchema> dst, Schema s) { | ||||
| 		if(s instanceof ClassSchema) { | ||||
| 			ClassSchema cs = (ClassSchema)s; | ||||
| 			if(!dst.contains(cs)) { dst.add(cs); } | ||||
| 			for(FieldSchema f : cs.getFields()) { | ||||
| 				findSubclassSchema(dst, f.getSchema()); | ||||
| 			} | ||||
| 		} else if(s instanceof ArraySchema) { | ||||
| 			ArraySchema as = (ArraySchema)s; | ||||
| 			findSubclassSchema(dst, as.getElementSchema(0)); | ||||
| 		} else if(s instanceof MapSchema) { | ||||
| 			MapSchema as = (MapSchema)s; | ||||
| 			findSubclassSchema(dst, as.getKeySchema()); | ||||
| 			findSubclassSchema(dst, as.getValueSchema()); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private void writeHeader() throws IOException { | ||||
| 		if(schema.getNamespace() != null) { | ||||
| 			line("package "+schema.getNamespace()+";"); | ||||
| 			line(); | ||||
| 		} | ||||
| 		line("import java.util.*;"); | ||||
| 		line("import java.io.*;"); | ||||
| 		line("import org.msgpack.*;"); | ||||
| 		line("import org.msgpack.schema.ClassSchema;"); | ||||
| 		line("import org.msgpack.schema.FieldSchema;"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeFooter() throws IOException { | ||||
| 		line(); | ||||
| 	} | ||||
|  | ||||
| 	private void writeClass() throws IOException { | ||||
| 		line(); | ||||
| 		line("public final class "+schema.getName()+" implements MessagePackable, MessageConvertable"); | ||||
| 		line("{"); | ||||
| 		pushIndent(); | ||||
| 			writeSchema(); | ||||
| 			writeMemberVariables(); | ||||
| 			writeMemberFunctions(); | ||||
| 		popIndent(); | ||||
| 		line("}"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeSubclass() throws IOException { | ||||
| 		line(); | ||||
| 		line("final class "+schema.getName()+" implements MessagePackable, MessageConvertable"); | ||||
| 		line("{"); | ||||
| 		pushIndent(); | ||||
| 			writeSchema(); | ||||
| 			writeMemberVariables(); | ||||
| 			writeMemberFunctions(); | ||||
| 		popIndent(); | ||||
| 		line("}"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeSchema() throws IOException { | ||||
| 		line("private static final ClassSchema _SCHEMA = (ClassSchema)Schema.load(\""+schema.getExpression()+"\");"); | ||||
| 		line("public static ClassSchema getSchema() { return _SCHEMA; }"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeMemberVariables() throws IOException { | ||||
| 		line(); | ||||
| 		for(FieldSchema f : schema.getFields()) { | ||||
| 			line("public "+f.getSchema().getFullName()+" "+f.getName()+";"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private void writeMemberFunctions() throws IOException { | ||||
| 		// void messagePack(Packer pk) | ||||
| 		// boolean equals(Object obj) | ||||
| 		// int hashCode() | ||||
| 		// void set(int _index, Object _value) | ||||
| 		// Object get(int _index); | ||||
| 		// getXxx() | ||||
| 		// setXxx(Xxx xxx) | ||||
| 		writeConstructors(); | ||||
| 		writeAccessors(); | ||||
| 		writePackFunction(); | ||||
| 		writeConvertFunction(); | ||||
| 		writeFactoryFunction(); | ||||
| 	} | ||||
|  | ||||
| 	private void writeConstructors() throws IOException { | ||||
| 		line(); | ||||
| 		line("public "+schema.getName()+"() { }"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeAccessors() throws IOException { | ||||
| 		// FIXME | ||||
| 		//line(); | ||||
| 		//for(FieldSchema f : schema.getFields()) { | ||||
| 		//	line(""); | ||||
| 		//} | ||||
| 	} | ||||
|  | ||||
| 	private void writePackFunction() throws IOException { | ||||
| 		line(); | ||||
| 		line("@Override"); | ||||
| 		line("public void messagePack(Packer _pk) throws IOException"); | ||||
| 		line("{"); | ||||
| 		pushIndent(); | ||||
| 			line("_pk.packArray("+schema.getFields().length+");"); | ||||
| 			line("FieldSchema[] _fields = _SCHEMA.getFields();"); | ||||
| 			int i = 0; | ||||
| 			for(FieldSchema f : schema.getFields()) { | ||||
| 				line("_fields["+i+"].getSchema().pack(_pk, "+f.getName()+");"); | ||||
| 				++i; | ||||
| 			} | ||||
| 		popIndent(); | ||||
| 		line("}"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeConvertFunction() throws IOException { | ||||
| 		line(); | ||||
| 		line("@Override"); | ||||
| 		line("@SuppressWarnings(\"unchecked\")"); | ||||
| 		line("public void messageConvert(Object obj) throws MessageTypeException"); | ||||
| 		line("{"); | ||||
| 		pushIndent(); | ||||
| 			line("Object[] _source = ((List)obj).toArray();"); | ||||
| 			line("FieldSchema[] _fields = _SCHEMA.getFields();"); | ||||
| 			int i = 0; | ||||
| 			for(FieldSchema f : schema.getFields()) { | ||||
| 				line("if(_source.length <= "+i+") { return; } this."+f.getName()+" = ("+f.getSchema().getFullName()+")_fields["+i+"].getSchema().convert(_source["+i+"]);"); | ||||
| 				++i; | ||||
| 			} | ||||
| 		popIndent(); | ||||
| 		line("}"); | ||||
| 	} | ||||
|  | ||||
| 	private void writeFactoryFunction() throws IOException { | ||||
| 		line(); | ||||
| 		line("@SuppressWarnings(\"unchecked\")"); | ||||
| 		line("public static "+schema.getName()+" createFromMessage(Object[] _message)"); | ||||
| 		line("{"); | ||||
| 		pushIndent(); | ||||
| 			line(schema.getName()+" _self = new "+schema.getName()+"();"); | ||||
| 			int i = 0; | ||||
| 			for(FieldSchema f : schema.getFields()) { | ||||
| 				line("if(_message.length <= "+i+") { return _self; } _self."+f.getName()+" = ("+f.getSchema().getFullName()+")_message["+i+"];"); | ||||
| 				++i; | ||||
| 			} | ||||
| 			line("return _self;"); | ||||
| 		popIndent(); | ||||
| 		line("}"); | ||||
| 	} | ||||
|  | ||||
| 	private void line(String str) throws IOException { | ||||
| 		for(int i=0; i < indent; ++i) { | ||||
| 			writer.write("\t"); | ||||
| 		} | ||||
| 		writer.write(str+"\n"); | ||||
| 	} | ||||
|  | ||||
| 	private void line() throws IOException { | ||||
| 		writer.write("\n"); | ||||
| 	} | ||||
|  | ||||
| 	private void pushIndent() { | ||||
| 		indent += 1; | ||||
| 	} | ||||
|  | ||||
| 	private void popIndent() { | ||||
| 		indent -= 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,95 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public abstract class ClassSchema extends Schema implements IArraySchema { | ||||
| 	protected FieldSchema[] fields; | ||||
| 	protected List<String> imports; | ||||
| 	protected String namespace; | ||||
| 	protected String fqdn; | ||||
|  | ||||
| 	public ClassSchema( | ||||
| 			String name, String namespace, | ||||
| 			List<String> imports, List<FieldSchema> fields) { | ||||
| 		super(name); | ||||
| 		this.namespace = namespace; | ||||
| 		this.imports = imports;  // FIXME clone? | ||||
| 		this.fields = new FieldSchema[fields.size()]; | ||||
| 		System.arraycopy(fields.toArray(), 0, this.fields, 0, fields.size()); | ||||
| 		if(namespace == null) { | ||||
| 			this.fqdn = name; | ||||
| 		} else { | ||||
| 			this.fqdn = namespace+"."+name; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public final FieldSchema[] getFields() { | ||||
| 		return fields; | ||||
| 	} | ||||
|  | ||||
| 	String getNamespace() { | ||||
| 		return namespace; | ||||
| 	} | ||||
|  | ||||
| 	List<String> getImports() { | ||||
| 		return imports; | ||||
| 	} | ||||
|  | ||||
| 	void setNamespace(String namespace) { | ||||
| 		this.namespace = namespace; | ||||
| 	} | ||||
|  | ||||
| 	void setImports(List<String> imports) { | ||||
| 		this.imports = imports;  // FIXME clone? | ||||
| 	} | ||||
|  | ||||
| 	//@Override | ||||
| 	//public String getFullName() | ||||
| 	//{ | ||||
| 	//	if(namespace == null) { | ||||
| 	//		return getName(); | ||||
| 	//	} else { | ||||
| 	//		return namespace+"."+getName(); | ||||
| 	//	} | ||||
| 	//} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		StringBuffer b = new StringBuffer(); | ||||
| 		b.append("(class "); | ||||
| 		b.append(getName()); | ||||
| 		if(namespace != null) { | ||||
| 			b.append(" (package "+namespace+")"); | ||||
| 		} | ||||
| 		for(FieldSchema f : fields) { | ||||
| 			b.append(" "+f.getExpression()); | ||||
| 		} | ||||
| 		b.append(")"); | ||||
| 		return b.toString(); | ||||
| 	} | ||||
|  | ||||
| 	public boolean equals(SpecificClassSchema o) { | ||||
| 		return (namespace != null ? namespace.equals(o.getNamespace()) : o.getNamespace() == null) && | ||||
| 			getName().equals(o.getName()); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,84 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class DoubleSchema extends Schema { | ||||
| 	public DoubleSchema() { | ||||
| 		super("Double"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "double"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Number) { | ||||
| 			pk.packDouble( ((Number)obj).doubleValue() ); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Double) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof Number) { | ||||
| 			return ((Number)obj).doubleValue(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return (double)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return (double)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return (double)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return (double)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return (double)v; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,84 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class FloatSchema extends Schema { | ||||
| 	public FloatSchema() { | ||||
| 		super("Float"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "float"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Number) { | ||||
| 			pk.packFloat( ((Number)obj).floatValue() ); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Float) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof Number) { | ||||
| 			return ((Number)obj).floatValue(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return (float)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return (float)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return (float)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return (float)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return (float)v; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,91 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.HashMap; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class GenericClassSchema extends ClassSchema { | ||||
| 	public GenericClassSchema( | ||||
| 			String name, String namespace, | ||||
| 			List<String> imports, List<FieldSchema> fields) { | ||||
| 		super(name, namespace, imports, fields); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Map) { | ||||
| 			Map d = (Map)obj; | ||||
| 			pk.packArray(fields.length); | ||||
| 			for(int i=0; i < fields.length; ++i) { | ||||
| 				FieldSchema f = fields[i]; | ||||
| 				f.getSchema().pack(pk, d.get(f.getName())); | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Collection) { | ||||
| 			// FIXME optimize | ||||
| 			return createFromArray( ((Collection)obj).toArray() ); | ||||
|  | ||||
| 		} else if(obj instanceof Map) { | ||||
| 			HashMap<String,Object> m = new HashMap<String,Object>(fields.length); | ||||
| 			Map d = (Map)obj; | ||||
| 			for(int i=0; i < fields.length; ++i) { | ||||
| 				FieldSchema f = fields[i]; | ||||
| 				String fieldName = f.getName(); | ||||
| 				m.put(fieldName, f.getSchema().convert(d.get(fieldName))); | ||||
| 			} | ||||
| 			return m; | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public Schema getElementSchema(int index) { | ||||
| 		// FIXME check index < fields.length | ||||
| 		return fields[index].getSchema(); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromArray(Object[] obj) { | ||||
| 		HashMap<String,Object> m = new HashMap<String,Object>(fields.length); | ||||
| 		int i=0; | ||||
| 		for(; i < obj.length; ++i) { | ||||
| 			m.put(fields[i].getName(), obj[i]); | ||||
| 		} | ||||
| 		for(; i < fields.length; ++i) { | ||||
| 			m.put(fields[i].getName(), null); | ||||
| 		} | ||||
| 		return m; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,192 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.util.HashMap; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
| //import org.msgpack.generic.*; | ||||
|  | ||||
| public class GenericSchema extends Schema implements IArraySchema, IMapSchema { | ||||
| 	public GenericSchema() { | ||||
| 		super("Object"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "object"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		pk.pack(obj); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		return obj; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Schema getElementSchema(int index) { | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Schema getKeySchema() { | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Schema getValueSchema() { | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromNil() { | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromBoolean(boolean v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromLong(long v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromRaw(byte[] b, int offset, int length) { | ||||
| 		byte[] bytes = new byte[length]; | ||||
| 		System.arraycopy(b, offset, bytes, 0, length); | ||||
| 		return bytes; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromArray(Object[] obj) { | ||||
| 		return Arrays.asList(obj); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public Object createFromMap(Object[] obj) { | ||||
| 		HashMap m = new HashMap(obj.length / 2); | ||||
| 		int i = 0; | ||||
| 		while(i < obj.length) { | ||||
| 			Object k = obj[i++]; | ||||
| 			Object v = obj[i++]; | ||||
| 			m.put(k, v); | ||||
| 		} | ||||
| 		return m; | ||||
| 	} | ||||
|  | ||||
| 	/* | ||||
|    @Override | ||||
| 	public Object createFromNil() { | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromBoolean(boolean v) { | ||||
| 		return new GenericBoolean(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFromByte(byte v) { | ||||
| 		return new GenericByte(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return new GenericShort(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return new GenericInt(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromLong(long v) { | ||||
| 		return new GenericLong(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return new GenericFloat(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return new GenericDouble(v); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromRaw(byte[] b, int offset, int length) { | ||||
| 		return new GenericRaw(b, offset, length); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromArray(Object[] obj) { | ||||
| 		// FIXME GenericArray | ||||
| 		return Arrays.asList(obj); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromMap(Object[] obj) { | ||||
| 		GenericMap m = new GenericMap(obj.length / 2); | ||||
| 		int i = 0; | ||||
| 		while(i < obj.length) { | ||||
| 			Object k = obj[i++]; | ||||
| 			Object v = obj[i++]; | ||||
| 			m.put(k, v); | ||||
| 		} | ||||
| 		return m; | ||||
| 	} | ||||
| 	*/ | ||||
| } | ||||
|  | ||||
| @@ -1,27 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import org.msgpack.Schema; | ||||
|  | ||||
| public interface IMapSchema { | ||||
| 	public Schema getKeySchema(); | ||||
| 	public Schema getValueSchema(); | ||||
| 	public Object createFromMap(Object[] obj); | ||||
| } | ||||
|  | ||||
| @@ -1,89 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class IntSchema extends Schema { | ||||
| 	public IntSchema() { | ||||
| 		super("Integer"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "int"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Number) { | ||||
| 			pk.packInt( ((Number)obj).intValue() ); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Integer) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof Number) { | ||||
| 			return ((Number)obj).intValue(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return (int)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return (int)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return (int)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromLong(long v) { | ||||
| 		return (int)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return (int)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return (int)v; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,89 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class LongSchema extends Schema { | ||||
| 	public LongSchema() { | ||||
| 		super("Long"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "long"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Number) { | ||||
| 			pk.packLong( ((Number)obj).longValue() ); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Long) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof Number) { | ||||
| 			return ((Number)obj).longValue(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return (long)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return (long)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return (long)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromLong(long v) { | ||||
| 		return (long)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return (long)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return (long)v; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,103 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Map; | ||||
| import java.util.HashMap; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class MapSchema extends Schema implements IMapSchema { | ||||
| 	private Schema keySchema; | ||||
| 	private Schema valueSchema; | ||||
|  | ||||
| 	public MapSchema(Schema keySchema, Schema valueSchema) { | ||||
| 		super("map"); | ||||
| 		this.keySchema = keySchema; | ||||
| 		this.valueSchema = valueSchema; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getFullName() { | ||||
| 		return "HashList<"+keySchema.getFullName()+", "+valueSchema.getFullName()+">"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "(map "+keySchema.getExpression()+" "+valueSchema.getExpression()+")"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Map) { | ||||
| 			Map<Object,Object> d = (Map<Object,Object>)obj; | ||||
| 			pk.packMap(d.size()); | ||||
| 			for(Map.Entry<Object,Object> e : d.entrySet()) { | ||||
| 				keySchema.pack(pk, e.getKey()); | ||||
| 				valueSchema.pack(pk, e.getValue()); | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Map) { | ||||
| 			Map<Object,Object> d = (Map<Object,Object>)obj; | ||||
| 			Map<Object,Object> m = new HashMap<Object,Object>(); | ||||
| 			for(Map.Entry<Object,Object> e : d.entrySet()) { | ||||
| 				m.put(keySchema.convert(e.getKey()), valueSchema.convert(e.getValue())); | ||||
| 			} | ||||
| 			return m; | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Schema getKeySchema() { | ||||
| 		return keySchema; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Schema getValueSchema() { | ||||
| 		return valueSchema; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public Object createFromMap(Object[] obj) { | ||||
| 		HashMap m = new HashMap(obj.length / 2); | ||||
| 		int i = 0; | ||||
| 		while(i < obj.length) { | ||||
| 			Object k = obj[i++]; | ||||
| 			Object v = obj[i++]; | ||||
| 			m.put(k, v); | ||||
| 		} | ||||
| 		return m; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,105 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.nio.ByteBuffer; | ||||
| import java.io.IOException; | ||||
| import java.io.UnsupportedEncodingException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class RawSchema extends Schema { | ||||
| 	public RawSchema() { | ||||
| 		super("raw"); | ||||
| 	} | ||||
|  | ||||
| 	public String getFullName() { | ||||
| 		return "byte[]"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		// FIXME instanceof GenericObject | ||||
| 		if(obj instanceof byte[]) { | ||||
| 			byte[] d = (byte[])obj; | ||||
| 			pk.packRaw(d.length); | ||||
| 			pk.packRawBody(d); | ||||
|  | ||||
| 		} else if(obj instanceof ByteBuffer) { | ||||
| 			ByteBuffer d = (ByteBuffer)obj; | ||||
| 			if(!d.hasArray()) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
| 			pk.packRaw(d.capacity()); | ||||
| 			pk.packRawBody(d.array(), d.position(), d.capacity()); | ||||
|  | ||||
| 		} else if(obj instanceof String) { | ||||
| 			try { | ||||
| 				byte[] d = ((String)obj).getBytes("UTF-8"); | ||||
| 				pk.packRaw(d.length); | ||||
| 				pk.packRawBody(d); | ||||
| 			} catch (UnsupportedEncodingException e) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		// FIXME instanceof GenericObject | ||||
| 		if(obj instanceof byte[]) { | ||||
| 			// FIXME copy? | ||||
| 			//byte[] d = (byte[])obj; | ||||
| 			//byte[] v = new byte[d.length]; | ||||
| 			//System.arraycopy(d, 0, v, 0, d.length); | ||||
| 			//return v; | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof ByteBuffer) { | ||||
| 			ByteBuffer d = (ByteBuffer)obj; | ||||
| 			byte[] v = new byte[d.capacity()]; | ||||
| 			int pos = d.position(); | ||||
| 			d.get(v); | ||||
| 			d.position(pos); | ||||
| 			return v; | ||||
|  | ||||
| 		} else if(obj instanceof String) { | ||||
| 			try { | ||||
| 				return ((String)obj).getBytes("UTF-8"); | ||||
| 			} catch (UnsupportedEncodingException e) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromRaw(byte[] b, int offset, int length) { | ||||
| 		byte[] d = new byte[length]; | ||||
| 		System.arraycopy(b, offset, d, 0, length); | ||||
| 		return d; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,64 +0,0 @@ | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.lang.reflect.*; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| // FIXME | ||||
| public abstract class ReflectionClassSchema extends ClassSchema { | ||||
| 	private Constructor constructorCache; | ||||
|  | ||||
| 	public ReflectionClassSchema(String name, List<FieldSchema> fields, String namespace, List<String> imports) { | ||||
| 		super(name, namespace, imports, fields); | ||||
| 	} | ||||
|  | ||||
| 	/* | ||||
| 	Schema getElementSchema(int index) | ||||
| 	{ | ||||
| 		// FIXME check index < fields.length | ||||
| 		fields[index].getSchema(); | ||||
| 	} | ||||
|  | ||||
| 	Object createFromArray(Object[] obj) | ||||
| 	{ | ||||
| 		Object o = newInstance(); | ||||
| 		((MessageConvertable)o).messageConvert(obj); | ||||
| 		return o; | ||||
| 	} | ||||
|  | ||||
| 	Object newInstance() | ||||
| 	{ | ||||
| 		if(constructorCache == null) { | ||||
| 			cacheConstructor(); | ||||
| 		} | ||||
| 		try { | ||||
| 			return constructorCache.newInstance((Object[])null); | ||||
| 		} catch (InvocationTargetException e) { | ||||
| 			throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 		} catch (InstantiationException e) { | ||||
| 			throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 		} catch (IllegalAccessException e) { | ||||
| 			throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private void cacheConstructor() | ||||
| 	{ | ||||
| 		try { | ||||
| 			Class c = Class.forName(fqdn); | ||||
| 			int index = 0; | ||||
| 			for(SpecificFieldSchema f : fields) { | ||||
| 				f.cacheField(c, index++); | ||||
| 			} | ||||
| 			constructorCache = c.getDeclaredConstructor((Class[])null); | ||||
| 			constructorCache.setAccessible(true); | ||||
| 		} catch(ClassNotFoundException e) { | ||||
| 			throw new RuntimeException("class not found: "+fqdn); | ||||
| 		} catch (NoSuchMethodException e) { | ||||
| 			throw new RuntimeException("class not found: "+fqdn+": "+e.getMessage()); | ||||
| 		} | ||||
| 	} | ||||
| 	*/ | ||||
| } | ||||
|  | ||||
| @@ -1,254 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Stack; | ||||
| import java.util.regex.Pattern; | ||||
| import java.util.regex.Matcher; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| // FIXME exception class | ||||
|  | ||||
| public class SSchemaParser { | ||||
| 	public static Schema parse(String source) { | ||||
| 		return new SSchemaParser(false).run(source); | ||||
| 	} | ||||
|  | ||||
| 	public static Schema load(String source) { | ||||
| 		return new SSchemaParser(true).run(source); | ||||
| 	} | ||||
|  | ||||
| 	private static abstract class SExp { | ||||
| 		boolean isAtom() { return false; } | ||||
| 		public String getAtom() { return null; } | ||||
|  | ||||
| 		boolean isTuple() { return false; } | ||||
| 		public SExp getTuple(int i) { return null; } | ||||
| 		public int size() { return 0; } | ||||
| 		public boolean empty() { return size() == 0; } | ||||
| 		Iterator<SExp> iterator(int offset) { return null; } | ||||
| 	} | ||||
|  | ||||
| 	private static class SAtom extends SExp { | ||||
| 		private String atom; | ||||
|  | ||||
| 		SAtom(String atom) { this.atom = atom; } | ||||
|  | ||||
| 		boolean isAtom() { return true; } | ||||
| 		public String getAtom() { return atom; } | ||||
|  | ||||
| 		public String toString() { return atom; } | ||||
| 	} | ||||
|  | ||||
| 	private static class STuple extends SExp { | ||||
| 		private List<SExp> tuple; | ||||
|  | ||||
| 		STuple() { this.tuple = new ArrayList<SExp>(); } | ||||
|  | ||||
| 		public void add(SExp e) { tuple.add(e); } | ||||
|  | ||||
| 		boolean isTuple() { return true; } | ||||
| 		public SExp getTuple(int i) { return tuple.get(i); } | ||||
| 		public int size() { return tuple.size(); } | ||||
|  | ||||
| 		Iterator<SExp> iterator(int skip) { | ||||
| 			Iterator<SExp> i = tuple.iterator(); | ||||
| 			for(int s=0; s < skip; ++s) { i.next(); } | ||||
| 			return i; | ||||
| 		} | ||||
|  | ||||
| 		public String toString() { | ||||
| 			if(tuple.isEmpty()) { return "()"; } | ||||
| 			Iterator<SExp> i = tuple.iterator(); | ||||
| 			StringBuffer o = new StringBuffer(); | ||||
| 			o.append("(").append(i.next()); | ||||
| 			while(i.hasNext()) { o.append(" ").append(i.next()); } | ||||
| 			o.append(")"); | ||||
| 			return o.toString(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	boolean specificClass; | ||||
|  | ||||
| 	private SSchemaParser(boolean specificClass) { | ||||
| 		this.specificClass = specificClass; | ||||
| 	} | ||||
|  | ||||
| 	private static Pattern pattern = Pattern.compile( | ||||
| 			"(?:\\s+)|([\\(\\)]|[\\d\\w\\.]+)"); | ||||
|  | ||||
| 	private Schema run(String source) { | ||||
| 		Matcher m = pattern.matcher(source); | ||||
|  | ||||
| 		Stack<STuple> stack = new Stack<STuple>(); | ||||
| 		String token; | ||||
|  | ||||
| 		while(true) { | ||||
| 			while(true) { | ||||
| 				if(!m.find()) { throw new RuntimeException("unexpected end of file"); } | ||||
| 				token = m.group(1); | ||||
| 				if(token != null) { break; } | ||||
| 			} | ||||
|  | ||||
| 			if(token.equals("(")) { | ||||
| 				stack.push(new STuple()); | ||||
| 			} else if(token.equals(")")) { | ||||
| 				STuple top = stack.pop(); | ||||
| 				if(stack.empty()) { | ||||
| 					stack.push(top); | ||||
| 					break; | ||||
| 				} | ||||
| 				stack.peek().add(top); | ||||
| 			} else { | ||||
| 				if(stack.empty()) { | ||||
| 					throw new RuntimeException("unexpected token '"+token+"'"); | ||||
| 				} | ||||
| 				stack.peek().add(new SAtom(token)); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		while(true) { | ||||
| 			if(!m.find()) { break; } | ||||
| 			token = m.group(1); | ||||
| 			if(token != null) { throw new RuntimeException("unexpected token '"+token+"'"); } | ||||
| 		} | ||||
|  | ||||
| 		return readType( stack.pop() ); | ||||
| 	} | ||||
|  | ||||
| 	private Schema readType(SExp exp) { | ||||
| 		if(exp.isAtom()) { | ||||
| 			String type = exp.getAtom(); | ||||
| 			if(type.equals("string")) { | ||||
| 				return new StringSchema(); | ||||
| 			} else if(type.equals("raw")) { | ||||
| 				return new RawSchema(); | ||||
| 			} else if(type.equals("byte")) { | ||||
| 				return new ByteSchema(); | ||||
| 			} else if(type.equals("short")) { | ||||
| 				return new ShortSchema(); | ||||
| 			} else if(type.equals("int")) { | ||||
| 				return new IntSchema(); | ||||
| 			} else if(type.equals("long")) { | ||||
| 				return new LongSchema(); | ||||
| 			} else if(type.equals("float")) { | ||||
| 				return new FloatSchema(); | ||||
| 			} else if(type.equals("double")) { | ||||
| 				return new DoubleSchema(); | ||||
| 			} else if(type.equals("object")) { | ||||
| 				return new GenericSchema(); | ||||
| 			} else { | ||||
| 				throw new RuntimeException("byte, short, int, long, float, double, raw, string or object is expected but got '"+type+"': "+exp); | ||||
| 			} | ||||
| 		} else { | ||||
| 			String type = exp.getTuple(0).getAtom(); | ||||
| 			if(type.equals("class")) { | ||||
| 				return parseClass(exp); | ||||
| 			} else if(type.equals("array")) { | ||||
| 				return parseArray(exp); | ||||
| 			} else if(type.equals("map")) { | ||||
| 				return parseMap(exp); | ||||
| 			} else { | ||||
| 				throw new RuntimeException("class, array or map is expected but got '"+type+"': "+exp); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private ClassSchema parseClass(SExp exp) { | ||||
| 		if(exp.size() < 3 || !exp.getTuple(1).isAtom()) { | ||||
| 			throw new RuntimeException("class is (class NAME CLASS_BODY): "+exp); | ||||
| 		} | ||||
|  | ||||
| 		String namespace = null; | ||||
| 		List<String> imports = new ArrayList<String>(); | ||||
| 		String name = exp.getTuple(1).getAtom(); | ||||
| 		List<FieldSchema> fields = new ArrayList<FieldSchema>(); | ||||
|  | ||||
| 		for(Iterator<SExp> i=exp.iterator(2); i.hasNext();) { | ||||
| 			SExp subexp = i.next(); | ||||
| 			if(!subexp.isTuple() || subexp.empty() || !subexp.getTuple(0).isAtom()) { | ||||
| 				throw new RuntimeException("field, package or import is expected: "+subexp); | ||||
| 			} | ||||
| 			String type = subexp.getTuple(0).getAtom(); | ||||
| 			if(type.equals("field")) { | ||||
| 				fields.add( parseField(subexp) ); | ||||
| 			} else if(type.equals("package")) { | ||||
| 				if(namespace != null) { | ||||
| 					throw new RuntimeException("duplicated package definition: "+subexp); | ||||
| 				} | ||||
| 				namespace = parseNamespace(subexp); | ||||
| 			} else if(type.equals("import")) { | ||||
| 				imports.add( parseImport(subexp) ); | ||||
| 			} else { | ||||
| 				throw new RuntimeException("field, package or import is expected but got '"+type+"': "+subexp); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if(specificClass) { | ||||
| 			return new SpecificClassSchema(name, namespace, imports, fields); | ||||
| 		} else { | ||||
| 			return new GenericClassSchema(name, namespace, imports, fields); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private ArraySchema parseArray(SExp exp) { | ||||
| 		if(exp.size() != 2) { | ||||
| 			throw new RuntimeException("array is (array ELEMENT_TYPE): "+exp); | ||||
| 		} | ||||
| 		Schema elementType = readType(exp.getTuple(1)); | ||||
| 		return new ArraySchema(elementType); | ||||
| 	} | ||||
|  | ||||
| 	private MapSchema parseMap(SExp exp) { | ||||
| 		if(exp.size() != 3 || !exp.getTuple(1).isAtom()) { | ||||
| 			throw new RuntimeException("map is (map KEY_TYPE VALUE_TYPE): "+exp); | ||||
| 		} | ||||
| 		Schema keyType   = readType(exp.getTuple(1)); | ||||
| 		Schema valueType = readType(exp.getTuple(2)); | ||||
| 		return new MapSchema(keyType, valueType); | ||||
| 	} | ||||
|  | ||||
| 	private String parseNamespace(SExp exp) { | ||||
| 		if(exp.size() != 2 || !exp.getTuple(1).isAtom()) { | ||||
| 			throw new RuntimeException("package is (package NAME): "+exp); | ||||
| 		} | ||||
| 		String name = exp.getTuple(1).getAtom(); | ||||
| 		return name; | ||||
| 	} | ||||
|  | ||||
| 	private String parseImport(SExp exp) { | ||||
| 		if(exp.size() != 2 || !exp.getTuple(1).isAtom()) { | ||||
| 			throw new RuntimeException("import is (import NAME): "+exp); | ||||
| 		} | ||||
| 		String name = exp.getTuple(1).getAtom(); | ||||
| 		return name; | ||||
| 	} | ||||
|  | ||||
| 	private FieldSchema parseField(SExp exp) { | ||||
| 		if(exp.size() != 3 || !exp.getTuple(1).isAtom()) { | ||||
| 			throw new RuntimeException("field is (field NAME TYPE): "+exp); | ||||
| 		} | ||||
| 		String name = exp.getTuple(1).getAtom(); | ||||
| 		Schema type = readType(exp.getTuple(2)); | ||||
| 		return new FieldSchema(name, type); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,89 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class ShortSchema extends Schema { | ||||
| 	public ShortSchema() { | ||||
| 		super("Short"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getExpression() { | ||||
| 		return "short"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj instanceof Number) { | ||||
| 			pk.packShort( ((Number)obj).shortValue() ); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Short) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof Number) { | ||||
| 			return ((Number)obj).shortValue(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromByte(byte v) { | ||||
| 		return (short)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromShort(short v) { | ||||
| 		return (short)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromInt(int v) { | ||||
| 		return (short)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromLong(long v) { | ||||
| 		return (short)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromFloat(float v) { | ||||
| 		return (short)v; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromDouble(double v) { | ||||
| 		return (short)v; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,122 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.lang.reflect.*; | ||||
| import java.io.IOException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class SpecificClassSchema extends ClassSchema { | ||||
| 	private Class classCache; | ||||
| 	private Method factoryCache; | ||||
| 	private Constructor constructorCache; | ||||
|  | ||||
| 	public SpecificClassSchema( | ||||
| 			String name, String namespace, | ||||
| 			List<String> imports, List<FieldSchema> fields) { | ||||
| 		super(name, namespace, imports, fields); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		if(obj == null) { | ||||
| 			pk.packNil(); | ||||
| 			return; | ||||
| 		} | ||||
| 		if(classCache == null) { | ||||
| 			cacheFactory(); | ||||
| 		} | ||||
| 		if(classCache.isInstance(obj)) { | ||||
| 			((MessagePackable)obj).messagePack(pk); | ||||
| 		} else { | ||||
| 			// FIXME Map<String,Object> | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		if(obj instanceof Collection) { | ||||
| 			if(constructorCache == null) { | ||||
| 				cacheConstructor(); | ||||
| 			} | ||||
| 			try { | ||||
| 				MessageConvertable o = (MessageConvertable)constructorCache.newInstance((Object[])null); | ||||
| 				o.messageConvert(obj); | ||||
| 				return o; | ||||
| 			} catch (InvocationTargetException e) { | ||||
| 				throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 			} catch (InstantiationException e) { | ||||
| 				throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 			} catch (IllegalAccessException e) { | ||||
| 				throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 			} | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public Schema getElementSchema(int index) { | ||||
| 		// FIXME check index < fields.length | ||||
| 		return fields[index].getSchema(); | ||||
| 	} | ||||
|  | ||||
| 	public Object createFromArray(Object[] obj) { | ||||
| 		if(factoryCache == null) { | ||||
| 			cacheFactory(); | ||||
| 		} | ||||
| 		try { | ||||
| 			return factoryCache.invoke(null, new Object[]{obj}); | ||||
| 		} catch (InvocationTargetException e) { | ||||
| 			throw new RuntimeException("can't instantiate "+fqdn+": "+e.getCause()); | ||||
| 		} catch (IllegalAccessException e) { | ||||
| 			throw new RuntimeException("can't instantiate "+fqdn+": "+e.getMessage()); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	private void cacheFactory() { | ||||
| 		try { | ||||
| 			classCache = Class.forName(fqdn); | ||||
| 			factoryCache = classCache.getDeclaredMethod("createFromMessage", new Class[]{Object[].class}); | ||||
| 			factoryCache.setAccessible(true); | ||||
| 		} catch(ClassNotFoundException e) { | ||||
| 			throw new RuntimeException("class not found: "+fqdn); | ||||
| 		} catch (NoSuchMethodException e) { | ||||
| 			throw new RuntimeException("class not found: "+fqdn+": "+e.getMessage()); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@SuppressWarnings("unchecked") | ||||
| 	private void cacheConstructor() { | ||||
| 		try { | ||||
| 			classCache = Class.forName(fqdn); | ||||
| 			constructorCache = classCache.getDeclaredConstructor((Class[])null); | ||||
| 			constructorCache.setAccessible(true); | ||||
| 		} catch(ClassNotFoundException e) { | ||||
| 			throw new RuntimeException("class not found: "+fqdn); | ||||
| 		} catch (NoSuchMethodException e) { | ||||
| 			throw new RuntimeException("class not found: "+fqdn+": "+e.getMessage()); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,111 +0,0 @@ | ||||
| // | ||||
| // MessagePack for Java | ||||
| // | ||||
| // Copyright (C) 2009-2010 FURUHASHI Sadayuki | ||||
| // | ||||
| //    Licensed 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. | ||||
| // | ||||
| package org.msgpack.schema; | ||||
|  | ||||
| import java.nio.ByteBuffer; | ||||
| import java.io.IOException; | ||||
| import java.io.UnsupportedEncodingException; | ||||
| import org.msgpack.*; | ||||
|  | ||||
| public class StringSchema extends Schema { | ||||
| 	public StringSchema() { | ||||
| 		super("string"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getFullName() { | ||||
| 		return "String"; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void pack(Packer pk, Object obj) throws IOException { | ||||
| 		// FIXME instanceof GenericObject | ||||
| 		if(obj instanceof String) { | ||||
| 			try { | ||||
| 				byte[] d = ((String)obj).getBytes("UTF-8"); | ||||
| 				pk.packRaw(d.length); | ||||
| 				pk.packRawBody(d); | ||||
| 			} catch (UnsupportedEncodingException e) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj instanceof byte[]) { | ||||
| 			byte[] d = (byte[])obj; | ||||
| 			pk.packRaw(d.length); | ||||
| 			pk.packRawBody(d); | ||||
|  | ||||
| 		} else if(obj instanceof ByteBuffer) { | ||||
| 			ByteBuffer d = (ByteBuffer)obj; | ||||
| 			if(!d.hasArray()) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
| 			pk.packRaw(d.capacity()); | ||||
| 			pk.packRawBody(d.array(), d.position(), d.capacity()); | ||||
|  | ||||
| 		} else if(obj == null) { | ||||
| 			pk.packNil(); | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object convert(Object obj) throws MessageTypeException { | ||||
| 		// FIXME instanceof GenericObject | ||||
| 		if(obj instanceof String) { | ||||
| 			return obj; | ||||
|  | ||||
| 		} else if(obj instanceof byte[]) { | ||||
| 			try { | ||||
| 				return new String((byte[])obj, "UTF-8"); | ||||
| 			} catch (UnsupportedEncodingException e) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
|  | ||||
| 		} else if(obj instanceof ByteBuffer) { | ||||
| 			ByteBuffer d = (ByteBuffer)obj; | ||||
| 			try { | ||||
| 				if(d.hasArray()) { | ||||
| 					return new String(d.array(), d.position(), d.capacity(), "UTF-8"); | ||||
| 				} else { | ||||
| 					byte[] v = new byte[d.capacity()]; | ||||
| 					int pos = d.position(); | ||||
| 					d.get(v); | ||||
| 					d.position(pos); | ||||
| 					return new String(v, "UTF-8"); | ||||
| 				} | ||||
| 			} catch (UnsupportedEncodingException e) { | ||||
| 				throw MessageTypeException.invalidConvert(obj, this); | ||||
| 			} | ||||
|  | ||||
| 		} else { | ||||
| 			throw MessageTypeException.invalidConvert(obj, this); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object createFromRaw(byte[] b, int offset, int length) { | ||||
| 		try { | ||||
| 			return new String(b, offset, length, "UTF-8"); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new RuntimeException(e.getMessage()); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 advect
					advect