transactd 1.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (218) hide show
  1. checksums.yaml +7 -0
  2. data/BUILD_UNIX-JA +174 -0
  3. data/BUILD_WIN-JA +256 -0
  4. data/CMakeLists.txt +96 -0
  5. data/COPYING +339 -0
  6. data/README +406 -0
  7. data/README-JA +424 -0
  8. data/bin/common/tdclc_32_1_0.dll +0 -0
  9. data/bin/common/tdclc_64_1_0.dll +0 -0
  10. data/build/common/check_for_link_iconv.cmake +73 -0
  11. data/build/common/copyifgreater.cmd +30 -0
  12. data/build/common/copyifgreater.js +290 -0
  13. data/build/common/get_boost_libs.cmake +106 -0
  14. data/build/common/get_ruby_path.cmake +115 -0
  15. data/build/common/options.cmake +127 -0
  16. data/build/common/smart_install.cmake +263 -0
  17. data/build/common/system.cmake +122 -0
  18. data/build/common/transactd.rc.in +52 -0
  19. data/build/common/transactd_cl_common.cmake +101 -0
  20. data/build/common/transactd_cl_output.cmake +93 -0
  21. data/build/common/transactd_common.cmake +237 -0
  22. data/build/common/transactd_required.cmake +28 -0
  23. data/build/swig/ruby/generate.cmake.in +35 -0
  24. data/build/swig/ruby/generate.cmd.in +19 -0
  25. data/build/swig/ruby/ruby.swg +101 -0
  26. data/build/swig/tdcl.i +188 -0
  27. data/build/tdclc/BUILDNUMBER.txt +1 -0
  28. data/build/tdclc/CMakeLists.txt +170 -0
  29. data/build/tdclc/tdclc_32.cbproj +181 -0
  30. data/build/tdclc/tdclc_64.cbproj +205 -0
  31. data/build/tdclcpp/BUILDNUMBER.txt +1 -0
  32. data/build/tdclcpp/CMakeLists.txt +142 -0
  33. data/build/tdclcpp/tdclcpp_bcb_32.cbproj +239 -0
  34. data/build/tdclcpp/tdclcpp_bcb_64.cbproj +304 -0
  35. data/build/tdclrb/BUILDNUMBER.txt +1 -0
  36. data/build/tdclrb/CMakeLists.txt +258 -0
  37. data/build/tdclrb/GEM_VERSION +3 -0
  38. data/build/tdclrb/bldgem/extconf.rb +123 -0
  39. data/build/tdclrb/gem/INSTALLLOG.win32 +9 -0
  40. data/build/tdclrb/gem/Makefile.win32-VS +65 -0
  41. data/build/tdclrb/gem/Makefile.win32-prebuilt +48 -0
  42. data/build/tdclrb/gem/detect.rb +31 -0
  43. data/build/tdclrb/gem/helper.rb +113 -0
  44. data/build/tdclrb/gem/transactd.rb +22 -0
  45. data/build/tdclrb/gem_output.cmake +44 -0
  46. data/source/bzs/db/IBlobBuffer.h +51 -0
  47. data/source/bzs/db/blobBuffer.h +177 -0
  48. data/source/bzs/db/blobStructs.h +85 -0
  49. data/source/bzs/db/engine/mysql/IReadRecords.h +52 -0
  50. data/source/bzs/db/engine/mysql/bookmark.h +195 -0
  51. data/source/bzs/db/engine/mysql/database.cpp +1882 -0
  52. data/source/bzs/db/engine/mysql/database.h +465 -0
  53. data/source/bzs/db/engine/mysql/dbManager.cpp +303 -0
  54. data/source/bzs/db/engine/mysql/dbManager.h +143 -0
  55. data/source/bzs/db/engine/mysql/errorMessage.cpp +75 -0
  56. data/source/bzs/db/engine/mysql/errorMessage.h +43 -0
  57. data/source/bzs/db/engine/mysql/fieldAccess.h +158 -0
  58. data/source/bzs/db/engine/mysql/mydebuglog.cpp +349 -0
  59. data/source/bzs/db/engine/mysql/mydebuglog.h +89 -0
  60. data/source/bzs/db/engine/mysql/mysqlInternal.h +171 -0
  61. data/source/bzs/db/engine/mysql/mysqlThd.cpp +169 -0
  62. data/source/bzs/db/engine/mysql/mysqlThd.h +35 -0
  63. data/source/bzs/db/engine/mysql/percentageKey.h +260 -0
  64. data/source/bzs/db/protocol/ICommandExecuter.h +49 -0
  65. data/source/bzs/db/protocol/hs/hsCommandExecuter.cpp +689 -0
  66. data/source/bzs/db/protocol/hs/hsCommandExecuter.h +228 -0
  67. data/source/bzs/db/protocol/tdap/btrDate.cpp +437 -0
  68. data/source/bzs/db/protocol/tdap/btrDate.h +227 -0
  69. data/source/bzs/db/protocol/tdap/client/bulkInsert.h +127 -0
  70. data/source/bzs/db/protocol/tdap/client/client.cpp +106 -0
  71. data/source/bzs/db/protocol/tdap/client/client.h +292 -0
  72. data/source/bzs/db/protocol/tdap/client/connMgr.cpp +144 -0
  73. data/source/bzs/db/protocol/tdap/client/connMgr.h +82 -0
  74. data/source/bzs/db/protocol/tdap/client/database.cpp +863 -0
  75. data/source/bzs/db/protocol/tdap/client/database.h +118 -0
  76. data/source/bzs/db/protocol/tdap/client/databaseFactory.cpp +100 -0
  77. data/source/bzs/db/protocol/tdap/client/dbDef.cpp +1640 -0
  78. data/source/bzs/db/protocol/tdap/client/dbDef.h +135 -0
  79. data/source/bzs/db/protocol/tdap/client/dllmain.cpp +434 -0
  80. data/source/bzs/db/protocol/tdap/client/errorMessage.cpp +92 -0
  81. data/source/bzs/db/protocol/tdap/client/errorMessage_ja.cpp +98 -0
  82. data/source/bzs/db/protocol/tdap/client/fieldDDF.cpp +174 -0
  83. data/source/bzs/db/protocol/tdap/client/fieldDDF.h +91 -0
  84. data/source/bzs/db/protocol/tdap/client/fileDDF.cpp +140 -0
  85. data/source/bzs/db/protocol/tdap/client/fileDDF.h +86 -0
  86. data/source/bzs/db/protocol/tdap/client/filter.cpp +527 -0
  87. data/source/bzs/db/protocol/tdap/client/filter.h +154 -0
  88. data/source/bzs/db/protocol/tdap/client/indexDDF.cpp +137 -0
  89. data/source/bzs/db/protocol/tdap/client/indexDDF.h +84 -0
  90. data/source/bzs/db/protocol/tdap/client/nsDatabase.cpp +724 -0
  91. data/source/bzs/db/protocol/tdap/client/nsDatabase.h +123 -0
  92. data/source/bzs/db/protocol/tdap/client/nsTable.cpp +899 -0
  93. data/source/bzs/db/protocol/tdap/client/nsTable.h +199 -0
  94. data/source/bzs/db/protocol/tdap/client/request.h +198 -0
  95. data/source/bzs/db/protocol/tdap/client/sharedData.cpp +58 -0
  96. data/source/bzs/db/protocol/tdap/client/sharedData.h +56 -0
  97. data/source/bzs/db/protocol/tdap/client/sqlBuilder.cpp +574 -0
  98. data/source/bzs/db/protocol/tdap/client/sqlBuilder.h +53 -0
  99. data/source/bzs/db/protocol/tdap/client/stringConverter.h +627 -0
  100. data/source/bzs/db/protocol/tdap/client/table.cpp +2613 -0
  101. data/source/bzs/db/protocol/tdap/client/table.h +221 -0
  102. data/source/bzs/db/protocol/tdap/client/trdboostapi.h +1096 -0
  103. data/source/bzs/db/protocol/tdap/client/trdboostapiInternal.h +179 -0
  104. data/source/bzs/db/protocol/tdap/client/trdclcppautolink.h +40 -0
  105. data/source/bzs/db/protocol/tdap/client/trnsctcl.def +11 -0
  106. data/source/bzs/db/protocol/tdap/myDateTime.cpp +500 -0
  107. data/source/bzs/db/protocol/tdap/mysql/characterset.cpp +184 -0
  108. data/source/bzs/db/protocol/tdap/mysql/characterset.h +60 -0
  109. data/source/bzs/db/protocol/tdap/mysql/databaseSchema.cpp +284 -0
  110. data/source/bzs/db/protocol/tdap/mysql/databaseSchema.h +53 -0
  111. data/source/bzs/db/protocol/tdap/mysql/debuglog.cpp +383 -0
  112. data/source/bzs/db/protocol/tdap/mysql/debuglog.h +106 -0
  113. data/source/bzs/db/protocol/tdap/mysql/recordsetReader.h +680 -0
  114. data/source/bzs/db/protocol/tdap/mysql/request.h +202 -0
  115. data/source/bzs/db/protocol/tdap/mysql/tdapCommandExecuter.cpp +1020 -0
  116. data/source/bzs/db/protocol/tdap/mysql/tdapCommandExecuter.h +141 -0
  117. data/source/bzs/db/protocol/tdap/tdapRequest.h +190 -0
  118. data/source/bzs/db/protocol/tdap/tdapSchema.cpp +295 -0
  119. data/source/bzs/db/protocol/tdap/tdapSchema.h +558 -0
  120. data/source/bzs/db/protocol/tdap/tdapcapi.h +423 -0
  121. data/source/bzs/db/transactd/appBuilderImple.h +55 -0
  122. data/source/bzs/db/transactd/appModule.cpp +183 -0
  123. data/source/bzs/db/transactd/appModule.h +80 -0
  124. data/source/bzs/db/transactd/connManager.cpp +201 -0
  125. data/source/bzs/db/transactd/connManager.h +60 -0
  126. data/source/bzs/db/transactd/connectionRecord.h +69 -0
  127. data/source/bzs/db/transactd/transactd.cpp +325 -0
  128. data/source/bzs/env/compiler.h +135 -0
  129. data/source/bzs/env/crosscompile.cpp +130 -0
  130. data/source/bzs/env/crosscompile.h +150 -0
  131. data/source/bzs/env/fileopen.h +36 -0
  132. data/source/bzs/env/mbcswchrLinux.cpp +40 -0
  133. data/source/bzs/env/mbcswchrLinux.h +88 -0
  134. data/source/bzs/env/tstring.h +183 -0
  135. data/source/bzs/example/changeSchema.cpp +117 -0
  136. data/source/bzs/example/changeSchema_c.cpp +78 -0
  137. data/source/bzs/example/connection_pool_c.cpp +171 -0
  138. data/source/bzs/example/createDatabase.cpp +305 -0
  139. data/source/bzs/example/createDatabase_c.cpp +202 -0
  140. data/source/bzs/example/deleteRecords.cpp +87 -0
  141. data/source/bzs/example/deleteRecords_c.cpp +57 -0
  142. data/source/bzs/example/dropDatabase.cpp +59 -0
  143. data/source/bzs/example/dropDatabase_c.cpp +34 -0
  144. data/source/bzs/example/insertRecords.cpp +212 -0
  145. data/source/bzs/example/insertRecords_c.cpp +153 -0
  146. data/source/bzs/example/readRecords.cpp +141 -0
  147. data/source/bzs/example/readRecords_c.cpp +107 -0
  148. data/source/bzs/example/updateRecords.cpp +99 -0
  149. data/source/bzs/example/updateRecords_c.cpp +71 -0
  150. data/source/bzs/example/update_with_transaction.cpp +104 -0
  151. data/source/bzs/example/update_with_transaction_c.cpp +80 -0
  152. data/source/bzs/netsvc/client/tcpClient.cpp +226 -0
  153. data/source/bzs/netsvc/client/tcpClient.h +489 -0
  154. data/source/bzs/netsvc/server/IAppModule.h +94 -0
  155. data/source/bzs/netsvc/server/iserver.h +65 -0
  156. data/source/bzs/netsvc/server/serverCpt.cpp +522 -0
  157. data/source/bzs/netsvc/server/serverCpt.h +88 -0
  158. data/source/bzs/netsvc/server/serverPipe.cpp +705 -0
  159. data/source/bzs/netsvc/server/serverPipe.h +96 -0
  160. data/source/bzs/netsvc/server/serverTpool.cpp +416 -0
  161. data/source/bzs/netsvc/server/serverTpool.h +84 -0
  162. data/source/bzs/rtl/benchmark.cpp +96 -0
  163. data/source/bzs/rtl/benchmark.h +65 -0
  164. data/source/bzs/rtl/datetime.cpp +375 -0
  165. data/source/bzs/rtl/datetime.h +53 -0
  166. data/source/bzs/rtl/debuglog.cpp +106 -0
  167. data/source/bzs/rtl/debuglog.h +97 -0
  168. data/source/bzs/rtl/exception.h +116 -0
  169. data/source/bzs/rtl/stl_uty.cpp +35 -0
  170. data/source/bzs/rtl/stl_uty.h +29 -0
  171. data/source/bzs/rtl/stringBuffers.cpp +101 -0
  172. data/source/bzs/rtl/stringBuffers.h +58 -0
  173. data/source/bzs/rtl/strtrim.cpp +135 -0
  174. data/source/bzs/rtl/strtrim.h +46 -0
  175. data/source/bzs/test/tdclatl/bench_tdclatl.js +445 -0
  176. data/source/bzs/test/tdclrb/bench_tdclcpp.rb +375 -0
  177. data/source/bzs/test/tdclrb/prepare.rb +226 -0
  178. data/source/bzs/test/tdclrb/transactd_datetime_spec.rb +172 -0
  179. data/source/bzs/test/tdclrb/transactd_kanjischema_spec.rb +208 -0
  180. data/source/bzs/test/tdclrb/transactd_spec.rb +1536 -0
  181. data/source/bzs/test/transactdBench/transactdBench.cpp +430 -0
  182. data/source/bzs/test/transactdBench/transactdBench2.cpp +342 -0
  183. data/source/bzs/test/trdclengn/test_trdclengn.cpp +2030 -0
  184. data/source/global/tdclatl/Database.cpp +503 -0
  185. data/source/global/tdclatl/Database.h +139 -0
  186. data/source/global/tdclatl/DbDef.cpp +242 -0
  187. data/source/global/tdclatl/DbDef.h +79 -0
  188. data/source/global/tdclatl/Field.cpp +92 -0
  189. data/source/global/tdclatl/Field.h +59 -0
  190. data/source/global/tdclatl/FieldDef.cpp +238 -0
  191. data/source/global/tdclatl/FieldDef.h +87 -0
  192. data/source/global/tdclatl/Flags.cpp +111 -0
  193. data/source/global/tdclatl/Flags.h +55 -0
  194. data/source/global/tdclatl/KeyDef.cpp +51 -0
  195. data/source/global/tdclatl/KeyDef.h +55 -0
  196. data/source/global/tdclatl/KeySegment.cpp +55 -0
  197. data/source/global/tdclatl/Table.cpp +600 -0
  198. data/source/global/tdclatl/Table.h +144 -0
  199. data/source/global/tdclatl/TableDef.cpp +291 -0
  200. data/source/global/tdclatl/TableDef.h +86 -0
  201. data/source/global/tdclatl/TdVersion.cpp +74 -0
  202. data/source/global/tdclatl/TdVersion.h +54 -0
  203. data/source/global/tdclatl/_IDatabaseEvents_CP.h +113 -0
  204. data/source/global/tdclatl/dllmain.cpp +30 -0
  205. data/source/global/tdclatl/dllmain.h +27 -0
  206. data/source/global/tdclatl/keySegment.h +57 -0
  207. data/source/global/tdclatl/resource.h +0 -0
  208. data/source/global/tdclatl/stdafx.cpp +2 -0
  209. data/source/global/tdclatl/stdafx.h +25 -0
  210. data/source/global/tdclatl/targetver.h +4 -0
  211. data/source/global/tdclatl/tdclatl.cpp +68 -0
  212. data/source/global/tdclatl/tdclatl.def +10 -0
  213. data/source/global/tdclatl/tdclatl.idl +1035 -0
  214. data/source/linux/charsetConvert.h +112 -0
  215. data/source/linux/linuxTypes.h +33 -0
  216. data/source/linux/tchar.h +428 -0
  217. data/transactd.gemspec +97 -0
  218. metadata +267 -0
@@ -0,0 +1,94 @@
1
+ #ifndef BZS_NETSVC_IAPPMODULE_H
2
+ #define BZS_NETSVC_IAPPMODULE_H
3
+ /*=================================================================
4
+ Copyright (C) 2012 2013 BizStation Corp All rights reserved.
5
+
6
+ This program is free software; you can redistribute it and/or
7
+ modify it under the terms of the GNU General Public License
8
+ as published by the Free Software Foundation; either version 2
9
+ of the License, or (at your option) any later version.
10
+
11
+ This program is distributed in the hope that it will be useful,
12
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
13
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
+ GNU General Public License for more details.
15
+
16
+ You should have received a copy of the GNU General Public License
17
+ along with this program; if not, write to the Free Software
18
+ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19
+ 02111-1307, USA.
20
+ =================================================================*/
21
+
22
+ #include <vector>
23
+ #include <boost/noncopyable.hpp>
24
+ #include <boost/asio.hpp>
25
+ #include <boost/thread/mutex.hpp>
26
+
27
+ /** Application framework interface
28
+ * IModuleBuilder: module builder.
29
+ * IAppModule :The module created for every session.
30
+ * This receives received data and returns a result.
31
+ * It will be done, if these interfaces are implement and a server's constructor is passed.
32
+ */
33
+
34
+ namespace bzs
35
+ {
36
+ namespace netsvc
37
+ {
38
+ namespace server
39
+ {
40
+
41
+ class iconnection;
42
+
43
+ #define EXECUTE_RESULT_SUCCESS 0
44
+ #define EXECUTE_RESULT_QUIT 1
45
+ #define EXECUTE_RESULT_FORCSE_ASYNC 2
46
+ #define EXECUTE_RESULT_FORCSE_SYNC 3
47
+
48
+ typedef std::vector<boost::asio::const_buffer> buffers;
49
+
50
+ class IAppModule
51
+ {
52
+ public:
53
+
54
+ virtual ~IAppModule() {};
55
+ virtual int execute(char* result, size_t& size, buffers* optionalData) = 0;
56
+ virtual size_t onRead(const char* data, size_t size, bool& complete)=0;
57
+ virtual size_t onAccept(char* message, size_t bufsize)=0;
58
+ virtual void reset() = 0;
59
+ virtual bool isShutDown() = 0;
60
+ virtual bool checkHost(const char* hostCheckname) = 0;
61
+ virtual void cleanup() = 0;
62
+ virtual boost::mutex& mutex()const = 0;
63
+
64
+
65
+ };
66
+
67
+ //Defines at Implementing of IAppModule , two variables below.
68
+ extern boost::mutex modulesMutex;
69
+ extern std::vector<IAppModule*> modules;
70
+
71
+
72
+
73
+ #define SERVER_TYPE_TPOOL 1
74
+ #define SERVER_TYPE_CPT 0
75
+
76
+ /**
77
+ * used by worker thread
78
+ * use with dynamic_cast for your application
79
+ */
80
+ class IAppModuleBuilder
81
+ {
82
+ public:
83
+ virtual ~IAppModuleBuilder(){};
84
+ virtual IAppModule* createSessionModule(const boost::asio::ip::tcp::endpoint& endpoint
85
+ , iconnection* connection
86
+ , bool tpool) = 0;
87
+ };
88
+
89
+
90
+
91
+ }//namespace server
92
+ }//namespace netsvc
93
+ }//namespace bzs
94
+ #endif//BZS_NETSVC_IAPPMODULE_H
@@ -0,0 +1,65 @@
1
+ /*=================================================================
2
+ Copyright (C) 2013 BizStation Corp All rights reserved.
3
+
4
+ This program is free software; you can redistribute it and/or
5
+ modify it under the terms of the GNU General Public License
6
+ as published by the Free Software Foundation; either version 2
7
+ of the License, or (at your option) any later version.
8
+
9
+ This program is distributed in the hope that it will be useful,
10
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
11
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
+ GNU General Public License for more details.
13
+
14
+ You should have received a copy of the GNU General Public License
15
+ along with this program; if not, write to the Free Software
16
+ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
17
+ 02111-1307, USA.
18
+ =================================================================*/
19
+ #ifndef BZS_NETSVC_SERVER_ISERVER_H
20
+ #define BZS_NETSVC_SERVER_ISERVER_H
21
+
22
+ namespace bzs
23
+ {
24
+ namespace netsvc
25
+ {
26
+ namespace server
27
+ {
28
+
29
+ /** Super class have to implement multithread safety.
30
+ */
31
+ class inotifyHandler
32
+ {
33
+
34
+ public:
35
+ virtual ~inotifyHandler(){};
36
+ virtual void printError(const char* msg) = 0;
37
+ virtual void printInfo(const char* msg) = 0;
38
+ };
39
+
40
+ class iserver
41
+ {
42
+ public:
43
+ virtual ~iserver(){};
44
+ virtual void start()=0;
45
+ virtual void stop()=0;
46
+ virtual void registerErrorHandler(inotifyHandler* ) = 0;
47
+ };
48
+
49
+ /** connection close method for application
50
+ */
51
+ class iconnection
52
+ {
53
+
54
+ public:
55
+ virtual ~iconnection(){};
56
+ virtual void close() = 0;
57
+ };
58
+
59
+
60
+ }//namespace sever
61
+ }//namespace netsvc
62
+ }//namespace bzs
63
+
64
+ #endif //BZS_NETSVC_SERVER_ISERVER_H
65
+
@@ -0,0 +1,522 @@
1
+ /*=================================================================
2
+ Copyright (C) 2012 2013 BizStation Corp All rights reserved.
3
+
4
+ This program is free software; you can redistribute it and/or
5
+ modify it under the terms of the GNU General Public License
6
+ as published by the Free Software Foundation; either version 2
7
+ of the License, or (at your option) any later version.
8
+
9
+ This program is distributed in the hope that it will be useful,
10
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
11
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
+ GNU General Public License for more details.
13
+
14
+ You should have received a copy of the GNU General Public License
15
+ along with this program; if not, write to the Free Software
16
+ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
17
+ 02111-1307, USA.
18
+ =================================================================*/
19
+ #include "serverCpt.h"
20
+ #include "IAppModule.h"
21
+ #include <boost/bind.hpp>
22
+ #include <boost/array.hpp>
23
+ #include <boost/shared_ptr.hpp>
24
+ #include <boost/scoped_array.hpp>
25
+ #include <boost/thread.hpp>
26
+ #include <boost/asio/write.hpp>
27
+ #include <boost/thread/xtime.hpp>
28
+ #include <algorithm>
29
+ #include <boost/enable_shared_from_this.hpp>
30
+ #include <bzs/rtl/debuglog.h>
31
+ #include <bzs/env/crosscompile.h>
32
+ #include <bzs/rtl/exception.h>
33
+
34
+ using namespace boost;
35
+ using namespace boost::asio;
36
+ using namespace boost::asio::ip;
37
+
38
+
39
+
40
+ namespace bzs
41
+ {
42
+ namespace netsvc
43
+ {
44
+ namespace server
45
+ {
46
+ namespace cpt //connection per thread
47
+ {
48
+
49
+ unsigned int g_connections = 0;
50
+ unsigned int g_waitThread = 0;
51
+ // ---------------------------------------------------------------------------
52
+ // connection
53
+ // ---------------------------------------------------------------------------
54
+ #define READBUF_SIZE 66000
55
+ #define WRITEBUF_SIZE 66000
56
+
57
+ class connection : public iconnection, private boost::noncopyable
58
+ {
59
+ mutable io_service m_ios;
60
+
61
+ static mutex m_mutex;
62
+ tcp::socket m_socket;
63
+ std::vector<char> m_buffer;
64
+ std::vector<char> m_result;
65
+ buffers m_optionalBuffes;
66
+ shared_ptr<IAppModule> m_module;
67
+
68
+ size_t m_readLen;
69
+ void handle_read(const boost::system::error_code& e, std::size_t bytes_transferred)
70
+ {
71
+ DEBUG_PROFILE_START(m_readLen==0)
72
+ if (!e)
73
+ {
74
+ bool complete = false;
75
+
76
+ if (bytes_transferred==0)
77
+ return ;
78
+ m_readLen += bytes_transferred;
79
+ size_t n = m_module->onRead(&m_buffer[0], m_readLen, complete);
80
+ if (complete)
81
+ {
82
+ DEBUG_PROFILE_END(1, "handle_read")
83
+ size_t size=0;
84
+ if (m_optionalBuffes.size())
85
+ m_optionalBuffes.clear();
86
+ if (m_module->execute(&m_result[0], size, &m_optionalBuffes) == EXECUTE_RESULT_QUIT)
87
+ return ;
88
+ else
89
+ {
90
+ m_readLen = 0;
91
+ DEBUG_PROFILE_START(1)
92
+ {
93
+ if (m_optionalBuffes.size())
94
+ {
95
+ m_optionalBuffes.insert(m_optionalBuffes.begin(), buffer(&m_result[0], size));
96
+ async_write(m_socket, m_optionalBuffes, boost::bind(&connection::handle_write, this,
97
+ boost::asio::placeholders::error));
98
+ }else
99
+ {
100
+ async_write(m_socket, buffer(&m_result[0], size), boost::bind(&connection::handle_write, this,
101
+ boost::asio::placeholders::error));
102
+ /*boost::asio::write(m_socket, buffer(&m_result[0], size), boost::asio::transfer_all());
103
+ m_socket.async_read_some(buffer(&m_buffer[0], m_buffer.size()),
104
+ boost::bind(&connection::handle_read, this,
105
+ boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred));*/
106
+ }
107
+ }
108
+ m_module->cleanup();
109
+ return;
110
+ }
111
+ }
112
+
113
+ if (n > m_buffer.size())
114
+ m_buffer.resize(n);
115
+ m_socket.async_read_some(buffer(&m_buffer[m_readLen], m_buffer.size()-m_readLen),
116
+ boost::bind(&connection::handle_read, this,
117
+ boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred));
118
+ }
119
+ }
120
+
121
+ void handle_write(const boost::system::error_code& e)
122
+ {
123
+ if (!e)
124
+ {
125
+
126
+ DEBUG_PROFILE_END(1, "write")
127
+ m_socket.async_read_some(buffer(&m_buffer[0], m_buffer.size()),
128
+ boost::bind(&connection::handle_read, this,
129
+ boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred));
130
+ }
131
+
132
+ }
133
+ public:
134
+
135
+ connection(): m_socket(m_ios)
136
+ ,m_readLen(0)
137
+ {
138
+ m_buffer.resize(READBUF_SIZE);
139
+ m_result.resize(WRITEBUF_SIZE);
140
+ mutex::scoped_lock lck(m_mutex);
141
+ connections.push_back(this);
142
+ ++g_connections;
143
+
144
+ }
145
+
146
+ ~connection()
147
+ {
148
+
149
+ if (connections.size())
150
+ {
151
+ mutex::scoped_lock lck(m_mutex);
152
+ std::vector<connection*>::iterator it = find(connections.begin(), connections.end(), this);
153
+ if (it != connections.end())
154
+ {
155
+ connections.erase( it);
156
+ --g_connections;
157
+ }
158
+ }
159
+ }
160
+
161
+ void close()
162
+ {
163
+ m_ios.post(boost::bind(&connection::doClose, this));
164
+ }
165
+
166
+ void doClose()
167
+ {
168
+ boost::system::error_code ec;
169
+ m_socket.close(ec);
170
+ }
171
+
172
+ io_service& ios()const{return m_ios;};
173
+
174
+ tcp::socket& socket(){return m_socket;}
175
+
176
+ void setModule(boost::shared_ptr<IAppModule> p)
177
+ {
178
+ m_module = p;
179
+ }
180
+
181
+ void sendConnectAccept()
182
+ {
183
+ m_ios.reset();
184
+ const boost::asio::ip::tcp::no_delay nodelay(true);
185
+ m_socket.set_option(nodelay);
186
+
187
+ size_t n = m_module->onAccept(&m_result[0], WRITEBUF_SIZE);
188
+ if (n)
189
+ boost::asio::write(m_socket, buffer(&m_result[0], n), boost::asio::transfer_all());
190
+ }
191
+
192
+ void start()
193
+ {
194
+ m_socket.async_read_some(buffer(&m_buffer[0], m_buffer.size()),
195
+ boost::bind(&connection::handle_read, this,
196
+ boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred));
197
+ m_ios.run();
198
+ }
199
+
200
+ static std::vector<connection* > connections;
201
+
202
+ static void stop()
203
+ {
204
+ mutex::scoped_lock lck(m_mutex);
205
+ try
206
+ {
207
+ for (size_t i=0;i<connections.size();i++)
208
+ connections[i]->ios().stop();
209
+ }
210
+ catch(boost::system::system_error &){};
211
+ }
212
+
213
+ };
214
+ std::vector<connection* > connection::connections;
215
+ mutex connection::m_mutex;
216
+
217
+ // ---------------------------------------------------------------------------
218
+ // worker
219
+ // ---------------------------------------------------------------------------
220
+
221
+ class worker : private boost::noncopyable
222
+ {
223
+
224
+ static boost::mutex m_mutex;
225
+ static std::vector< shared_ptr<boost::thread> > m_threads;
226
+ static std::vector<worker*> m_workers;
227
+ static worker* findWaitThread()
228
+ {
229
+ mutex::scoped_lock lck(m_mutex);
230
+ for (size_t i=0;i<m_workers.size();i++)
231
+ if (m_workers[i]->m_connection==NULL)
232
+ return m_workers[i];
233
+ return NULL;
234
+ }
235
+
236
+ boost::shared_ptr<connection> m_connection;
237
+
238
+ bool resume(){return shutdown || m_connection;}
239
+
240
+ ~worker()
241
+ {
242
+ mutex::scoped_lock lck(m_mutex);
243
+ m_workers.erase( find(m_workers.begin(), m_workers.end(), this));
244
+ }
245
+
246
+ public:
247
+ static bool shutdown;
248
+ static const char* hostCheckName;
249
+ static boost::condition_variable condition;
250
+
251
+ static void registThread(shared_ptr<boost::thread> t)
252
+ {
253
+ mutex::scoped_lock lck(m_mutex);
254
+ m_threads.push_back(t);
255
+ }
256
+
257
+ static void join()
258
+ {
259
+ for (size_t i=0;i<m_threads.size();i++)
260
+ m_threads[i]->join();
261
+ //delete thread
262
+ m_threads.erase(m_threads.begin(), m_threads.end());
263
+ }
264
+
265
+ static worker* get(const IAppModuleBuilder* app)
266
+ {
267
+ worker* p = findWaitThread();
268
+ if (p==NULL)
269
+ {
270
+ boost::thread::attributes attr;
271
+ attr.set_stack_size( 125 * 1024);
272
+ p = new worker();
273
+ shared_ptr<boost::thread> t(new boost::thread(attr, bind(&worker::run, p, app)));
274
+ registThread(t);
275
+ }
276
+ return p;
277
+ }
278
+
279
+ worker()
280
+ {
281
+ mutex::scoped_lock lck(m_mutex);
282
+ m_workers.push_back(this);
283
+ }
284
+
285
+ void setConnection(boost::shared_ptr<connection> conn)
286
+ {
287
+ m_connection = conn;
288
+ }
289
+
290
+ void run(const IAppModuleBuilder* app)
291
+ {
292
+ try
293
+ {
294
+ while (!shutdown)
295
+ {
296
+ if (m_connection)
297
+ {
298
+ boost::system::error_code ec;
299
+ tcp::endpoint endpoint = m_connection->socket().remote_endpoint(ec);
300
+ boost::shared_ptr<IAppModule> mod(((IAppModuleBuilder*)app)->createSessionModule(endpoint, m_connection.get(), SERVER_TYPE_CPT));
301
+ m_connection->setModule(mod);
302
+ if (mod->checkHost(hostCheckName))
303
+ {
304
+ m_connection->sendConnectAccept();
305
+ m_connection->start(); //It does not return, unless a connection is close.
306
+ }
307
+ m_connection.reset();
308
+ }
309
+ //TODO A used thread -- it is all held.
310
+ //The number of maintenance is decided and it is made not to hold any more.
311
+ mutex::scoped_lock lck(m_mutex);
312
+ ++g_waitThread;
313
+ condition.wait(lck, bind(&worker::resume, this));
314
+ --g_waitThread;
315
+ }
316
+ }
317
+ catch(bzs::rtl::exception &e)
318
+ {
319
+ if (server::erh)
320
+ {
321
+ if(const std::string *msg = getMsg(e))
322
+ {
323
+ std::string s = "Cpt server " + *msg;
324
+ server::erh->printError(s.c_str());
325
+ }else
326
+ server::erh->printError(boost::diagnostic_information(e).c_str());
327
+ }
328
+ }
329
+ catch(...)
330
+ {
331
+ if (server::erh)
332
+ server::erh->printError("Cpt server Unknown");
333
+ }
334
+ //An end of this thread will delete self.
335
+ delete this;
336
+ }
337
+
338
+ };
339
+
340
+ bool worker::shutdown = false;
341
+ const char* worker::hostCheckName;
342
+ boost::condition_variable worker::condition;
343
+ boost::mutex worker::m_mutex;
344
+ std::vector< boost::shared_ptr<boost::thread> > worker::m_threads;
345
+ std::vector<worker*> worker::m_workers;
346
+
347
+ // ---------------------------------------------------------------------------
348
+ // listener
349
+ // ---------------------------------------------------------------------------
350
+
351
+ class listener
352
+ {
353
+ boost::shared_ptr<IAppModuleBuilder> m_app;
354
+ boost::asio::ip::tcp::acceptor m_acceptor;
355
+ boost::shared_ptr<connection> m_newConnection;
356
+ server* m_srv;
357
+ public:
358
+ listener(server* srv,
359
+ shared_ptr<IAppModuleBuilder> app, const std::string& address, const std::string& port)
360
+ :m_srv(srv), m_app(app),m_acceptor(srv->ios())
361
+ {
362
+
363
+ m_newConnection.reset(new connection());
364
+ tcp::resolver resolver(m_newConnection->ios());
365
+ tcp::resolver::query query(address, port, resolver_query_base::numeric_service);
366
+ tcp::endpoint endpoint = *resolver.resolve(query);
367
+
368
+ m_acceptor.open(endpoint.protocol());
369
+ m_acceptor.set_option(tcp::acceptor::reuse_address(true));
370
+ m_acceptor.bind(endpoint);
371
+ m_acceptor.listen((int)m_srv->maxConnections());
372
+ }
373
+
374
+ void startAsyncAccept()
375
+ {
376
+ if (m_srv->checkConnections())
377
+ m_acceptor.async_accept(m_newConnection->socket(), boost::bind(&listener::onAccept, this, placeholders::error));
378
+ }
379
+
380
+ void onAccept(const boost::system::error_code& e)
381
+ {
382
+ //connection is passed to a thread and it resumes.
383
+ if (!e)
384
+ {
385
+ worker* w = worker::get(m_app.get());
386
+ w->setConnection(m_newConnection);
387
+ worker::condition.notify_all();
388
+ m_newConnection.reset(new connection());
389
+ startAsyncAccept();
390
+ }
391
+ }
392
+
393
+ void cancel()
394
+ {
395
+ boost::system::error_code ec;
396
+ m_acceptor.cancel(ec);
397
+ }
398
+ };
399
+
400
+
401
+ // ---------------------------------------------------------------------------
402
+ // server
403
+ // ---------------------------------------------------------------------------
404
+ inotifyHandler* server::erh=NULL;
405
+
406
+ /** server
407
+ * If it starts, a server will create the exclusive thread for accpter
408
+ * and will go into an infinite loop.
409
+ */
410
+ server::server(std::size_t max_connections, const char* hostCheckName)
411
+ : m_maxConnections(max_connections) ,m_timer(m_ios),m_stopped(false)
412
+ {
413
+ worker::hostCheckName = hostCheckName;
414
+ }
415
+
416
+ server::~server()
417
+ {
418
+ m_listeners.clear();
419
+ }
420
+
421
+ void server::addApplication(boost::shared_ptr<IAppModuleBuilder> app, const std::string& address
422
+ , const std::string& port)
423
+ {
424
+ m_listeners.push_back(boost::shared_ptr<listener>(new listener(this, app, address, port)));
425
+ }
426
+
427
+ /** Start the server
428
+ */
429
+ void server::start()
430
+ {
431
+ shared_ptr<boost::thread> t(new boost::thread(bind(&server::run, this)));
432
+ worker::registThread(t);
433
+ }
434
+
435
+ bool server::checkConnections()
436
+ {
437
+ while (connection::connections.size() > m_maxConnections)
438
+ {
439
+ Sleep(100*MCRTOMM);
440
+ if (m_stopped)
441
+ return false;
442
+ }
443
+ return true;
444
+ }
445
+
446
+ void server::startAsyncAccept()
447
+ {
448
+ for (size_t i = 0;i < m_listeners.size();i++)
449
+ m_listeners[i]->startAsyncAccept();
450
+ }
451
+
452
+ /** Strat the time.
453
+ * If the timer times out then call shoutdown ceck function
454
+ */
455
+ void server::startTimer()
456
+ {
457
+ m_timer.expires_from_now(posix_time::seconds(10));
458
+ m_timer.async_wait(bind(&server::onCheckInternlShutdown, this, placeholders::error));
459
+
460
+ }
461
+
462
+ void server::run()
463
+ {
464
+ if (erh)
465
+ erh->printInfo("Start Cpt server.");
466
+
467
+ startAsyncAccept();
468
+ startTimer();
469
+ m_ios.run();
470
+ }
471
+
472
+ void server::onCheckInternlShutdown(const boost::system::error_code& e)
473
+ {
474
+ bool shutdown = false;
475
+ {
476
+ mutex::scoped_lock lck(modulesMutex);
477
+ for (size_t i=0;i<modules.size();i++)
478
+ {
479
+ IAppModule* mod = modules[i];
480
+ if (mod->isShutDown())
481
+ {
482
+ shutdown = true;
483
+ break;
484
+ }
485
+ }
486
+ }
487
+
488
+ if (shutdown)
489
+ doStop();
490
+ else
491
+ startTimer();
492
+ }
493
+
494
+ void server::doStop()
495
+ {
496
+ if (!m_stopped)
497
+ {
498
+ m_stopped = true;
499
+ if (erh)
500
+ erh->printInfo("Stopping Cpt server ...");
501
+
502
+ for (size_t i = 0;i < m_listeners.size();i++)
503
+ m_listeners[i]->cancel();
504
+ worker::shutdown = true;
505
+ connection::stop();
506
+ m_ios.stop();
507
+ worker::condition.notify_all();
508
+ }
509
+ }
510
+
511
+ void server::stop()
512
+ {
513
+ doStop();
514
+ worker::join();
515
+
516
+ }
517
+
518
+
519
+ }//namespace cpt
520
+ }//namespace server
521
+ }//namesapce netsvc
522
+ }//namespace bzs