prestogres 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (393) hide show
  1. data/.gitignore +4 -0
  2. data/Gemfile +2 -0
  3. data/Gemfile.lock +20 -0
  4. data/LICENSE +202 -0
  5. data/NOTICE +22 -0
  6. data/README.md +217 -0
  7. data/Rakefile +13 -0
  8. data/VERSION +1 -0
  9. data/bin/prestogres +254 -0
  10. data/config/pcp.conf.sample +28 -0
  11. data/config/pgpool.conf +678 -0
  12. data/config/pool_hba.conf +84 -0
  13. data/config/pool_passwd +0 -0
  14. data/config/postgresql.conf +2 -0
  15. data/ext/.gitignore +6 -0
  16. data/ext/depend +26 -0
  17. data/ext/extconf.rb +4 -0
  18. data/ext/prestogres_config.c +12 -0
  19. data/pgpool2/.gitignore +36 -0
  20. data/pgpool2/AUTHORS +4 -0
  21. data/pgpool2/COPYING +12 -0
  22. data/pgpool2/ChangeLog +1 -0
  23. data/pgpool2/INSTALL +1 -0
  24. data/pgpool2/Makefile.am +159 -0
  25. data/pgpool2/Makefile.in +1187 -0
  26. data/pgpool2/NEWS +4960 -0
  27. data/pgpool2/README +1 -0
  28. data/pgpool2/README.euc_jp +1 -0
  29. data/pgpool2/README.online-recovery +62 -0
  30. data/pgpool2/TODO +103 -0
  31. data/pgpool2/ac_func_accept_argtypes.m4 +85 -0
  32. data/pgpool2/aclocal.m4 +1088 -0
  33. data/pgpool2/c-compiler.m4 +134 -0
  34. data/pgpool2/c-library.m4 +325 -0
  35. data/pgpool2/child.c +2097 -0
  36. data/pgpool2/config.guess +1532 -0
  37. data/pgpool2/config.h.in +332 -0
  38. data/pgpool2/config.sub +1640 -0
  39. data/pgpool2/configure +15752 -0
  40. data/pgpool2/configure.in +392 -0
  41. data/pgpool2/depcomp +522 -0
  42. data/pgpool2/doc/basebackup.sh +17 -0
  43. data/pgpool2/doc/pgpool-de.html +4220 -0
  44. data/pgpool2/doc/pgpool-en.html +5738 -0
  45. data/pgpool2/doc/pgpool-fr.html +4118 -0
  46. data/pgpool2/doc/pgpool-ja.css +198 -0
  47. data/pgpool2/doc/pgpool-ja.html +11279 -0
  48. data/pgpool2/doc/pgpool-zh_cn.html +4445 -0
  49. data/pgpool2/doc/pgpool.css +280 -0
  50. data/pgpool2/doc/pgpool_remote_start +13 -0
  51. data/pgpool2/doc/recovery.conf.sample +117 -0
  52. data/pgpool2/doc/tutorial-en.html +707 -0
  53. data/pgpool2/doc/tutorial-ja.html +422 -0
  54. data/pgpool2/doc/tutorial-memqcache-en.html +325 -0
  55. data/pgpool2/doc/tutorial-memqcache-ja.html +370 -0
  56. data/pgpool2/doc/tutorial-memqcache-zh_cn.html +322 -0
  57. data/pgpool2/doc/tutorial-watchdog-en.html +306 -0
  58. data/pgpool2/doc/tutorial-watchdog-ja.html +343 -0
  59. data/pgpool2/doc/tutorial-watchdog-zh_cn.html +301 -0
  60. data/pgpool2/doc/tutorial-zh_cn.html +537 -0
  61. data/pgpool2/doc/watchdog.png +0 -0
  62. data/pgpool2/doc/wd-en.html +236 -0
  63. data/pgpool2/doc/wd-en.jpg +0 -0
  64. data/pgpool2/doc/wd-ja.html +219 -0
  65. data/pgpool2/doc/wd-ja.jpg +0 -0
  66. data/pgpool2/doc/wd-zh_cn.html +201 -0
  67. data/pgpool2/doc/where_to_send_queries.odg +0 -0
  68. data/pgpool2/doc/where_to_send_queries.pdf +0 -0
  69. data/pgpool2/general.m4 +166 -0
  70. data/pgpool2/getopt_long.c +200 -0
  71. data/pgpool2/getopt_long.h +44 -0
  72. data/pgpool2/install-sh +251 -0
  73. data/pgpool2/ltmain.sh +8406 -0
  74. data/pgpool2/m4/libtool.m4 +7360 -0
  75. data/pgpool2/m4/ltoptions.m4 +368 -0
  76. data/pgpool2/m4/ltsugar.m4 +123 -0
  77. data/pgpool2/m4/ltversion.m4 +23 -0
  78. data/pgpool2/m4/lt~obsolete.m4 +92 -0
  79. data/pgpool2/main.c +2971 -0
  80. data/pgpool2/md5.c +444 -0
  81. data/pgpool2/md5.h +28 -0
  82. data/pgpool2/missing +360 -0
  83. data/pgpool2/mkinstalldirs +40 -0
  84. data/pgpool2/parser/Makefile.am +50 -0
  85. data/pgpool2/parser/Makefile.in +559 -0
  86. data/pgpool2/parser/copyfuncs.c +3310 -0
  87. data/pgpool2/parser/gram.c +39100 -0
  88. data/pgpool2/parser/gram.h +940 -0
  89. data/pgpool2/parser/gram.y +13408 -0
  90. data/pgpool2/parser/gramparse.h +74 -0
  91. data/pgpool2/parser/keywords.c +32 -0
  92. data/pgpool2/parser/keywords.h +39 -0
  93. data/pgpool2/parser/kwlist.h +425 -0
  94. data/pgpool2/parser/kwlookup.c +88 -0
  95. data/pgpool2/parser/list.c +1156 -0
  96. data/pgpool2/parser/makefuncs.c +518 -0
  97. data/pgpool2/parser/makefuncs.h +83 -0
  98. data/pgpool2/parser/memnodes.h +79 -0
  99. data/pgpool2/parser/nodes.c +29 -0
  100. data/pgpool2/parser/nodes.h +609 -0
  101. data/pgpool2/parser/outfuncs.c +5790 -0
  102. data/pgpool2/parser/parsenodes.h +2615 -0
  103. data/pgpool2/parser/parser.c +262 -0
  104. data/pgpool2/parser/parser.h +46 -0
  105. data/pgpool2/parser/pg_class.h +158 -0
  106. data/pgpool2/parser/pg_config_manual.h +273 -0
  107. data/pgpool2/parser/pg_list.h +352 -0
  108. data/pgpool2/parser/pg_trigger.h +147 -0
  109. data/pgpool2/parser/pg_wchar.h +492 -0
  110. data/pgpool2/parser/pool_memory.c +342 -0
  111. data/pgpool2/parser/pool_memory.h +77 -0
  112. data/pgpool2/parser/pool_parser.h +222 -0
  113. data/pgpool2/parser/pool_string.c +121 -0
  114. data/pgpool2/parser/pool_string.h +37 -0
  115. data/pgpool2/parser/primnodes.h +1280 -0
  116. data/pgpool2/parser/scan.c +4094 -0
  117. data/pgpool2/parser/scan.l +1451 -0
  118. data/pgpool2/parser/scanner.h +120 -0
  119. data/pgpool2/parser/scansup.c +221 -0
  120. data/pgpool2/parser/scansup.h +28 -0
  121. data/pgpool2/parser/snprintf.c +1102 -0
  122. data/pgpool2/parser/stringinfo.c +294 -0
  123. data/pgpool2/parser/stringinfo.h +178 -0
  124. data/pgpool2/parser/value.c +78 -0
  125. data/pgpool2/parser/value.h +62 -0
  126. data/pgpool2/parser/wchar.c +2048 -0
  127. data/pgpool2/pcp.conf.sample +28 -0
  128. data/pgpool2/pcp/Makefile.am +40 -0
  129. data/pgpool2/pcp/Makefile.in +771 -0
  130. data/pgpool2/pcp/libpcp_ext.h +250 -0
  131. data/pgpool2/pcp/md5.c +444 -0
  132. data/pgpool2/pcp/md5.h +28 -0
  133. data/pgpool2/pcp/pcp.c +1652 -0
  134. data/pgpool2/pcp/pcp.h +61 -0
  135. data/pgpool2/pcp/pcp_attach_node.c +172 -0
  136. data/pgpool2/pcp/pcp_detach_node.c +185 -0
  137. data/pgpool2/pcp/pcp_error.c +87 -0
  138. data/pgpool2/pcp/pcp_node_count.c +160 -0
  139. data/pgpool2/pcp/pcp_node_info.c +198 -0
  140. data/pgpool2/pcp/pcp_pool_status.c +166 -0
  141. data/pgpool2/pcp/pcp_proc_count.c +166 -0
  142. data/pgpool2/pcp/pcp_proc_info.c +261 -0
  143. data/pgpool2/pcp/pcp_promote_node.c +185 -0
  144. data/pgpool2/pcp/pcp_recovery_node.c +172 -0
  145. data/pgpool2/pcp/pcp_stop_pgpool.c +179 -0
  146. data/pgpool2/pcp/pcp_stream.c +385 -0
  147. data/pgpool2/pcp/pcp_stream.h +52 -0
  148. data/pgpool2/pcp/pcp_systemdb_info.c +194 -0
  149. data/pgpool2/pcp/pcp_watchdog_info.c +211 -0
  150. data/pgpool2/pcp_child.c +1493 -0
  151. data/pgpool2/pg_md5.c +305 -0
  152. data/pgpool2/pgpool.8.in +121 -0
  153. data/pgpool2/pgpool.conf +553 -0
  154. data/pgpool2/pgpool.conf.sample +666 -0
  155. data/pgpool2/pgpool.conf.sample-master-slave +665 -0
  156. data/pgpool2/pgpool.conf.sample-replication +664 -0
  157. data/pgpool2/pgpool.conf.sample-stream +664 -0
  158. data/pgpool2/pgpool.spec +264 -0
  159. data/pgpool2/pgpool_adm/TODO +7 -0
  160. data/pgpool2/pgpool_adm/pgpool_adm--1.0.sql +85 -0
  161. data/pgpool2/pgpool_adm/pgpool_adm.c +558 -0
  162. data/pgpool2/pgpool_adm/pgpool_adm.control +5 -0
  163. data/pgpool2/pgpool_adm/pgpool_adm.h +46 -0
  164. data/pgpool2/pgpool_adm/pgpool_adm.sql.in +85 -0
  165. data/pgpool2/pool.h +655 -0
  166. data/pgpool2/pool_auth.c +1390 -0
  167. data/pgpool2/pool_config.c +5007 -0
  168. data/pgpool2/pool_config.h +284 -0
  169. data/pgpool2/pool_config.l +3281 -0
  170. data/pgpool2/pool_config_md5.c +29 -0
  171. data/pgpool2/pool_connection_pool.c +812 -0
  172. data/pgpool2/pool_error.c +242 -0
  173. data/pgpool2/pool_globals.c +27 -0
  174. data/pgpool2/pool_hba.c +1723 -0
  175. data/pgpool2/pool_hba.conf.sample +67 -0
  176. data/pgpool2/pool_ip.c +567 -0
  177. data/pgpool2/pool_ip.h +65 -0
  178. data/pgpool2/pool_ipc.h +38 -0
  179. data/pgpool2/pool_lobj.c +242 -0
  180. data/pgpool2/pool_lobj.h +32 -0
  181. data/pgpool2/pool_memqcache.c +3818 -0
  182. data/pgpool2/pool_memqcache.h +268 -0
  183. data/pgpool2/pool_params.c +163 -0
  184. data/pgpool2/pool_passwd.c +249 -0
  185. data/pgpool2/pool_passwd.h +41 -0
  186. data/pgpool2/pool_path.c +193 -0
  187. data/pgpool2/pool_path.h +81 -0
  188. data/pgpool2/pool_process_context.c +247 -0
  189. data/pgpool2/pool_process_context.h +62 -0
  190. data/pgpool2/pool_process_query.c +5001 -0
  191. data/pgpool2/pool_process_reporting.c +1671 -0
  192. data/pgpool2/pool_process_reporting.h +44 -0
  193. data/pgpool2/pool_proto2.c +671 -0
  194. data/pgpool2/pool_proto_modules.c +3524 -0
  195. data/pgpool2/pool_proto_modules.h +185 -0
  196. data/pgpool2/pool_query_cache.c +1020 -0
  197. data/pgpool2/pool_query_context.c +1871 -0
  198. data/pgpool2/pool_query_context.h +105 -0
  199. data/pgpool2/pool_relcache.c +284 -0
  200. data/pgpool2/pool_relcache.h +78 -0
  201. data/pgpool2/pool_rewrite_outfuncs.c +9060 -0
  202. data/pgpool2/pool_rewrite_query.c +715 -0
  203. data/pgpool2/pool_rewrite_query.h +192 -0
  204. data/pgpool2/pool_select_walker.c +1150 -0
  205. data/pgpool2/pool_select_walker.h +68 -0
  206. data/pgpool2/pool_sema.c +161 -0
  207. data/pgpool2/pool_session_context.c +952 -0
  208. data/pgpool2/pool_session_context.h +203 -0
  209. data/pgpool2/pool_shmem.c +185 -0
  210. data/pgpool2/pool_signal.c +158 -0
  211. data/pgpool2/pool_signal.h +61 -0
  212. data/pgpool2/pool_ssl.c +339 -0
  213. data/pgpool2/pool_stream.c +962 -0
  214. data/pgpool2/pool_stream.h +61 -0
  215. data/pgpool2/pool_system.c +659 -0
  216. data/pgpool2/pool_timestamp.c +1215 -0
  217. data/pgpool2/pool_timestamp.h +38 -0
  218. data/pgpool2/pool_type.h +171 -0
  219. data/pgpool2/pool_worker_child.c +384 -0
  220. data/pgpool2/ps_status.c +404 -0
  221. data/pgpool2/recovery.c +435 -0
  222. data/pgpool2/redhat/pgpool.conf.sample.patch +52 -0
  223. data/pgpool2/redhat/pgpool.init +201 -0
  224. data/pgpool2/redhat/pgpool.sysconfig +7 -0
  225. data/pgpool2/redhat/rpm_installer/basebackup-replication.sh +53 -0
  226. data/pgpool2/redhat/rpm_installer/basebackup-stream.sh +55 -0
  227. data/pgpool2/redhat/rpm_installer/config_for_script +17 -0
  228. data/pgpool2/redhat/rpm_installer/failover.sh +64 -0
  229. data/pgpool2/redhat/rpm_installer/getsources.sh +141 -0
  230. data/pgpool2/redhat/rpm_installer/install.sh +1363 -0
  231. data/pgpool2/redhat/rpm_installer/pgpool_recovery_pitr +47 -0
  232. data/pgpool2/redhat/rpm_installer/pgpool_remote_start +15 -0
  233. data/pgpool2/redhat/rpm_installer/recovery.conf +4 -0
  234. data/pgpool2/redhat/rpm_installer/uninstall.sh +57 -0
  235. data/pgpool2/sample/dist_def_pgbench.sql +73 -0
  236. data/pgpool2/sample/pgpool.pam +3 -0
  237. data/pgpool2/sample/pgpool_recovery +20 -0
  238. data/pgpool2/sample/pgpool_recovery_pitr +19 -0
  239. data/pgpool2/sample/pgpool_remote_start +13 -0
  240. data/pgpool2/sample/replicate_def_pgbench.sql +18 -0
  241. data/pgpool2/sql/insert_lock.sql +15 -0
  242. data/pgpool2/sql/pgpool-recovery/pgpool-recovery.c +280 -0
  243. data/pgpool2/sql/pgpool-recovery/pgpool-recovery.sql.in +19 -0
  244. data/pgpool2/sql/pgpool-recovery/pgpool_recovery--1.0.sql +24 -0
  245. data/pgpool2/sql/pgpool-recovery/pgpool_recovery.control +5 -0
  246. data/pgpool2/sql/pgpool-recovery/uninstall_pgpool-recovery.sql +3 -0
  247. data/pgpool2/sql/pgpool-regclass/pgpool-regclass.c +206 -0
  248. data/pgpool2/sql/pgpool-regclass/pgpool-regclass.sql.in +4 -0
  249. data/pgpool2/sql/pgpool-regclass/pgpool_regclass--1.0.sql +7 -0
  250. data/pgpool2/sql/pgpool-regclass/pgpool_regclass.control +5 -0
  251. data/pgpool2/sql/pgpool-regclass/uninstall_pgpool-regclass.sql +1 -0
  252. data/pgpool2/sql/system_db.sql +38 -0
  253. data/pgpool2/strlcpy.c +85 -0
  254. data/pgpool2/test/C/test_extended.c +98 -0
  255. data/pgpool2/test/jdbc/.cvsignore +2 -0
  256. data/pgpool2/test/jdbc/AutoCommitTest.java +45 -0
  257. data/pgpool2/test/jdbc/BatchTest.java +55 -0
  258. data/pgpool2/test/jdbc/ColumnTest.java +60 -0
  259. data/pgpool2/test/jdbc/CreateTempTableTest.java +48 -0
  260. data/pgpool2/test/jdbc/InsertTest.java +34 -0
  261. data/pgpool2/test/jdbc/LockTest.java +36 -0
  262. data/pgpool2/test/jdbc/PgpoolTest.java +75 -0
  263. data/pgpool2/test/jdbc/README.euc_jp +73 -0
  264. data/pgpool2/test/jdbc/RunTest.java +83 -0
  265. data/pgpool2/test/jdbc/SelectTest.java +37 -0
  266. data/pgpool2/test/jdbc/UpdateTest.java +32 -0
  267. data/pgpool2/test/jdbc/expected/CreateTempTable +1 -0
  268. data/pgpool2/test/jdbc/expected/autocommit +10 -0
  269. data/pgpool2/test/jdbc/expected/batch +1 -0
  270. data/pgpool2/test/jdbc/expected/column +100 -0
  271. data/pgpool2/test/jdbc/expected/insert +1 -0
  272. data/pgpool2/test/jdbc/expected/lock +100 -0
  273. data/pgpool2/test/jdbc/expected/select +2 -0
  274. data/pgpool2/test/jdbc/expected/update +1 -0
  275. data/pgpool2/test/jdbc/pgpool.properties +7 -0
  276. data/pgpool2/test/jdbc/prepare.sql +54 -0
  277. data/pgpool2/test/jdbc/run.sh +6 -0
  278. data/pgpool2/test/parser/.cvsignore +6 -0
  279. data/pgpool2/test/parser/README +32 -0
  280. data/pgpool2/test/parser/expected/copy.out +17 -0
  281. data/pgpool2/test/parser/expected/create.out +64 -0
  282. data/pgpool2/test/parser/expected/cursor.out +37 -0
  283. data/pgpool2/test/parser/expected/delete.out +10 -0
  284. data/pgpool2/test/parser/expected/drop.out +12 -0
  285. data/pgpool2/test/parser/expected/insert.out +13 -0
  286. data/pgpool2/test/parser/expected/misc.out +28 -0
  287. data/pgpool2/test/parser/expected/prepare.out +4 -0
  288. data/pgpool2/test/parser/expected/privileges.out +31 -0
  289. data/pgpool2/test/parser/expected/scanner.out +30 -0
  290. data/pgpool2/test/parser/expected/select.out +89 -0
  291. data/pgpool2/test/parser/expected/transaction.out +38 -0
  292. data/pgpool2/test/parser/expected/update.out +11 -0
  293. data/pgpool2/test/parser/expected/v84.out +37 -0
  294. data/pgpool2/test/parser/expected/v90.out +25 -0
  295. data/pgpool2/test/parser/expected/var.out +22 -0
  296. data/pgpool2/test/parser/input/alter.sql +2 -0
  297. data/pgpool2/test/parser/input/copy.sql +17 -0
  298. data/pgpool2/test/parser/input/create.sql +64 -0
  299. data/pgpool2/test/parser/input/cursor.sql +37 -0
  300. data/pgpool2/test/parser/input/delete.sql +10 -0
  301. data/pgpool2/test/parser/input/drop.sql +12 -0
  302. data/pgpool2/test/parser/input/insert.sql +13 -0
  303. data/pgpool2/test/parser/input/misc.sql +28 -0
  304. data/pgpool2/test/parser/input/prepare.sql +4 -0
  305. data/pgpool2/test/parser/input/privileges.sql +31 -0
  306. data/pgpool2/test/parser/input/scanner.sql +34 -0
  307. data/pgpool2/test/parser/input/select.sql +89 -0
  308. data/pgpool2/test/parser/input/transaction.sql +38 -0
  309. data/pgpool2/test/parser/input/update.sql +11 -0
  310. data/pgpool2/test/parser/input/v84.sql +37 -0
  311. data/pgpool2/test/parser/input/v90.sql +38 -0
  312. data/pgpool2/test/parser/input/var.sql +22 -0
  313. data/pgpool2/test/parser/main.c +96 -0
  314. data/pgpool2/test/parser/parse_schedule +16 -0
  315. data/pgpool2/test/parser/pool.h +13 -0
  316. data/pgpool2/test/parser/run-test +62 -0
  317. data/pgpool2/test/pdo-test/README.euc_jp +58 -0
  318. data/pgpool2/test/pdo-test/SQLlist/test1.sql +3 -0
  319. data/pgpool2/test/pdo-test/SQLlist/test2.sql +3 -0
  320. data/pgpool2/test/pdo-test/collections.inc +11 -0
  321. data/pgpool2/test/pdo-test/def.inc +7 -0
  322. data/pgpool2/test/pdo-test/log.txt +0 -0
  323. data/pgpool2/test/pdo-test/mod/database.inc +36 -0
  324. data/pgpool2/test/pdo-test/mod/def.inc +0 -0
  325. data/pgpool2/test/pdo-test/mod/errorhandler.inc +27 -0
  326. data/pgpool2/test/pdo-test/pdotest.php +11 -0
  327. data/pgpool2/test/pdo-test/regsql.inc +56 -0
  328. data/pgpool2/test/pgpool_setup +898 -0
  329. data/pgpool2/test/regression/README +39 -0
  330. data/pgpool2/test/regression/clean.sh +21 -0
  331. data/pgpool2/test/regression/libs.sh +16 -0
  332. data/pgpool2/test/regression/regress.sh +166 -0
  333. data/pgpool2/test/regression/tests/001.load_balance/test.sh +128 -0
  334. data/pgpool2/test/regression/tests/002.native_replication/PgTester.java +47 -0
  335. data/pgpool2/test/regression/tests/002.native_replication/create.sql +6 -0
  336. data/pgpool2/test/regression/tests/002.native_replication/test.sh +71 -0
  337. data/pgpool2/test/regression/tests/003.failover/expected.r +6 -0
  338. data/pgpool2/test/regression/tests/003.failover/expected.s +6 -0
  339. data/pgpool2/test/regression/tests/003.failover/test.sh +45 -0
  340. data/pgpool2/test/regression/tests/004.watchdog/master.conf +12 -0
  341. data/pgpool2/test/regression/tests/004.watchdog/standby.conf +19 -0
  342. data/pgpool2/test/regression/tests/004.watchdog/test.sh +52 -0
  343. data/pgpool2/test/regression/tests/050.bug58/test.sh +50 -0
  344. data/pgpool2/test/regression/tests/051.bug60/bug.sql +12 -0
  345. data/pgpool2/test/regression/tests/051.bug60/database-clean.sql +6 -0
  346. data/pgpool2/test/regression/tests/051.bug60/database-setup.sql +28 -0
  347. data/pgpool2/test/regression/tests/051.bug60/test.sh +79 -0
  348. data/pgpool2/test/regression/tests/052.do_query/test.sh +44 -0
  349. data/pgpool2/test/regression/tests/053.insert_lock_hangs/test.sh +81 -0
  350. data/pgpool2/test/regression/tests/054.postgres_fdw/test.sh +67 -0
  351. data/pgpool2/test/regression/tests/055.backend_all_down/test.sh +52 -0
  352. data/pgpool2/test/regression/tests/056.bug63/jdbctest2.java +66 -0
  353. data/pgpool2/test/regression/tests/056.bug63/test.sh +47 -0
  354. data/pgpool2/test/regression/tests/057.bug61/test.sh +40 -0
  355. data/pgpool2/test/regression/tests/058.bug68/jdbctest3.java +45 -0
  356. data/pgpool2/test/regression/tests/058.bug68/test.sh +47 -0
  357. data/pgpool2/test/timestamp/expected/insert.out +16 -0
  358. data/pgpool2/test/timestamp/expected/misc.out +3 -0
  359. data/pgpool2/test/timestamp/expected/update.out +6 -0
  360. data/pgpool2/test/timestamp/input/insert.sql +16 -0
  361. data/pgpool2/test/timestamp/input/misc.sql +3 -0
  362. data/pgpool2/test/timestamp/input/update.sql +6 -0
  363. data/pgpool2/test/timestamp/main.c +129 -0
  364. data/pgpool2/test/timestamp/parse_schedule +3 -0
  365. data/pgpool2/test/timestamp/run-test +69 -0
  366. data/pgpool2/version.h +1 -0
  367. data/pgpool2/watchdog/Makefile.am +17 -0
  368. data/pgpool2/watchdog/Makefile.in +505 -0
  369. data/pgpool2/watchdog/test/stab.c +266 -0
  370. data/pgpool2/watchdog/test/test.c +85 -0
  371. data/pgpool2/watchdog/test/wd_child_t.c +87 -0
  372. data/pgpool2/watchdog/test/wd_lifecheck_t.c +87 -0
  373. data/pgpool2/watchdog/test/wd_packet_t.c +87 -0
  374. data/pgpool2/watchdog/test/wd_ping_t.c +20 -0
  375. data/pgpool2/watchdog/watchdog.c +408 -0
  376. data/pgpool2/watchdog/watchdog.h +209 -0
  377. data/pgpool2/watchdog/wd_child.c +444 -0
  378. data/pgpool2/watchdog/wd_ext.h +123 -0
  379. data/pgpool2/watchdog/wd_heartbeat.c +577 -0
  380. data/pgpool2/watchdog/wd_if.c +216 -0
  381. data/pgpool2/watchdog/wd_init.c +126 -0
  382. data/pgpool2/watchdog/wd_interlock.c +347 -0
  383. data/pgpool2/watchdog/wd_lifecheck.c +512 -0
  384. data/pgpool2/watchdog/wd_list.c +429 -0
  385. data/pgpool2/watchdog/wd_packet.c +1159 -0
  386. data/pgpool2/watchdog/wd_ping.c +330 -0
  387. data/pgpool2/ylwrap +223 -0
  388. data/pgsql/presto_client.py +346 -0
  389. data/pgsql/prestogres.py +156 -0
  390. data/pgsql/setup_functions.sql +21 -0
  391. data/pgsql/setup_language.sql +3 -0
  392. data/prestogres.gemspec +23 -0
  393. metadata +496 -0
@@ -0,0 +1,4445 @@
1
+ <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
2
+ <html>
3
+ <head>
4
+ <meta http-equiv="content-type" content="text/html; charset=UTF-8">
5
+ <link rel="stylesheet" href="./pgpool.css" type="text/css">
6
+ <title>pgpool-II 用户手册</title>
7
+ </head>
8
+
9
+ <!-- hhmts start -->
10
+ Last modified: Wed Jun 29 09:50:01 JST 2011
11
+ <!-- hhmts end -->
12
+
13
+ <body bgcolor="#ffffff">
14
+ <div id="top" class="header_text">欢迎使用 pgpool-II 手册</div>
15
+
16
+ <div id="menu">
17
+ <div id="navcontainer">
18
+ <ul id="navlist">
19
+ <li id="active"><a href="#Whatis" id="current">什么是 pgpool</a></li>
20
+ <li><a href="#license">许可协议</a></li>
21
+ <li><a href="#platform">支持的平台</a></li>
22
+ <li><a href="#install">pgpool-II 的安装</a></li>
23
+ <li><a href="#config">配置 pgpool-II</a></li>
24
+ <li><a href="#common">配置通用部分</a></li>
25
+ <li><a href="#connection_pool_mode">连接池模式</a></li>
26
+ <li><a href="#replication_mode">复制模式</a></li>
27
+ <li><a href="#master_slave_mode">主备模式</a></li>
28
+ <li><a href="#stream">流复制</a></li>
29
+ <li><a href="#parallel">并行模式</a></li>
30
+ <li><a href="#hba">客户端认证</a></li>
31
+ <li><a href="#query_cache">设置查询缓存方法</a></li>
32
+ <li><a href="#memqcache">基于内存的查询缓存</a></li>
33
+ <li><a href="#start">启动/停止 pgpool-II</a></li>
34
+ <li><a href="#reload">重新加载 pgpool-II 配置文件</a></li>
35
+ <li><a href="#show-commands">显示命令</a></li>
36
+ <li><a href="#online-recovery">在线恢复</a></li>
37
+ <li><a href="#backup">备份</a></li>
38
+ <li><a href="#deploy">部署 pgpool-II</a></li>
39
+ <li><a href="#watchdog">看门狗</a></li>
40
+ <li><a href="#troubleshooting">故障排除</a></li>
41
+ <li><a href="#restriction">限制</a></li>
42
+ <li><a href="#reference">参考信息</a></li>
43
+ <li><a href="#internal">内部信息</a></li>
44
+ </ul>
45
+ </div>
46
+ <br />
47
+
48
+ <div class="header_small" align="center">
49
+ [<a href="pgpool-en.html">英文页面</a>]
50
+ </div>
51
+ </div>
52
+
53
+
54
+ <div id="manual">
55
+
56
+ <!-- ================================================================================ -->
57
+
58
+ <h1 style="margin-top: 20px">什么是 pgpool-II?<a name="whatis"></a></h1>
59
+
60
+ <p> pgpool-II 是一个位于 PostgreSQL 服务器和 PostgreSQL
61
+ 数据库客户端之间的中间件,它提供以下功能:</p>
62
+
63
+ <p>
64
+ <ul>
65
+
66
+ <li>连接池</li>
67
+ <p>pgpool-II 保持已经连接到 PostgreSQL 服务器的连接,
68
+ 并在使用相同参数(例如:用户名,数据库,协议版本)
69
+ 连接进来时重用它们。
70
+ 它减少了连接开销,并增加了系统的总体吞吐量。</p>
71
+
72
+ <li>复制</li>
73
+ <p>pgpool-II 可以管理多个 PostgreSQL 服务器。
74
+ 激活复制功能并使在2台或者更多 PostgreSQL
75
+ 节点中建立一个实时备份成为可能,
76
+ 这样,如果其中一台节点失效,服务可以不被中断继续运行。</p>
77
+
78
+ <li>负载均衡</li>
79
+ <p>如果数据库进行了复制(可能运行在复制模式或者主备模式下),
80
+ 则在任何一台服务器中执行一个 SELECT 查询将返回相同的结果。
81
+ pgpool-II 利用了复制的功能以降低每台 PostgreSQL 服务器的负载。
82
+ 它通过分发 SELECT 查询到所有可用的服务器中,增强了系统的整体吞吐量。
83
+ 在理想的情况下,读性能应该和 PostgreSQL 服务器的数量成正比。
84
+ 负载均很功能在有大量用户同时执行很多只读查询的场景中工作的效果最好。</p>
85
+
86
+ <li>限制超过限度的连接</li>
87
+ <p>PostgreSQL 会限制当前的最大连接数,当到达这个数量时,新的连接将被拒绝。
88
+ 增加这个最大连接数会增加资源消耗并且对系统的全局性能有一定的负面影响。
89
+ pgpoo-II 也支持限制最大连接数,但它的做法是将连接放入队列,而不是立即返回一个错误。</p>
90
+
91
+ <li>并行查询</li>
92
+ <p>使用并行查询时,数据可以被分割到多台服务器上,
93
+ 所以一个查询可以在多台服务器上同时执行,以减少总体执行时间。
94
+ 并行查询在查询大规模数据的时候非常有效。</p>
95
+
96
+ </ul>
97
+ </p>
98
+
99
+ <p>pgpool-II 使用 PostgreSQL 的前后台程序之间的协议,并且在前后台之间传递消息。
100
+ 因此,一个(前端的)数据库应用程序认为 pgpool-II 就是实际的 PostgreSQL 数据库,
101
+ 而后端的服务进程则认为 pgpool-II 是它的一个客户端。
102
+ 因为 pgpool-II 对于服务器和客户端来说是透明的,
103
+ 现有的数据库应用程序基本上可以不需要修改就可以使用 pgpool-II 了。</p>
104
+
105
+ <p>
106
+ <strong>通过 pgpool-II 使用 SQL 有一些限制条件。参考 <a href="#restriction">限制</a> 获得详细信息。</strong>
107
+ </p>
108
+
109
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
110
+
111
+ <!-- ================================================================================ -->
112
+
113
+ <h1>许可协议<a name="license"></a></h1>
114
+
115
+ <p>
116
+ 版权所有 (c) 2003-2013 PgPool 全球开发小组
117
+
118
+ 授权出于任何目的任意使用、拷贝、修改和分发本软件以及文档,
119
+ 但必须在所有软件拷贝的支持文档中提供以上的版权信息,包括版权通知和许可,
120
+ 并且在没有事先明确的书面授权前,作者的名字不能用于发布软件的广告和宣传。
121
+ 作者不对本软件的任何用途负责。
122
+ 本软件以“现状”来提供,不提供任何明文或暗示的保证。
123
+ </p>
124
+
125
+ <p class="top_link"><a href="#Top">back to top</a></p>
126
+
127
+ <!-- ================================================================================ -->
128
+
129
+ <h1>支持的平台<a name="platform"></a></h1>
130
+
131
+ <p>pgpool-II 可以运行在 Linux,Solaris,FreeBSD 以及基本上所有的类 UNIX 架构的平台上。
132
+ 不支持 Windows。支持 6.4 以上版本的 PostgreSQL 服务器。
133
+ 然而,如果要使用并行查询功能,需要 7.4 或更高版本。</p>
134
+
135
+ <p>如果你在使用 7.3 或者更老版本的 PostgreSQL,一些 pgpool-II 的功能将无法使用。
136
+ 但无论如何你不应该还在用这么老的版本了。</p>
137
+
138
+ <p>你还要确保你所有的 PostgreSQL 服务器运行相同主版本号的 PostgreSQL 程序。另外,如果你想要使用在线恢复,硬件架构和操作系统必须一致。
139
+ 另外,我们不推荐在使用不同编译参数编译的 PostgreSQL:包括是否支持 SSL,是否使用了 --disable-integer-datetimes 参数或者不同的块大小。
140
+ 这些可能对 pgpool-II 的部分功能有影响。
141
+ 通常情况下,小版本号不同没有什么影响。但是我们没有针对小版本的区别做全面测试,因此我们建议还是使用相同的版本的 PostgreSQL。</p>
142
+
143
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
144
+
145
+ <!-- ================================================================================ -->
146
+
147
+ <h1>pgpool-II 的安装<a name="install"></a></h1>
148
+
149
+ <p>
150
+ pgpool-II 可以在 <a href="http://pgpool.net/">pgpool 开发页面</a>下载到。
151
+ 而且也提供包括 CentOS,RedHat Enterprise Linux,Fedora 和 Debian 在内的大量平台的二进制包。
152
+ 请检查相关的软件库。
153
+ </p>
154
+
155
+ <p>
156
+ 可以在以下位置下载 pgpool-II 的源码:
157
+ <a href="http://pgpool.net/mediawiki/index.php/Downloads">pgpool 开发页面</a>。
158
+ </p>
159
+
160
+ <p>从源码安装 pgpool-II 需要 2.9 甚至或更高版本的 gcc,以及 GNU make。
161
+ 而且,pgpool-II 需要链接到 libpq 库,所以在构建 pgpool-II 的机器上必须安装 libpq 库和它的开发头文件。
162
+ 另外,还需要 OpenSSL 库和它的头文件以便在 pgpool-II 中提供 OpenSSL 支持。</p>
163
+
164
+ <dl>
165
+ <dt>configure</dt>
166
+ <dd>
167
+ <p>在解压源码包后,执行以下配置脚本。</p>
168
+ <pre>
169
+ ./configure
170
+ </pre>
171
+
172
+ <p>如果你需要非默认的值,有以下选项可以设置:</p>
173
+ <table border>
174
+ <tr><th class="nodec"><code>--prefix=path</code></th>
175
+ <td>pgpool-II 的二进制程序和文档将被安装到这个目录。默认值为 <code>/usr/local</code></td></tr>
176
+ <tr><th class="nodec"><code>--with-pgsql=path</code></th>
177
+ <td>PostgreSQL 的客户端库安装的顶层目录。默认值由 <code>pg_config</code> 提供</td></tr>
178
+ <tr><th class="nodec"><code>--with-openssl</code></th>
179
+ <td>pgpool-II 程序将提供 OpenSSL 支持。默认是禁用 OpenSSL 支持的。<span class="version">V2.3 -</span></td></tr>
180
+ <tr><th class="nodec"><code>--enable-sequence-lock</code></th>
181
+ <td>在 pgpool-II 3.0 系列中使用 insert_lock 兼容。pgpool-II 针对序列表中的一行进行加锁。PostgreSQL 8.2 或2011年六月以后发布的版本无法使用这种加锁方法。
182
+ <span class="version">V3.1 -</span></td></tr>
183
+ <tr><th class="nodec"><code>--enable-table-lock</code></th>
184
+ <td>在 pgpool-II 2.2 和 2.3 系列中使用 insert_lock 兼容。pgpool-II 针对被插入的表进行加锁。这种锁因为和 VACUUM 冲突,已被废弃。<span class="version">V3.1 -</span></td></tr>
185
+ <tr><th class="nodec"><code>--with-memcached=path</code></th>
186
+ <td>pgpool-II 的二进制程序将使用 <a href="#memcached_params">memcached</a> 作为 <a href="#memqcache">基于内存的查询缓存</a>。你必须先安装 <a href="http://libmemcached.org/libMemcached.html">libmemcached</a>。
187
+ <span class="version">V3.2 -</span></td>
188
+ </tr>
189
+ </table>
190
+ </dd>
191
+
192
+ <dt>make</dt>
193
+ <dd>
194
+ <pre>
195
+ make
196
+ make install
197
+ </pre>
198
+ <p>
199
+ 这将安装 install pgpool-II. (如果你使用的是 Solaris 或 FreeBSD,需要用 gmake 替换 make )
200
+ </p>
201
+ </dd>
202
+
203
+ <dt>安装 pgpool_regclass <span class="version">V3.0 -</span></dt>
204
+ <dd>
205
+ <p>
206
+ 如果你在使用 PostgreSQL 8.0 或之后的版本,强烈推荐在需要访问的 PostgreSQL
207
+ 中安装 pgpool_regclass 函数,因为它被 pgpool-II 内部使用。
208
+ 如果不这样做,在不同的 schema 中处理相同的表名会出现问题(临时表不会出问题)。
209
+ </p>
210
+ <pre>
211
+ cd pgpool-II-x.x.x/sql/pgpool-regclass
212
+ make
213
+ make install
214
+ </pre>
215
+ </p>
216
+ <p>
217
+ 在这之后:
218
+ <p>
219
+ <pre>
220
+ psql -f pgpool-regclass.sql template1
221
+ </pre>
222
+ </p>
223
+ <p>
224
+ 或者
225
+ <p>
226
+ <pre>
227
+ psql template1
228
+ CREATE EXTENSION pgpool_regclass;
229
+ </pre>
230
+ </p>
231
+ <p>
232
+ 应在每台通过 pgpool-II 访问的数据库中执行 pgpool-regclass.sql 或者 CREATE EXTENSION。
233
+ 你不需要在你执行“psql -f pgpool-regclass.sql template1” 或者 CREATE EXTENSION 后建立的数据库中这么做,
234
+ 因为这个模板数据库将被克隆成新建的数据库。
235
+ </p>
236
+ </dd>
237
+ <dt>建立 insert_lock 表<span class="version">V3.0 -</span></dt>
238
+ <dd>
239
+ <p>
240
+ 如果你在复制模式中使用了 insert_lock ,强烈推荐建立 pgpool_catalog.insert_lock 表,用于互斥。
241
+ 到现在为止,insert_lock 还能够工作。但是,在这种情况下,pgpool-II 需要锁定插入的目标表。
242
+ 这种行为和 pgpool-II 2.2 和 2.3 系列类似。由于表锁与 VACUUM 冲突,所以 INSERT 操作可能因而等待很长时间。
243
+ </p>
244
+
245
+ <pre>
246
+ cd pgpool-II-x.x.x/sql
247
+ psql -f insert_lock.sql template1
248
+ </pre>
249
+ <p>
250
+ 应在在每台通过 pgpool-II 访问的数据库中执行 insert_lock.sql。
251
+ 你不需要在你执行“psql -f insert_lock.sql template1”后建立的数据库中这么做,
252
+ 因为这个模板数据库将被克隆成新建的数据库。
253
+ </p>
254
+ </dd>
255
+ </dl>
256
+
257
+ <p>
258
+ 安装过程至此完成。如果你是使用 Solaris 或者 FreeBSD,
259
+ 你需要在以上的描述中用 “gmake” 代替 “make”,因为这些操作系统需要 GNU make。
260
+ </p>
261
+
262
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
263
+
264
+ <!-- ================================================================================ -->
265
+
266
+ <h1>配置 pgpool-II<a name="config"></a></h1>
267
+
268
+ <p>pgpool-II 的默认配置文件为 <code>/usr/local/etc/pgpool.conf</code> 和
269
+ <code>/usr/local/etc/pcp.conf</code>。pgpool-II 中有很多操作模式。
270
+ 每种模式都有相关的可被开启或者禁用的功能,而且有相关的配置项用于控制它们的行为。</p>
271
+
272
+ <table border>
273
+
274
+ <tr class="header">
275
+ <th>功能/模式</th>
276
+ <th>原始模式 (*3)</th>
277
+ <th>复制模式</th>
278
+ <th>主/备模式</th>
279
+ <th>并行查询模式</th>
280
+ </tr>
281
+
282
+ <tr>
283
+ <th>连接池</th>
284
+ <td align="center">X</td>
285
+ <td align="center">O</td>
286
+ <td align="center">O</td>
287
+ <td align="center">O</td>
288
+ </tr>
289
+
290
+ <tr>
291
+ <th>复制</th>
292
+ <td align="center">X</td>
293
+ <td align="center">O</td>
294
+ <td align="center">X</td>
295
+ <td align="center">(*1)</td>
296
+ </tr>
297
+
298
+ <tr>
299
+ <th>负载均衡</th>
300
+ <td align="center">X</td>
301
+ <td align="center">O</td>
302
+ <td align="center">O</td>
303
+ <td align="center">(*1)</td>
304
+ </tr>
305
+
306
+ <tr>
307
+ <th>故障恢复</th>
308
+ <td align="center">O</td>
309
+ <td align="center">O</td>
310
+ <td align="center">O</td>
311
+ <td align="center">X</td>
312
+ </tr>
313
+
314
+ <tr>
315
+ <th>在线恢复</th>
316
+ <td align="center">X</td>
317
+ <td align="center">O</td>
318
+ <td align="center">(*2)</td>
319
+ <td align="center">X</td>
320
+ </tr>
321
+
322
+ <tr>
323
+ <th>并行查询</th>
324
+ <td align="center">X</td>
325
+ <td align="center">X</td>
326
+ <td align="center">X</td>
327
+ <td align="center">O</td>
328
+ </tr>
329
+
330
+ <tr>
331
+ <th>需要的服务器数</th>
332
+ <td align="center">1 或更多</td>
333
+ <td align="center">2 或更多</td>
334
+ <td align="center">2 或更多</td>
335
+ <td align="center">2 或更多</td>
336
+ </tr>
337
+
338
+ <tr>
339
+ <th>是否需要系统数据库</th>
340
+ <td align="center">否</td>
341
+ <td align="center">否</td>
342
+ <td align="center">否</td>
343
+ <td align="center">是</td>
344
+ </tr>
345
+
346
+ </table>
347
+
348
+ <ul>
349
+ <li> O 意味着“可用”, X 意味着“不可用
350
+ <li>(*1) 并行查询模式需要同时打开复制和负载均衡,但是复制和负载均衡无法用于并行查询模式中的分布式表。
351
+ <li>(*2) 在线恢复可以和流复制同时使用。
352
+ <li>(*3) 客户端仅仅是通过 pgpool-II 连接到 PostgreSQL 服务器。这种模式仅仅用于限制到服务器的连接数,或者在多台机器上启用故障恢复。
353
+ </ul>
354
+
355
+ <h2 id="pcp_conf">配置 <code>pcp.conf</code></h2>
356
+
357
+ <p>pgpool-II 有一个控制接口,管理员可以通过它远程收集 pgpool-II 的状态信息或者终止 pgpool-II 进程。
358
+ <code>pcp.conf</code> 是用于这个接口认证的用户/密码文件。
359
+ 所有的模式都需要先设置 <code>pcp.conf</code> 文件。
360
+ 一个 <code>$prefix/etc/pcp.conf.sample</code> 文件在 pgpool-II 安装时已经被创建。
361
+ 重命名这个文件为 <code>pcp.conf</code> 并添加你要的用户名和密码到其中。
362
+ </p>
363
+
364
+ <pre>
365
+ cp $prefix/etc/pcp.conf.sample $prefix/etc/pcp.conf
366
+ </pre>
367
+ <p>空行或者以“<code>#</code>”开始的行将被认为是注释,会被忽略掉。一个用户名和对应的密码必须以以下的方式写成一行。</p>
368
+ <pre>
369
+ username:[password encrypted in md5]
370
+ </pre>
371
+ <p>
372
+ <code>[password encrypted in md5]</code> 可以通过 <code>$prefix/bin/pg_md5</code> 命令生成。
373
+ </p>
374
+
375
+ <pre>
376
+ pg_md5 -p
377
+ password: &lt;your password&gt;
378
+ </pre>
379
+ <p>
380
+ 或者
381
+ </p>
382
+ <pre>
383
+ ./pg_md5 foo
384
+ acbd18db4cc2f85cedef654fccc4a4d8
385
+ </pre>
386
+ <p>
387
+ <code>pcp.conf</code> 对于运行 pgpool-II 的用户必须可读。</p>
388
+
389
+ <h2>配置 <code>pgpool.conf</code></h2>
390
+
391
+ <p>就像之前说的,每种操作模式在 <code>pgpool.conf</code> 文件中有它对应的配置项。一个 <code>$prefix/etc/pgpool.conf.sample</code> 文件在 pgpool-II 安装时已经被创建。重命名这个文件为 <code>pgpool.conf</code> 并修改它的内容。
392
+ </p>
393
+
394
+ <pre>
395
+ cp $prefix/etc/pgpool.conf.sample $prefix/etc/pgpool.conf
396
+ </pre>
397
+
398
+ <p>
399
+ 针对每种不同的模式,提供了附加的示例 pgpool.conf。<span class="version">V2.3 -</span>
400
+ </p>
401
+ <p>
402
+ <table border>
403
+ <tr class="header">
404
+ <th>模式</th>
405
+ <th>示例文件</th>
406
+ </tr>
407
+
408
+ <tr>
409
+ <td>复制模式</td>
410
+ <td>pgpool.conf.sample-replication</td>
411
+ </tr>
412
+
413
+ <tr>
414
+ <td>主/备模式(Slony-I)</td>
415
+ <td>pgpool.conf.sample-master-slave</td>
416
+ </tr>
417
+
418
+ <tr>
419
+ <td>主/备模式(流复制)</td>
420
+ <td>pgpool.conf.sample-stream</td>
421
+ </tr>
422
+ </table>
423
+
424
+ <p>
425
+ 空行或者以“<code>#</code>”开始的行将被认为是注释,会被忽略掉。</p>
426
+ </p>
427
+
428
+ <h3>连接</h3>
429
+
430
+ <dl>
431
+ <dt><a name="LISTEN_ADDRESS"></a>listen_addresses</dt>
432
+ <dd>
433
+ <p>指定 pgpool-II 将接受 TCP/IP 连接的主机名或者IP地址。<code>'*'</code> 将接受所有的连接。<code>''</code> 将禁用 TCP/IP 连接。默认为 <code>'localhost'</code>。总是支持接受通过 UNIX 域套接字发起的连接。需要重启 pgpool-II 以使改动生效。</p>
434
+ </dd>
435
+
436
+ <dt><a name="PORT"></a>port</dt>
437
+ <dd>
438
+ <p>pgpool-II 监听 TCP/IP 连接的端口号。默认为 9999。需要重启 pgpool-II 以使改动生效。</p>
439
+ </dd>
440
+
441
+ <dt><a name="SOCKET_DIR"></a>socket_dir</dt>
442
+ <dd>
443
+ <p>pgpool-II 建立用于建立接受 UNIX 域套接字连接的目录。默认为 <code>'/tmp'</code>。注意,这个套接字可能被 cron 任务删除。我们建议设置这个值为 <code>'/var/run'</code> 或类似目录。需要重启 pgpool-II 以使改动生效。</p>
444
+ </dd>
445
+
446
+ <dt><a name="PCP_PORT"></a>pcp_port</dt>
447
+ <dd>
448
+ <p>PCP 进程接受连接的端口号。默认为 9898。本参数必须在服务器启动前设置。</p>
449
+ </dd>
450
+
451
+ <dt><a name="PCP_SOCKET_DIR"></a>pcp_socket_dir</dt>
452
+ <dd>
453
+ <p>PCP 进程用于建立接受 UNIX 域套接字连接的目录。默认为 <code>'/tmp'</code>。注意,这个套接字可能被 cron 任务删除。我们建议设置这个值为 <code>'/var/run'</code> 或类似目录。需要重启 pgpool-II 以使改动生效。</p>
454
+ </dd>
455
+
456
+ <dt><a name="BACKEND_SOCKET_DIR"></a>backend_socket_dir<span class="version">- V3.0</span></dt>
457
+ <dd>
458
+ <p class="version_notice">
459
+ <em class="caution">不推荐使用</em> :
460
+ 为了保持和 libpq 策略的一致性,反对使用本参数。参考 backend_hostname 参数来定义你对应的配置。
461
+ </p>
462
+ <p>本参数定义了 PostgreSQL 服务器的 UNIX 域套接字目录。
463
+ 默认为 <code>'/tmp'</code>.
464
+ </p>
465
+ <p>
466
+ 本参数必须在服务器启动前设置。</p>
467
+ </dd>
468
+ </dl>
469
+
470
+ <h3>池</h3>
471
+
472
+ <dl>
473
+ <dt><a name="NUM_INIT_CHILDREN"></a>num_init_children</dt>
474
+ <dd>
475
+ <p>预先生成的 pgpool-II 服务进程数。默认为 32。num_init_children 也是 pgpool-II 支持的从客户端发起的最大并发连接数。如果超过 num_init_children 数的客户端尝试连接到 pgpool-II,它们将被阻塞(而不是拒绝连接),直到到任何一个 pgpool-II 进程的连接被关闭为止。最多有 2*num_init_children 可以被放入等待队列。</p>
476
+
477
+ <p>对于以上内容的一些提示:</p>
478
+ <p>
479
+ <ul>
480
+ <li>取消一个执行中的查询将导致建立另一个到后端的连接;因此,如果所有的连接都在使用,则查询无法被取消。如果你想确保查询可以被取消,设置本值为预期最大连接数的两倍。</li>
481
+ <li>PostgreSQL 允许最多 max_connections - superuser_reserved_connections 个非超级用户的并发连接。</li>
482
+ </ul>
483
+ </p>
484
+ <p>归纳起来,max_pool,num_init_children,max_connections 和 superuser_reserved_connections 必须符合以下规则:
485
+ </p>
486
+ <p>
487
+ <pre>
488
+ max_pool*num_init_children <= (max_connections - superuser_reserved_connections) (不需要取消查询)
489
+ max_pool*num_init_children*2 <= (max_connections - superuser_reserved_connections) (需要取消查询)</pre>
490
+ <p>
491
+ 本参数必须在服务器启动前设置。
492
+ </p>
493
+ </dd>
494
+
495
+ <dt><a name="CHILD_LIFE_TIME"></a>child_life_time</dt>
496
+ <dd>
497
+ <p>pgpool-II 子进程的生命周期,单位为秒。如果子进程空闲了这么多秒,它将被终止,一个新的子进程将被创建。这个参数是用于应对内存泄露和其他不可预料错误的一个措施。默认值为 300 (5分钟)。0 将禁用本功能。注意它不影响尚未接受任何连接的进程。如果改变了这个值,你需要重新加载 pgpool.conf 以使变动生效。
498
+ </p>
499
+ </dd>
500
+
501
+ <dt><a name="CHILD_MAX_CONNECTIONS"></a>child_max_connections</dt>
502
+ <dd>
503
+ <p>当 pgpool-II 子进程处理这么多个客户端连接后,它将被终止。这个参数在繁忙到 child_life_time 和 connection_life_time 永远不会触发的服务器上有效。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
504
+ </p>
505
+ </dd>
506
+
507
+ <dt><a name="CLIENT_IDLE_LIMIT"></a>client_idle_limit</dt>
508
+ <dd>
509
+ <p>当一个客户端在执行最后一条查询后如果空闲到了 client_idle_limit 秒数,到这个客户端的连接将被断开。这在避免 pgpool 子进程被懒客户端占用或者探测断开的客户端和 pgpool 之间的 TCP/IP 连接非常有用。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
510
+ </p>
511
+
512
+ <dt><a name="ENABLE_POOL_HBA"></a>enable_pool_hba</dt>
513
+ <dd>
514
+ <p>
515
+ 如果为 true,则使用 pgpool_hba.conf 来进行客户端认证。
516
+ 参考<a href="#hba">设置用于客户端认证的 pool_hba.conf</a>。
517
+ </p>
518
+ <p>
519
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
520
+ </p>
521
+ </dd>
522
+
523
+ <dt><a name="POOL_PASSWD"></a>pool_passwd</dt>
524
+ <dd>
525
+ <p>
526
+ 指定用于 md5 认证的文件名。默认值为"pool_passwd"。"" 表示禁用。
527
+ 参考 <a href="#md5">认证 / 访问控制</a> 获得更多信息。
528
+ </p>
529
+ <p>
530
+ 如果你改变了这个值,需要重启 pgpool-II 以生效。
531
+ </p>
532
+ </dd>
533
+
534
+ <dt><a name="AUTHENTICATION_TIMEOUT"></a>authentication_timeout</dt>
535
+ <dd>
536
+ <p>指定 pgpool 认证超时的时长。0 指禁用超时,默认值为 60 。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
537
+ </p>
538
+ </dl>
539
+
540
+ <h3>日志</h3>
541
+
542
+ <dl>
543
+ <dt><a name="LOG_DESTINATION"></a>log_destination <span class="version">V3.1 -</span></dt>
544
+ <dd>
545
+ <p>pgpool-II 支持多种记录服务器消息的方式,包括 stderr 和 syslog。默认为记录到 stderr。
546
+ </p>
547
+ <p>注:要使用syslog 作为 log_destination 的选项,你将需要更改你系统的 syslog 守护进程的配置。pgpool-II 可以记录到 syslog 设备 LOCAL0 到 LOCAL7 (参考 syslog_facility),
548
+ 但是大部分平台的默认的 syslog 配置将忽略这些消息。你需要添加如下一些内容
549
+ </p>
550
+ <pre>
551
+ local0.* /var/log/pgpool.log </pre>
552
+ <p>到 syslog 守护进程的配置文件以使它生效。
553
+ </p>
554
+ </dd>
555
+
556
+ <dt><a name="PRINT_TIMESTAMP"></a>print_timestamp</dt>
557
+ <dd>
558
+ <p>如果本值被设置为 true ,则将在日志中添加时间戳。默认值为 true。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
559
+ </p>
560
+ </dd>
561
+
562
+ <dt><a name="LOG_CONNECTIONS"></a>log_connections</dt>
563
+ <dd>
564
+ <p>如果为 true,进入的连接将被打印到日志中。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
565
+ </p>
566
+ </dd>
567
+
568
+ <dt><a name="LOG_HOSTNAME"></a>log_hostname</dt>
569
+ <dd>
570
+ <p>如果为 true,ps 命令将显示客户端的主机名而不是 IP 地址。而且,如果 log_connections 被开启,也会将主机名写入日志。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
571
+ </p>
572
+ </dd>
573
+
574
+ <dt><a name="LOG_STATEMENT"></a>log_statement</dt>
575
+ <dd>
576
+ <p>当设置为 true 时生成 SQL 日志消息。这类似于 PostgreSQL 中的 log_statement 参数。即使调试选项没有在启动的时候传递到 pgpool-II,它也会产生日志。</p>
577
+ </dd>
578
+
579
+ <dt><a name="LOG_PER_NODE_STATEMENT"></a>log_per_node_statement <span class="version">V2.3 -</span></dt>
580
+ <dd>
581
+ <p>类似于 log_statement,除了它是针对每个 DB 节点产生日志外。例如它对于确定复制是否正常运行非常有用。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
582
+ </p>
583
+ </dd>
584
+
585
+ <dt><a name="SYSLOG_FACILITY"></a>syslog_facility <span class="version">V3.1 -</span></dt>
586
+ <dd>
587
+ <p>当记录日志到 syslog 被启用,本参数确定被使用的 syslog “设备”。你可以使用 LOCAL0, LOCAL1, LOCAL2, LOCAL3, LOCAL4, LOCAL5, LOCAL6, LOCAL7;默认为 LOCAL0。还请参考你系统的 syslog 守护进程的文档。
588
+ </p>
589
+ </dd>
590
+
591
+ <dt><a name="SYSLOG_IDENT"></a>syslog_ident <span class="version">V3.1 -</span></dt>
592
+ <dd>
593
+ <p>当记录日志到 syslog 被启用,本参数确定用于标记 syslog 中 pgpool-II 消息的程序名。默认为“pgpool”。
594
+ </p>
595
+ </dd>
596
+
597
+ <dt><a name="DEBUG_LEVEL"></a>debug_level <span class="version">V3.0 -</span></dt>
598
+ <dd>
599
+ <p>调试消息详细程度级别。0 表示没有消息。大于 1 表示更详细的消息。默认值为 0。
600
+ </p>
601
+ </dd>
602
+ </dl>
603
+
604
+ <h3>文件位置</h3>
605
+
606
+ <dl>
607
+ <dt><a name="PID_FILE_NAME"></a>pid_file_name</dt>
608
+ <dd>
609
+ <p>到包含 pgpool-II 进程 ID 的文件的完整路径名。默认为 <code>'/var/run/pgpool/pgpool.pid'</code>。需要重启 pgpool-II 以使改动生效。
610
+ </p>
611
+ </dd>
612
+ <dt><a name="LOGDIR"></a>logdir</dt>
613
+ <dd>
614
+ <p>保存日志文件的目录。 pgpool_status 将被写入这个目录。
615
+ </p>
616
+ </dd>
617
+ </dl>
618
+
619
+ <h3>连接池</h3>
620
+
621
+ <dl>
622
+ <dt><a name="CONNECTION_CACHE"></a>connection_cache</dt>
623
+ <dd>
624
+ <p>如果本值被设置为 true,则缓存到 PostgreSQL 的连接。默认为 true。需要重启 pgpool-II 以使改动生效。</p>
625
+ </dd>
626
+ </dl>
627
+
628
+ <h3>健康检查</h3>
629
+
630
+ <dl>
631
+ <dt><a name="HEALTH_CHECK_TIMEOUT"></a>health_check_timeout</dt>
632
+ <dd>
633
+ <p>pgpool-II 定期尝试连接到后台以检测服务器是否在服务器或网络上有问题。这种错误检测过程被称为“健康检查”。如果检测到错误,则 pgpool-II 会尝试进行故障恢复或者退化操作。<br>
634
+ 本参数用于避免健康检查在例如网线断开等情况下等待很长时间。超时值的单位为秒。默认值为 20 。0 禁用超时(一直等待到 TCP/IP 超时)。<br>
635
+ 健康检查需要额外的到后端程序的连接,所以 <code>postgresql.conf</code> 中的 <code>max_connections</code> 需要加一。<br>
636
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
637
+ </p>
638
+ </dd>
639
+
640
+ <dt><a name="HEALTH_CHECK_PERIOD"></a>health_check_period</dt>
641
+ <dd>
642
+ <p>本参数指出健康检查的间隔,单位为秒。默认值为 0 ,代表禁用健康检查。<br>
643
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
644
+ </p>
645
+ </dd>
646
+
647
+ <dt><a name="HEALTH_CHECK_USER"></a>health_check_user</dt>
648
+ <dd>
649
+ <p>用于执行健康检查的用户。用户必须存在于 PostgreSQL 后台中。<br>
650
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
651
+ </p>
652
+ </dd>
653
+
654
+ <dt><a name="HEALTH_CHECK_PASSWORD"></a>health_check_password <span class="version">V3.1 -</span></dt>
655
+ <dd>
656
+ <p>用于执行健康检查的用户的密码。<br>
657
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
658
+ </p>
659
+ </dd>
660
+ <dt><a name="HEALTH_CHECK_MAX_RETRIES"></a>health_check_max_retries <span class="version">V3.2 -</span></dt>
661
+ <dd>
662
+ <p>在执行失效故障切换前尝试的最大失效健康检查次数。这个参数对于网络不稳定的时,健康检查失败但主节点依旧正常的情况下非常有效。默认值为 0,也就是不重试。如果你想启用 health_check_max_retries,建议你禁用 <a href="#FAIL_OVER_ON_BACKEND_ERROR">fail_over_on_backend_error</a>。<br>
663
+ 如果你改变了 health_check_max_retries,需要重新加载 pgpool.conf。
664
+ </p>
665
+ </dd>
666
+
667
+ <dt><a name="HEALTH_CHECK_RETRY_DELAY"></a>health_check_retry_delay <span class="version">V3.2 -</span></dt>
668
+ <dd>
669
+ <p>失效健康检查重试的间隔时间(单位为秒)( health_check_max_retries > 0 时有效 )。如果为 0 则立即重试(不延迟)。<br>
670
+ 如果你改变了 health_check_retry_delay,需要重新加载 pgpool.conf。
671
+ </p>
672
+ </dd>
673
+
674
+ <dt><a name="SEARCH_PRIMARY_NODE_TIMEOUT"></a>search_primary_node_timeout <span class="version">V3.3 -</span></dt>
675
+ <dd>
676
+ <p>本参数指定在发生故障切换的时候查找一个主节点的最长时间,单位为秒。默认值为 10。pgpool-II 将在发生故障切换的时候在设置的时间内尝试搜索主节点,如果到达这么长时间未搜索到则放弃搜索。0 表示一直尝试。本参数在流复制模式之外的情况下被忽略。<br>
677
+ </p>
678
+ <p>
679
+ 如果你改变了 search_primary_node_timeout,需要重新加载 pgpool.conf。
680
+ </p>
681
+ </dd>
682
+ </dl>
683
+
684
+ <h3>故障切换和恢复</h3>
685
+
686
+ <dl>
687
+ <dt><a name="FAILOVER_COMMAND"></a>failover_command</dt>
688
+ <dd>
689
+ <p>本参数指定当一个节点断开连接时执行的命令。pgpool-II 使用后台对应的信息代替以下的特别字符。
690
+ </p>
691
+
692
+
693
+ <table border>
694
+ <tr><td>特殊字符</td><td>描述</td></tr>
695
+ <tr><td>%d</td><td>断开连接的节点的后台 ID。</td></tr>
696
+ <tr><td>%h</td><td>断开连接的节点的主机名。</td></tr>
697
+ <tr><td>%p</td><td>断开连接的节点的端口号。</td></tr>
698
+ <tr><td>%D</td><td>断开连接的节点的数据库实例所在目录。</td></tr>
699
+ <tr><td>%M</td><td>旧的主节点 ID。</td></tr>
700
+ <tr><td>%m</td><td>新的主节点 ID。</td></tr>
701
+ <tr><td>%H</td><td>新的主节点主机名。</td></tr>
702
+ <tr><td>%P</td><td>旧的第一节点 ID。</td></tr>
703
+ <tr><td>%r</td><td>新的主节点的端口号。</td></tr>
704
+ <tr><td>%R</td><td>新的主节点的数据库实例所在目录。</td></tr>
705
+ <tr><td>%%</td><td>'%' 字符</td></tr>
706
+ </table>
707
+
708
+ <p>
709
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
710
+ </p>
711
+
712
+ <p>
713
+ 当进行故障切换时,pgpool 杀掉它的所有子进程,这将顺序终止所有的到 pgpool 的会话。然后,pgpool 调用 failover_command 并等待它完成。然后,pgpool 启动新的子进程并再次开始从客户端接受连接。
714
+ </p>
715
+
716
+ <dt><a name="FAILBACK_COMMAND"></a>failback_command</dt>
717
+ <dd>
718
+ <p>
719
+ 本参数指定当一个节点连接时执行的命令。pgpool-II 使用后台对应的信息代替以下的特别字符。
720
+
721
+ </p>
722
+
723
+ <table border>
724
+ <tr><td>特殊字符</td><td>描述</td></tr>
725
+ <tr><td>%d</td><td>新连接上的节点的后台 ID。</td></tr>
726
+ <tr><td>%h</td><td>新连接上的节点的主机名。</td></tr>
727
+ <tr><td>%p</td><td>新连接上的节点的端口号。</td></tr>
728
+ <tr><td>%D</td><td>新连接上的节点的数据库实例所在目录。</td></tr>
729
+ <tr><td>%M</td><td>旧的主节点 ID。</td></tr>
730
+ <tr><td>%m</td><td>新的主节点 ID。</td></tr>
731
+ <tr><td>%H</td><td>新的主节点主机名。</td></tr>
732
+ <tr><td>%P</td><td>旧的第一节点 ID。</td></tr>
733
+ <tr><td>%r</td><td>新的主节点的端口号。</td></tr>
734
+ <tr><td>%R</td><td>新的主节点的数据库实例所在目录。</td></tr>
735
+ <tr><td>%%</td><td>'%' 字符</td></tr>
736
+ </table>
737
+
738
+ <p>
739
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
740
+ </p>
741
+
742
+ <dt><a name="FOLLOW_MASTER_COMMAND"></a>follow_master_command <span class="version">V3.1 -</span></dt>
743
+ <dd>
744
+ <p>
745
+ 本参数指定一个在主备流复制模式中发生主节点故障恢复后执行的命令。pgpool-II 使用后台对应的信息代替以下的特别字符。
746
+ </p>
747
+
748
+ <table border>
749
+ <tr><td>特殊字符</td><td>描述</td></tr>
750
+ <tr><td>%d</td><td>断开连接的节点的后台 ID。</td></tr>
751
+ <tr><td>%h</td><td>断开连接的节点的主机名。</td></tr>
752
+ <tr><td>%p</td><td>断开连接的节点的端口号。</td></tr>
753
+ <tr><td>%D</td><td>断开连接的节点的数据库实例所在目录。</td></tr>
754
+ <tr><td>%M</td><td>旧的主节点 ID。</td></tr>
755
+ <tr><td>%m</td><td>新的主节点 ID。</td></tr>
756
+ <tr><td>%H</td><td>新的主节点主机名。</td></tr>
757
+ <tr><td>%P</td><td>旧的第一节点 ID。</td></tr>
758
+ <tr><td>%r</td><td>新的主节点的端口号。</td></tr>
759
+ <tr><td>%R</td><td>新的主节点的数据库实例所在目录。</td></tr>
760
+ <tr><td>%%</td><td>'%' 字符</td></tr>
761
+ </table>
762
+
763
+ <p>
764
+ 如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
765
+ </p>
766
+
767
+ <p>
768
+ 如果 follow_master_commnd 不为空,当一个主备流复制中的主节点的故障切换完成,pgpool 退化所有的除新的主节点外的所有节点并启动一个新的子进程,再次准备好接受客户端的连接。在这之后,pgpool 针对每个退化的节点运行 ‘follow_master_command’ 指定的命令。通常,这个命令应该用于调用例如 pcp_recovery_node 命令来从新的主节点恢复备节点。
769
+ </p>
770
+
771
+ <dt><a name="FAIL_OVER_ON_BACKEND_ERROR"></a>fail_over_on_backend_error <span class="version">V2.3 -</span></dt>
772
+ <dd>
773
+ <p>
774
+ 如果为 true,当往后台进程的通信中写入数据时发生错误,pgpool-II 将触发故障处理过程。这和 pgpool-II 2.2.x 甚至以前版本的行为一样。如果设置为 false,则 pgpool 将报告错误并断开该连接。请注意如果设置为 true,当连接到一个后台进程失败或者 pgpool 探测到 postmaster 由于管理原因关闭,pgpool 也会执行故障恢复过程。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
775
+ </p>
776
+ </dd>
777
+ </dl>
778
+
779
+ <h3>负载均衡模式</h3>
780
+
781
+ <dl>
782
+ <dt><a name="IGNORE_LEADING_WHITE_SPACE"></a>ignore_leading_white_space</dt>
783
+ <dd>
784
+ <p>在负载均衡模式中 pgpool-II 忽略 SQL 查询语句前面的空白字符。如果使用类似于 DBI/DBD:Pg 一类的在用户的查询前增加空白的 API 中非常有用。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。</p>
785
+ </dd>
786
+ </dl>
787
+
788
+ <h3>后端(PostgreSQL服务器)</h3>
789
+
790
+ <dl>
791
+ <dt><a name="BACKEND_HOSTNAME"></a>backend_hostname</dt>
792
+ <dd>
793
+ <p>指出连接到 PostgreSQL 后台程序的地址。它用于 pgpool-II 与服务器通信。如果你改变了这个值,需要重新加载 pgpool.conf 以使变动生效。
794
+ </p>
795
+ <p>
796
+ 对于 TCP/IP 通信,本参数可以是一个主机名或者IP地址。如果它是从斜线开始的,它指出是通过 UNIX 域套接字通信,而不是 TCP/IP 协议;它的值为存储套接字文件所在的目录。如果 backend_host 为空,则它的默认行为是通过 <code>/tmp</code> 中的 UNIX 域套接字连接。
797
+ </p>
798
+ <p>
799
+ 可以通过在本参数名的末尾添加一个数字来指定多个后台程序(例如<code>backend_hostname0</code>)。这个数字对应为“数据库节点 ID”,是从 0 开始的正整数。被设置数据库节点ID为 0 的后台程序后台程序将被叫做“主数据库”。当定义了多个后台程序时,即使主数据库当机后依然能继续(某些模式下不行)。在这种情况下,存活的最小的数据库节点编号的数据库将被变成新的主数据库。
800
+ <p> 请注意有编号为 0 的节点在流复制中没有其他意义。但是,你需要注意数据库节点是不是“主节点”。请参考 <a href="#stream">流复制</a> 获得更多细节。</p>
801
+ </p>
802
+ <p>如果你只计划使用一台 PostgreSQL 服务器,可以通过 <code>backend_hostname0</code> 指定。</p>
803
+ <p>
804
+ 可以通过配置本参数后重新加载配置文件添加新的节点。但是,对已有的值无法更新,所以这种情况下你必须重启 pgpool-II。
805
+ </p>
806
+ </dd>
807
+
808
+ <dt><a name="BACKEND_PORT"></a>backend_port</dt>
809
+ <dd>
810
+ <p>
811
+ 指定后台程序的端口号。可以通过在本参数名的末尾添加一个数字来指定多个后台程序(例如<code>backend_port0</code>)。如果你只计划使用一台 PostgreSQL 服务器,可以通过 <code>backend_port0</code> 指定。</p>
812
+ <p>
813
+ 可以通过配置本参数后重新加载配置文件添加新的后台端口。但是,对已有的值无法更新,所以这种情况下你必须重启 pgpool-II。
814
+ </p>
815
+ </dd>
816
+
817
+ <dt><a name="BACKEND_WEIGHT"></a>backend_weight</dt>
818
+ <dd>
819
+ <p>指定后台程序的负载均衡权重。可以通过在本参数名的末尾添加一个数字来指定多个后台程序(例如<code>backend_weight0</code>)。如果你只计划使用一台 PostgreSQL 服务器,可以通过 <code>backend_weight0</code> 指定。</p><p>在原始模式中,请将本值设置为 1。</p>
820
+ <p>
821
+ 可以通过配置本参数后重新加载配置文件添加新的负载均衡权重。但是,对已有的值无法更新,所以这种情况下你必须重启 pgpool-II。
822
+ </p>
823
+ <p>
824
+ 在 pgpool-II 2.2.6/2.3 或者更新的版本中,你可以通过重新加载配置文件来改变本值。但这只对新连接的客户会话生效。这在主备模式中可以避免任何执行一些管理工作的查询被发送到备用节点上。
825
+ </p>
826
+ </dd>
827
+
828
+ <dt><a name="BACKEND_DATA_DIRECTORY"></a>backend_data_directory</dt>
829
+ <dd>
830
+ <p>指定后台的数据库实例的目录。可以通过在本参数名的末尾添加一个数字来指定多个后台程序(例如<code>backend_data_directory0</code>)。如果你不打算使用在线恢复,你可以不设置本参数。</p>
831
+
832
+ <p>
833
+ 可以通过配置本参数后重新加载配置文件添加新的后台的数据库实例目录。但是,对已有的值无法更新,所以这种情况下你必须重启 pgpool-II。
834
+ </p>
835
+ </dd>
836
+
837
+ <dt><a name="BACKEND_FLAG"></a>backend_flag <span class="version">V3.1 -</span></dt>
838
+ <dd>
839
+ <p>控制大量的后台程序的行为。可以通过在本参数名的末尾添加一个数字来指定多个后台程序(例如<code>backend_flag0</code>)
840
+ </p>
841
+ <p>
842
+ 当前支持以下的内容。多个标志可以通过“|”来分隔。
843
+ </p>
844
+
845
+ <table border>
846
+ <tr><th class="nodec">ALLOW_TO_FAILOVER</th>
847
+ <td>允许故障切换或者从后台程序断开。本值为默认值。指定本值后,不能同时指定 DISALLOW_TO_FAILOVER 。
848
+ </td></tr>
849
+ <tr><th class="nodec">DISALLOW_TO_FAILOVER</th>
850
+ <td>不允许故障切换或者从后台程序断开。本值在你使用 HA(高可用性)软件例如 Heartbeat 或者 Packmaker 来保护后台程序时非常有用。
851
+ 本值为默认值。指定本值后,不能同时指定 DISALLOW_TO_FAILOVER 。
852
+ </td></tr>
853
+ </table>
854
+ </dt>
855
+ </dl>
856
+
857
+ <h3>SSL</h3>
858
+
859
+ <dl>
860
+ <dt><a name="SSL"></a>ssl <span class="version">V2.3 -</span></dt>
861
+ <dd>
862
+ <p>如果设置为 ture,则启用了到前端程序和后端程序的连接的 ssl 支持。注意为了能与前端程序进行 SSL 连接,必须设置 <code>ssl_key</code> 和 <code>ssl_cert</code>。</p>
863
+ <p>SSL 默认被关闭。就像在 <a href="#install">pgpool-II 的安装</a> 小节所说的,注意必须在编译时配置 OpenSSL 支持才能打开 SSL 支持。</p>
864
+ <p>如果修改了 SSL 相关的设置, pgpool-II 守护进程必须重启。</p>
865
+ </dd>
866
+
867
+ <dt><a name="SSL_KEY"></a>ssl_key <span class="version">V2.3 -</span></dt>
868
+ <dd>
869
+ <p>对于进入的连接使用的私钥文件所在路径。
870
+ </p>
871
+ <p>本选项没有默认值,如果本值不设置,则对于进入的连接将禁用 SSL。
872
+ </p>
873
+ </dd>
874
+
875
+ <dt><a name="SSL_CERT"></a>ssl_cert <span class="version">V2.3 -</span></dt>
876
+ <dd>
877
+ <p>
878
+ 对于进入的连接使用的公共 x509 证书文件所在的路径。
879
+ </p>
880
+ <p>本选项没有默认值,如果本值不设置,则对于进入的连接将禁用 SSL。
881
+ </p>
882
+ </dd>
883
+ </dl>
884
+
885
+ <h3>其他</h3>
886
+
887
+ <dl>
888
+ <dt><a name="RELCACHE_EXPIRE"></a>relcache_expire <span class="version">V3.1 -</span></dt>
889
+ <dd>
890
+ <p>
891
+ 关系缓存的生命周期。0(默认值)表示没有缓冲区过期。
892
+ 关系缓存用于缓存用来获取包含表结构信息或一个表是不是一个临时表等大量信息的相关的
893
+ PostgreSQL 系统 catalog 的查询结果。缓存位于 pgpool 子进程的本地,并被保存到它的生命结束。
894
+ 如果某些人使用了 ALTER TABLE 修改了表结构或其他类似内容,关系缓存不再一致。
895
+ 为了这个目的,relcache_expire 控制缓存的生命周期。
896
+ </p>
897
+ </dd>
898
+
899
+ <dt><a name="RELCACHE_SIZE"></a>relcache_size <span class="version">V3.2 -</span></dt>
900
+ <dd>
901
+ <p>
902
+ relcache 的条目数。默认为 256。
903
+ 如果你频繁看到以下信息,请增加此数量。
904
+ </p>
905
+ <pre>
906
+ "pool_search_relcache: cache replacement happened"
907
+ </pre>
908
+ </dd>
909
+
910
+ <dt><a name="CHECK_TEMP_TABLE"></a>check_temp_table <span class="version">V3.2 -</span></dt>
911
+ <dd>
912
+ <p>
913
+ 如果为 on,在 SELECT 语句中启用临时表检查。这会在启动查询前查询主节点上的系统对象,
914
+ 因此增加了主节点上的负载。如果你确定你的系统不会使用临时表,并且你想降低对主节点的访问,
915
+ 弄可以将它设置为 off。默认为 on。
916
+ </p>
917
+ </dd>
918
+
919
+ </dl>
920
+
921
+ <h2>生成 SSL 证书</h2>
922
+ <p>证书处理不在本文档讨论范围。postgresql.org 的 <a href="http://developer.postgresql.org/pgdocs/postgres/ssl-tcp.html">
923
+ Secure TCP/IP Connections with SSL</a> 页面指出了一些用来生成自签名证书的示例命令。
924
+ </p>
925
+
926
+ <h2 id="failover_in_raw_mode">在原始模式中的故障切换</h2>
927
+
928
+ <p>如果定义了多个服务器,可以在原始模式中进行故障切换。
929
+ pgpool-II 在普通操作中通常访问 <code>backend_hostname0</code> 指定的后台程序。
930
+ 如果 backend_hostname0 因为某些原因不能正常工作,pgpool-II 尝试访问 backend_hostname1 指定的后台程序。
931
+ 如果它也不能正常工作,pgpool-II 尝试访问 backend_hostname2,3 等等。</p>
932
+
933
+ <p class="top_link"><a href="#Top">back to top</a></p>
934
+
935
+ <!-- ================================================================================ -->
936
+
937
+ <h1><a name="connection_pool_mode"></a>连接池模式</h1>
938
+
939
+ <p>在连接池模式中,所有在原始模式中的功能以及连接池功能都可以使用。要启用本模式,你需要设置 "<a href="#CONNECTION_CACHE">connection_cache</a>" 为 on。
940
+ 以下参数会对连接池产生影响。</p>
941
+
942
+ <dl>
943
+ <dt><a name="MAX_POOL"></a>max_pool</dt>
944
+ <dd>
945
+ <p>在 pgpool-II 子进程中缓存的最大连接数。当有新的连接使用相同的用户名连接到相同的数据库,pgpool-II 将重用缓存的连接。如果不是,则 pgpool-II 建立一个新的连接到 PostgreSQL。如果缓存的连接数达到了 max_pool,则最老的连接将被抛弃,并使用这个槽位来保存新的连接。默认值为 4。请小心通过 pgpool-II 进程到后台的连接数可能达到
946
+ <code><a href="#NUM_INIT_CHILDREN">num_init_children</a></code> *
947
+ <code><a href="#MAX_POOL">max_pool</a></code> 个。需要重启 pgpool-II 以使改动生效。
948
+ </p>
949
+ </dd>
950
+ <dt><a name="CONNECTION_LIFE_TIME"></a>connection_life_time</dt>
951
+ <dd>
952
+ <p>缓存的连接的过期时长,单位为秒。过期的缓存连接将被关闭。默认值为 0,表示缓存的连接将不被关闭。</p>
953
+ </dd>
954
+ <dt><a name="RESET_QUERY_LIST"></a>reset_query_list</dt>
955
+ <dd>
956
+ <p>指定在推出一个会话时发送到后台程序的SQL命令。多个命令可以通过“;”隔开。默认为以下的设置但你可以根据你的需求改变。
957
+ <pre>
958
+ reset_query_list = 'ABORT; DISCARD ALL'</pre>
959
+ <p>
960
+ 不同版本的 PostgreSQL 需要使用不同的命令。以下为推荐的设置。
961
+ </p>
962
+ <p>
963
+ <table border>
964
+ <tr class="header"><th>PostgreSQL 版本</th><th>reset_query_list 的值</th></tr>
965
+ <tr><td>7.1 或更老的版本</td><td>ABORT</td></tr>
966
+ <tr><td>7.2 到 8.2</td><td>ABORT; RESET ALL; SET SESSION AUTHORIZATION DEFAULT</td></tr>
967
+ <tr><td>8.3 或更新的版本</td><td>ABORT; DISCARD ALL</td></tr>
968
+ </table>
969
+ </p>
970
+ <ul>
971
+ <li>在 7.4 或更新的版本中,当不是在一个事务块中的时候,“ABORT”将不会被发出。
972
+ </ul>
973
+
974
+ <p>
975
+ 修改本参数后需要重新加载 pgpool.conf 以使改变生效。
976
+ </p>
977
+ </dd>
978
+ </dl>
979
+
980
+ <h2><p>连接池模式中的故障切换</p></h2>
981
+
982
+ <p>连接池模式中的故障切换和原始模式的相同。</p>
983
+
984
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
985
+
986
+ <!-- ================================================================================ -->
987
+
988
+ <h1><a name="replication_mode"></a>复制模式</h1>
989
+
990
+ <p>本模式在后台程序间启用了数据复制。以下配置参数必须在设置以上参数之外另外设置。</p>
991
+
992
+ <dl>
993
+ <dt><a name="REPLICATION_MODE"></a>replication_mode</dt>
994
+ <dd>
995
+ <p>设置为 true 以启用复制模式。默认值为 false。</p>
996
+ </dd>
997
+
998
+ <dt><a name="LOAD_BALANCE_MODE"></a>load_balance_mode</dt>
999
+ <dd>
1000
+ <p>当设置为 true 时,SELECT 查询将被分发到每个后台程序上用于负载均衡。默认值为 false。本参数必须在服务器启动前设置。</p>
1001
+ </dd>
1002
+
1003
+ <dt><a name="REPLICATION_STOP_ON_MISMATCH"></a>replication_stop_on_mismatch</dt>
1004
+ <dd>
1005
+ <p>当设置为 true 时,当不同的后台程序返回不同的包类型时,则和其他后台程序差别最大的后台程序将被退化。</p>
1006
+ <p>一个典型的用例为一个事务中的 SELECT 语句,在 <a href="#REPLICATE_SELECT">replicate_select</a> 设置为 true 的情况下,一个 SELECT 语句从不同的后台程序中返回不同的行数。非 SELECT 语句也可能触发这种情况。
1007
+ 例如,一个后台程序执行 UPDATE 成功但其他的失败。注意 pgpool 不会检查 SELECT 返回的记录的内容。如果设置为 false,则会话被终止但后台程序不被退化。默认值为 false。</p>
1008
+ </dd>
1009
+
1010
+ <dt><a name="FAILOVER_IF_AFFECTED_TUPLES_MISMATCH"></a>failover_if_affected_tuples_mismatch <span class="version">V3.0 -</span></dt>
1011
+ <dd>
1012
+ <p>当设置为 true 时,在执行 INSERT/UPDATE/DELETE 后不同的后台程序返回不同的生效记录数,则和其他后台程序差别最大的后台程序将被退化。如果设置为 false,则会话被终止但后台程序不被退化。默认值为 false。</p>
1013
+ </dd>
1014
+
1015
+ <dt><a name="WHITE_FUNCTION_LIST"></a>white_function_list <span class="version">V3.0 -</span></dt>
1016
+ <dd>
1017
+ <p>指定一系列用逗号隔开的<strong>不会</strong>更新数据库的函数名。在复制模式中,不在本列表中指定的函数将即不会被负载均衡,也不会被复制。在主备模式中,这些 SELECT 语句只被发送到主节点。</p>
1018
+ <p>你可以使用正则表达式来匹配函数名,例如你通过前缀“get_”或“select_”来作为你只读函数的开头:
1019
+ </p>
1020
+ <pre>
1021
+ white_function_list = 'get_.*,select_.*'
1022
+ </pre>
1023
+ </dd>
1024
+
1025
+ <dt><a name="BLACK_FUNCTION_LIST"></a>black_function_list <span class="version">V3.0 -</span></dt>
1026
+ <dd>
1027
+ <p>指定一系列用逗号隔开的<strong>会</strong>更新数据库的函数名。在复制模式中,在本列表中指定的函数将即不会被负载均衡,也不会被复制。在主备模式中,这些 SELECT 语句只被发送到主节点。</p>
1028
+ <p>你可以使用正则表达式来匹配函数名,例如你通过前缀“set_”、“update_”、“delete_”或“insert_”来作为你只读函数的开头:</p>
1029
+ <pre>
1030
+ black_function_list = 'nextval,setval,set_.*,update_.*,delete_.*,insert_.*'
1031
+ </pre>
1032
+ <p>
1033
+ 以上两项不能同时配置。
1034
+ </p>
1035
+ <p>在 pgpool-II 3.0 之前,nextval() 和 setval() 是已知的会写入数据库的函数。你可以通使用 white_function_list 好 balck_function_list 来做到:
1036
+ </p>
1037
+ <p>
1038
+ <pre>
1039
+ white_function_list = ''
1040
+ black_function_list = 'nextval,setval,lastval,currval'
1041
+ </pre>
1042
+ </p>
1043
+ <p>注意我们在 nextval 和 setval 后面追加了 lastval 和 currval。虽然 lastval() 和 currval() 不是会写入的函数,但添加 lastval() 和 currval() 可以避免这些函数被无意地被负载均衡到其他的数据库节点而导致错误。添加到 black_function_list 将避免它们被负载均衡。
1044
+ </p>
1045
+ </dd>
1046
+
1047
+ <dt><a name="replicate_select"></a>replicate_select</dt>
1048
+ <dd>
1049
+ <p>当设置为 true,pgpool-II 在复制模式中将复制 SELECT 语句。如果为 false,则 pgpool-II 只发送它们到主数据库。默认为 false。
1050
+ </p>
1051
+ <p>
1052
+ 如果 SELECT 查询是在一个显式的事务块中,<a href="#REPLICATE_SELECT">replicate_select</a> 和 <a href="#LOAD_BALANCE_MODE">load_balance_mode</a> 将影响复制的工作模式。以下为具体的细节。
1053
+ </p>
1054
+
1055
+ <p>
1056
+ <table border>
1057
+
1058
+ <tr>
1059
+ <th class="nodec">SELECT 在一个事务块中</th>
1060
+ <td>Y</td>
1061
+ <td>Y</td>
1062
+ <td>Y</td>
1063
+ <td>N</td>
1064
+ <td>N</td>
1065
+ <td>N</td>
1066
+ <td>Y</td>
1067
+ <td>N</td>
1068
+ </tr>
1069
+
1070
+ <tr>
1071
+ <th class="nodec">replicate_select 为 true</th>
1072
+ <td>Y</td>
1073
+ <td>Y</td>
1074
+ <td>N</td>
1075
+ <td>N</td>
1076
+ <td>Y</td>
1077
+ <td>Y</td>
1078
+ <td>N</td>
1079
+ <td>N</td>
1080
+ </tr>
1081
+
1082
+ <tr>
1083
+ <th class="nodec">load_balance_mode 为 true</th>
1084
+ <td>Y</td>
1085
+ <td>N</td>
1086
+ <td>N</td>
1087
+ <td>N</td>
1088
+ <td>Y</td>
1089
+ <td>N</td>
1090
+ <td>Y</td>
1091
+ <td>Y</td>
1092
+ </tr>
1093
+
1094
+ <tr>
1095
+ <th class="nodec">结果(R: 负载均衡, M: 只发送到主节点, L: 负载均衡)</th>
1096
+ <td>R</td>
1097
+ <td>R</td>
1098
+ <td>M</td>
1099
+ <td>M</td>
1100
+ <td>R</td>
1101
+ <td>R</td>
1102
+ <td>M</td>
1103
+ <td>L</td>
1104
+ </tr>
1105
+ </table>
1106
+ </p>
1107
+ </dd>
1108
+
1109
+ <dt><a name="INSERT_LOCK"></a>insert_lock</dt>
1110
+ <dd>
1111
+ <p>如果在包含 SERIAL 类型的表中做复制, SERIAL 列的值在不同后台间可能不同。这个问题可以通过显式的锁表解决(当然,事务的并发性将被严重退化)。为了达到这个目的,必须做以下的改变:</p>
1112
+
1113
+ <pre>
1114
+ INSERT INTO ...
1115
+ </pre>
1116
+
1117
+ <p>
1118
+ 改变为
1119
+ </p>
1120
+
1121
+ <pre>
1122
+ BEGIN;
1123
+ LOCK TABLE ...
1124
+ INSERT INTO ...
1125
+ COMMIT;
1126
+ </pre>
1127
+
1128
+ <p>当 <code>insert_lock</code> 为 true 时,pgpool-II 自动在每次执行 INSERT 时添加以上的查询(如果已经在事务中,它只是简单地添加 LOCK TABLE ... )。
1129
+ </p>
1130
+ <p>pgpool-II 2.2 或更高的版本中,可以自动探测是否表拥有 SERIAL 类型的列,所以如果没有 SERIAL 类型的列,则将不会锁表。
1131
+ </p>
1132
+ <p>
1133
+ pgpool-II 3.0 系列直到 3.0.4 为止针对串行的关系使用一个行锁,而不是表锁。这使在 VACUUM(包括 autovacuum)时的锁冲突最小。但这会导致另一个问题。如果发生了嵌套事务,对串行的关系使用行所会导致 PostgreSQL 的内部错误(确切地说,保存事务状态的 pg_clog 会发生访问错误)。为了避免这个问题,PostgreSQL 核心开发者决定禁止对串行的关系加锁,当然这也会让 pgpool-II 无法工作(&quot;修复&quot;后的 PostgreSQL 版本为 9.0.5, 8.4.9, 8.3.16 和 8.2.22)。
1134
+ </p>
1135
+ <p>
1136
+ 由于新版的 PostgreSQL 不允许对串行的关系加锁,pgpool-II 3.0.5 或更新的版本针对 pgpool_catalog.insert_lock 使用行锁。所以需要预先在通过 pgpool-II 访问的数据库中建立 insert_lock 表。详细内容请参考<a href="#install">建立 insert_lock 表</a>。如果不存在 insert_lock 表,pgpool-II 将锁定插入的目标表。这种行为和 pgpool-II 2.2 和 2.3 系列相同。如果你希望使用与旧版本兼容的 insert_lock,你可以在配置脚本中指定锁定模式。详细内容请参考 <a href="#install">configure</a> 。
1137
+ </p>
1138
+ <p>
1139
+ 你也许需要更好(针对每个事务)的控制手段:
1140
+ </p>
1141
+
1142
+ <ol>
1143
+ <li>设置 <code>insert_lock</code> 为 true,并添加 <code>/*NO INSERT LOCK*/</code> 代码到你不想要表锁的 INSERT 语句的开始位置。</li>
1144
+
1145
+ <li>设置 <code>insert_lock</code> 为 false,并添加 <code>/*INSERT LOCK*/</code> 到你需要表锁的 INSERT 语句的开始位置。</li>
1146
+ </ol>
1147
+
1148
+ <p>
1149
+ 默认值为 false。如果 <code>insert_lock</code> 被启用,则(通过 pgpool-II 运行的) PostgreSQL 8.0 的事务、权限、规则和 alter_table 的回归测试会失败。原因是 pgpool-II 会尝试 LOCK 这些规则测试的 VIEW ,并产生以下的错误消息:
1150
+ </p>
1151
+
1152
+ <pre>
1153
+ ! ERROR: current transaction is aborted, commands ignored until
1154
+ end of transaction block
1155
+ </pre>
1156
+
1157
+ <p>例如,事务测试尝试 INSERT 到一个不存在的表,而 pgpool-II 导致 PostgreSQL 在这之前请求锁。事务将被终止,而之后的 INSERT 语句会产生以上的错误消息。
1158
+ </p>
1159
+
1160
+
1161
+ <dt><a name="RECOVERY_USER"></a>recovery_user</dt>
1162
+ <dd>
1163
+ <p>
1164
+ 本参数指定一个用于在线恢复的 PostgreSQL 用户名。改变本参数不需要重启。
1165
+ </p>
1166
+
1167
+ <dt><a name="RECOVERY_PASSWORD"></a>recovery_password</dt>
1168
+ <dd>
1169
+ <p>
1170
+ 本参数指定一个用于在线恢复的 PostgreSQL 密码。改变本参数不需要重启。
1171
+ </p>
1172
+
1173
+ <dt><a name="RECOVERY_1ST_STAGE_COMMAND"></a>recovery_1st_stage_command</dt>
1174
+ <dd>
1175
+ <p>
1176
+ 本参数指定一个在在线恢复第一阶段在主(Primary)PostgreSQL 服务器上运行的命令。处于安全原因,本命令必须被放置在数据库实例目录中。例如,如果 recovery_1st_stage_command = 'sync-command',那么 pgpool-II 将执行 $PGDATA/sync-command。
1177
+ </p>
1178
+ <p>
1179
+ recovery_1st_stage_command 将接受以下 3 个参数:
1180
+ </p>
1181
+ <ol>
1182
+ <li>到主(Primary)数据库实例的路径</li>
1183
+ <li>需要恢复的 PostgreSQL 主机名</li>
1184
+ <li>需要恢复的数据库实例路径</li>
1185
+ </ol>
1186
+ </p>
1187
+ <p>
1188
+ 注意 pgpool-II 在执行 recovery_1st_stage_command 时<b>接收</b>连接和查询。在本阶段中,你可以查询和更新数据。
1189
+ </p>
1190
+ <p>
1191
+ 改变本参数不需要重启。
1192
+ </p>
1193
+
1194
+ <dt><a name="RECOVERY_2ND_STAGE_COMMAND"></a>recovery_2nd_stage_command</dt>
1195
+ <dd>
1196
+ <p>
1197
+ 本参数指定一个在在线恢复第二阶段在主(Primary)PostgreSQL 服务器上运行的命令。处于安全原因,本命令必须被放置在数据库实例目录中。例如,如果 recovery_2st_stage_command = 'sync-command',那么 pgpool-II 将执行 $PGDATA/sync-command。
1198
+ </p>
1199
+ <p>
1200
+ recovery_2nd_stage_command 将接受以下 3 个参数:
1201
+ </p>
1202
+ <ol>
1203
+ <li>到主(Primary)数据库实例的路径</li>
1204
+ <li>需要恢复的 PostgreSQL 主机名</li>
1205
+ <li>需要恢复的数据库实例路径</li>
1206
+ </ol>
1207
+ </p>
1208
+ <p>
1209
+ 注意 pgpool-II 在运行 recovery_2nd_stage_command 时<b>不接收</b>连接和查询。因此如果一个客户端长时间持有一个连接,则恢复命令不会被执行。pgpool-II 等待所有的客户端关闭它们的连接。这个命令只在没有任何客户端连接到 pgpool-II 时才执行。
1210
+ </p>
1211
+ <p>
1212
+ 改变本参数不需要重启。
1213
+ </p>
1214
+
1215
+ <dt><a name="RECOVERY_TIMEOUT"></a>recovery_timeout</dt>
1216
+ <dd>
1217
+ <p>
1218
+ pgpool 在第二阶段不接受新的连接。如果一个客户端在恢复过程中连接到 pgpool,它必须等待到恢复结束。
1219
+ </p>
1220
+ <p>
1221
+ 本参数指定恢复超时的时间,单位为秒。如果到达了本超时值,则 pgpool 取消在线恢复并接受连接。0 表示不等待。
1222
+ </p>
1223
+ <p>
1224
+ 改变本参数不需要重启。
1225
+ </p>
1226
+
1227
+ <dt><a name="CLIENT_IDLE_LIMIT_IN_RECOVERY"></a>client_idle_limit_in_recovery <span class="version">V2.2 -</span></dt>
1228
+ <dd>
1229
+ <p> 类似于 client_idle_limit 但是只在恢复的第二阶段生效。从执行最后一个查询后空闲到 client_idle_limit_in_recovery 秒的客户端将被断开连接。
1230
+ 这对避免 pgpool 的恢复被懒客户端扰乱或者客户机和 pgpool 之间的 TCP/IP 连接被意外断开(例如网线断开)非常有用。如果设置为 -1 ,则立即断开客户端连接。
1231
+ client_idle_limit_in_recovery 的默认值为 0,表示本功能不启用。
1232
+ </p>
1233
+ <p> 如果你的客户端非常繁忙,则无论你将 client_idle_limit_in_recovery 设置为多少 pgpool-II 都无法进入恢复的第二阶段。在这种情况下,
1234
+ 你可以设置 client_idle_limit_in_recovery 为 -1 因而 pgpool-II 在进入第二阶段前立即断开这些繁忙的客户端的连接。
1235
+ </p>
1236
+ <p>
1237
+ 如果你改变了 client_idle_limit_in_recovery 你需要重新加载 pgpool.conf 。</p>
1238
+
1239
+ <dt><a name="LOBJ_LOCK_TABLE"></a>lobj_lock_table <span class="version">V2.2 -</span></dt>
1240
+ <dd>
1241
+ <p>
1242
+ 本参数指定一个表名用于大对象的复制控制。如果它被指定,pgpool 将锁定由 lobj_lock_table 指定的表并通过查找 pg_largeobject 系统 catalog 生产一个大对象 id,并调用 lo_create 来建立这个大对象。这个过程保证 pgpool 在复制模式中在所有的数据库节点中获得相同的大对象 id。注意 PostgreSQL 8.0 或者更老的版本没有 lo_create,因此本功能将无法工作。
1243
+ </p>
1244
+ <p>
1245
+ 对 libpq 的 lo_creat() 函数的调用将触发本功能。通过 Java API(JDBC 驱动),PHP API(pg_lo_create,或者 PHP 库中类似的 API 例如 PDO)进行的大对象创建,以及其他各种编程语言中相同的 API 使用相同的协议,因此也应该能够运行。
1246
+ </p>
1247
+ <p>
1248
+ 以下的大对象建立操作将无法运行:
1249
+ <p>
1250
+ <ul>
1251
+ <li>libpq 中的 lo_create
1252
+ <li>任何语言中使用 lo_create 的任何 API
1253
+ <li>后台程序的 lo_import 函数
1254
+ <li>SELECT lo_creat
1255
+ </ul>
1256
+ </p>
1257
+ <p>
1258
+ lobj_lock_table 存储在哪个 schema 并不重要,但是这个表必须对任何用户都可以写入。以下为如何建立这样一个表的示例:
1259
+ </p>
1260
+ <p>
1261
+ <pre>
1262
+ CREATE TABLE public.my_lock_table ();
1263
+ GRANT ALL ON public.my_lock_table TO PUBLIC;
1264
+ </pre>
1265
+ </p>
1266
+ <p>
1267
+ lobj_lock_table 指定的表必须被预先建立。如果你在 template1 中建立这个表,之后建立的任何数据库都将有这个表。
1268
+ </p>
1269
+ <p>
1270
+ 如果 lobj_lock_table 为空字符串(''),这个功能被禁用(大对象的复制将无法工作)。lobj_lock_table is 的默认值为''。
1271
+ </p>
1272
+
1273
+ </dl>
1274
+
1275
+ <h2 id="condition_for_load_balance">负载均衡的条件</h2>
1276
+ <p>
1277
+ 需要对一个查询使用负载均衡,需要满足以下的所有条件:
1278
+ </p>
1279
+
1280
+ <ul>
1281
+ <li>PostgreSQL 7.4 或更高版本</li>
1282
+ <li>即可以在复制模式中,也可以在主备模式中</li>
1283
+ <li>在复制模式中,查询必须不是在一个显式的事务中(例如,不在 BEGIN ~ END 块中)</li>
1284
+ <li>不能是 SELECT INTO</li>
1285
+ <li>不能是 SELECT FOR UPDATE 或者 FOR SHARE</li>
1286
+ <li>以 "SELECT" 开始或者为 COPY TO STDOUT, EXPLAIN, EXPLAIN ANALYZE SELECT... 其中一个,ignore_leading_white_space = true 将忽略开头的空格。
1287
+ (除非是在 <a href="#BLACK_FUNCTION_LIST">black_list</a> 或者 <a href="#WHITE_FUNCTION_LIST">white_list</a> 指明的有写动作的函数)
1288
+ <li><span class="version">V3.0 -</span>在主备模式中,除了以上条件,还包含以下条件:
1289
+ <ul>
1290
+ <li>不能使用临时表</li>
1291
+ <li>不能使用不写日志的表</li>
1292
+ <li>不能使用系统表</li>
1293
+ <li>不过,如果碰到以下条件,即使是显式事务,依然有可能进行负载均衡</li>
1294
+ <ul>
1295
+ <li>事务隔离级别不是SERIALIZABLE</li>
1296
+ <li>事务一直没有进行写查询(直到发生写入查询前,都可能进行负载均衡)</li>
1297
+ </ul>
1298
+ </ul>
1299
+ </li>
1300
+ </ul>
1301
+
1302
+ <p>
1303
+ 注意你可以通过在 SELECT 语句之前插入任意的注释来禁止负载均衡:
1304
+ <pre>
1305
+ /*REPLICATION*/ SELECT ...
1306
+ </pre>
1307
+ </p>
1308
+
1309
+ <p>
1310
+ 请参考<a href="#replicate_select">replicate_select</a>。也可以参考<a href="where_to_send_queries.pdf">flow chart</a>。
1311
+ </p>
1312
+
1313
+ <p>
1314
+ <font color="red">
1315
+ 注:JDBC 驱动有自动提交的选项。如果自动提交为 false,则 JDBC 驱动将自己发送 "BEGIN" 和 "COMMIT"。因此 pgpool-II 无法做任何负载均衡。你需要调用 setAutoCommit(true) 来启用自动提交。
1316
+ </font>
1317
+ </p>
1318
+
1319
+ <h2 id="failover_in_replication_mode">复制模式中的故障切换</h2>
1320
+
1321
+ <p> pgpool-II 退化一个死掉的后台并继续提供服务。只要最少还有一个后台还或者,服务就可以继续。</p>
1322
+
1323
+ <h2 id="errors_in_replication_mode">复制模式中的特有错误</h2>
1324
+ <p>
1325
+ 在复制模式中,如果 pgpool 发现 INSERT,UPDATE 和 DELETE 生效的行数不同,如果 failover_if_affected_tuples_mismatch 被设置为 false,则 pgpool 将发送错误的 SQL 语句到所有的数据库节点来取消当前当前事务(如果为 false 则发生退化)。
1326
+ 在这种情况下,你将在客户端终端中看到以下错误信息:
1327
+ <p>
1328
+ <pre>
1329
+ =# UPDATE t SET a = a + 1;
1330
+ ERROR: pgpool detected difference of the number of update tuples Possible last query was: "update t1 set i = 1;"
1331
+ HINT: check data consistency between master and other db node
1332
+ </pre>
1333
+
1334
+ <p>
1335
+ 你将在 PostgreSQL 的日志中看到更新的行数(在本例中,数据库节点 0 更新了 0 行而数据库节点 1 更新了 1 行)。
1336
+ </p>
1337
+ <pre>
1338
+ 2010-07-22 13:23:25 LOG: pid 5490: SimpleForwardToFrontend: Number of affected tuples are: 0 1
1339
+ 2010-07-22 13:23:25 LOG: pid 5490: ReadyForQuery: Degenerate backends: 1
1340
+ 2010-07-22 13:23:25 LOG: pid 5490: ReadyForQuery: Number of affected tuples are: 0 1
1341
+ </pre>
1342
+
1343
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
1344
+
1345
+ <!-- ================================================================================ -->
1346
+
1347
+ <h1><a name="master_slave_mode"></a>主/备模式</h1>
1348
+
1349
+ <p>
1350
+ 本模式用于使用其他负责完成实际的数据复制的主/备复制软件(类似于 Slong-I 和 基于流复制)来连接 pgpool-II。
1351
+ 必须设置数据库节点的信息(如果你需要在线恢复功能,<a href="#BACKEND_HOSTNAME">backend_hostname</a>、<a href="#BACKEND_PORT">backend_port</a>、
1352
+ <a href="#BACKEND_WEIGHT">backend_weight</a>、<a href="#BACKEND_FLAG">backend_flag</a> 和 <a href="#BACKEND_DATA_DIRECTORY">backend_data_directory</a>),
1353
+ 这和复制模式中的方法相同。另外,还需要设置 <code><a href="#MASTER_SLAVE_MODE">master_slave_mode</a></code> 和 <code><a href="#LOAD_BALANCE_MODE">load_balance_mode</a></code> 为 true。
1354
+ </p>
1355
+ <p>
1356
+ pgpool-II 将发送需要复制的查询到主数据库,并在必要时将其他的查询将被负载均衡。不能被负载均衡而发送到主数据库的查询当然也是受负载均衡逻辑控制的。
1357
+ </p>
1358
+
1359
+
1360
+ <p>在主/备模式中,对于临时表的 DDL 和 DML 操作只能在主节点上被执行。SELECT 也可以被强制在主节点上执行,但这需要你在 SELECT 语句前添加一个 /*NO LOAD BALANCE*/ 注释。
1361
+ </p>
1362
+
1363
+ <p>在主/备模式中, <code><a href="#REPLICATION_MODE">replication_mode</a></code> 必须被设置为 false ,并且 <code><a href="#MASTER_SLAVE_MODE">master_slave_mode</a></code> 为 true。</p>
1364
+
1365
+ <p>主/备模式有一个“master_slave_sub mode”。默认值为 'slony',用于 Slony-I。你也可以设置它为 'stream',它在你想使用 PostgreSQL 内置的复制系统(基于流复制)时被设置。用于 Slony-I 子模式的示例配置文件为 pgpool.conf.sample-master-slave,用于基于流复制的子模式的示例文件为 sub-module is pgpool.conf.sample-stream。
1366
+ </p>
1367
+ <p>
1368
+ 修改以上任何参数都需要重新启动 pgpool-II。
1369
+ </p>
1370
+ <p>
1371
+ 在主/备模式中,你可以通过设置 <a href="#WHITE_FUNCTION_LIST">white_function_list</a> 和 <a href="#BLACK_FUNCTION_LIST">black_function_list</a> 来控制负载均衡。参考 <a href="#WHITE_FUNCTION_LIST">white_function_list</a> 获得详细信息。
1372
+ </p>
1373
+
1374
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
1375
+
1376
+ <!-- ================================================================================ -->
1377
+
1378
+ <h1><a name="stream"></a>流复制 <span class="version">V3.1 -</span></h1>
1379
+ <p>
1380
+ 就像以上规定的,pgpool-II 可以与 PostgreSQL 9.0 带来的基于流复制协同工作。要使用它,启用“<a href="#MASTER_SLAVE_MODE">master_slave_mode</a>”并设置“<a href="#MASTER_SLAVE_SUB_MODE">master_slave_sub_mode</a>”为“stream”。
1381
+ pgpool-II 认为基于流复制启用了热备,也就是说备库是以只读方式打开的。以下参数可以用于本模式:
1382
+ </p>
1383
+
1384
+ <dl>
1385
+ <dt id="DELAY_THRESHOLD">delay_threshold <span class="version">V3.0 -</span></dt>
1386
+ <dd>
1387
+ <p>
1388
+ 指定能够容忍的备机上相对于主服务器上的 WAL 的复制延迟,单位为字节。
1389
+ 如果延迟到达了 delay_threshold,pgpool-II 不再发送 SELECT 查询到备机。
1390
+ 所有的东西都被发送到主服务器,即使启用了负载均衡模式,直到备机追赶上来。
1391
+ 如果 delay_threshold 为 0 或者流复制检查被禁用,则延迟检查不被执行。
1392
+ 这个检查在每“<a href=#SR_CHECK_PERIOD">sr_check_period</a>”周期执行一次。
1393
+ delay_threshold 的默认值为 0。
1394
+ </p>
1395
+ <p>
1396
+ 要使对本参数的改动生效,你需要重新加载 pgpool.conf。
1397
+ </p>
1398
+ </dd>
1399
+
1400
+ <dt id="SR_CHECK_PERIOD">sr_check_period <span class="version">V3.1 -</span></dt>
1401
+ <dd>
1402
+ <p>
1403
+ 本参数指出基于流复制的延迟检查的间隔,单位为秒。
1404
+ 默认为 0,表示禁用这个检查。
1405
+ </p>
1406
+ <p>
1407
+ 如果你修改了 sr_check_period,需要重新加载 pgpool.conf 以使变动生效。
1408
+ </p>
1409
+ </dd>
1410
+
1411
+ <dt id="SR_CHECK_USER">sr_check_user <span class="version">V3.1 -</span></dt>
1412
+ <dd>
1413
+ <p>
1414
+ 执行基于流复制检查的用户名。用户必须存在于所有的 PostgreSQL 后端上,否则,检查将出错。
1415
+ 注意即使 sr_check_period 为 0, sr_check_user 和 sr_check_password 也会被使用。
1416
+ 要识别主服务器,pgpool-II 发送函数调用请求到每个后端。
1417
+ sr_check_user 和 sr_check_password 用于这个会话。
1418
+ </p>
1419
+ <p>
1420
+ 如果你修改了 sr_check_user,需要重新加载 pgpool.conf 以使变动生效。
1421
+ </p>
1422
+ </dd>
1423
+
1424
+ <dt id="SR_CHECK_PASSWORD">sr_check_password <span class="version">V3.1 -</span></dt>
1425
+ <dd>
1426
+ <p>
1427
+ 指出如何记录复制延迟。如果指定 'none',则不写入日志。
1428
+ 如果为 'always',在每次执行健康检查时记录延迟。如果 'if_over_threshold' 被指定,
1429
+ 只有当延迟到达 delay_threshold 时记录日志。log_standby_delay 的默认值为 'none'。
1430
+ </p>
1431
+ <p>
1432
+ 要使对本参数的改动生效,你需要重新加载 pgpool.conf。
1433
+ </p>
1434
+ </dd>
1435
+
1436
+ <dt id="LOG_STANDBY_DELAY">log_standby_delay <span class="version">V3.1 -</span></dt>
1437
+ <dd>
1438
+ <p>
1439
+ 指出如何记录复制延迟。如果指定 'none',则不写入日志。
1440
+ 如果为 'always',在每次执行健康检查时记录延迟。
1441
+ 如果 'if_over_threshold' 被指定,只有当延迟到达 <a href="#DELAY_THRESHOLD">delay_threshold</a> 时记录日志。
1442
+ log_standby_delay 的默认值为 'none'。
1443
+ 要使对本参数的改动生效,你需要重新加载 pgpool.conf。
1444
+ </p>
1445
+ <p>
1446
+ 你也可以使用“<a href="#pool_status">show pool_status</a>”命令监控复制延迟。
1447
+ 列名为“standby_delay#”(其中 '#' 需要用数据库节点编号代替)。
1448
+ </p>
1449
+ </dd>
1450
+ </dl>
1451
+
1452
+ <h2>流复制下的故障切换</h2>
1453
+ <p>
1454
+ 在使用流复制的主/备模式中,如果主节点或者备节点失效,pgpool-II 可以被设置为触发一个故障切换。节点可以被自动断开而不需要进行更多设置。
1455
+ 当进行流复制的时候,备节点检查一个“触发文件”的存在,一旦发现它,则备节点停止持续的恢复并进入读写模式。通过使用这种功能,我们可以使备数据库在主节点失效的时候进行替换。
1456
+ </p>
1457
+ <p>
1458
+ <strong>警告:如果你计划使用多个备节点,我们建议设置一个 delay_threshold 值来避免任何查询由于查询被发送到其他备节点而导致获取旧数据。
1459
+ </p>
1460
+ <p>
1461
+ 如果第二个备节点在第一个备节点已经发生替换的时候替换主节点,你会从第二备节点获取错误的数据。我们不推荐计划使用这种配置。
1462
+ </strong>
1463
+ </p>
1464
+ <p>
1465
+ 以下例举了如何设置一个故障切换的配置。
1466
+ </p>
1467
+ <p>
1468
+ <ol>
1469
+ <li>将一个故障切换脚本放置到某个地方(例如 /usr/local/pgsql/bin )并给它执行权限。
1470
+ <pre>
1471
+ $ cd /usr/loca/pgsql/bin
1472
+ $ cat failover_stream.sh
1473
+ #! /bin/sh
1474
+ # Failover command for streaming replication.
1475
+ # This script assumes that DB node 0 is primary, and 1 is standby.
1476
+ #
1477
+ # If standby goes down, do nothing. If primary goes down, create a
1478
+ # trigger file so that standby takes over primary node.
1479
+ #
1480
+ # Arguments: $1: failed node id. $2: new master hostname. $3: path to
1481
+ # trigger file.
1482
+
1483
+ failed_node=$1
1484
+ new_master=$2
1485
+ trigger_file=$3
1486
+
1487
+ # Do nothing if standby goes down.
1488
+ if [ $failed_node = 1 ]; then
1489
+ exit 0;
1490
+ fi
1491
+
1492
+ # Create the trigger file.
1493
+ /usr/bin/ssh -T $new_master /bin/touch $trigger_file
1494
+
1495
+ exit 0;
1496
+
1497
+ chmod 755 failover_stream.sh
1498
+ </pre>
1499
+ <li>在 pgpool.conf 中设置 failover_commmand。
1500
+ <pre>
1501
+ failover_command = '/usr/local/src/pgsql/9.0-beta/bin/failover_stream.sh %d %H /tmp/trigger_file0'
1502
+ </pre>
1503
+
1504
+ <li>在备节点中设置 recovery.conf。
1505
+ <a href="recovery.conf.sample">一个 recovery.conf 示例</a> 可以在 PostgreSQL 安装目录中找到。它的名字为 "share/recovery.conf.sample"。拷贝 recovery.conf.sample 为 recovery.conf 到数据库节点目录并编辑它。
1506
+ <pre>
1507
+ standby_mode = 'on'
1508
+ primary_conninfo = 'host=name of primary_host user=postgres'
1509
+ trigger_file = '/tmp/trigger_file0'
1510
+ </pre>
1511
+
1512
+ <li>设置主节点上的 postgresql.conf 。以下仅仅是一个示例。你需要根据你自己的环境做调整。
1513
+ <pre>
1514
+ wal_level = hot_standby
1515
+ max_wal_senders = 1
1516
+ </pre>
1517
+
1518
+ <li>设置主节点上的 pg_hba.conf 。以下仅仅是一个示例。你需要根据你自己的环境做调整。
1519
+ <pre>
1520
+ host replication postgres 192.168.0.10/32 trust
1521
+ </pre>
1522
+
1523
+ </ol>
1524
+
1525
+ <p>
1526
+ 启动首要 PostgreSQL 节点和第二 PostgreSQL 节点来初始化基于流复制。如果主节点失效,备节点将自动启动为普通 PostgreSQL 并准备好接受写查询。
1527
+ </p>
1528
+
1529
+ <h2>流复制</h2>
1530
+ <p>
1531
+ 当使用流复制和热备的时候,确定哪个查询可以被发送到主节点或备节点或者不能被发送到备节点非常重要。pgpool-II 的流复制模式可以很好的处理这种情况。在本章,我们将解释 pgpool-II 如何做到这一点的。
1532
+ </p>
1533
+ <p>
1534
+ 我们通过检查查询来辨别哪个查询应该被发送到哪个节点。
1535
+ </p>
1536
+ <p>
1537
+ <ul>
1538
+ <li>这些查询只允许被发送到主节点
1539
+ <ul>
1540
+ <li>INSERT, UPDATE, DELETE, COPY FROM, TRUNCATE, CREATE, DROP, ALTER, COMMENT
1541
+ <li>SELECT ... FOR SHARE | UPDATE
1542
+ <li>在事务隔离级别为 SERIALIZABLE 的 SELECT
1543
+ <li>比 ROW EXCLUSIVE MODE 更严厉的 LOCK 命令
1544
+ <li>一些事务相关命令:
1545
+ <ul>
1546
+ <li>BEGIN READ WRITE, START TRANSACTION READ WRITE
1547
+ <li>SET TRANSACTION READ WRITE, SET SESSION CHARACTERISTICS AS TRANSACTION READ WRITE
1548
+ <li>SET transaction_read_only = off
1549
+ </ul>
1550
+ <li>两步提交命令:PREPARE TRANSACTION, COMMIT PREPARED, ROLLBACK PREPARED
1551
+ <li>LISTEN, UNLISTEN, NOTIFY
1552
+ <li>VACUUM
1553
+ <li>一些序列生成器操作函数(nextval 和 setval)
1554
+ <li>大对象建立命令
1555
+ </ul>
1556
+ <li>这些查询可以被发送到主节点和备节点。如果启用了负载均衡,这些查询可以被发送到备节点。但是,如果设置了 delay_threshold 且复制延迟大于这个值,则查询被发送到主节点。
1557
+ <ul>
1558
+ <li>SELECT not listed above
1559
+ <li>COPY TO
1560
+ <li>DECLARE, FETCH, CLOSE
1561
+ <li>SHOW
1562
+ </ul>
1563
+ <li>以下查询被同时发送到主节点和备节点
1564
+ <ul>
1565
+ <li>SET
1566
+ <li>DISCARD
1567
+ <li>DEALLOCATE ALL
1568
+ </ul>
1569
+ </ul>
1570
+ </p>
1571
+
1572
+ <p>
1573
+ 在一个显式的事务中:
1574
+ <ul>
1575
+ <li>启动事务的命令例如 BEGIN 只被发送到主节点。
1576
+ <li>接下来的 SELECT 和一些可以被发送到主节点或备节点的其他查询会在事务中执行或者在备节点中执行。
1577
+ <li>无法在备节点中执行的命令例如 INSERT 被发送到主节点。在这些命令之后的命令,即使是 SELECT 也被发送到主节点。这是因为这些 SELECT 语句可能需要立即查看 INSERT 的结果。这种行为一直持续到事务关闭或者终止。
1578
+ </ul>
1579
+ </p>
1580
+
1581
+ <p>
1582
+ 在扩展协议中,在负载均衡模式中在分析查询时,有可能探测是否查询可以被发送到备节点。规则和非扩展协议下相同。例如,INSERT 被发送到主节点。接下来的 bind,describe 和 execute 也将被发送到主节点。
1583
+ </p>
1584
+
1585
+ <p>
1586
+ [注:如果对 SELECT 语句的分析由于负载均衡被发送到备节点,然后一个 DML 语句,例如一个 INSERT ,被发送到 pgpool-II,那么,被分析的 SELECT 必须在主节点上执行。因此,我们会在主节点上重新分析这个 SELECT 语句。]
1587
+ </p>
1588
+ <p>
1589
+ 最后,pgpool-II 的分析认为有错误的查询将被发送到主节点。
1590
+ </p>
1591
+
1592
+ <h2>流复制中的在线恢复</h2>
1593
+ <p>
1594
+ 在流复制的主/备模式中,可以执行在线恢复。在在线恢复过程中,首要务器扮演了主服务器的角色并恢复到指定的备服务器。因此恢复过程需要首要服务器启动并运行。
1595
+ 如果第一服务器失效且没有备用服务器被提升,你需要停止 pgpool-II 和所有的 PostgreSQL 服务器并手动恢复它们。
1596
+ </p>
1597
+ <p>
1598
+ <ol>
1599
+ <li>设置 recovery_user。通常为 "postgres"。
1600
+ <pre>
1601
+ recovery_user = 'postgres'
1602
+ </pre>
1603
+ <li>设置登录到数据库的 recover_user 的 recovery_password。
1604
+ <pre>
1605
+ recovery_password = 't-ishii'
1606
+ </pre>
1607
+
1608
+ <li>设置 recovery_1st_stage_command。这个阶段的这个脚本用来执行一个首要数据库的备份并还原它到备用节点。将此脚本放置在首要数据库示例的目录中并给它可执行权限。这里有一个用于配置了一个主节点和一个备节点的示例脚本 <a href="basebackup.sh">(basebackup.sh)</a> 。你需要设置 ssh 让 recovery_user 可以从首要节点登录到备用节点而不需要提供密码。
1609
+
1610
+ <pre>
1611
+ recovery_1st_stage_command = 'basebackup.sh'
1612
+ </pre>
1613
+ <li>让 recovery_2nd_stage_command 保留为空。
1614
+ <pre>
1615
+ recovery_2nd_stage_command = ''
1616
+ </pre>
1617
+
1618
+ <li>在每个数据库节点中安装必须的执行在线恢复的 C 和 SQL 函数。
1619
+ </p>
1620
+
1621
+ <pre>
1622
+ # cd pgpool-II-x.x.x/sql/pgpool-recovery
1623
+ # make
1624
+ # make install
1625
+ # psql -f pgpool-recovery.sql template1
1626
+ </pre>
1627
+
1628
+ <li>在完成在线恢复后,pgpool-II 将在备节点启动 PostgreSQL。在每个数据库节点中安装用于本用途的脚本。
1629
+ <a href="pgpool_remote_start">示例脚本</a> 包含在源码的“sample”目录中。这个脚本使用了 ssh。你需要允许 recover_user 从首要节点登录到备用节点而不需要输入密码。
1630
+ </ol>
1631
+ </p>
1632
+
1633
+ <p>
1634
+ 以上未全部内容。现在你可以使用 pcp_recovery_node (作为备用节点的步骤)或者点击 pgpoolAdmin 的“恢复”按钮来执行在线恢复了。如果出现问题,请检查 pgpool-II 的日子,首要服务器的日志和备用服务器的日志。
1635
+ </p>
1636
+ <p>
1637
+ 作为参考,以下为恢复过程的步骤。
1638
+ <ol>
1639
+ <li>Pgpool-II 使用 user = recovery_user, password = recovery_password 连接到首要服务器的 template1 数据库。
1640
+ <li>首要服务器执行 pgpool_recovery 函数。
1641
+ <li>pgpool_recovery 函数执行 recovery_1st_stage_command。注意 PostgreSQL 在数据库实例的当前目录中执行函数。因此,recovery_1st_stage_command 在数据库实例的目录中执行。
1642
+ <li>首要服务器执行 pgpool_remote_start 函数。本函数执行一个在数据库实例路径中名为“pgpool_remote_start”的脚本,它通过 ssh 在备用服务器上执行 pg_ctl 命令来进行恢复。pg_ctl 将在后台启动 postmaster。所以我们需要确保备用节点上的 postmaster 真正启动了。
1643
+ <li>pgpool-II 尝试使用 user = recovery_user 和 password = recovery_password 连接到备用 PostgreSQL。如果可能,连接到的数据库为“postgres”。否则,使用“template1”。pgpool-II 尝试 <a href="#RECOVERY_TIMEOUT">recovery_timeout</a> 秒。如果成功,进行下一步。
1644
+ <li>如果 <a href="#FAILBACK_COMMAND">failback_command</a> 不为空,pgpool-II 父进程执行这个脚本。
1645
+ <li>在 failback_command 完成后,pgpool-II 重新启动所有的子进程。
1646
+ </ol>
1647
+ </p>
1648
+
1649
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
1650
+
1651
+ <!-- ================================================================================ -->
1652
+
1653
+
1654
+ <h1 id="parallel">并行模式</h3>
1655
+
1656
+ <p>本模式实现了查询的并行执行。表可以被分割,数据分布在每个节点中。而且,复制和负载均衡功能也可以同时使用。在并行模式中,pgpool.conf 中的 <a href="#REPLICATION_MODE">replication_mode</a> 和 <a href="#LOAD_BALANCE_MODE">load_balance_mode</a> 被设置为 ture,<a href="#MASTER_SLAVE_MODE">master_slave</a> 被设置为 false,<a href="#PARALLEL_MODE">parallel_mode</a> 被设置为 false。当你改变这些参数后,需要重启 pgpool-II。
1657
+ </p>
1658
+
1659
+ <h2 id="system_db">配置 System DB</h2>
1660
+
1661
+ <p>如果想使用并行模式,System DB 需要被正确配置。System DB 包含存储在一个表中的,用于选择被分区的数据将被发送到的恰当后端程序的规则。
1662
+ System DB 不一定要和 pgpool-II 在同一台主机上。System DB 的配置由 <code>pgpool.conf</code> 完成。</p>
1663
+
1664
+ <dl>
1665
+ <dt><a name="SYSTEM_DB_HOSTNAME"></a>system_db_hostname</dt>
1666
+ <dd>
1667
+ <p>System DB 所在的主机名。给出空字符串('')表示 SystemDB 和 pgpool-II 在同一台主机,且将通过 UNIX 域套接字访问。</p>
1668
+ </dd>
1669
+
1670
+ <dt><a name="SYSTEM_DB_PORT"></a>system_db_port</dt>
1671
+ <dd>
1672
+ <p>System DB 使用的端口号</p>
1673
+ </dd>
1674
+
1675
+ <dt><a name="SYSTEM_DBNAME"></a>system_dbname</dt>
1676
+ <dd>
1677
+ <p>分区规则和其他的信息将被定义到这里指定的数据库中。默认值为:<code>'pgpool'</code>。</p>
1678
+ </dd>
1679
+
1680
+ <dt><a name="SYSTEM_DB_SCHEMA"></a>system_db_schema</dt>
1681
+ <dd>
1682
+ <p>分区规则和其他的信息将被定义到这里指定的模式中。默认值为:<code>'pgpool_catalog'</code>。</p>
1683
+ </dd>
1684
+
1685
+ <dt><a name="SYSTEM_DB_USER"></a>system_db_user</dt>
1686
+ <dd>
1687
+ <p>连接到 System DB 的用户名。</p>
1688
+ </dd>
1689
+
1690
+ <dt><a name="SYSTEM_DB_PASSWORD"></a>system_db_password</dt>
1691
+ <dd>
1692
+ <p>连接到 System DB 的密码,如果不需要密码,则设置为空字符串('')。</p>
1693
+ </dd>
1694
+
1695
+ <dt><a name="SSL_CA_CERT"></a>ssl_ca_cert</dt>
1696
+ <dd>
1697
+ <p>
1698
+ 到达用于校验后台服务器的 PEM 格式的包含一个或者多个 CA 根证书的文件的路径。这类似于 OpenSSL 的 <code>verify(1)</code> 命令的 <code>-CAfile</code> 选项。
1699
+ </p>
1700
+ <p>
1701
+ 默认值为未设置,也就是不进行认证。如果本选项未设置但是 <code>ssl_ca_cert_dir</code>被设置了,则认证过程依旧会发生。
1702
+ </p>
1703
+ </dd>
1704
+
1705
+ <dt><a name="SSL_CA_CERT_DIR"></a>ssl_ca_cert_dir</dt>
1706
+ <dd>
1707
+ <p>
1708
+ 到达包含用于校验后台服务器的 PEM 格式的 CA 证书的目录的路径。这类似于 OpenSSL 的 <code>verify(1)</code> 命令的 <code>-CApath</code> 选项。
1709
+ </p>
1710
+ <p>
1711
+ 默认值为未设置,也就是不进行认证。如果本选项未设置但是 <code>ssl_ca_cert</code> 被设置了,则认证过程依旧会发生。
1712
+ </p>
1713
+ </dd>
1714
+
1715
+ </dl>
1716
+
1717
+ <h2 id="system_db_sql">初始化 System DB 的配置</h2>
1718
+
1719
+ <p>首先,需要建立 <code>pgpool.conf</code> 文件中指定的数据库和模式。可以在 <code>$prefix/share/system_db.sql</code> 找到一个示例脚本。如果你指定了不同的数据库名和模式,在脚本中相应地修改它们。
1720
+ </p>
1721
+ <pre>
1722
+ psql -f $prefix/share/system_db.sql pgpool
1723
+ </pre>
1724
+
1725
+ </p>
1726
+
1727
+ <h3 id="distdef">注册一条分区规则</h3>
1728
+
1729
+ <p>用于数据分区的规则必须被注册到 <code>pgpool_catalog.dist_def</code> 表中。</p>
1730
+
1731
+ <pre>
1732
+ CREATE TABLE pgpool_catalog.dist_def(
1733
+ dbname TEXT, -- database name
1734
+ schema_name TEXT, -- schema name
1735
+ table_name TEXT, -- table name
1736
+ col_name TEXT NOT NULL CHECK (col_name = ANY (col_list)), -- partitioning key column name
1737
+ col_list TEXT[] NOT NULL, -- names of table attributes
1738
+ type_list TEXT[] NOT NULL, -- types of table attributes
1739
+ dist_def_func TEXT NOT NULL, -- name of the partitioning rule function
1740
+ PRIMARY KEY (dbname,schema_name,table_name)
1741
+ );
1742
+ </pre>
1743
+
1744
+ <h3 id="replicate_def">注册一条复制规则</h3>
1745
+ <p>
1746
+ 未被分发的表必须被复制. 当一个查询对一个分发表和另外一个表进行连接时,pgpool-II 从 pgpool_catalog.replicate_def 表获取复制信息。一个表只能被复制或被分发。
1747
+ </p>
1748
+
1749
+ <pre>
1750
+ CREATE TABLE pgpool_catalog.replicate_def(
1751
+ dbname TEXT, --database name
1752
+ schema_name TEXT, --schema name
1753
+ table_name TEXT, --table name
1754
+ col_list TEXT[] NOT NULL, -- names of table attributes
1755
+ type_list TEXT[] NOT NULL, -- types of table attributes
1756
+ PRIMARY KEY (dbname,schema_name,table_name)
1757
+ );
1758
+ </pre>
1759
+
1760
+ <h2 id="example_partitioning">对 pgbench 表的分区示例</h2>
1761
+
1762
+ <p>
1763
+ 在这个示例中,accounts 表被分区,branches 和 tellers 表被复制。accounts 表和 branches 表通过 bid 进行连接。branches 表注册到复制表中,若这三个表(accounts, branches 和 tellers)将被连接的话,有必要也为 tellers 表注册一条复制规则。
1764
+ </p>
1765
+ <pre>
1766
+ INSERT INTO pgpool_catalog.dist_def VALUES (
1767
+ 'pgpool',
1768
+ 'public',
1769
+ 'accounts',
1770
+ 'aid',
1771
+ ARRAY['aid','bid','abalance','filler'],
1772
+ ARRAY['integer','integer','integer','character(84)'],
1773
+ 'pgpool_catalog.dist_def_accounts'
1774
+ );
1775
+
1776
+ INSERT INTO pgpool_catalog.replicate_def VALUES (
1777
+ 'pgpool',
1778
+ 'public',
1779
+ 'branches',
1780
+ ARRAY['bid','bbalance','filler'],
1781
+ ARRAY['integer','integer','character(84)']
1782
+ );
1783
+ </pre>
1784
+
1785
+ <p>分区规则函数(此处是 pgpool_catalog.dist_def_accounts )需要一个值作为分区的关键字列,并返回相应的DB节点ID。注意节点ID必须从0开始,
1786
+ 下面是为pgbench写的函数示例。
1787
+ </p>
1788
+ <pre>
1789
+ CREATE OR REPLACE FUNCTION pgpool_catalog.dist_def_accounts (val ANYELEMENT) RETURNS INTEGER AS '
1790
+ SELECT CASE WHEN $1 >= 1 and $1 <= 30000 THEN 0
1791
+ WHEN $1 > 30000 and $1 <= 60000 THEN 1
1792
+ ELSE 2
1793
+ </pre>
1794
+
1795
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
1796
+
1797
+ <!-- ================================================================================ -->
1798
+
1799
+ <h1><a name="hba"></a>设置用于客户端认证(HBA)的 pool_hba.conf</h1>
1800
+
1801
+ <p>
1802
+ 和 PostgreSQL 中使用的 pg_hba.conf 文件类似,pgpool-II 使用一个称之为 "pool_hba.conf" 的配置文件来支持类似的客户端认证功能。
1803
+ </p>
1804
+ <p>
1805
+ 当安装 pgpool 的时候,pool_hba.conf.sample 文件将被安装在"/usr/local/etc"目录下,该位置也是配置文件的默认目录。拷贝 pool_hba.conf.sample 为 pool_hba.conf,如果必要的话并修改它。默认的情况下,<a href="#ENABLE_POOL_HBA">enable_pool_hba</a> 认证被开启。
1806
+ </p>
1807
+ <p>
1808
+ pool_hba.conf 文件的格式和 PostgreSQL 的 pg_hba.conf 的格式遵循的非常相近。
1809
+ </p>
1810
+ <pre>
1811
+ local DATABASE USER METHOD [OPTION]
1812
+ host DATABASE USER CIDR-ADDRESS METHOD [OPTION]
1813
+ </pre>
1814
+ <p>
1815
+ 查看 "pool_hba.conf.sample" 文件获取每个字段详细的解释。
1816
+ </p>
1817
+ <p>
1818
+ 下面是 pool_hba 的一些限制。
1819
+ <ul>
1820
+ <li>"hostssl" 连接类型不被支持</li>
1821
+ <p>
1822
+ 尽管"hostssl"不能被使用,pgpool-II 2.3或更高版本支持SSL,详见<a href="#ssl">SSL</a>。
1823
+ </p>
1824
+ <li>DATABASE 字段使用的"samegroup" 不被支持</li>
1825
+ <p>
1826
+ 尽管 pgpool 并不知道后端服务器的用户的任何信息,但是将通过 pool_hba.conf 中的 DATABASE 字段项对数据库名进行简单的检查。
1827
+ </p>
1828
+ <li>USER 字段使用的 group 名字后面跟个"+"不被支持 </li>
1829
+ <p>
1830
+ 这与上面介绍的 "samegroup" 原因相同,将通过 pool_hba.conf 中 USER 字段项对用户名进行简单的检查。
1831
+ </p>
1832
+ <li>为 IP address/mask 使用的 IPv6 不被支持</li>
1833
+ <p>
1834
+ pgpool 当前不支持 IPv6.
1835
+ </p>
1836
+ <li>METHOD 字段仅仅支持 "trust", "reject", "md5" 和 "pam" </li>
1837
+ <p>
1838
+ 再次,这与上面介绍的 "samegroup" 原因相同, pgpool 不能够访问 user/password 信息。
1839
+ </p>
1840
+ <p>
1841
+ 要使用md5认证,你需要在 "pool_passwd" 中注册你的名字和密码。详见<a href="#md5">认证/访问控制</a>。
1842
+ </ul>
1843
+ <p>
1844
+ 注意本节描述的所有认证发生在客户端和 pgpool-II 之间;客户端仍然需要继续通过 PostgreSQL 的认证过程。
1845
+ pool_hba 并不关心客户端提供的用户名/数据库名(例如 psql -U testuser testdb)是否真实存在于后端服务器中。pool_hba 仅仅关心是否在 pool_hba.conf 中存在匹配。
1846
+ </p>
1847
+
1848
+ <p>
1849
+ PAM 认证使用 pgpool 运行的主机上的用户信息来获得支持。若让 pgpool 支持PAM,需要在 configure 时指定"--with-pam"选项。
1850
+ </p>
1851
+ <pre>
1852
+ configure --with-pam
1853
+ </pre>
1854
+ <p>
1855
+ 若启用 PAM 认证,你需要为 pgpool 在系统的 PAM 配置目录(通常是在"/etc/pam.d")中创建一个 service-configuration 文件。
1856
+ 一个 service-configuration 的示例文件被安装为安装目录下的"share/pgpool.pam"。
1857
+ </p>
1858
+
1859
+ <h1 id="query_cache">设置 Query cache 方法 <span class="version">- V3.1 (已废弃)</span></h1>
1860
+
1861
+ <strong>注意:这个(基于磁盘的)查询缓存功能将在不久后被移除。
1862
+ 清使用<a href="#memqcache">基于内存的查询缓存</a>代替。
1863
+ </strong>
1864
+
1865
+ <p> Query cache 可以在 pgpool-II 的所有模式中使用。在 pgpool.conf 中启用它如下:</p>
1866
+ <pre>
1867
+ enable_query_cache = true
1868
+ </pre>
1869
+
1870
+ <p>
1871
+ 你还需要在 System DB 中创建下面的表:
1872
+ </p>
1873
+ <pre>
1874
+ CREATE TABLE pgpool_catalog.query_cache (
1875
+ hash TEXT,
1876
+ query TEXT,
1877
+ value bytea,
1878
+ dbname TEXT,
1879
+ create_time TIMESTAMP WITH TIME ZONE,
1880
+ PRIMARY KEY(hash, dbname)
1881
+ );
1882
+ </pre>
1883
+ <p>
1884
+ 然而,如果你不使用"pgpool_catalog"的话,你可能需要修改该语句中的 schema。
1885
+ </p>
1886
+ <p>
1887
+ <strong>注意:当前的查询缓存的实现方法为在数据库中建立缓存数据。因此启用查询缓存可能会导致达不到最高的性能。即使相关的表被更新了,查询缓存的内容不会更新。你需要从缓存的表中删除缓存的数据或者通过 -c(删除缓存) 参数重启 pgpool-II。
1888
+ </strong>
1889
+ </p>
1890
+
1891
+ <h1><a name="memqcache"></a> 基于内存的查询缓存 <span class="version">V3.2 -</span></h1>
1892
+
1893
+ <p>你可以在任何模式中使用基于内存的查询缓存。它不同于以上的查询缓存,因为基于内存的查询缓存会快很多,因为缓存存储于内存中。
1894
+ 另外,如果缓存事小了,你不需要重启 pgpool-II 因为相关的表已经得到更新了。
1895
+ </p>
1896
+
1897
+ <p>基于内存的缓存保存 SELECT 语句(以及它绑定的参数,如果 SELECT 是一个扩展的查询)以及对应的数据。
1898
+ 如果是相同的 SELECT 语句,则直接返回缓存的值。因为不再有 SQL 分析或者到 PostgreSQL 的调用,实际上它会非常快。
1899
+ </p>
1900
+
1901
+ <p>
1902
+ 其他方面,它会比较慢,因为它增加了一些负载用于缓存。另外,当一个表被更新,pgpool 自动删除相关的表的缓存。
1903
+ 因此,在有很多更新的系统中,性能会降低。如果 cache_hit_ratio 低于 70%,建议你关闭基于内存的缓存。
1904
+ </p>
1905
+
1906
+ <h2 id="MEMORY_CACHE_RESTRICTIONS">限制</h2>
1907
+ <ul>
1908
+ <li>
1909
+ 基于内存的查询缓存通过监视 UPDATE,INSERT,ALTER TABLE一类的查询语句来自动删除缓存的数据。
1910
+ 但 pgpool-II 无法发现通过触发器、外键和 DROP TABLE CASCADE 产生的非显式的更新。
1911
+ 你可以通过配置 <a href="#MEMQCACHE_EXPIRE">memqcache_expire</a> 让 pgpool 在固定时间周期内自动删除缓存来避免这个问题,
1912
+ 你也可以通过配置 <a href="#BLACK_MEMQCACHE_TABLE_LIST">black_memqcache_table_list</a> 来让 pgpool 的基于内存的缓存忽略指定的表。
1913
+ </li>
1914
+
1915
+ <li>
1916
+ 如果你使用 pgpool-II 的多个实例来使用共享内存进行缓存,可能出现一个 pgpool 发现表被更新了因而删除了缓存,但另一个依旧使用旧的缓存。
1917
+ 对于这种情况,使用 memcached 是一个更好的策略。
1918
+ </li>
1919
+ </ul>
1920
+
1921
+ <h2 id="MEMORY_CACHE_ENABLED">启用基于内存的查询缓存</h2>
1922
+ <p>
1923
+ 要启用基于内存的查询缓存,设置以下选项为 on(默认为 off)。
1924
+ </p>
1925
+
1926
+ <pre>
1927
+ memory_cache_enabled = on
1928
+ </pre>
1929
+
1930
+ <h2 id="MEMQCACHE_METHOD">选择缓存存储</h2>
1931
+ <p>
1932
+ 你可以选择一个缓存策略:共享内存或者 <a href="http://memcached.org">memcached</a>(不能同时使用)。
1933
+ 使用共享内存的查询缓存很快且简单,因为你不需要安装和配置 memcached,但缓存的最大数量限制于共享内存。
1934
+ 使用 memcached 的查询缓存需要更多的负载用于访问网络,但你可以任意设置你需要的大小。
1935
+ </p>
1936
+ <p>
1937
+ 可以通过 memqcache_method 指定内存缓存的行为。可以是 “shmem”(共享内存) 或者 “memcached”。默认为 shmem。
1938
+ </p>
1939
+ <pre>
1940
+ memqcache_method = 'shmem'
1941
+ </pre>
1942
+
1943
+
1944
+ <h2 id="memqcache_cases">基于内存缓存被禁用的情况</h2>
1945
+ <p>
1946
+ 并非所有的 SELECT 和 WITH 语句可以被缓存。在包括以下列表的一些情况,为了保证数据库和缓存的一致性,缓存被避免了。
1947
+ </p>
1948
+ <ul>
1949
+ <li>SELECT 语句以 "/*NO QUERY CACHE*/" 注释开始</li>
1950
+ <li>SELECT 包含以下表的 <a href="#BLACK_MEMQCACHE_TABLE_LIST">black_memqcache_table_list</a></li>
1951
+ <li>SELECT FOR SHARE / UPDATE</li>
1952
+ <li>SELECT 包含 un-immutable 函数</li>
1953
+ <li>SELECT 包含 TEMP TABLE</li>
1954
+ <li>SELECT 包含 系统对象</li>
1955
+ <li>SELECT 包含 VIEW 和 不记录日志的表。但如果表在 <a href="#WHITE_MEMQCACHE_TABLE_LIST">white_memqcache_table_list</a> 中,结果还是会被缓存的。</li>
1956
+
1957
+ <li>SELECT 包含 VIEW </li>
1958
+ <li>SELECT 在一个被终止的显示事务中</li>
1959
+ <li>SELECT 结果超出 <a href="#MEMQCACHE_MAXCACHE">memqcache_maxcache</a></li>
1960
+ </ul>
1961
+
1962
+ <h2 id="non_memqcache_case">缓存不被使用的情况</h2>
1963
+ <p>
1964
+ 存在一些情况,及时匹配的查询缓存存在,pgpool 也不返回结果。
1965
+ </p>
1966
+ <ul>
1967
+ <li>如果一个更新擦做在一个显式的事务中执行,在事务中,pgpool 不使用查询缓存。</li>
1968
+ <li>匹配的查询是由其他用户(因为安全原因)生成的。</li>
1969
+ <li>匹配的查询由于 <a href="#MEMQCACHE_EXPIRE">memqcache_expire</a> 设置而需要被删除。</li>
1970
+ </ul>
1971
+
1972
+ <h2 id="memqcache_params">配置</h2>
1973
+
1974
+ <p>
1975
+ 以下参数可以同时用于 shmem 和 memcached。
1976
+ </p>
1977
+
1978
+ <dl>
1979
+ <dt id="MEMQCACHE_EXPIRE">memqcache_expire <span class="version">V3.2 -</span></dt>
1980
+ <dd>
1981
+ <p>
1982
+ 查询缓存的生命周期,默认为 0。0 表示没有缓存超时,而且缓存被启用直到表被更新。
1983
+ 本参数和 <a href="#MEMQCACHE_AUTO_CACHE_INVALIDATION">memqcache_auto_cache_invalidation</a> 是相交的。
1984
+ </p>
1985
+ </dd>
1986
+
1987
+ <dt id="MEMQCACHE_AUTO_CACHE_INVALIDATION">memqcache_auto_cache_invalidation
1988
+ <span class="version">V3.2 -</span></dt>
1989
+ <dd>
1990
+ <p>
1991
+ 如果为 on,则在表被更新的时候自动删除相关的缓存。
1992
+ 如果为 off,则不删除缓存。默认为 on。
1993
+ 本参数和 <a href="#MEMQCACHE_EXPIRE">memqcache_expire</a> 相交。
1994
+ </p>
1995
+ </dd>
1996
+
1997
+ <dt id="MEMQCACHE_MAXCACHE">memqcache_maxcache <span class="version">V3.2 -</span></dt>
1998
+ <dd>
1999
+ <p>
2000
+ 如果 SELECT 结果集的大小超出了 memqcache_maxcache 字节,则不缓存且显示以下消息:
2001
+ </p>
2002
+ <pre>
2003
+ 2012-05-02 15:08:17 LOG: pid 13756: pool_add_temp_query_cache: data size exceeds memqcache_maxcache. current:4095 requested:111 memq_maxcache:4096
2004
+ </pre>
2005
+ <p>
2006
+ 要避免这个问题,你需要将 memqcache_maxcache 设置得大一些。
2007
+ 但如果你使用共享内存作为缓存策略,它必须小于 <a href="#MEMQCACHE_CACHE_BLOCK_SIZE">memqcache_cache_block_size</a>。
2008
+ 如果是 memchached,她必须小于 slab 的大小(默认为 1 MB)。
2009
+ </p>
2010
+ </dd>
2011
+
2012
+ <dt id="WHITE_MEMQCACHE_TABLE_LIST">white_memqcache_table_list <span class="version">V3.2 -</span></dt>
2013
+ <dd>
2014
+ <p>
2015
+ 指定一个以逗号分隔的表名的列表,用于使 SELECT 的结果被缓存,也可以是视图或者不写日志的表。可以使用正则表达式。
2016
+ </p>
2017
+ <p>
2018
+ 同时存在于 white_memqcache_table_list 和 <a href="#BLACK_MEMQCACHE_TABLE_LIST">black_memqcache_table_list</a> 的表和视图将被缓存。
2019
+ </p>
2020
+ </dd>
2021
+
2022
+ <dt id="BLACK_MEMQCACHE_TABLE_LIST">black_memqcache_table_list <span class="version">V3.2 -</span></dt>
2023
+ <dd>
2024
+ <p>
2025
+ 指定一个以逗号分隔的表名的列表,用于使 SELECT 的结果不被缓存,也可以是视图或者不写日志的表。可以使用正则表达式。
2026
+ </p>
2027
+ </dd>
2028
+
2029
+ <dt id="MEMQCACHE_OIDDIR">memqcache_oiddir <span class="version">V3.2 -</span></dt>
2030
+ <dd>
2031
+ <p>
2032
+ 用于 SELECT 的存储表的 OID 的目录的完整路径。在 memqcache_oiddir 下有使用数据库 oid 命名的目录,
2033
+ 每个目录之下是在 SELECT 中使用的以表的 oid 命名的文件。在文件中,存储了查询缓存。它们是用于删除缓存的关键。
2034
+ </p>
2035
+ <p>
2036
+ 在 memqcache_oiddir 下的目录和文件不会被删除,即使 pgpool-II 重启。
2037
+ 如果你使用 "<a href="#start">pgpool -C</a>" 启动 pgpool,则 pgpool 使用旧的 oid 映射。
2038
+ </p>
2039
+ </dd>
2040
+ </dl>
2041
+
2042
+ <h2 id="monitoring_memqcache">监控缓存</h2>
2043
+ <p>
2044
+ 这里讲述如何监控查询缓存。要知道一个 SELECT 的结果是不是从查询缓存获得,需要启用 <a href="#LOG_PER_NODE_STATEMENT">log_per_node_statement</a>。
2045
+ </p>
2046
+ <pre>
2047
+ 2012-05-01 15:42:09 LOG: pid 20181: query result fetched from cache. statement: select * from t1;
2048
+ </pre>
2049
+
2050
+ <p>
2051
+ <a href="#pool_status">pool_status</a> 命令显示缓存的命中率。
2052
+ </p>
2053
+ <pre>
2054
+ memqcache_stats_start_time | Tue May 1 15:41:59 2012 | Start time of query cache stats
2055
+ memqcache_no_cache_hits | 80471 | Number of SELECTs not hitting query cache
2056
+ memqcache_cache_hits | 36717 | Number of SELECTs hitting query cache
2057
+ </pre>
2058
+
2059
+
2060
+ <p>
2061
+ 在本例中,你可以通过以下方法计算:
2062
+ </p>
2063
+ <p>
2064
+ <pre>
2065
+ (memqcache_cache_hits) / (memqcache_no_cache_hits+memqcache_cache_hits) = 36717 / (36717 + 80471) = 31.3%
2066
+ </pre>
2067
+
2068
+ <p>
2069
+ <a href="#pool_cache">show pool_cache</a> 也显示相同的结果。
2070
+ </p>
2071
+
2072
+ <h2 id="shmem_params">配置使用共享内存</h2>
2073
+ <p>
2074
+ 以下为用于共享内存缓存策略的参数。
2075
+ </p>
2076
+
2077
+ <dl>
2078
+ <dt id="MEMQCACHE_TOTAL_SIZE">memqcache_total_size <span class="version">V3.2 -</span></dt>
2079
+ <dd>
2080
+ <p>
2081
+ 指定用于缓存的共享内存的大小,单位为字节。
2082
+ </p>
2083
+ </dd>
2084
+
2085
+ <dt id="MEMQCACHE_MAX_NUM_CACHE">memqcache_max_num_cache <span class="version">V3.2 -</span></dt>
2086
+ <dd>
2087
+ <p>
2088
+ 制定缓存的项目数。这用于定义缓存管理空间的大小。(你需要它来协助 <a href="#MEMQCACHE_TOTAL_SIZE">memqcache_total_size</a> 参数)。
2089
+ 可以使用以下方法计算缓存管理空间的大小:<a href="#MEMQCACHE_MAX_NUM_CACHE">memqcache_max_num_cache</a> * 48 字节。
2090
+ 如果数量太小,则注册缓存的时候会报错。但如果太大则浪费空间。
2091
+ </p>
2092
+ </dd>
2093
+
2094
+ <dt id="MEMQCACHE_CACHE_BLOCK_SIZE">memqcache_cache_block_size <span class="version">V3.2 -</span></dt>
2095
+ <dd>
2096
+ <p>
2097
+ 如果缓存存储是共享内存,pgpool 使用内存除以 memqcache_cache_block_size。SELECT 的结果被放入块中。
2098
+ 但是因为 SELECT 的结果不能被放入多个块中,它无法缓存大于 memqcache_cache_block_size 的结果。
2099
+ memqcache_cache_block_size 必须大于或等于 512。
2100
+ </p>
2101
+ </dd>
2102
+ </dl>
2103
+
2104
+ <h2 id="memcached_params">配置使用 memcached</h2>
2105
+
2106
+ <p>
2107
+ 以下为用于 memcached 缓存策略的参数。
2108
+ </p>
2109
+
2110
+ <dl>
2111
+ <dt id="MEMQCACHE_MEMCACHED_HOST">memqcache_memcached_host <span class="version">V3.2 -</span></dt>
2112
+ <dd>
2113
+ <p>
2114
+ 指定 memcached 工作的主机的主机名或 IP 地址。如果和 pgpool-II 在同一台机器,设置为 'localhost'。
2115
+ </p>
2116
+ </dd>
2117
+
2118
+ <dt id="MEMQCACHE_MEMCACHED_PORT">memqcache_memcached_port <span class="version">V3.2 -</span></dt>
2119
+ <dd>
2120
+ <p>
2121
+ 指定 memcached 的端口。默认为 11211。
2122
+ </p>
2123
+ </dd>
2124
+ </dl>
2125
+
2126
+
2127
+ <h3 id="install_memcached">memcached 安装</h3>
2128
+
2129
+ <p>
2130
+ 要使用 memcached 作为缓存策略,pgpool-II 需要配合 memcached 和其客户端 libmemcached 使用。
2131
+ 通过 rpm 安装很简单。这里讲解如何使用源码安装。
2132
+ </p>
2133
+
2134
+ <p>
2135
+ 可以在以下位置获得 memcached 的源码:
2136
+ <a href="http://memcached.org/">memcached development page</a>
2137
+ </p>
2138
+
2139
+ <dl>
2140
+ <dt>configure</dt>
2141
+ <dd>
2142
+ <p>
2143
+ 在解压源码包后,执行配置脚本。
2144
+ </p>
2145
+ <pre>
2146
+ ./configure
2147
+ </pre>
2148
+ </dd>
2149
+
2150
+ <dt>make</dt>
2151
+ <dd>
2152
+ <pre>
2153
+ make
2154
+ make install
2155
+ </pre>
2156
+ </dd>
2157
+ </dl>
2158
+
2159
+ <h3 id="install_libmemcached">libmemcached 安装</h3>
2160
+
2161
+ <p>
2162
+ Libmemcached 是 memcached 的客户端库。你需要在安装 memcached 后安装 libmemcached。
2163
+ </p>
2164
+
2165
+ <p>
2166
+ 可以在以下位置获得 libmemcached 的源码:
2167
+ <a href="http://libmemcached.org/libMemcached.html">libmemcached development page</a>
2168
+ </p>
2169
+
2170
+ <dl>
2171
+ <dt>configure</dt>
2172
+ <dd>
2173
+ <p>
2174
+ 在解压源码包后,执行配置脚本。
2175
+ </p>
2176
+ <pre>
2177
+ ./configure
2178
+ </pre>
2179
+
2180
+ <p>
2181
+ 如果你不想使用默认值,一些可选项为:
2182
+ </p>
2183
+
2184
+ <ul>
2185
+ <li><code>--with-memcached=path</code><br/>
2186
+ 安装 Memcached 的顶层目录。</li>
2187
+ </ul>
2188
+ </dd>
2189
+
2190
+ <dt>make</dt>
2191
+ <dd>
2192
+ <pre>
2193
+ make
2194
+ make install
2195
+ </pre>
2196
+ </dd>
2197
+ </dl>
2198
+
2199
+ <p class="top_link"><a href="#Top">回到顶部</a></p>
2200
+
2201
+ <!-- ================================================================================ -->
2202
+
2203
+ <h1>启动/停止 pgpool-II<a name="start"></a></h1>
2204
+
2205
+ <h2 id="start_pgpool">启动 pgpool-II</h2>
2206
+
2207
+ <p>所有后端服务器和System DB(若需要)都必须在启动 pgpool-II 之前启动。
2208
+ </p>
2209
+
2210
+ <pre>
2211
+ pgpool [-c][-f config_file][-a hba_file][-F pcp_config_file][-n][-D][-d]
2212
+ </pre>
2213
+
2214
+ <table border>
2215
+ <tr><td>-c</td><td>--clear-cache</td>
2216
+ <td>删除 query cache</td></tr>
2217
+ <tr><td>-f config_file</td><td>--config-file config-file</td>
2218
+ <td>指定 pgpool.conf</td></tr>
2219
+ <tr><td>-a hba_file</td><td>--hba-file hba_file</td>
2220
+ <td>指定 pool_hba.conf</tr>
2221
+ <tr><td>-F pcp_config_file</td><td>--pcp-password-file</td>
2222
+ <td>指定 pcp.conf</td></tr>
2223
+ <tr><td>-n</td><td>--no-daemon</td>
2224
+ <td>非守护进程模式(不脱离终端运行)</td></tr>
2225
+ <tr><td>-D</td><td>--discard-status</td>
2226
+ <td>忽略 pgpool_status 文件且不恢复到先前的状态
2227
+ <span class="version">V3.0 -</span></td></tr>
2228
+ <tr><td>-C</td><td>--clear-oidmaps</td>
2229
+ <td>忽略用于基于内存的查询缓存的 <a href="#MEMQCACHE_OIDDIR">memqcache_oiddir</a> 中的 oid 映射文件
2230
+ (仅当 <a href="#MEMQCACHE_METHOD">memqcache_method</a> 为 'memcached'时有效),
2231
+ 如果是 shm,则总是忽略。
2232
+ <span class="version">V3.2 -</span></td></tr>
2233
+ <tr><td>-d</td><td>--debug</td><td>调试模式</tr>
2234
+ </table>
2235
+
2236
+ <h2 id="stop_pgpool">Stop pgpool-II</h2>
2237
+ <p>
2238
+ 有两种方式来关闭 pgpool-II。一种是使用 PCP 命令(后面介绍),另外一种是使用 pgpool-II 命令。下面是使用 pgpool-II 命令的示例。
2239
+ </p>
2240
+
2241
+ <pre>
2242
+ pgpool [-f config_file][-F pcp_config_file] [-m {s[mart]|f[ast]|i[mmediate]}] stop
2243
+ </pre>
2244
+
2245
+ <table border>
2246
+ <tr><td><code>-m s[mart]</code></td><td><code>--mode s[mart]</code></td>
2247
+ <td>等待客户端断开连接,然后关闭(默认)</td></tr>
2248
+ <tr><td><code>-m f[ast]</code></td><td><code>--mode f[ast]</code></td>
2249
+ <td>并不等待客户端; 立即关闭</td></tr>
2250
+ <tr><td><code>-m i[mmediate]</code></td><td><code>--mode i[mmediate]</code></td>
2251
+ <td>等同于 <code>'-m f'</code></td></tr>
2252
+ </table>
2253
+
2254
+ <p>
2255
+ pgpool 记录后端服务器的状态到 [logdir]/pgpool_status 文件中,当 pgpool 重启时,它读该文件并恢复后端服务器的状态。
2256
+ 这将避免可能由于下述原因造成的DB节点间的具有不同的数据:
2257
+ <ol>
2258
+ <li>一个后端服务器突然停止,然后pgpool执行恢复程序
2259
+ <li>一个update通过pgpool出现在一个活动的DBs上
2260
+ <li>管理员决定停止pgpool
2261
+ <li>其它人决定重启已停止的DB但并没通知管理员
2262
+ <li>管理员重启pgpool
2263
+ </ol>
2264
+ </p>
2265
+ <p>
2266
+ 如果由于某些原因,例如,停止的DB已经通过其它另外的方式已经获得了同步,则 pgpool_status 可在启动 pgpool 之前被安全的移除。
2267
+ </p>
2268
+
2269
+ <p class="top_link"><a href="#Top">回到顶部</a></p>
2270
+
2271
+ <!-- ================================================================================ -->
2272
+
2273
+ <h1>重新加载 pgpool-II 配置文件<a name="reload"></a></h1>
2274
+ <p>pgpool-II 能够不需要重启而重新加载配置文件。
2275
+ </p>
2276
+
2277
+ <pre>
2278
+ pgpool [-c][-f config_file][-a hba_file][-F pcp_config_file] reload
2279
+ </pre>
2280
+ <p>
2281
+
2282
+ <table border>
2283
+ <tr><td>-f config_file</td><td>--config-file config-file</td><td>指定 pgpool.conf</tr>
2284
+ <tr><td>-a hba_file</td><td>--hba-file hba_file</td><td>指定 pool_hba.conf</tr>
2285
+ <tr><td>-F pcp_config_file</td><td>--pcp-password-file</td><td>指定 pcp.conf</tr>
2286
+ </table>
2287
+
2288
+ <p>
2289
+ 需要指出的是,一些配置项并不能通过重新加载来改变。新的修改后的配置将在新会话上生效。
2290
+ </p>
2291
+
2292
+ <p class="top_link"><a href="#Top">回到顶部</a></p>
2293
+
2294
+ <!-- ================================================================================ -->
2295
+
2296
+ <h1><a name="show-commands"></a>SHOW 命令</h1>
2297
+ <h2>概述</h2>
2298
+ <p>
2299
+ pgpool-II 通过 SHOW 命令提供一些信息。SHOW 是一个真实的 SQL 语句, 但是如果该命令查询 pgpool-II 信息的话,pgpool-II 解释了该命令。可用选项如下:
2300
+ <ul>
2301
+ <li>pool_status, 获取配置</li>
2302
+ <li>pool_nodes, 获取节点信息 <span class="version">V3.0 -</span></li>
2303
+ <li>pool_processes, 获取pgPool-II 进程信息 <span class="version">V3.0 -</span></li>
2304
+ <li>pool_pools, 获取pgPool-II 所有的连接池信息 <span class="version">V3.0 -</span></li>
2305
+ <li>pool_version, 获取pgPool_II 版本信息 <span class="version">V3.0 -</span></li>
2306
+ </ul>
2307
+ <p>除 "pool_status" 之外的选项都是从 pgpool-II 3.0 开始添加的。
2308
+ <p>
2309
+ <u>注意</u>:术语 'pool' 指的是一个 pgpool 进程所拥有的 PostgreSQL 会话池,并非指所有的 pgpool 所拥有的会话。
2310
+ </p>
2311
+ </p>
2312
+ <p>SQL语句中的 "pool_status" 在以前的版本中已经存在,但是其它可选项在 3.0 中才出现。
2313
+ </p>
2314
+ <h2>pool_status</h2>
2315
+ <p>"SHOW pool_status" 返回配置参数列表,包含参数的 name, value, 和 description。下面是返回结果的一个摘录:
2316
+ <pre>
2317
+ benchs2=# show pool_status;
2318
+ item | value | description
2319
+ --------------------------------------+--------------------------------+------------------------------------------------------------------
2320
+ listen_addresses | localhost | host name(s) or IP address(es) to listen to
2321
+ port | 9999 | pgpool accepting port number
2322
+ socket_dir | /tmp | pgpool socket directory
2323
+ pcp_port | 9898 | PCP port # to bind
2324
+ pcp_socket_dir | /tmp | PCP socket directory
2325
+ </pre>
2326
+ </p>
2327
+ <h2>pool_nodes <span class="version">V3.0 -</span></h2>
2328
+ <p>"SHOW pool_nodes" 返回所有配置节点的列表.它显示了节点的 id, hostname, port, status, 以及权重 weight (仅在使用负载均衡模式时有意义)。status 列可能的值可在 <a href="#pcp_node_info">pcp_node_info reference</a> 里获得解释。
2329
+ <pre>
2330
+ benchs2=# show pool_nodes;
2331
+ id | hostname | port | status | lb_weight | role
2332
+ ------+-------------+------+--------+-----------+---------
2333
+ 0 | 127.0.0.1 | 5432 | 2 | 0.5 | primary
2334
+ 1 | 192.168.1.7 | 5432 | 3 | 0.5 | standby
2335
+ (2 lignes)
2336
+ </pre>
2337
+ </p>
2338
+ <h2>pool_processes <span class="version">V3.0 -</span></h2>
2339
+ <p>"SHOW pool_processes" 返回 pgpool-II 所有进程的信息,这些进程正等待连接或在处理一个连接。
2340
+ </p>
2341
+ <p>
2342
+ 结果有 6 列:
2343
+ <ul>
2344
+ <li>pool_pid 是显示的pgPool-II进程的PID</li>
2345
+ <li>start_time 是该进程启动时的时间戳</li>
2346
+ <li>database 是该进程当前连接的活跃的后台数据库名</li>
2347
+ <li>username 是该进程当前连接的活跃的后台用户名</li>
2348
+ <li>create_time 是连接的的创建时间和日期</li>
2349
+ <li>pool_counter 计数客户端使用该连接池(进程)的次数</li>
2350
+ </ul>
2351
+ </p>
2352
+ <p>结果将一直返回 num_init_children 行。</p>
2353
+ <pre>
2354
+ benchs2=# show pool_processes;
2355
+ pool_pid | start_time | database | username | create_time | pool_counter
2356
+ ----------+---------------------+----------+-----------+---------------------+--------------
2357
+ 8465 | 2010-08-14 08:35:40 | | | |
2358
+ 8466 | 2010-08-14 08:35:40 | benchs | guillaume | 2010-08-14 08:35:43 | 1
2359
+ 8467 | 2010-08-14 08:35:40 | | | |
2360
+ 8468 | 2010-08-14 08:35:40 | | | |
2361
+ 8469 | 2010-08-14 08:35:40 | | | |
2362
+ (5 lines)
2363
+ </pre>
2364
+ <h2>pool_pools <span class="version">V3.0 -</span></h2>
2365
+ <p>"SHOW pool_pools" 返回 pgpool-II 所处理的连接池列表,包含它们的 name, value, 和 description,下面是结果的一个摘录:
2366
+ </p>
2367
+ <p>
2368
+ 共 11 列:
2369
+ <ul>
2370
+ <li>pool_pid 是 pgpool-II 进程的PID</li>
2371
+ <li>start_time 是该进程启动时的时间和日期</li>
2372
+ <li>pool_id is 是连接池的标识符 (应该在 0 到 max_pool-1之间)</li>
2373
+ <li>backend_id 是后端服务器的标识符(应该在0到最大配置的后端服务器数-1之间)</li>
2374
+ <li>database 是该进程的连接池所连接的数据库名</li>
2375
+ <li>username 是该进程的连接池所连接的用户名</li>
2376
+ <li>create_time 是该连接的创建时间和日期</li>
2377
+ <li>majorversion and minorversion 是该连接使用的协议版本</li>
2378
+ <li>pool_counter 计数客户端使用该连接的次数</li>
2379
+ <li>pool_backendpid 是PostgreSQL 进程的PID </li>
2380
+ <li>pool_connected 如果前端正使用该后端服务器时是true (1).</li>
2381
+ </ul>
2382
+ </p>
2383
+ <p>结果将一直返回 <a href="#NUM_INIT_CHILDREN">num_init_children</a> * <a href="#MAX_POOL">max_pool</a> 行.</p>
2384
+ <pre>
2385
+ pool_pid | start_time | pool_id | backend_id | database | username | create_time | majorversion | minorversion | pool_counter | pool_backendpid | pool_connected
2386
+ ----------+---------------------+---------+------------+----------+-----------+---------------------+--------------+--------------+--------------+-----------------+----------------
2387
+ 8465 | 2010-08-14 08:35:40 | 0 | 0 | | | | | | | |
2388
+ 8465 | 2010-08-14 08:35:40 | 1 | 0 | | | | | | | |
2389
+ 8465 | 2010-08-14 08:35:40 | 2 | 0 | | | | | | | |
2390
+ 8465 | 2010-08-14 08:35:40 | 3 | 0 | | | | | | | |
2391
+ 8466 | 2010-08-14 08:35:40 | 0 | 0 | benchs | guillaume | 2010-08-14 08:35:43 | 3 | 0 | 1 | 8473 | 1
2392
+ 8466 | 2010-08-14 08:35:40 | 1 | 0 | | | | | | | |
2393
+ 8466 | 2010-08-14 08:35:40 | 2 | 0 | | | | | | | |
2394
+ 8466 | 2010-08-14 08:35:40 | 3 | 0 | | | | | | | |
2395
+ 8467 | 2010-08-14 08:35:40 | 0 | 0 | | | | | | | |
2396
+ 8467 | 2010-08-14 08:35:40 | 1 | 0 | | | | | | | |
2397
+ 8467 | 2010-08-14 08:35:40 | 2 | 0 | | | | | | | |
2398
+ 8467 | 2010-08-14 08:35:40 | 3 | 0 | | | | | | | |
2399
+ 8468 | 2010-08-14 08:35:40 | 0 | 0 | | | | | | | |
2400
+ 8468 | 2010-08-14 08:35:40 | 1 | 0 | | | | | | | |
2401
+ 8468 | 2010-08-14 08:35:40 | 2 | 0 | | | | | | | |
2402
+ 8468 | 2010-08-14 08:35:40 | 3 | 0 | | | | | | | |
2403
+ 8469 | 2010-08-14 08:35:40 | 0 | 0 | | | | | | | |
2404
+ 8469 | 2010-08-14 08:35:40 | 1 | 0 | | | | | | | |
2405
+ 8469 | 2010-08-14 08:35:40 | 2 | 0 | | | | | | | |
2406
+ 8469 | 2010-08-14 08:35:40 | 3 | 0 | | | | | | | |
2407
+ (20 lines)
2408
+ </pre>
2409
+
2410
+ <h2 id="pool_version">pool_version <span class="version">V3.0 -</span></h2>
2411
+ <p>"SHOW pool_version" 将显示包含 pgpool-II 版本号的一个字符串。下面是一个示例:</p>
2412
+ <pre>
2413
+ benchs2=# show pool_version;
2414
+ pool_version
2415
+ ------------------------
2416
+ 3.0-dev (umiyameboshi)
2417
+ (1 line)
2418
+ </pre>
2419
+
2420
+ <h2 id="pool_cache">pool_cache <span class="version">V3.0 -</span></h2>
2421
+ <p>"SHOW pool_cache" 显示启用 <a href="#memqcache">on memory query cache</a> 时缓存的存储统计。以下为一个示例:
2422
+ </p>
2423
+
2424
+ <pre>
2425
+ test=# \x
2426
+ \x
2427
+ Expanded display is on.
2428
+ test=# show pool_cache;
2429
+ show pool_cache;
2430
+ -[ RECORD 1 ]---------------+---------
2431
+ num_cache_hits | 891703
2432
+ num_selects | 99995
2433
+ cache_hit_ratio | 0.90
2434
+ num_hash_entries | 131072
2435
+ used_hash_entries | 99992
2436
+ num_cache_entries | 99992
2437
+ used_cache_enrties_size | 12482600
2438
+ free_cache_entries_size | 54626264
2439
+ fragment_cache_entries_size | 0
2440
+ </pre>
2441
+
2442
+ <ul>
2443
+ <li>num_cache_hits 表示命中缓存的 SELECT 语句数量。</li>
2444
+ <li>num_selects 表示未命中缓存的 SELECT 语句数量。</li>
2445
+ <li>
2446
+ cache_hit_ratio 表示缓存的命中率,算法为 num_cache_hits/(num_cache_hits+num_selects)。
2447
+ <br>num_hash_entries 之下的任何项目仅仅在缓存为共享内存的情况下有效。
2448
+ </li>
2449
+ <li>
2450
+ num_hash_entries 表示哈希表中的项目数,它用于索引缓存存储,其应该和 pgpool.conf 中的
2451
+ <a href="#MEMQCACHE_MAX_NUM_CACHE">memqcache_max_num_cache</a> 相同。这是缓存条目的上限。
2452
+ </li>
2453
+ <li>used_hash_entries 表示 num_hash_entries 中已经使用的项目。</li>
2454
+ <li>num_cache_entries 表示缓存存储中有效的缓存项目数,应该等于 used_hash_entries。</li>
2455
+ <li>used_cache_entries_size 表示已经使用的总共缓存存储字节数。</li>
2456
+ <li>free_cache_entries_size 表示未使用或者说可用的总共缓存存储字节数。</li>
2457
+ <li>fragment_cache_entries_size 表示由于是碎片而无法使用的总共缓存存储字节数。</li>
2458
+ <li>如果 free_cache_entries_size 变成 0(或者没有足够的空间存储 SELECT 的结果),则碎片区域可以被重用。</li>
2459
+ </ul>
2460
+
2461
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
2462
+
2463
+ <!-- ================================================================================ -->
2464
+
2465
+ <h1><a name="online-recovery"></a>在线恢复</h1>
2466
+ <h2>概述</h2>
2467
+ <p>
2468
+ pgpool-II,在复制模式下在给客户端提供服务的同时,能够同步一个数据库并关联(attach)一个节点,我们称这项特性为"在线恢复"。
2469
+ </p>
2470
+
2471
+ <p>
2472
+ 在执行在线恢复之前,一个恢复的目标节点必须处于 detached 状态。如果你想动态的增加一个 PostgreSQL server,则在pgpool.conf中增加'backend_hostname'及其相关的参数并重新加载该配置文件。 pgpool-II 把该新的节点注册为 detached 节点。
2473
+ </p>
2474
+
2475
+ <p>
2476
+ <font color="red">
2477
+ 注意:停止 master 节点(第一个启动和运行的节点)上的autovacuum。如果Autovacuum运行的话,它可能会改变数据的内容并可能造成在线恢复后的不一致性。这仅适用于使用简单的复制机制的恢复,例如下文将会讨论的 rsync 方式,而对使用 PostgreSQL 的PITR机制进行在线恢复的并不适用。
2478
+ </font>
2479
+ </p>
2480
+
2481
+ <p>
2482
+ 若目标 PostgreSQL 服务器已经启动开,你需要关闭它。
2483
+ </p>
2484
+
2485
+ <p>
2486
+ pgpool-II 以两个阶段来执行在线恢复。当一个恢复节点同步数据库时,客户端可能需要等待几秒到几分钟不等的时间来连接 pgpool-II。下面是这些步骤:
2487
+ <ol>
2488
+ <li> CHECKPOINT
2489
+ <li> 在线恢复的第一阶段
2490
+ <li> 等待所有的客户端连接断开
2491
+ <li> CHECKPOINT
2492
+ <li> 在线恢复的第二阶段
2493
+ <li> 启动 postmaster (执行 <a href="#pool_remote_start">pgpool_remote_start</a>)
2494
+ <li> 节点连接上来
2495
+ </ol>
2496
+ </p>
2497
+
2498
+ <p>
2499
+ 数据同步的第一个步称之为"第一阶段"。 数据在第一阶段中被同步,在第一阶段中,数据<b>能够</b>被并行的从任意表更新或获取。
2500
+ </p>
2501
+
2502
+ <p>
2503
+ 你可以指定第一阶段中执行的脚本。pgpool-II传递三个参数到该脚本中。
2504
+ <ol>
2505
+ <li>主节点的数据库实例路径。
2506
+ <li>被恢复的目标节点的主机名
2507
+ <li>被恢复的目标节点的数据库实例路径
2508
+ </ol>
2509
+ </p>
2510
+
2511
+ <p>
2512
+ 数据同步在称之为"第二阶段"中完成。在进入第二阶段前,pgpool-II 等待所有的客户端断开连接。它阻塞所有新来的连接直到第二阶段完成。当所有的连接被中断后,pgpool-II合并在第一和第二阶段中更新的数据。这是同步的最终数据。
2513
+ </p>
2514
+
2515
+ <p>
2516
+ <font color="red">
2517
+ 注意:关于在线恢复有一个限制。如果 pgpool-II 本身安装在多个主机上,在线恢复则不能正确的工作,这是因为 pgpool-II需要在在线恢复的第二阶段中断所有客户端连接。如果存在多个 pgpool-II 主机,只有一个主机能收到在线恢复命令,然后阻塞连接。
2518
+ </font>
2519
+ </p>
2520
+
2521
+ <h2 id="online_recovery_params">配置在线恢复</h2>
2522
+ <p>
2523
+ 为在线恢复在pgpool.conf中设置如下参数:
2524
+ </p>
2525
+
2526
+ <ul>
2527
+ <li><a href="#BACKEND_DATA_DIRECTORY">backend_data_directory</a></li>
2528
+ <li><a href="#RECOVERY_USER">recovery_user</a></li>
2529
+ <li><a href="#RECOVERY_PASSWORD">recovery_password</a></li>
2530
+ <li><a href="#RECOVERY_1ST_STAGE_COMMAND">recovery_1st_stage_command</a></li>
2531
+ <li><a href="#RECOVERY_2ND_STAGE_COMMAND">recovery_2nd_stage_command</a></li>
2532
+ </ul>
2533
+
2534
+
2535
+ <h2><a name="installing-c-functions"></a>安装c语言函数</h2>
2536
+ <p>
2537
+ 你需要为在线恢复在所有的后端节点中的"template1"数据库中安装c语言函数。源代码在pgpool-II的压缩包中。
2538
+ </p>
2539
+
2540
+ <pre>
2541
+ pgpool-II-x.x.x/sql/pgpool-recovery/
2542
+ </pre>
2543
+
2544
+ <p>
2545
+ 改变相应的目录,并执行"make install"。
2546
+ </p>
2547
+
2548
+ <pre>
2549
+ % cd pgpool-II-x.x.x/sql/pgpool-recovery/
2550
+ % make install
2551
+ </pre>
2552
+
2553
+ <p>
2554
+ 然后安装 SQL 函数。
2555
+ </p>
2556
+
2557
+ <pre>
2558
+ % cd pgpool-II-x.x.x/sql/pgpool-recovery/
2559
+ % psql -f pgpool-recovery.sql template1
2560
+ </pre>
2561
+
2562
+
2563
+ <h2 id="recovery_script"> 恢复脚本部署 </h2>
2564
+ <p>
2565
+ 我们必须在数据库实例目录($PGDATA)中部署一些数据同步脚本和一个远程启动脚本。在pgpool-II-x.x.x/sample 目录中可找到一些示例脚本文件。
2566
+ </p>
2567
+
2568
+ <h3 id="recovery_pitr">通过PITR在线恢复</h3>
2569
+ <p>
2570
+ 本节说明如何通过基于时间点的恢复(Point In Time Recovery,PITR)执行在线恢复。PITR 在 PostgreSQL 8.2 以及后续版本可用。注意所有的涉及的 PostgreSQL 服务器都需要支持 PITR。
2571
+ </p>
2572
+ <h4>第一阶段</h4>
2573
+ <p>
2574
+ 需要一个脚本从主节点获取一个备份库,然后把它拷贝到恢复目标节点中去。该脚本可命名为如"copy-base-backup"。下面是个示例:
2575
+ </p>
2576
+
2577
+ <pre>
2578
+ #! /bin/sh
2579
+ DATA=$1
2580
+ RECOVERY_TARGET=$2
2581
+ RECOVERY_DATA=$3
2582
+
2583
+ psql -c "select pg_start_backup('pgpool-recovery')" postgres
2584
+ echo "restore_command = 'scp $HOSTNAME:/data/archive_log/%f %p'" &gt; /data/recovery.conf
2585
+ tar -C /data -zcf pgsql.tar.gz pgsql
2586
+ psql -c 'select pg_stop_backup()' postgres
2587
+ scp pgsql.tar.gz $RECOVERY_TARGET:$RECOVERY_DATA
2588
+ </pre>
2589
+
2590
+ <p>
2591
+ 该脚本把主数据库设成backup模式,生成如下的 recovery.conf:
2592
+ </p>
2593
+ <pre>
2594
+ restore_command = 'scp master:/data/archive_log/%f %p'
2595
+ </pre>
2596
+ <p>
2597
+ 执行备份,接着把主数据库不再设成备份模式,然后拷贝备份库到选择的目标节点。
2598
+ </p>
2599
+ <h4>第二阶段</h4>
2600
+ <p>
2601
+ 该过程的第二阶段是一个脚本进行强制 XLOG 文件转换。该脚本此处命名为"pgpool_recovery_pitr"。它强制执行事务日志的一个转换。为此,pg_switch_xlog可能会被用到。</p>
2602
+ <p><span class="version">V3.1 -</span>然而它可能在转换执行完<b>前</b>返回,这样可能会导致在线恢复过程的失败。Pgpool-II提供了一个更加安全的称之为"pgpool_switch_xlog"的函数,该函数等待一直到事务日志转换实际被完成后。pgpool_switch_xlog 在<a href="#installing-c-functions">安装 C 函数</a>小节过程执行时被安装。
2603
+ </p>
2604
+ <p>
2605
+ 下面是示例脚本。
2606
+ </p>
2607
+ <p>
2608
+ <pre>
2609
+ #! /bin/sh
2610
+ # Online recovery 2nd stage script
2611
+ #
2612
+ datadir=$1 # master dabatase cluster
2613
+ DEST=$2 # hostname of the DB node to be recovered
2614
+ DESTDIR=$3 # database cluster of the DB node to be recovered
2615
+ port=5432 # PostgreSQL port number
2616
+ archdir=/data/archive_log # archive log directory
2617
+
2618
+ # Force to flush current value of sequences to xlog
2619
+ psql -p $port -t -c 'SELECT datname FROM pg_database WHERE NOT datistemplate AND datallowconn' template1|
2620
+ while read i
2621
+ do
2622
+ if [ "$i" != "" ];then
2623
+ psql -p $port -c "SELECT setval(oid, nextval(oid)) FROM pg_class WHERE relkind = 'S'" $i
2624
+ fi
2625
+ done
2626
+
2627
+ psql -p $port -c "SELECT pgpool_switch_xlog('$archdir')" template1
2628
+ </pre>
2629
+ </p>
2630
+
2631
+ <p>
2632
+ 该序列注入(flushing of sequences)仅在复制模式下有效:在该情况下,序列不得不在所有节点上具有同一个起点。在主/备模式下并不可用。脚本中的循环强制 PostgreSQL 发出所有数据库中序列生成器的当前值到事务日志中,这样它就传播到恢复的目标节点中去。
2633
+ </p>
2634
+
2635
+ <p>
2636
+ 我们部署这些脚本到 $PGDATA 目录中去。
2637
+ </p>
2638
+ <p>
2639
+ 最后,我们修改pgpool.conf。
2640
+
2641
+ <pre>
2642
+ recovery_1st_stage_command = 'copy-base-backup'
2643
+ recovery_2nd_stage_command = 'pgpool_recovery_pitr'
2644
+ </pre>
2645
+
2646
+ </p>
2647
+
2648
+ <p>
2649
+ 我们已完成通过PITR进行在线恢复的准备工作。
2650
+ </p>
2651
+
2652
+ <h4 id="pool_remote_start">pgpool_remote_start</h4>
2653
+ <p>
2654
+ 该脚本启动远程主机的 postmaster 进程。pgpool-II 以如下方式执行。
2655
+ </p>
2656
+
2657
+ <pre>
2658
+ % pgpool_remote_start remote_host remote_datadir
2659
+ remote_host: Hostname of a recovery target.
2660
+ remote_datadir: Database cluster path of a recovery target.
2661
+ </pre>
2662
+
2663
+ <p>
2664
+ 在该示例脚本中,我们通过 ssh 启动 postmaster 进程。所以如果你想要它能够运行,你需要提供不需要密码输入的 ssh 连接。
2665
+ </p>
2666
+
2667
+ <p>
2668
+ 如果你使用PITR进行恢复,你需要部署一个库备份。PostgreSQL 将自动启动一个 PITR 恢复,然后它开始接受连接。
2669
+ </p>
2670
+
2671
+ <pre>
2672
+ #! /bin/sh
2673
+ DEST=$1
2674
+ DESTDIR=$2
2675
+ PGCTL=/usr/local/pgsql/bin/pg_ctl
2676
+
2677
+ # Deploy a base backup
2678
+ ssh -T $DEST 'cd /data/; tar zxf pgsql.tar.gz' 2>/dev/null 1>/dev/null < /dev/null
2679
+ # Startup PostgreSQL server
2680
+ ssh -T $DEST $PGCTL -w -D $DESTDIR start 2>/dev/null 1>/dev/null < /dev/null &
2681
+ </pre>
2682
+
2683
+ <h3>通过rsync在线恢复</h3>
2684
+ <p>
2685
+ PostgreSQL 7.4 没有PITR ,rsync 可被用来进行在线恢复。在 pgpool-II 的压缩包中的"sample"目录中,有一个"pgpool_recovery"交恢复脚本。该脚本使用rsync命令,pgpool-II使用三个参数调用该脚本。
2686
+ </p>
2687
+
2688
+ <pre>
2689
+ % pgpool_recovery datadir remote_host remote_datadir
2690
+ datadir: Database cluster path of a master node.
2691
+ remote_host: Hostname of a recovery target node.
2692
+ remote_datadir: Database cluster path of a recovery target node.
2693
+ </pre>
2694
+
2695
+ <p>
2696
+ 该脚本通过 ssh 使用 rsync 拷贝物理文件。所以你需要提供不需要密码输入的 ssh 连接。
2697
+ </p>
2698
+
2699
+ <p>
2700
+ 关于rsync的一些笔记:
2701
+ <ul>
2702
+ <li>-c (or --checksum) 当可靠文件传输时需要设置该可选项
2703
+ <li>-z (or --compress) 该可选项用于传输数据时数据压缩。这可能对低速连接非常棒,但是对于一个 100Mbit 或更快的连接可能会增加太多的 CPU 负担,在这种情况下你可能不愿意使用该可选项。
2704
+ <li>rsync 3.0.5 有着巨大的速度性能提升(根据 pgpool-general 邮件列表里的一份报告,会快50%)
2705
+ </ul>
2706
+
2707
+ </p>
2708
+
2709
+ <p>
2710
+ 如果你使用 pgpool_recovery,在 pgpool.conf 里增加如下几行。
2711
+ <pre>
2712
+ recovery_1st_stage_command = 'pgpool_recovery'
2713
+ recovery_2nd_stage_command = 'pgpool_recovery'
2714
+ </pre>
2715
+ </p>
2716
+
2717
+ <h2>如何执行在线恢复</h2>
2718
+ <p>
2719
+ 为了执行在线恢复,使用 <a href="#PCP_RECOVERY_NODE">pcp_recovery_node</a> 命令或 pgpoolAdmin。
2720
+ </p>
2721
+
2722
+ <p>
2723
+ 注意你需要给 <a href="#pcp_recovery_node">pcp_recovery_node</a> 第一个参数传递一个大数。该数是一个以秒为单位的定时参数。如果你使用 pgpoolAdmin,在pgmgt.conf.php中设置 "_PGPOOL2_PCP_TIMEOUT "参数为一个大数。
2724
+ </p>
2725
+
2726
+ <h2 id="minorup_w_online_recovery">使用在线恢复实现 PostgreSQL 版本升级</h2>
2727
+ <h3>复制模式的情况下</h3>
2728
+ <p>
2729
+ 如果 pgpool-II 运行在复制模式,你可以不停止 pgpool-II 实现对每个 PostgreSQL节点的升级。
2730
+ 注意在断开和重连数据库节点时,从客户端到 pgpool-II 的激活的会话会被关闭。
2731
+ 还请注意这种情况下你不能用以下描述的方法做大版本升级
2732
+ (例如,需要 dump/restore 的版本升级)。
2733
+ </p>
2734
+
2735
+ <p>
2736
+ <ol>
2737
+ <li><p>
2738
+ 准备在线恢复</p>
2739
+ </li>
2740
+
2741
+ <li><p>版本升级应该先从不是主节点的节点开始。
2742
+ 在非主节点上停止 PostgreSQL。
2743
+ Pgpool-II 将探测到 PostgreSQL 停止并生成以下日志。
2744
+ 这种情况下,到 pgpool-II 的会话都将被关闭。
2745
+ </p>
2746
+ <pre>
2747
+ 2010-07-27 16:32:29 LOG: pid 10215: set 1 th backend down status
2748
+ 2010-07-27 16:32:29 LOG: pid 10215: starting degeneration. shutdown host localhost(5433)
2749
+ 2010-07-27 16:32:29 LOG: pid 10215: failover_handler: set new master node: 0
2750
+ 2010-07-27 16:32:29 LOG: pid 10215: failover done. shutdown host localhost(5433)
2751
+ </pre>
2752
+ </li>
2753
+
2754
+ <li><p>
2755
+ 在停止的节点上执行 PostgreSQL 的版本升级。
2756
+ 你可以覆盖旧的 PostgreSQL,我们建议你将旧版本的 PostgreSQL 移动到某处
2757
+ 以便在必要情况下恢复它。
2758
+ </p>
2759
+ </li>
2760
+
2761
+ <li><p>
2762
+ 如果你安装新的 PostgreSQL 在和旧版本不同的位置且你不想更新你的恢复脚本,
2763
+ 你需要使用软连接一类的工具来修正路径。
2764
+ 如果你选择覆盖,你可以跳过以下步骤,直到安装 C 函数的步骤。
2765
+ 你可以立即执行在线恢复。
2766
+ </p>
2767
+ </li>
2768
+
2769
+ <li><p>
2770
+ 改变旧的 PostgreSQL 的安装目录。
2771
+ 在以下示例中,PostgreSQL 的安装目录被假设为 /user/local/pgsql。
2772
+ </p>
2773
+ <pre>
2774
+ $ mv /usr/local/pgsql /usr/local/pgsql-old
2775
+ </pre>
2776
+ </li>
2777
+
2778
+ <li><p>
2779
+ 建立一个到新版本 PostgreSQL 安装位置的软连接。
2780
+ 这允许你使用你当前的命令搜索路径。
2781
+ 在以下示例中,新的 PostgreSQL 的安装路径被假设为 /usr/local/pgsql-new。
2782
+ </p>
2783
+ <pre>
2784
+ $ ln -s /usr/local/pgsql-new /usr/local/pgsql
2785
+ </pre>
2786
+ </li>
2787
+
2788
+ <li><p>
2789
+ 如果数据库目录位于旧的 PostgreSQL 安装目录中,
2790
+ 你应该建立或者拷贝一份因而新的 PostgreSQL 可以访问它。
2791
+ 在以下示例中我们使用软连接。
2792
+ </p>
2793
+ <pre>
2794
+ $ ln -s /usr/local/pgsql-old/data /usr/local/pgsql/data
2795
+ </pre>
2796
+ </li>
2797
+
2798
+ <li><p>
2799
+ 安装 C 函数到 PostgreSQL 中。参考“安装 C 函数”小节。
2800
+ 因为在线恢复复制了数据库集群,最后一步使用 psql 安装函数不是必要的。
2801
+ 只需要 make install。
2802
+ </p>
2803
+ </li>
2804
+
2805
+ <li><p>
2806
+ 执行在线恢复。这样,你完成了一个节点的版本升级。
2807
+ 要只需在线恢复,你需要使用 <a href="#pcp_recovery_node">pcp_recovery_node</a> 或 pgpoolAdmin。
2808
+ </p>
2809
+ </li>
2810
+
2811
+ <li><p>
2812
+ 针对每个节点重复以上步骤。到最后一步才是升级主节点。然后你就完成了。
2813
+ </p>
2814
+ </li>
2815
+ </ol>
2816
+
2817
+ <h3>如果你使用流复制</h3>
2818
+ <p>
2819
+ 可以升级备节点而不停止 pgpool-II。
2820
+ </p>
2821
+
2822
+ <p>
2823
+ 升级 PostgreSQL 备节点的步骤和以上复制模式的步骤一样。
2824
+ 请参考“流复制模式中的在线恢复”设置 recovery_1st_stage_command 和 recovery_2nd_stage_command。
2825
+ </p>
2826
+
2827
+ <p>
2828
+ 无法实现不停止 pgpool-II 升级主节点。
2829
+ 你需要在升级主节点时停止 pgpool-II。
2830
+ 升级主 PostgreSQL 服务器的过程和和备节点相同。
2831
+ 升级主 PostgreSQL 服务器的过程如下:
2832
+ </p>
2833
+ <ol>
2834
+ <li>停止 pgpool-II</li>
2835
+ <li>停止主 PostgreSQL</li>
2836
+ <li>升级主 PostgreSQL</li>
2837
+ <li>启动主 PostgreSQL</li>
2838
+ <li>启动 pgpool-II</li>
2839
+ </ol>
2840
+
2841
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
2842
+
2843
+ <!-- ================================================================================ -->
2844
+
2845
+ <h1><a name="backup"></a>备份</h1>
2846
+ <p>
2847
+ 要备份后端的 PostgreSQL 服务器和系统数据库,和备份 PostgreSQL一样,
2848
+ 你可以使用物理备份,逻辑备份(pg_dump,pg_dumpall)和 PITR。
2849
+ 注意使用逻辑备份和 PITR 应该直接在 PostgreSQL 上执行,
2850
+ 而不是通过 pgpool-II,这样才能避免由于 <a href="#LOAD_BALANCE_MODE">load_balance_mode</a> 和
2851
+ <a href="#REPLICATE_SELECT">replicate_select</a> 引起的错误。
2852
+ </p>
2853
+
2854
+ <h2 id="backup_rep_or_ms_mode">复制模式和主/备模式</h2>
2855
+ <p>
2856
+ 如果 pgpool-II 运行在复制模式或主/备模式,只要备份集群中的一个数据库节点。
2857
+ </p>
2858
+
2859
+ <p>
2860
+ 如果你使用的是主/备模式并且使用了异步复制系统(Slony-I和流复制)且想要最新的备份,
2861
+ 你应该在主节点做备份。
2862
+ </p>
2863
+
2864
+ <p>
2865
+ pg_dump 在数据库中使用共享锁(ACCESS SHARE lock)。使用排他锁(ACCESS EXECUTE lock)
2866
+ 的命令例如 ALTER TABLE, DROP TABLE, TRUNCATE, REINDEX,CLUSTER 和
2867
+ VACUUM FULL 将等待 pg_dump 的完成,因为存在锁冲突。这甚至会影响主节点,
2868
+ 即使你在备节点执行 pg_dump。
2869
+ </p>
2870
+
2871
+ <h2 id="backup_parallel">并行模式</h2>
2872
+ <p>
2873
+ 如果你在使用并行模式且希望得到连续的备份,你需要停止 pgpool-II。
2874
+ </p>
2875
+
2876
+ <p>
2877
+ 要使用逻辑备份,停止应用程序和 pgpool-II,然后在所有节点上执行 pg_dump 或 pg_dumpall。
2878
+ 在完成备份后,启动 pgpool-II,然后启动应用程序。
2879
+ </p>
2880
+
2881
+ <p>
2882
+ 如果使用 PITR,请确保所有节点的系统时间一致。
2883
+ 准备归档日志并执行基础备份。
2884
+ 完成备份后,停止和重启应用程序和 pgpool-II。
2885
+ 记录停止和启动的时间点。
2886
+ 这种临时停止将使各集群之间保持一致的状态。
2887
+ 如果你需要从基础备份和归档日志中恢复,
2888
+ 在启停之间设置 recovery.conf 的 recovery_target_time。
2889
+ </p>
2890
+
2891
+ <h2 id="backup_system_db">备份系统数据库</h2>
2892
+ <p>
2893
+ 如果 pgpool-II 运行在并行查询模式或者你在使用查询缓存,你需要备份 pgpool-II 的系统数据库。
2894
+ 只需要备份 pgpool.conf 中 system_db_dbname 指定的数据库。
2895
+ </p>
2896
+
2897
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
2898
+
2899
+ <!-- ================================================================================ -->
2900
+
2901
+ <h1><a name="deploy"></a>部署 pgpool-II</a></h1>
2902
+ <p>
2903
+ pgpool-II 可以运行在专用的服务器上,应用服务运行的服务器上或者其他服务器上。
2904
+ 本章我们将讲述如何实现这些部署以及相应的利弊。
2905
+ </p>
2906
+
2907
+ <p>
2908
+ <dl>
2909
+ <dt>专用服务器</dt>
2910
+ <dd>
2911
+ <p>
2912
+ pgpool-II 运行在专用服务器上。结构简单且 pgpool-II 不受其他服务软件的影响。
2913
+ 很明显,缺点是你需要买更多硬件。
2914
+ 而且这种配置下,pgpool-II 可能发生单点故障
2915
+ (你可以使用以下提到的 pgpool-HA 避免这个问题)。
2916
+ </p>
2917
+ </dd>
2918
+
2919
+ <dt>部署在网站服务器或应用服务器上</dt>
2920
+ <dd>
2921
+ <p>
2922
+ 将 pgpool-II 部署到一台运行 Apache, JBoss, Tomcat 或其他网站服务器和应用服务器上。
2923
+ 由于 pgpool-II 和网站服务器或者应用服务器之间的通信是本地的,网络通信速度比服务器间的通信更快。
2924
+ 而且如果你使用多个网站服务器或者应用服务器,你可以避免单点故障
2925
+ (这种情况下,你必须为每个 pgpool-II 实例配置相同的 pgpool.conf)。
2926
+ 这种配置情况下,你必须注意几项事项:
2927
+ </p>
2928
+ <ul>
2929
+ <li>如果 pgpool-II 和数据库服务器之间的通信部稳定,
2930
+ 有可能数据库节点 #1 到一个 pgpool-II 实例的连接断开
2931
+ 但到另一个 pgpool-II 实例的连接是正常的。
2932
+ 要避免这种问题,你可以使用双网络链路。
2933
+ </li>
2934
+ <li>当在复制模式中执行在线恢复时,你需要停止除进行在线恢复外所有的 pgpool-II 实例。
2935
+ 否则,数据库会运行到不一致的状态。
2936
+ 在主备模式和流复制模式中,你不需要停止其他的 pgpool-II 实例。
2937
+ 然而,你不允许在多个 pgpool-II 实例运行的时候执行在线恢复。
2938
+ </li>
2939
+ </ul>
2940
+ </dd>
2941
+
2942
+ <dt>在数据库服务器上运行 pgpool-II</dt>
2943
+ <dd>
2944
+ <p>
2945
+ 在运行 PostgreSQL 的服务器上运行 pgpool-II。
2946
+ 这种配置下避免 pgpool-II 的单点故障的问题。
2947
+ 很明显你不需要另外购买专用服务器。
2948
+ 这种配置下的问题是,应用程序需要选择连接到哪台数据库服务器。
2949
+ 要解决这个问题你可以通过 pgpool-HA 使用虚拟 IP。
2950
+ </p>
2951
+ </dd>
2952
+ </dl>
2953
+
2954
+ <h2 id="pgpool_ha">关于 pgpool-HA</h2>
2955
+ <p>
2956
+ Pgpool-HA 是一个高可用性软件来让 pgpool-II 配合 hearbeat使用。
2957
+ Pgpool-HA 是一个 pgpool 以及 pgpool-II 项目的子项目。
2958
+ Pgpool-HA 可以在 pgpool 开发网站找到,它是一个开源软件。</p>
2959
+
2960
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
2961
+
2962
+ <!-- ================================================================================ -->
2963
+
2964
+ <h1 id="watchdog">看门狗<span class="version">V3.2 -</span></h1>
2965
+
2966
+ <p>
2967
+ “看门狗”是一个 pgpool-II 的子进程,用于添加高可用性功能。
2968
+ 这通过多个 pgpool-II 的合作解决了单点故障的问题。
2969
+ 看门狗为 pgpool-II 添加了以下功能:
2970
+ </p>
2971
+ <h3>pgpool-II 服务的生命检测</h3>
2972
+ <p>
2973
+ 看门狗通过两种方式监控 pgpool-II,“心跳”模式或者“查询”模式。
2974
+ </p>
2975
+ <ul>
2976
+ <li>
2977
+ 在心跳模式中,看门狗使用心跳信号监控其他 pgpool-II 进程。看门狗接收其他 pgpool-II 定期发送的心跳信号。
2978
+ 如果时间周期到了还没收到信号,则看门狗认为 pgpool-II 已经失效。
2979
+ 为了冗余,你可以使用多个网络接口设备来在 pgpool-II 之间实现心跳交换。这是默认的模式,且推荐使用。
2980
+ </li>
2981
+ <li>
2982
+ 在查询模式中,看门狗监控 pgpool-II 提供的服务,而不是进程。看门狗向其他 pgpool-II 发送查询并检查反馈。
2983
+ 注意这种模式需要连接到其他 pgpool-II,所以如果 <a href="#NUM_INIT_CHILDREN">num_init_children</a> 不够大则可能监控失败。
2984
+ 本模式不赞成使用,保留的原因是为了向下兼容。
2985
+ </li>
2986
+ </ul>
2987
+ <p>
2988
+ 看门狗还监控到从 pgpool-II 到前端服务器的连接(例如应用服务器),并检查 pgpool-II 能否为这些服务提供服务。如果监控到失败,则认为 pgpool-II 宕机了。
2989
+ </p>
2990
+
2991
+ <h3>协调多个 pgpool-II 共同工作</h3>
2992
+ <p>
2993
+ 看门狗通过互相交换信息来协调多个 pgpool-II 共同工作。
2994
+ </p>
2995
+ <ul>
2996
+ <li>
2997
+ 在后端节点例如由于故障切换而发生状态变化后,看门狗通知其他 pgpool-II 节点并同步信息。
2998
+ 在发生在线恢复时,看门狗限制客户端连接到其他 pgpool-II 节点,以避免后端节点的不一致。
2999
+ </li>
3000
+ <li>
3001
+ 故障切换或者故障恢复的命令 (<a href="#FAILOVER_COMMAND">failover_command</a>,
3002
+ <a href="#FAILBACK_COMMAND">failback_command</a>, <a href="#FOLLOW_MASTER_COMMAND">follow_master_command</a>)
3003
+ 通过内部锁机制只被一个 pgpool-II 节点执行。
3004
+ </li>
3005
+ </ul>
3006
+
3007
+ <h3>在检测到某些故障时交换活跃/备用状态</h3>
3008
+ <p>
3009
+ 当一个 pgpool-II 的故障被检测到,看门狗通知其他的看门狗这个消息。
3010
+ 如果是活跃的 pgpool-II 发生故障,看门狗通过投票确定新的活跃 pgpool-II 并更新活跃/备用状态。
3011
+ </p>
3012
+
3013
+ <h3>在服务器切换的时候实现自动虚拟 IP 地址分配</h3>
3014
+ <p>
3015
+ 当一个备用 pgpool 服务器提升为活跃的,新的活跃服务器启动虚拟 IP 接口。
3016
+ 也就是,之前的活跃服务器停用虚拟 IP 接口。
3017
+ 这确保活动的 pgpool 使用相同的 IP 地址,即使在发生服务器切换的时候。
3018
+ </p>
3019
+
3020
+ <h3>在恢复的时候自动注册服务器为备用服务器</h3>
3021
+ <p>
3022
+ 当失效的服务器恢复或者新的服务器连接上来,看门狗进程通知其他的看门狗进程关于新服务器的信息,
3023
+ 看门狗进程在活跃服务器和其他服务器上接收这些信息。
3024
+ 然后,新连接上的服务器注册为备用节点。
3025
+ </p>
3026
+
3027
+ <h2 id="deploy_watchdog">服务器组件</h2>
3028
+
3029
+ <p>
3030
+ 下图描述 pgpool-II 和看门狗进程是如何配置的。
3031
+ </p>
3032
+ <p><img src="wd-en.jpg" alt="看门狗服务组件" hight=70% width=70%></p>
3033
+
3034
+ <h2 id="start_watchdog">启动/停止看门狗</h2>
3035
+ <p>
3036
+ 看门狗进程由 pgpool-II 自动启动/停止,也就是说,没有单独的命令来启动/停止它。
3037
+ </p>
3038
+ <p>
3039
+ 看门狗功能启动时必须拥有<font color="red">管理员权限(root)</font> 来控制虚拟 IP 接口。
3040
+ 方法之一是使用 root 权限启动 pgpool-II。
3041
+ 不过,考虑安全原因,推荐通过设置设置自定义命令 <a href="#IF_UP_CMD">if_up_cmd</a>、
3042
+ <a href="#IF_DOWN_CMD">if_up_cmd</a> 和 <a href="#ARPING_CMD">if_up_cmd</a> 为使用 sudo 或者 setuid 的方法。
3043
+ </p>
3044
+ <p>
3045
+ 在等待到所有的 pgpool 启动后,生命监测将启动。
3046
+ </p>
3047
+
3048
+ <h2>配置看门狗 (pgpool.conf)</h2>
3049
+ <p>
3050
+ 看门狗的配置参数在 pgpool.conf 中配置。
3051
+ 在 pgpool.conf.sample 文件中的 WATCHDOG 小节是配置看门狗的示例。
3052
+ </p>
3053
+
3054
+ <p>
3055
+ 以下所有的选项都是使用看门狗进程时必须指定的。
3056
+ </p>
3057
+ <h3>启用</h3>
3058
+ <dl>
3059
+ <dt><a name="USE_WATCHDOG"></a>use_watchdog <span class="version">V3.2 -</span></dt>
3060
+ <dd>
3061
+ <p>
3062
+ 如果为 on,则激活看门狗。默认为 off 。
3063
+ </p>
3064
+ <p>
3065
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3066
+ </p>
3067
+ </dd>
3068
+ </dl>
3069
+
3070
+ <h3>Watchdog communication</h3>
3071
+ <dl>
3072
+ <dt><a name="WD_HOSTNAME"></a>wd_hostname <span class="version">V3.2 -</span></dt>
3073
+ <dd>
3074
+ <p>
3075
+ 指定 pgpool-II 的主机名或者 IP 地址。这用于发送/接收查询和通讯包,也用于标记看门狗。
3076
+ </p>
3077
+ <p>
3078
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3079
+ </p>
3080
+ </dd>
3081
+
3082
+ <dt><a name="WD_PORT"></a>wd_port <span class="version">V3.2 -</span></dt>
3083
+ <dd>
3084
+ <p>
3085
+ 指定看门狗的通信端口。
3086
+ </p>
3087
+ <p>
3088
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3089
+ </p>
3090
+ </dd>
3091
+
3092
+ <dt><a name="WD_AUTHKEY"></a>wd_authkey <span class="version">V3.3 -</span></dt>
3093
+ <dd>
3094
+ <p>
3095
+ 本选项指定用于看门狗通信的认证密钥。所有的 pgpool-II 必须有相同的密钥。
3096
+ 从使用不同密钥的看门狗发送过来的包将被拒绝。
3097
+ 本密钥也被用于生命检测被设置为心跳模式时的心跳信号。如果本值为空(默认)则看门狗不使用认证。
3098
+ </p>
3099
+ <p>
3100
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3101
+ </p>
3102
+ </dd>
3103
+ </dl>
3104
+
3105
+ <h3>连接到上游服务器</h3>
3106
+
3107
+ <dl>
3108
+ <dt><a name="TRUSTED_SERVERS"></a>trusted_servers <span class="version">V3.2 -</span></dt>
3109
+ <dd>
3110
+ <p>
3111
+ 用于检测上游连接的可信服务器列表。每台服务器都应能响应 ping。
3112
+ 指定一个用逗号分隔的服务器列表例如 "hostA,hostB,hostC"。
3113
+ 如果没有任何服务器可以 ping 通,则看门狗认为 pgpool-II 出故障了。
3114
+ </p>
3115
+ <p>
3116
+ 如果本选项为空,则看门狗不检查上游连接。
3117
+ </p>
3118
+ <p>
3119
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3120
+ </p>
3121
+ </dd>
3122
+
3123
+ <dt><a name="PING_PATH"></a>ping_path <span class="version">V3.2 -</span></dt>
3124
+ <dd>
3125
+ <p>
3126
+ 本参数指定用于监控上游服务器的 ping 命令的路径。只需要设置路径例如 "/bin" 。
3127
+ </p>
3128
+ <p>
3129
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3130
+ </p>
3131
+ </dd>
3132
+ </dl>
3133
+
3134
+ <h3>虚拟 IP 控制</h3>
3135
+
3136
+ <p>
3137
+ 配置虚拟 IP 接口相关控制
3138
+ </p>
3139
+
3140
+ <dl>
3141
+ <dt><a name="DELEGATE_IP"></a>delegate_IP <span class="version">V3.2 -</span></dt>
3142
+ <dd>
3143
+ <p>
3144
+ 指定客户端的服务(例如应用服务等)连接到的 pgpool-II 的虚拟 IP (VIP) 地址。
3145
+ 当一个 pgpool-II 从备节点切换为激活节点时,pgpool-II 接管这个 VIP。
3146
+ </p>
3147
+ <p>
3148
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3149
+ </p>
3150
+ </dd>
3151
+
3152
+ <dt><a name="IFCONFIG_PATH"></a>ifconfig_path <span class="version">V3.2 -</span></dt>
3153
+ <dd>
3154
+ <p>
3155
+ 本参数指定用于切换 IP 地址的命令的所在路径。
3156
+ 只设置路径,例如 "/sbin"。
3157
+ </p>
3158
+ <p>
3159
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3160
+ </p>
3161
+ </dd>
3162
+
3163
+ <dt><a name="IF_UP_CMD"></a>if_up_cmd <span class="version">V3.2 -</span></dt>
3164
+ <dd>
3165
+ <p>
3166
+ 本参数指定一个命令用以启用虚拟 IP。设置命令和参数,例如 "ifconfig eth0:0 inet $_IP_$ netmask 255.255.255.0"。
3167
+ 参数 $_IP_$ 会被 <a href="#DELEGATE_IP">delegate_IP</a> 设置的值替换。
3168
+ </p>
3169
+ <p>
3170
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3171
+ </p>
3172
+ </dd>
3173
+
3174
+ <dt><a name="IF_DOWN_CMD"></a>if_down_cmd <span class="version">V3.2 -</span></dt>
3175
+ <dd>
3176
+ <p>
3177
+ 本参数指定一个命令用以停用虚拟 IP。设置命令和参数,例如 "ifconfig eth0:0 down"。
3178
+ </p>
3179
+ <p>
3180
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3181
+ </p>
3182
+ </dd>
3183
+
3184
+ <dt><a name="ARPING_PATH"></a>arping_path <span class="version">V3.2 -</span></dt>
3185
+ <dd>
3186
+ <p>
3187
+ 本参数指定在发生虚拟 IP 切换后用于发送一个 ARP 请求的命令的所在路径。只设置路径,例如 "/sbin"。
3188
+ </p>
3189
+ <p>
3190
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3191
+ </p>
3192
+ </dd>
3193
+
3194
+ <dt><a name="ARPING_CMD"></a>arping_cmd <span class="version">V3.2 -</span></dt>
3195
+ <dd>
3196
+ <p>
3197
+ 本参数指定一个命令用以在发生虚拟 IP 切换后用于发送一个 ARP 请求的命令。
3198
+ 设置命令和参数,例如 "arping -U $_IP_$ -w 1"。
3199
+ 参数 $_IP_$ 会被 <a href="#DELEGATE_IP">delegate_IP</a> 设置的值替换。
3200
+ </p>
3201
+ <p>
3202
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3203
+ </p>
3204
+ </dd>
3205
+ </dl>
3206
+
3207
+ <h3>角色升级时的行为</h3>
3208
+
3209
+ <p>配置当 pgpool-II 角色升级为活跃(持有虚拟 IP )时的行为
3210
+
3211
+ <dl>
3212
+ <dt><a name="CLEAR_MEMQCACHE_ON_ESCALATION"></a>clear_memqcache_on_escalation <span class="version">V3.3 -</span></dt>
3213
+ <dd>
3214
+ <p>
3215
+ 如果本值为 on,pgpool-II 在升级为活跃状态时清空所有的共享内存中的查询缓存。
3216
+ 这会避免新激活的 pgpool-II 使用旧的 pgpool-II 而导致的不一致情况。默认值为 on。
3217
+ </p>
3218
+ <p>
3219
+ 本选项只有在 <a href="#MEMQCACHE_METHOD">memqcache_method</a> 为 'shmem' 时生效。
3220
+ </p>
3221
+ </dd>
3222
+
3223
+ <dt><a name="WD_ESCALATION_COMMAND"></a>wd_escalation_command <span class="version">V3.3 -</span></dt>
3224
+ <dd>
3225
+ <p>
3226
+ 在 pgpool-II 在升级为活跃状态时看门狗执行本命令。执行时间在虚拟 IP 启用之前。
3227
+ </p>
3228
+ </dd>
3229
+ </dl>
3230
+
3231
+ <h3>pgpool-II 存活情况查询</h3>
3232
+
3233
+ <p>看门狗定期检查 pgpool-II 的状态。这被称为“存活检查”。
3234
+
3235
+ <h4>通用</h4>
3236
+
3237
+ <dl>
3238
+ <dt><a name="WD_LIFECHECK_METHOD"></a>wd_lifecheck_method <span class="version">V3.3 -</span></dt>
3239
+ <dd>
3240
+ <p>
3241
+ 本参数指定存活检查的模式。可以是 'heartbeat' (默认) 或者 'query'。
3242
+ </p>
3243
+ <p>
3244
+ 在 'heartbeat' 模式中,看门狗定期发送心跳信号(UDP 包)到其他的 pgpool-II。看门狗也从其他 pgpool-II 获取信号。
3245
+ 如果在一段时间内没有收到信号,看门狗则认为那个 pgpool-II 发生了故障。
3246
+ 考虑冗余,你可以使用多个网络接口设备进行 pgpool-II 之间的心跳交换。
3247
+ 本模式是默认和推荐使用的。
3248
+ </p>
3249
+ <p>
3250
+ 在 'query' 模式中,看门狗发送监控查询到其他的 pgpool-II 并检查反馈情况。
3251
+ </p>
3252
+ <p>
3253
+ <font color="red">警告:如果你打算以查询模式使用看门狗,你需要设置 <a href="#NUM_INIT_CHILDREN">num_init_children</a>
3254
+ 到足够大的值。这是因为看门狗进程是以 pgpool-II 客户端的方式连接。</font>
3255
+ </p>
3256
+ <p>
3257
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3258
+ </p>
3259
+ </dd>
3260
+
3261
+ <dt><a name="WD_INTERVAL"></a>wd_interval <span class="version">V3.2 -</span></dt>
3262
+ <dd>
3263
+ <p>
3264
+ 本参数指定 pgpool-II 进行存活检查的间隔,单位为秒(大于或等于 1)。默认为 10.
3265
+ </p>
3266
+ <p>
3267
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3268
+ </p>
3269
+ </dd>
3270
+ </dl>
3271
+
3272
+ <h4>心跳模式的配置</h4>
3273
+
3274
+ <dl>
3275
+ <dt><a name="WD_HEARTBEAT_PORT"></a>wd_heartbeat_port <span class="version">V3.3 -</span></dt>
3276
+ <dd>
3277
+ <p>
3278
+ 本选项指定接收心跳信号的端口号。本选项仅用于心跳模式。
3279
+ </p>
3280
+ <p>
3281
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3282
+ </p>
3283
+ </dd>
3284
+
3285
+ <dt><a name="WD_HEARTBEAT_KEEPALIVE"></a>wd_heartbeat_keepalive <span class="version">V3.3 -</span></dt>
3286
+ <dd>
3287
+ <p>
3288
+ 本选项指定发送心跳信号的间隔(秒)。默认值为 2。本选项仅用于心跳模式。
3289
+ </p>
3290
+ <p>
3291
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3292
+ </p>
3293
+ </dd>
3294
+
3295
+ <dt><a name="WD_HEARTBEAT_DEADTIME"></a>wd_heartbeat_deadtime <span class="version">V3.3 -</span></dt>
3296
+ <dd>
3297
+ <p>
3298
+ 如果本选项指定的时间周期内没有收到心跳信号,则看门狗认为远端的 pgpool-II 发生故障。本选项仅用于心跳模式。
3299
+ </p>
3300
+ <p>
3301
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3302
+ </p>
3303
+ </dd>
3304
+ <dt><a name="HEARTBEAT_DESTINATION"></a>heartbeat_destination<font color="red">0</font> <span class="version">V3.3 -</span></dt>
3305
+ <dd>
3306
+ <p>
3307
+ 本选项指定心跳信号发送的目标,可以是 IP 地址或主机名。你可以指定多个目标。
3308
+ 本参数名后面的数字表示“目标序号”,从 0 开始。本参数仅用于心跳模式。
3309
+ </p>
3310
+ <p>
3311
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3312
+ </p>
3313
+ </dd>
3314
+ <dt><a name="HEARTBEAT_DESTINATION_PORT"></a>heartbeat_destination_port<font color="red">0</font> <span class="version">V3.3 -</span></dt>
3315
+ <dd>
3316
+ <p>
3317
+ 本选项指定由 <a href="HEARTBEAT_DESTINATION">heartbeat_destinationX</a> 指定的心跳信号目标的端口号。
3318
+ 这个值通常等于 <a href="WD_HEARTBEAT_PORT">wd_heartbeat_port</a>。
3319
+ 但如果这个端口号在某些主机上不能使用或者一个主机上有多个 pgpool-II 的时候你必须指定另一个值。
3320
+ 本参数名后面的数字表示“目标序号”,从 0 开始。本参数仅用于心跳模式。
3321
+ </p>
3322
+ <p>
3323
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3324
+ </p>
3325
+ </dd>
3326
+ <dt><a name="HEARTBEAT_DEVICE"></a>heartbeat_device<font color="red">0</font> <span class="version">V3.3 -</span></dt>
3327
+ <dd>
3328
+ <p>
3329
+ 本选项指定用于发送心跳信号到由 <a href="HEARTBEAT_DESTINATION">heartbeat_destinationX</a>指定的目标的设备名。
3330
+ 你可以向不同的目标使用相同的设备。
3331
+ 本参数名后面的数字表示“设备序号”,从 0 开始。
3332
+ 本选项仅用于心跳模式。另外,这只有在运行在 Linux 下的 root 权限中的 pgpool-II 才能生效,因为这使用了 SO_BINDTODEVICE 套接字选项。
3333
+ </p>
3334
+ <p>
3335
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3336
+ </p>
3337
+ </dd>
3338
+ </dl>
3339
+
3340
+ <h4>查询模式的配置</h4>
3341
+
3342
+ <dl>
3343
+ <dt><a name="WD_LIFE_POINT"></a>wd_life_point <span class="version">V3.2 -</span></dt>
3344
+ <dd>
3345
+ <p>
3346
+ 在确认 pgpool-II 失效时的重试次数(一个大于或等于 1 的数字)。默认为 3。本选项仅用于查询模式。
3347
+ </p>
3348
+ <p>
3349
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3350
+ </p>
3351
+ </dd>
3352
+
3353
+ <dt><a name="WD_LIFECHECK_QUERY"></a>wd_lifecheck_query <span class="version">V3.2 -</span></dt>
3354
+ <dd>
3355
+ <p>
3356
+ 用于检查 pgpool-II 的查询。默认为 "SELECT 1"。本选项仅用于查询模式。
3357
+ </p>
3358
+ <p>
3359
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3360
+ </p>
3361
+ </dd>
3362
+
3363
+ <dt><a name="WD_LIFECHECK_DBNAME"></a>wd_lifecheck_dbname <span class="version">V3.3 -</span></dt>
3364
+ <dd>
3365
+ <p>
3366
+ 用于检查 pgpool-II 时连接到的数据库名。默认为 "template1"。本选项仅用于查询模式。
3367
+ </p>
3368
+ </dd>
3369
+
3370
+ <dt><a name="WD_LIFECHECK_USER"></a>wd_lifecheck_user <span class="version">V3.3 -</span></dt>
3371
+ <dd>
3372
+ <p>
3373
+ 用于检查 pgpool-II 的用户名。用户名必须存在于后端的 PostgreSQL 中。默认为 "nobody"。
3374
+ 本选项仅用于查询模式。
3375
+ </p>
3376
+ </dd>
3377
+
3378
+ <dt><a name="WD_LIFECHECK_PASSWORD"></a>wd_lifecheck_password <span class="version">V3.3 -</span></dt>
3379
+ <dd>
3380
+ <p>
3381
+ 用于检查 pgpool-II 的密码。默认为 ""。
3382
+ 本选项仅用于查询模式。
3383
+ </p>
3384
+ </dd>
3385
+ </dl>
3386
+
3387
+ <h3>服务监控</h3>
3388
+ <dl>
3389
+ <dt><a name="OTHER_PGPOOL_HOSTNAME"></a>other_pgpool_hostname<font color="red">0</font>
3390
+ <span class="version">V3.2 -</span></dt>
3391
+ <dd>
3392
+ <p>
3393
+ 指定需要监控的 pgpool-II 服务器主机。这用于发送/接收查询和数据包,同时也是看门狗的标识。
3394
+ 本参数名后面的数字表示“服务序号”,从 0 开始。
3395
+ </p>
3396
+ <p>
3397
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3398
+ </p>
3399
+ </dd>
3400
+
3401
+ <dt><a name="OTHER_PGPOOL_PORT"></a>other_pgpool_port<font color="red">0</font>
3402
+ <span class="version">V3.2 -</span></dt>
3403
+ <dd>
3404
+ <p>
3405
+ 指定需要监控的 pgpool-II 服务器的 pgpool 服务的端口。
3406
+ 对于查询模式,<a href="#WD_LIFECHECK_QUERY">wd_lifecheck_query</a> 指定的查询语句将被发送到这个端口。
3407
+ 本参数名后面的数字表示“服务序号”,从 0 开始。
3408
+ </p>
3409
+ <p>
3410
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3411
+ </p>
3412
+ </dd>
3413
+
3414
+ <dt><a name="OTHER_WD_PORT"></a>other_wd_port<font color="red">0</font> <span class="version">V3.2 -</span></dt>
3415
+ <dd>
3416
+ <p>
3417
+ 指定需要监控的 pgpool-II 服务器的看门狗的端口。
3418
+ 本参数名后面的数字表示“服务序号”,从 0 开始。
3419
+ </p>
3420
+ <p>
3421
+ 修改本值后,你需要重启 pgpool-II 以使改动生效。
3422
+ </p>
3423
+ </dd>
3424
+ </dl>
3425
+
3426
+ <h2 id="watchdog_restrictions">看门狗的限制</h2>
3427
+ <ul>
3428
+ <li>
3429
+ 在查询模式中,如果所有的数据库节点由于 PostgreSQL 服务出现故障或者由于调用了 pcp_detach_node,
3430
+ 看门狗会认为 pgpool-II 服务为故障状态并会剥夺虚拟 IP。这样,pgpool-II 的客户端就再也无法通过
3431
+ 虚拟 IP 连接。这对于避免<i>脑裂</i>很重要,也就是说,这种情况下可能出先多个激活的 pgpool-II。
3432
+ </li>
3433
+ <li>
3434
+ 不要通过真实 IP 连接到状态为下线的 pgpool-II。因为状态为下线的 pgpool-II 无法从看门狗获取信息,
3435
+ 因此它的后端(数据库)信息可能与其他 pgpool-II 不同。
3436
+ </li>
3437
+ <li>
3438
+ 为下线状态的 pgpool-I 不能变为激活或者备用的 pgpool-II。从下线状态恢复需要重启 pgpool-II。
3439
+ </li>
3440
+ <li>
3441
+ 注意激活状态的 pgpool-II 停机,需要好几秒钟才能等到备用 pgpool-II 提升为激活状态,以确保之前的
3442
+ 虚拟 IP 下线前的下线通知包被送到其他的 pgpool-II 节点。
3443
+ </li>
3444
+ </ul>
3445
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
3446
+
3447
+ <!-- ================================================================================ -->
3448
+
3449
+ <h1><a name="troubleshooting"></a>故障解决</h1>
3450
+ <p>
3451
+ 本节描述你使用pgpool-II时碰到的问题及其解决办法。
3452
+ </p>
3453
+ <p>
3454
+ <dl>
3455
+
3456
+ <dt id="health_check_failed">健康检查失败</dt>
3457
+ <dd>
3458
+ <p>
3459
+ pgpool-II 的健康检查特性检测到DB节点失效。
3460
+ </p>
3461
+
3462
+ <pre>
3463
+ 2010-07-23 16:42:57 ERROR: pid 20031: health check failed. 1 th host foo at port 5432 is down
3464
+ 2010-07-23 16:42:57 LOG: pid 20031: set 1 th backend down status
3465
+ 2010-07-23 16:42:57 LOG: pid 20031: starting degeneration. shutdown host foo(5432)
3466
+ 2010-07-23 16:42:58 LOG: pid 20031: failover_handler: set new master node: 0
3467
+ 2010-07-23 16:42:58 LOG: pid 20031: failover done. shutdown host foo(5432)
3468
+ </pre>
3469
+
3470
+ <p>
3471
+ 该日志表明DB节点1 (主机foo) 与pgpool中断并失去连接(关闭),于是DB节点0成为新的master。检查DB节点1,处理掉产生故障的原因,之后如果可能则会在DB节点1上执行一个在线恢复。
3472
+ </p>
3473
+ </dd>
3474
+
3475
+ <dt>Failed to read kind from frontend
3476
+ <dd>
3477
+ <p>
3478
+ <pre>
3479
+ 2010-07-26 18:43:24 LOG: pid 24161: ProcessFrontendResponse: failed to read kind from frontend. frontend abnormally exited
3480
+ </pre>
3481
+ <p>
3482
+ 该日志表明前端程序未能与 pgpool-II 正确的断开。可能的原因是:客户端应的 bug,客户端应用的强制结束(kill)或者是临时的网络故障。该类型的事件不会导致一个DB的破坏或者数据一致性问题。其仅仅是一个关于违反协议的的警告。如果该消息一直出现,建议你检查应用和网络。
3483
+ </p>
3484
+
3485
+ <dt>Kind mismatch errors
3486
+ <dd>
3487
+ <p>
3488
+ pgpool-II操作在复制模式时你可能得到该消息。
3489
+ </p>
3490
+ <pre>
3491
+ 2010-07-22 14:18:32 ERROR: pid 9966: kind mismatch among backends. Possible last query was: "FETCH ALL FROM c;" kind details are: 0[T] 1[E: cursor "c" does not exist]
3492
+ </pre>
3493
+ <p>
3494
+ pgpool-II把 SQL 命令发送给数据库节点后便会等待响应。该错误消息指示并非所有的数据库节点返回同样的响应。你可能会在"Possible last query
3495
+ was:"之后看到引起错误的 SQL 语句,接着是响应类型。如果响应指示是一个错误,则从 PostgreSQL 返回的错误消息将被显示出来。这里你看到"0[T]" 显示DB节点响应:0[T]"(开始发送行描述),以及"1[E"
3496
+ 表示DB节点1返回一个错误,消息是"cursor "c" does not exist",然而数据库节点 0 发送了一个行记录。
3497
+ <p>
3498
+ 注意:在主/备模式时你同样会看到这类错误。例如即使在主/备模式下,SET 命令将会被发送给所有的数据库节点以保持所有的节点处于同一个状态。
3499
+ </p>
3500
+ <p>
3501
+ 如果你发现数据库不再同步,检查数据库并使用在线复制异步它们。
3502
+ </p>
3503
+
3504
+ <p>
3505
+ <dt>pgpool-II 检查插入、更新或者删除的不同的行数
3506
+ <dd>
3507
+ <p>
3508
+ 在复制模式下,pgpool-II 检测到 INSERT/UPDATE/DELETE 在影响的节点上产生不同的行数。
3509
+ </p>
3510
+ <p>
3511
+ <pre>
3512
+ 2010-07-22 11:49:28 ERROR: pid 30710: pgpool detected difference of the number of inserted, updated or deleted tuples. Possible last query was: "update t1 set i = 1;"
3513
+ 2010-07-22 11:49:28 LOG: pid 30710: ReadyForQuery: Degenerate backends: 1
3514
+ 2010-07-22 11:49:28 LOG: pid 30710: ReadyForQuery: Affected tuples are: 0 1
3515
+ </pre>
3516
+ <p>
3517
+ 在上面的示例中,在不同的数据库节点上执行"update t1 set i = 1"返回更新的行数不同。下一行表明 DB 1 接着出现退化 (断开,disconnected),
3518
+ 然后是DB节点0受影响的行数是0,然而节点1受影响的行数是1.
3519
+ </p>
3520
+ <p>
3521
+ 停止怀疑有错误数据的数据库节点并执行在线恢复。
3522
+ </p>
3523
+ </dl>
3524
+
3525
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
3526
+
3527
+
3528
+ <!-- ================================================================================ -->
3529
+
3530
+ <h1>限制<a name="restriction"></a></h1>
3531
+
3532
+ <p>
3533
+ <h2>PostgreSQL的函数</h2>
3534
+ <p>
3535
+ <ul>
3536
+ <li>
3537
+ 如果你使用 pg_terminate_backend() 终止后台数据库,这会触发一个故障。原因是 PostgreSQL 使用完全相同的消息来关闭 postmaster 以终止一个后台数据库,目前还没有办法解决它,请不要使用该函数。
3538
+ </ul>
3539
+
3540
+ <p>
3541
+ <h2><a name="md5"></a>认证/访问控制</h2>
3542
+
3543
+ <p>
3544
+ <ul>
3545
+ <li>在复制模式或主/备模式中,trust, clear text password, 以及 pam 方法被支持,md5 从 pgpool-II 3.0 之后开始被支持。
3546
+
3547
+ 通过使用"pool_passwd"来支持 md5。下面是启用 md5 的步骤:
3548
+
3549
+ <ol>
3550
+ <li>以数据库的操作系统用户(database's operating system user)登入,并键入 "pg_md5 --md5auth --username=<your_username> <your_passwd>",
3551
+ 用户名和md5加密的密码被注册进 pool_passwd。如果 pool_passwd 还不存在,pg_md5 将会自动的为你创建。
3552
+ </li>
3553
+ <li> pool_passwd 格式是 "username:encrypted_passwd".</li>
3554
+ <li>你还需要在 pool_hba.conf 中增加一个相应的 md5 项。详见<a href="#hba">为客户端认证设置 pool_hba.conf (HBA)</a>。
3555
+ </li>
3556
+ <li>请注意的是用户名和密码必须和 PostgreSQL 中注册的一样</li>
3557
+ </ol>
3558
+
3559
+ </li>
3560
+ <li>在所有的其它模式中,trust, clear text password, crypt, md5, 以及 pam都被支持。</li>
3561
+ <li>pgpool-II 并不支持类似 pg_hba.conf 的访问控制。如果 TCP/IP 连接启用的话,pgpool-II 接受来自任何主机的所有连接。若必要,使用iptable来控制其它主机的访问。
3562
+ (当然PostgreSQL服务器能够使用 pg_hba.conf 来接受 pgpool-II 的连接)
3563
+ </li>
3564
+ </ul>
3565
+ </p>
3566
+
3567
+ <h2>大对象 (Large objects)</h2>
3568
+ <p>
3569
+ pgpool-II 2.3.2 或以上版本支持大对象复制,如果后台服务器是 PostgreSQL 8.1 或以上版本。有鉴于此,你需要启用 pgpool.conf 中的<a href="#lobj_lock_table">lobj_lock_table</a>。然而,使用后台函数 lo_import 的大对象复制不被支持。
3570
+ </p>
3571
+
3572
+ <h2>主/备模式中的临时表</h2>
3573
+ <p>
3574
+ Creating/inserting/updating/deleting 临时表将会永远在主(首要)节点上执行。pgpool-II 3.0 或以上版本中,也可以在这些主节点的表上执行 SELECT。然而,如果在 SELECT 中临时表名被用作一个符号,就没有办法来检测到它,这样 SELECT 将被负载均衡。这会产生一个 "not found the table" 错误或者会找到另外一个同名的表。为避免该问题,使用/*NO LOAD BALANCE*/ SQL 注释。
3575
+ </p>
3576
+ <p>
3577
+ <pre>
3578
+ 将会产生问题的示例 SELECT:
3579
+ SELECT 't1'::regclass::oid;
3580
+ </pre>
3581
+ </p>
3582
+ <p>
3583
+ psql 的 \d命令使用符号表名。pgpool-II 3.0 或更高版本检查是否 SELECT 包含任何对系统表(system catalogs)的访问,并永远把这些查询发送到主节点,这样我们避免了该问题。
3584
+ </p>
3585
+
3586
+ <h2>复制模式中的函数等</h2>
3587
+
3588
+ <p>不保证使用依赖上下文的机制所产生的数据(如 random number, transaction ID, OID, SERIAL, sequence),都会被在多个后台服务器上被正确的复制。
3589
+ </p>
3590
+ <p>对于SERIAL,启用insert_lock在数据复制中很有用。insert_lock 还有助于 SELECT setval() 和 SELECT nextval()。
3591
+ </p>
3592
+
3593
+ <p>在 pgpool-II 2.3 或更高版本中,使用 CURRENT_TIMESTAMP,CURRENT_DATE, now()的 INSERT/UPDATE 将会被正确的复制。INSERT/UPDATE 使用 CURRENT_TIMESTAMP, CURRENT_DATE, now() 作为表的默认值时,也将会正确的复制。这通过执行时把这些函数替换成从master节点上获取的常量值来完成。然而仍有一些限制:
3594
+
3595
+ <ul>
3596
+ <li>在 pgpool-II 3.0 或更低的以前的版本中,计算表中默认值的临时数据在一些情况下并不准确,例如,下面的表定义:
3597
+ <pre>
3598
+ CREATE TABLE rel1(
3599
+ d1 date DEFAULT CURRENT_DATE + 1
3600
+ )
3601
+ </pre>
3602
+ 被看成如下的表:
3603
+ <pre>
3604
+ CREATE TABLE rel1(
3605
+ d1 date DEFAULT CURRENT_DATE
3606
+ )
3607
+ </pre>
3608
+ <p> pgpool-II 3.1 后更高版本能够正确的处理这些情况。这样"d1"将会把明天(CURRENT_DATE + 1)作为默认值。然而当扩展的协议(例如在JDBC, PHP PDO中使用的)或者 PREPARE 被使用时,这个改进并不适用。
3609
+ </p>
3610
+ <p>
3611
+ 请注意如果列类型不是临时的,重写并不会被执行,这儿有个例子:
3612
+ <pre>
3613
+ foo bigint default (date_part('epoch'::text,('now'::text)::timestamp(3) with time zone) * (1000)::double precision)
3614
+ </pre>
3615
+ </p>
3616
+ <li>假定我们有如下表:
3617
+ <pre>
3618
+ CREATE TABLE rel1(
3619
+ c1 int,
3620
+ c2 timestamp default now()
3621
+ )
3622
+ </pre>
3623
+ 我们可以复制:
3624
+ <pre>
3625
+ INSERT INTO rel1(c1) VALUES(1)
3626
+ </pre>
3627
+ 因为这将变为:
3628
+ <pre>
3629
+ INSERT INTO rel1(c1, c2) VALUES(1, '2009-01-01 23:59:59.123456+09')
3630
+ </pre>
3631
+ 然而,
3632
+ <pre>
3633
+ INSERT INTO rel1(c1) SELECT 1
3634
+ </pre>
3635
+ 不能够被转换,这样在当前的实现中不能被正确的复制,值将仍然会被插入,但是没有进行任何的转换。
3636
+ </ul>
3637
+ </p>
3638
+ <p>如果你在使用 8.3 或更高的版本 PostgreSQL ,并且在 reset_query_list 中指定 DISCARD ALL,那么将在会话的末尾处删除 <code>CREATE TEMP TABLE</code> 创建的表。
3639
+
3640
+ </p>
3641
+ <p>
3642
+ 对于8.2.x或更早的版本,退出会话时并不会删除 <code>CREATE TEMP TABLE</code> 创建的表。这是由于连接池的原因,从 PostgreSQL 后台的视角来看,要保持 session 处于活动状态。为避免这一点,你必须通过发出 <code>DROP TABLE</code> 命令显式的删除该临时表,或在事务块中使用 <code>CREATE TEMP TABLE ... ON COMMIT DROP</code> 语句。
3643
+ </p>
3644
+
3645
+ <h2>Queries</h2>
3646
+
3647
+ <p>下面是不能被 pgpool-II 处理的查询。</p>
3648
+
3649
+ <h3>INSERT (for parallel mode)</h3>
3650
+
3651
+ <p>你不能使用 <code>DEFAULT</code> 作为分区关键字列(partitioning key column)。例如如果表t中的列x是分区关键字列,
3652
+ <pre>
3653
+ INSERT INTO t(x) VALUES (DEFAULT);
3654
+ </pre>
3655
+ <p>
3656
+ 这将是无效的。而且该值也不能使用函数。
3657
+ </p>
3658
+
3659
+ <pre>
3660
+ INSERT INTO t(x) VALUES (func());
3661
+ </pre>
3662
+ <p>
3663
+ INSERT的分区值只能是常量值。<code>SELECT INTO</code> 和 <code>INSERT INTO ... SELECT</code> 也不被支持。
3664
+ </p>
3665
+
3666
+ <h3>UPDATE (for parallel mode)</h3>
3667
+
3668
+ <p>如果分区关键字列值被更新,后台数据库的数据一致性可能会丢失。pgpool-II 并不重新分区更新的数据。
3669
+ </p>
3670
+
3671
+ <p>如果约束被违反时一个查询在后台数据库中引起的了一个错误,一个事务有可能不能够被回滚。
3672
+ </p>
3673
+
3674
+ <p>如果在 <code>WHERE</code> 从句中一个函数被调用,该查询可能不能够被正确的执行。例如:
3675
+ <pre>
3676
+ UPDATE branches set bid = 100 where bid = (select max(bid) from beances);
3677
+ </pre>
3678
+ </p>
3679
+
3680
+ <h3>SELECT ... FOR UPDATE (for parallel mode)</h3>
3681
+
3682
+ <p>如果在 <code>WHERE</code> 从句中一个函数被调用,该查询可能不能够被正确的执行。例如:
3683
+ <pre>
3684
+ SELECT * FROM branches where bid = (select max(bid) from beances) FOR UPDATE;
3685
+ </pre>
3686
+ </p>
3687
+
3688
+ <h3>COPY (for parallel mode)</h3>
3689
+
3690
+ <p><code>COPY BINARY</code> 不被支持。从文件中拷贝也不被支持。只有 <code>COPY FROM STDIN</code> 和 <code>COPY TO STDOUT</code> 被支持。
3691
+ </p>
3692
+
3693
+ <h3>ALTER/CREATE TABLE (for parallel mode)</h3>
3694
+
3695
+ <p>为更新分区规则,pgpool-II 必须重启以从 System DB 读取它们。</p>
3696
+
3697
+ <h3>Transaction (for parallel mode)</h3>
3698
+
3699
+ <p>一个事务块里面执行的 <code>SELECT</code> 语句将被在一个单独的事务中执行,这有个示例:
3700
+
3701
+ <pre>
3702
+ BEGIN;
3703
+ INSERT INTO t(a) VALUES (1);
3704
+ SELECT * FROM t ORDER BY a; <-- INSERT above is not visible from this SELECT statement
3705
+ END;
3706
+ </pre>
3707
+
3708
+ <h3>Views / Rules (for parallel mode)</h3>
3709
+
3710
+ <p>
3711
+ 将会在所有的后台数据库中为 views 和 rules 创建同样的定义。
3712
+ </p>
3713
+
3714
+ <pre>
3715
+ SELECT * FROM a, b where a.i = b.i
3716
+ </pre>
3717
+ <p>像上面的 <code>JOIN</code> 将在后台数据库中执行,然后由每个后台数据库返回的结果进行合并。跨节点的 Views 和 Rules 不能够被创建。然而若连接的表的数据都在同一个节点中,一个 VIEW 能够被创建。该VIEW需要在 pgpool_catalog.dist_def 中注册。一个 col_name 和 一个 dist_def_func 也必须进行注册,它们当在视图上执行插入操作时被使用。
3718
+ </p>
3719
+
3720
+ <h3>Functions / Triggers (for parallel mode)</h3>
3721
+
3722
+ <p>同样的函数定义将在所有的后台数据库中被创建。函数内不能执行跨节点连接(Join),不能访问其它节点的数据。
3723
+ </p>
3724
+
3725
+ <h3>Extended Query Protocol (for parallel mode)</h3>
3726
+
3727
+ <p>JDBC驱动等等所使用的扩展的查询协议不被支持。必须使用简单的查询协议。这意味着你不能使用prepared statement。
3728
+ </p>
3729
+
3730
+ <h3>Natural Join (for parallel mode)</h3>
3731
+
3732
+ <p>自然连接(Natural Join)不被支持。必须使用"ON join condition" 或 "USING (join_column)"。
3733
+ </p>
3734
+
3735
+ <h3>USING CLAUSE (for parallel mode)</h3>
3736
+
3737
+ <p>USING CLAUSE 通过查询重写过程被转换为 ON CLAUSE。这样,当在目标列表中含有"*"时,连接的列会出现两次。
3738
+ </p>
3739
+ <p>
3740
+ 这儿有个示例:
3741
+ </p>
3742
+ <p><pre>
3743
+ =# SELECT * FROM t1 JOIN t2 USING(id);
3744
+ id | t | t
3745
+ ----+-----+-------
3746
+ 1 | 1st | first
3747
+ (1 row)
3748
+ </pre></p>
3749
+ <p>
3750
+ 在query rewrite过程中"USING"被转换成"ON"。这样下面是生效的重写结果:
3751
+ </p><p><pre>
3752
+ =# SELECT * FROM t1 JOIN t2 ON t1.id = t2.id;
3753
+ id | t | id | t
3754
+ ----+-----+----+-------
3755
+ 1 | 1st | 1 | first
3756
+ (1 row)
3757
+ </pre>
3758
+ </p>
3759
+ <p>
3760
+ 注意列"t"被重复。
3761
+ </p>
3762
+
3763
+ <h3>多字节字符 (for all modes)</h3>
3764
+
3765
+ <p>pgpool-II 不会在不同的多字节字符间进行转换。client, backend 以及 System DB 之间的编码必须相同。
3766
+ </p>
3767
+
3768
+ <h3>Multi-statement Query (for all modes)</h3>
3769
+ <p>
3770
+ pgpool-II 不处理 multi-statement queries。
3771
+ </p>
3772
+
3773
+ <h3>Deadlocks (for parallel mode)</h3>
3774
+
3775
+ <p>跨越多个后台数据库的死锁不能够被检测。例如:
3776
+ </p>
3777
+ <pre>
3778
+ (tellers table is partitioned using the following rule)
3779
+ tid <= 10 --> node 0
3780
+ tid >= 10 --> node 1
3781
+
3782
+ A) BEGIN;
3783
+ B) BEGIN;
3784
+ A) SELECT * FROM tellers WHERE tid = 11 FOR UPDATE;
3785
+ B) SELECT * FROM tellers WHERE tid = 1 FOR UPDATE;
3786
+ A) SELECT * FROM tellers WHERE tid = 1 FOR UPDATE;
3787
+ B) SELECT * FROM tellers WHERE tid = 11 FOR UPDATE;
3788
+ </pre>
3789
+ <p>
3790
+ 在上述情况下,单个的节点不能探测到死锁,这样pgpool-II将会无限期的等待响应。该现象可出现在任何获取行级别锁(row level lock)的query中。
3791
+ </p>
3792
+ <p>而且,如果一个节点上出现了一个死锁,每个节点的事务状态将不再具有一致性。这样如果死锁被探测到 pgpool-II 将终止该进程。
3793
+ </p>
3794
+ <pre>
3795
+ pool_read_kind: kind does not match between master(84) slot[1] (69)
3796
+ </pre>
3797
+
3798
+ <h3>Schemas (for parallel mode)</h3>
3799
+
3800
+ <p>schema 中非 public 下的对象必须使用完整名字,如:
3801
+ </p>
3802
+ <pre>
3803
+ schema.object
3804
+ </pre>
3805
+ <p>
3806
+ 当路径设置如下时,pgpool-II 不能找到正确的 schema:
3807
+ </p>
3808
+ <pre>
3809
+ set search_path = xxx
3810
+ </pre>
3811
+ <p>
3812
+ 且 schema 名在 query 中被省略。
3813
+ </p>
3814
+ <h3>table name - column name(for parallel mode)</h3>
3815
+ <p>
3816
+ Limitation object:Parallel mode
3817
+ </p>
3818
+ <p>
3819
+ 一个表或列名不能是以 pool_ 起头。当重写查询时,这些名字被保留为内部处理。
3820
+ </p>
3821
+
3822
+ </p>
3823
+ <h2>System DB</h2>
3824
+
3825
+ <h3>分区规则 (Partitioning Rules)</h3>
3826
+
3827
+ <p>仅可在一个分区规则中定义一个分区关键字列。如同'x or y'的条件式不被支持。</p>
3828
+
3829
+ <h2>环境要求 (Environment Requirements)</h2>
3830
+
3831
+ <h3>libpq</h3>
3832
+
3833
+ <p>
3834
+ 在编译构建pgpool-II时<code>libpq</code>被链接。libpq版本必须是3.0。使用2.0版本的libpq编译pgpool-II将会失败。而且System DB 必须是 PostgreSQL 7.4 或更高版本。
3835
+ </p>
3836
+
3837
+ <h2 id="caution_query_cache">查询缓存</h2>
3838
+
3839
+ <p>当前,<a href="#query_cache">查询缓存</a>必须被手动删除。
3840
+ 但这不影响 <a href="#memqcache">基于内存的查询缓存</a>。基于内存的查询缓存会自动清除无效的缓存。</p>
3841
+
3842
+ <h2>与pgpool的兼容性 (Compatibility with pgpool)</h2>
3843
+
3844
+
3845
+ <p class="top_link"><a href="#Top">back to top</a></p>
3846
+
3847
+ <!-- ================================================================================ -->
3848
+
3849
+ <h1>参考<a name="reference"></a></h1>
3850
+
3851
+ <h2>PCP 命令参考</h2>
3852
+
3853
+ <h3>PCP 命令列表</h3>
3854
+
3855
+ <p>PCP 命令是UNIX命令,通过网络操作pgpool-II。</p>
3856
+
3857
+ <table border>
3858
+ <tr><th><a href="#pcp_node_count">pcp_node_count</a></th>
3859
+ <td>获取节点数量</td></tr>
3860
+ <tr><th><a href="#pcp_node_info">pcp_node_info</a></th>
3861
+ <td>获取节点信息</td></tr>
3862
+ <tr><th><a href="#pcp_proc_count">pcp_proc_count</a></th>
3863
+ <td>获取进程列表</td></tr>
3864
+ <tr><th><a href="#pcp_proc_info">pcp_proc_info</a></th>
3865
+ <td>获取进程信息</td></tr>
3866
+ <tr><th><a href="#pcp_pool_status ">pcp_pool_status</a> <span class="version">V3.1 -</span></th>
3867
+ <td>获取 pgpool.conf 中的参数</td></tr>
3868
+ <tr><th><a href="#pcp_systemdb_info">pcp_systemdb_info</a></th>
3869
+ <td>获取 System DB 信息</td></tr>
3870
+ <tr><th><a href="#pcp_detach_node">pcp_detach_node</a></th>
3871
+ <td>从 pgpool-II 脱离一个节点</td></tr>
3872
+ <tr><th><a href="#pcp_attach_node">pcp_attach_node</a></th>
3873
+ <td>给 pgpool-II 关联一个节点</td></tr>
3874
+ <tr><th><a href="#pcp_promote_node">pcp_promote_node</a> <span class="version">V3.1 -</span></th>
3875
+ <td>给 pgpool-II 提升一个新的master节点</td></tr>
3876
+ <tr><th><a href="#pcp_stop_pgpool">pcp_stop_pgpool</a></th>
3877
+ <td>停止 pgpool-II</td></tr>
3878
+ </table>
3879
+
3880
+ <h2 id="pcp_comand_args">常用的命令行参数</h2>
3881
+
3882
+ <p>对于所有的PCP命令有5个常用的参数。它们给出了pgpool-II的信息和认证。对于有些命令必须需要多余的参数。
3883
+
3884
+ <pre>
3885
+ 例如) $ pcp_node_count 10 localhost 9898 postgres hogehoge
3886
+
3887
+ 第1个 argument - 以秒为单位的等待时间. 如果pgpool-II在这个数量的秒内没有响应,PCP将断开连接。
3888
+ 第2个 argument - pgpool-II的主机名
3889
+ 第3个 argument - PCP 端口号
3890
+ 第4个 argument - PCP 用户名
3891
+ 第5个 argument - PCP 密码
3892
+ </pre>
3893
+
3894
+ <p>PCP的用户名和密码必须在<code>$prefix/etc</code>目录中的<code>pcp.conf</code>中声明。
3895
+ 启动pgpool-II时<code>-F</code>选项可被用来指定<code>pcp.conf</code>所处的别的位置。当传递给PCP命令时,并不需要以md5格式的密码。
3896
+ </p>
3897
+
3898
+
3899
+ <h2>PCP 命令</h2>
3900
+
3901
+ <p>所有的PCP命令把结果显示在标准输出中。</p>
3902
+
3903
+
3904
+ <h3>pcp_node_count</h3>
3905
+
3906
+ <pre>
3907
+ Format:
3908
+ pcp_node_count _timeout_ _host_ _port_ _userid_ _passwd_
3909
+ </pre>
3910
+
3911
+ <p>
3912
+ 显示<code>pgpool.conf</code>中定义的所有的节点总数。它并不区别节点状态,例如attached/detached。所有节点都被算进去。
3913
+ </p>
3914
+
3915
+
3916
+ <h3><a name="pcp_node_info"/>pcp_node_info</a></h3>
3917
+
3918
+ <pre>
3919
+ Format:
3920
+ pcp_node_info _timeout_ _host_ _port_ _userid_ _passwd_ _nodeid_
3921
+ </pre>
3922
+
3923
+ <p>
3924
+ 显示给定节点 ID 的信息,这里是一个输出示例:
3925
+ </p>
3926
+
3927
+ <pre>
3928
+ $ pcp_node_info 10 localhost 9898 postgres hogehoge 0
3929
+ host1 5432 1 1073741823.500000
3930
+
3931
+ 结果以如下顺序被显示:
3932
+ 1. hostname
3933
+ 2. port number
3934
+ 3. status
3935
+ 4. load balance weight
3936
+
3937
+ Status 由数字 [0 - 3]来表示。
3938
+ 0 - 该状态仅仅用于初始化,PCP从不显示它。
3939
+ 1 - 节点已启动,还没有连接。
3940
+ 2 - 节点已启动,连接被缓冲。
3941
+ 3 - 节点已关闭。
3942
+ </pre>
3943
+ <p>
3944
+ 负载均衡权重以标准格式来显示。
3945
+ </p>
3946
+
3947
+ <p>
3948
+ --verbose 选项帮助理解输出. 例如:
3949
+ </p>
3950
+
3951
+ <pre>
3952
+ $ pcp_node_info --verbose 10 localhost 9898 postgres hogehoge 0
3953
+ Hostname: host1
3954
+ Port : 5432
3955
+ Status : 1
3956
+ Weight : 0.5
3957
+ </pre>
3958
+
3959
+ <p>指定一个无效的节点 ID 将会导致一个退出状态为 12 的错误,而且 BackendError 将被显示。</p>
3960
+
3961
+ <h3>pcp_proc_count</h3>
3962
+ <p>
3963
+ <pre>
3964
+ Format:
3965
+ pcp_proc_count _timeout_ _host_ _port_ _userid_ _passwd_
3966
+ </pre>
3967
+ <p>
3968
+ 显示 pgpool-II 子进程 ID 列表。如果有多于一个的进程,ID 间将用空格来隔开。
3969
+ </p>
3970
+ <h3>pcp_proc_info</h3>
3971
+ <p>
3972
+ <pre>
3973
+ Format:
3974
+ pcp_proc_info _timeout_ _host_ _port_ _userid_ _passwd_ _processid_
3975
+ </pre>
3976
+ <p>
3977
+ 显示给定 pgpool-II 孩子进程 ID 的信息。输出结果示例如下:
3978
+ </p>
3979
+ <pre>
3980
+ $ pcp_proc_info 10 localhost 9898 postgres hogehoge 3815
3981
+ postgres_db postgres 1150769932 1150767351 3 0 1 1467 1
3982
+ postgres_db postgres 1150769932 1150767351 3 0 1 1468 1
3983
+
3984
+ 结果以如下顺序被显示:
3985
+ 1. 连接的数据库名
3986
+ 2. 连接的用户名
3987
+ 3. 进程启动时间戳
3988
+ 4. 连接创建时间戳
3989
+ 5. 协议major版本
3990
+ 6. 协议minor版本
3991
+ 7. 连接重用计数
3992
+ 8. PostgreSQL backend 进程 ID
3993
+ 9. 前端连接时为1,否则为0
3994
+ </pre>
3995
+ <p>
3996
+ 如果没有与后台的连接,不显示任何内容。如果有多个连接,将显示多行,每行显示一个连接的信息,时间戳显示格式是 EPOCH。
3997
+ </p>
3998
+
3999
+ <p>
4000
+ --verbose 可选项帮助理解输出. 示例如下:
4001
+ </p>
4002
+
4003
+ <pre>
4004
+ $ pcp_proc_info --verbose 10 localhost 9898 postgres hogehoge 3815
4005
+ Database : postgres_db
4006
+ Username : postgres
4007
+ Start time : 1150769932
4008
+ Creation time: 1150767351
4009
+ Major : 3
4010
+ Minor : 0
4011
+ Counter : 1
4012
+ PID : 1467
4013
+ Connected : 1
4014
+ Database : postgres_db
4015
+ Username : postgres
4016
+ Start time : 1150769932
4017
+ Creation time: 1150767351
4018
+ Major : 3
4019
+ Minor : 0
4020
+ Counter : 1
4021
+ PID : 1468
4022
+ Connected : 1
4023
+ </pre>
4024
+
4025
+ <p>指定一个无效的节点 ID 将会导致一个退出状态为 12 的错误,而且 BackendError 将被显示。</p>
4026
+
4027
+ <h3 id="pcp_pool_status">pcp_pool_status <span class="version">V3.1 -</span></h3>
4028
+ <pre>
4029
+ 格式:
4030
+ pcp_pool_status _timeout_ _host_ _port_ _userid_ _passwd_
4031
+ </pre>
4032
+ <p>
4033
+ 显示 pgpool.conf 中的参数。以下是一个输出的示例:
4034
+ </p>
4035
+ <pre>
4036
+ $ pcp_pool_status 10 localhost 9898 postgres hogehoge
4037
+ name : listen_addresses
4038
+ value: localhost
4039
+ desc : host name(s) or IP address(es) to listen to
4040
+
4041
+ name : port
4042
+ value: 9999
4043
+ desc : pgpool accepting port number
4044
+
4045
+ name : socket_dir
4046
+ value: /tmp
4047
+ desc : pgpool socket directory
4048
+
4049
+ name : pcp_port
4050
+ value: 9898
4051
+ desc : PCP port # to bind
4052
+ </pre>
4053
+
4054
+ <h3>pcp_systemdb_info</h3>
4055
+ <p>
4056
+ <pre>
4057
+ Format:
4058
+ pcp_systemdb_info _timeout_ _host_ _port_ _userid_ _passwd_
4059
+ </pre>
4060
+ <p>
4061
+ 显示 System DB 信息。输出示例如下:
4062
+ </p>
4063
+ <pre>
4064
+ $ pcp_systemdb_info 10 localhost 9898 postgres hogehoge
4065
+ localhost 5432 yamaguti '' pgpool_catalog pgpool 3
4066
+ yamaguti public accounts aid 4 aid bid abalance filler integer integer integer character(84) dist_def_accounts
4067
+ yamaguti public branches bid 3 bid bbalance filler integer integer character(84) dist_def_branches
4068
+ yamaguti public tellers bid 4 tid bid tbalance filler integer integer integer character(84) dist_def_tellers
4069
+
4070
+ 首先,System DB 信息将被显示在第一行。结果以如下顺序被显示:
4071
+ 1. 主机名
4072
+ 2. 端口号
4073
+ 3. 用户名
4074
+ 4. 密码,若无显示为 ''
4075
+ 5. schema 名
4076
+ 6. 数据库名
4077
+ 7. 定义的分区规则数量
4078
+ </pre>
4079
+ <p>
4080
+ 然后,定义的分区规则将被在后续行显示。如果有多个定义,将显示多行,每行显示一个定义。结果以如下顺序被显示:
4081
+ </p>
4082
+ <pre>
4083
+ 1. 目标分区数据库名
4084
+ 2. 目标分区schema名
4085
+ 3. 目标分区表名
4086
+ 4. 分区关键字列名
4087
+ 5. 目标表中的列树
4088
+ 6. 列名 (最多显示5个)
4089
+ 7. 列类型 (最多显示5个)
4090
+ 8. 分区规则函数名
4091
+ </pre>
4092
+ <p>
4093
+ 如果 System DB 没有被定义(例如没有在 pgpool-II 模式,且 query cache 没有被启用),将会导致一个退出状态为12的错误,而且 BackendError 将被显示。
4094
+ </p>
4095
+
4096
+
4097
+ <h3>pcp_detach_node</h3>
4098
+ <pre>
4099
+ Format:
4100
+ pcp_detach_node [-g] _timeout_ _host_ _port_ _userid_ _passwd_ _nodeid_
4101
+ </pre>
4102
+ <p>
4103
+ 把指定的节点从 pgpool-II 分离。如果设定-g,将等待所有的客户端断开(除非client_idle_limit_in_recovery为-1,或者 recovery_timeout 过期)。
4104
+ </p>
4105
+
4106
+
4107
+ <h3>pcp_attach_node</h3>
4108
+ <p>
4109
+ <pre>
4110
+ Format:
4111
+ pcp_attach_node _timeout_ _host_ _port_ _userid_ _passwd_ _nodeid_
4112
+
4113
+ 把给定的节点加入到 pgpool-II。
4114
+ </pre>
4115
+ </p>
4116
+
4117
+ <h3 id="pcp_promote_node">pcp_promote_node <span class="version">V3.1 -</span></h3>
4118
+ <p>
4119
+ <pre>
4120
+ Format:
4121
+ pcp_promote_node [-g] _timeout_ _host_ _port_ _userid_ _passwd_ _nodeid_
4122
+
4123
+ 为 pgpool-II 提升指定的节点为 master 。只适用于 master/slave 流复制。如果设定-g,将等待所有的客户端断开(除非 client_idle_limit_in_recovery 为-1,或者 recovery_timeout 过期)。
4124
+ </pre>
4125
+ </p>
4126
+
4127
+ <h3>pcp_stop_pgpool</h3>
4128
+ <pre>
4129
+ Format:
4130
+ pcp_stop_pgpool _timeout_ _host_ _port_ _userid_ _passwd_ _mode_
4131
+ </pre>
4132
+
4133
+ <p>
4134
+ 使用指定的模式关闭 pgpool-II 进程。可用的模式如下:
4135
+ </p>
4136
+
4137
+ <pre>
4138
+ s - 智能模式 (smart mode)
4139
+ f - 快速模式 (fast mode)
4140
+ i - 立即模式 (immediate mode)
4141
+ </pre>
4142
+ <p>
4143
+ 如果 pgpool-II 进程不存在,将会导致一个退出状态为 8 的错误,而且 BackendError 将被显示。
4144
+ </p>
4145
+ <p>* 当前,快速和立即模式没有区别。pgpool-II 终止所有的进程无论是否还有客户端连接到后端服务器。
4146
+ </p>
4147
+
4148
+
4149
+ <h2>退出状态 (Exit Status)</h2>
4150
+
4151
+ <p>PCP 命令正确退出时状态为0.若由错误出现,将会具有如下的错误状态。
4152
+
4153
+ <table border>
4154
+ <tr><th>UNKNOWNERR</th>
4155
+ <td>1</td><td>Unknown Error (should not occur)</td></tr>
4156
+ <tr><th>EOFERR</th>
4157
+ <td>2</td><td>EOF Error</td></tr>
4158
+ <tr><th>NOMEMERR</th>
4159
+ <td>3</td><td>Memory shortage</td></tr>
4160
+ <tr><th>READERR</th>
4161
+ <td>4</td><td>Error while reading from the server</td></tr>
4162
+ <tr><th>WRITEERR</th>
4163
+ <td>5</td><td>Error while writing to the server</td></tr>
4164
+ <tr><th>TIMEOUTERR</th>
4165
+ <td>6</td><td>Timeout</td></tr>
4166
+ <tr><th>INVALERR</th>
4167
+ <td>7</td><td>Argument(s) to the PCP command was invalid</td></tr>
4168
+ <tr><th>CONNERR</th>
4169
+ <td>8</td><td>Server connection error</td></tr>
4170
+ <tr><th>NOCONNERR</th>
4171
+ <td>9</td><td>No connection exists</td></tr>
4172
+ <tr><th>SOCKERR</th>
4173
+ <td>10</td><td>Socket error</td></tr>
4174
+ <tr><th>HOSTERR</th>
4175
+ <td>11</td><td>Hostname resolution error</td></tr>
4176
+ <tr><th>BACKENDERR</th>
4177
+ <td>12</td><td>PCP process error on the server (specifying an invalid ID, etc.)</td></tr>
4178
+ <tr><th>AUTHERR</th>
4179
+ <td>13</td><td>Authorization failure</td></tr>
4180
+ </table>
4181
+
4182
+ </p>
4183
+
4184
+ <h1>内部信息 (Internal information)<a name="internal"></a></h1>
4185
+ <p>
4186
+ 相比于1.x版本,pgpool-II 2.0.x 带来大量的修改。请注意下面的内容不适用1.x版本。
4187
+ </p>
4188
+
4189
+ <h2>并行执行引擎 (Parallel execution engine)</h2>
4190
+ <p>
4191
+ 并行执行引擎内置于pgpool-II。殷勤在每一个节点上执行相同的查询(Query),根据节点的响应,传送结果到前端。
4192
+ </p>
4193
+
4194
+ <h2>查询重写 (Query Rewriting)</h2>
4195
+ <p>
4196
+ 本节解释了 pgpool-II 并行模式下的查询重写。
4197
+ </p>
4198
+ <p>
4199
+ 在并行模式下,由客户端发出的查询会经历两个阶段的处理:
4200
+ </p>
4201
+ <ul>
4202
+ <li>查询分析 (Analysis of Query)
4203
+ <li>查询重写 (Rewriting of Query)
4204
+ </ul>
4205
+ <p>
4206
+ 下面解释了这两个阶段:
4207
+ </p>
4208
+ <h3>查询分析 (Analysis of Query)</h3>
4209
+ <h4>概述</h4>
4210
+ <p>
4211
+ 客户端提交的获取查询(retrieval Query)将会通过 SQL 分析器。然后将会使用存储在system DB中的信息对其进行分析。将会使用该信息更新该查询的每一部分的执行状态。该执行状态存储了哪个节点可以处理。例如,如果一个表的数据分布在多个节点上(正如 catalog 里的 pgpool_catalog.dist_def 中所声明的那样),就需要从所有的节点上获取数据。另外一方面,注册在 pgpool_catalog.replicate_def 中的表的数据若已复制,则可从任意一个节点获取。当数据需要从所有节点上处理时,状态是'P',当需要从一个节点上处理时,状态是'L','S'状态是一个特殊的情况:意味着从所有节点获取后还需要执行另外一个步骤。例如需要对注册在 pgpool_catalog.dist_def 中表的数据进行排序。
4212
+ </p>
4213
+
4214
+ <p>
4215
+ 获取查询(retrieval Query)将被以如下顺序进行分析,且在此处理过程中会改变执行状态。查询最终将在哪儿处理取决于主选择的最终状态(main select)。
4216
+ </p>
4217
+
4218
+ <ol>
4219
+ <li>是否使用了UNION, EXTRACT, INTERSECT?
4220
+ <li>FROM语句的执行状态是什么?
4221
+ <li>通过TARGETLIST改变执行状态
4222
+ <li>根据WHERE语句对执行状态的改变
4223
+ <li>根据GROUP BY语句对执行状态的改变
4224
+ <li>根据HAVING语句对执行状态的改变
4225
+ <li>根据ORDER BY语句对执行状态的改变
4226
+ <li>执行状态中LIMIT OFFSET断言的改变
4227
+ <li>获取SELECT的最终执行状态
4228
+ </ol>
4229
+
4230
+ <p>
4231
+ SELECT最终执行状态(Execution status)和处理位置(Processed place)的关系如下:
4232
+ </p>
4233
+
4234
+ <p>
4235
+ <table border>
4236
+ <tr><td>Execution status</td><td>Processed place</td></tr>
4237
+ <tr><td align=center>L</td><td>查询可发送到任意一个节点上 </td></tr>
4238
+ <tr><td align=center>P</td><td>在所有节点上运行同样的查询,并返回数据到客户端,使用并行执行引擎</td></tr>
4239
+ <tr><td align=center>S</td><td>使用system DB处理完之后,返回数据到客户端</td></tr>
4240
+ </table>
4241
+ </p>
4242
+
4243
+ <p>
4244
+ 上面提到的规则也适用于子查询(Sub-Query)。在下面简单的查询中,如果 p1-table 注册在 system DB 的 pgpool_catalog.dist_def 表中(p1-table 是分布式的),最终的子查询的执行状态变为P,这样子查询的父查询,执行状态也变为了P。
4245
+ </p>
4246
+
4247
+ <pre>
4248
+ SELECT * FROM (SELECT * FROM P1-table) as P2-table;
4249
+ </pre>
4250
+
4251
+ <p>
4252
+ 接下来,让我们解释一下执行状态是如何具体改变的。让我们从一个例子开始,来解释 FROM 的状态。
4253
+ </p>
4254
+
4255
+ <h4><p>FROM 语句的执行状态</p></h4>
4256
+ <p>
4257
+ 这是一个获取查询(SELECT)。数据集和它的状态(P,L 和 S)根据FROM 语句进行定义。表的执行状态如下:当 FROM 语句中只有一个表时,整个数据集的执行状态和该表的执行状态一致。当 FROM 语句中有两个或多个或者有子查询时,根据JOIN的方法以及执行状态来执行查询,如下表所示。
4258
+ </p>
4259
+
4260
+ <p>
4261
+ <table border>
4262
+ <tr><td>JOIN type</td><td align = center colspan = 3> LEFT OUTER JOIN </td><td align = center colspan = 3> RIGHT OUTER JOIN </td><td align = center colspan = 3>FULL OUTER JOIN</td><td align = center colspan = 3>Other</td></tr>
4263
+ <tr><td align = center>left/right</td><td> P </td><td> L </td><td> S </td><td> P </td><td> L </td><td> S </td><td> P </td><td> L </td><td> S </td><td> P </td><td> L </td><td> S </td></tr>
4264
+
4265
+ <tr><td align = center> P </td><td> S </td><td> P </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> P </td><td> S </td></tr>
4266
+
4267
+ <tr><td align = center> L </td><td> S </td><td> L </td><td> S </td><td> P </td><td> L </td><td> S </td><td> S </td><td> L </td><td> S </td><td> P </td><td> L </td><td> S </td></tr>
4268
+
4269
+ <tr><td align = center> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td><td> S </td></tr>
4270
+
4271
+ </td></tr>
4272
+ </table>
4273
+ </p>
4274
+
4275
+ <p>
4276
+ 在下面的示例中,P1-table是P状态,L1-table和L2-table是L状态。
4277
+ <pre>
4278
+ SELECT * FROM P1-table,L1-table,L2-table;
4279
+ </pre>
4280
+ P1-table (左) and L1-table (右)被连接,根据上表它们获得了P状态。根据P状态,它们与L状态的L2-table进行连接,连接之后现在也是P状态。
4281
+ </p>
4282
+
4283
+
4284
+ <h4><p>由于 TARGETLIST 和 WHERE 从句导致执行状态的改变</p></h4>
4285
+ <p>
4286
+ 在一个基本的查询中,执行状态来自于 FROM 语句。然而,如果存在一个T ARGETLIST,WHERE 语句的执行状态能够以如下情况进行改变:
4287
+ </p>
4288
+ <ol>
4289
+ <li>当有一个子查询
4290
+ <li>当有一个聚集函数或者标记为'P'的 TARGETLIST 含 DISTINCT
4291
+ <li>当使用的一个列不存在于 FROM 语句中一个表(数据集)中时
4292
+ </ol>
4293
+ <p>
4294
+ 在这些情况下,如果初始状态是P 或 S,最终的子查询、TARGETLIST 和 WHERE 语句的执行状态是S。
4295
+ <p>
4296
+ 在下面的例子中,当子查询使用的表的状态是P,那么最终的子查询的执行状态是P。这样 WHERE 从句获得了S状态,无论L1的状态是什么,该查询将被运行在 system DB 中。
4297
+ <pre>
4298
+ SELECT * FROM L1-table where L1-table.column IN (SELECT * FROM P1-table);
4299
+ </pre>
4300
+ <p>
4301
+ 当一个状态为'P'的 TARGETLIST 含有一个聚集函数时,为了在数据获取之后执行聚集函数,FROM语句的状态改为S。在特定的条件下将对聚集函数做一些优化。
4302
+ </p>
4303
+ <p>若一个列不存在于一个表中时,也可以在一个查询中使用。例如在下面的子查询中:
4304
+ </p>
4305
+ <pre>
4306
+ SELECT * FROM L1-table WHERE L1-table.col1 IN (SELECT * FROM P1-table WHERE P1-table.col = L1-table.col1);
4307
+ </pre>
4308
+ <p>
4309
+ 该子查询从L1-table表中引用了L1-table.col1,含子查询的 WHERE 语句的执行状态为S。
4310
+ </p>
4311
+ <h4><p>由于 GROUP BY, HAVING, ORDER BY 和 LIMIT/OFFSET 导致执行状态的改变</p></h4>
4312
+
4313
+ <p>
4314
+ 执行状态是'P'的 WHERE 语句,若还存在GROUP BY, HAVING, ORDER BY 语句, 或 LIMIT/OFFSET 断言时,状态改为'S'。没有 GROUP BY 的查询从 WHERE 语句获取状态。类似的方法,若没有 HAVING 语句,将使用 GROUP BY 语句的执行状态。ORDER BY 语句和 LIMIT/OFFSET 断言具有类似的行为。
4315
+ </p>
4316
+
4317
+ <h4><p>当使用 UNION, EXTRACT, 及 INTERSECT 时</p></h4>
4318
+ <p>
4319
+ UNION, EXTRACT, 及 INTERSECT 查询的状态依赖于最终左侧和右侧 SELECT 语句的状态。若左右侧的状态都是L,组合的状态是L。若二者都是P,且查询是 UNION ALL,那么组合的状态是P。其它情况最终结果都是S。
4320
+
4321
+ </p>
4322
+ <h4><p>获取 SELECT 的最终执行状态</p></h4>
4323
+ <p>
4324
+ 如果 SELECT 中所有语句都具有状态L,那么最终的执行状态是L。同样规则使用P。对于任何其它的组合最终状态是S。如果状态是L,若 loadbalance_mode 为 true 时负载被分布在不同的节点中,否则只被发送到 MASTER 节点上。对于状态P,将使用并行执行引擎进行并行处理。对于S,将会进入下面讲述的查询重写。
4325
+ </p>
4326
+
4327
+ <h3>查询重写 (Query rewriting)</h3>
4328
+ <p>
4329
+ 通过使用查询分析获取的执行状态来重写查询。下面是一个例子,P1-table具有P状态,L1-table具有L状态。
4330
+ </p>
4331
+ <pre>
4332
+ SELECT P1-table.col, L1-table.col
4333
+ FROM P1-table,L1-table
4334
+ where P1-table.col = L1-table.col
4335
+ order by P1-table.col;
4336
+ </pre>
4337
+
4338
+ <p>
4339
+ 在该查询中,由于有一个 ORDER BY 语句,状态是S。FROM 语句, WHERE 语句, 和 TARGETLIST 都为P状态,查询被重写成:
4340
+ </p>
4341
+
4342
+ <pre>
4343
+ SELECT P1-table.col, L1-table.col FROM
4344
+ dblink(select pool_parallel(SELECT P1-table.col, L1-table.col FROM P1-table,L1-table where P1-table.col = L1-table.col))
4345
+ order by P1-table.col;
4346
+ </pre>
4347
+ <p>
4348
+ 此处dblink传送该查询到 pgpool-II。pool_parallel 函数负责发送该查询到并行执行引擎。
4349
+ </p>
4350
+ <p>在该示例中 FROM 和 WHERE 语句以及 TARGETLIST 被以并行模式执行。这不是一个真正的重写查询,只是因为想提供一个示例。
4351
+ </p>
4352
+ <p>
4353
+ 这儿有另一个例子:
4354
+ </p>
4355
+
4356
+ <pre>
4357
+ SELECT L1-table.col FROM L1-table WHERE L1-table.col % 2 = 0 AND L1-table.col IN (SELECT P1-table FROM P1-table) ;
4358
+ </pre>
4359
+
4360
+ <p>
4361
+ 在该示例中,FROM 和 WHERE 语句以及 TARGETLIST 都具有状态L。因为子查询是P状态,查询本身也为S状态。这样重写为如下:
4362
+ </p>
4363
+ <pre>
4364
+ SELECT L1-table.col
4365
+ FROM dblink(SELECT loadbalance(SELECT L1-table.col
4366
+ FROM L1-table
4367
+ WHERE L1-table.col % 2 = 0
4368
+ AND TRUE))
4369
+ WHERE
4370
+ L1-table.col %2 = 0 AND
4371
+ L1-table.col IN
4372
+ (
4373
+ SELECT P1-Table FROM
4374
+ dblink(select pool_parallel(SELECT P1-table FROM P1-table))
4375
+ ) ;
4376
+ </pre>
4377
+ <p>pool_loadbalance 是一个函数,负责分发查询到任一个节点。
4378
+ </p>
4379
+
4380
+ <h3>为聚集函数的查询重写</h3>
4381
+ <p>
4382
+ 对于分组查询(聚集函数和 GROUP BY),重写为了减少system DB上的负载将试图在每一个节点上执行第一次聚集。
4383
+ </p>
4384
+ <p>首先,我们看看 pgpool-II 为重写都做了那些事情。
4385
+ </p>
4386
+ <p>该查询的 FROM 语句具有P状态且 TARGETLIST 含有count(*),于是重写如下:
4387
+ </p>
4388
+ <pre>
4389
+ select count(*) from P1-table;
4390
+
4391
+ -> rewrite
4392
+
4393
+ SELECT
4394
+ sum(pool_c$1) as count
4395
+ FROM
4396
+ dblink(select pool_parallel('select count(*) from P1-table'))
4397
+ AS pool_$1g (pool_c$1 bigint);
4398
+ </pre>
4399
+ <p>在如下的条件下查询就会重写为上面类似的形式。
4400
+ </p>
4401
+ <ol>
4402
+ <li> FROM 语句具有P状态。
4403
+ <li>聚集函数(仅 count, sum, min, max, 和 avg)以及 GROUP BY 中的列存在于 target list 中。
4404
+ <li>WHERE 语句具有P状态。
4405
+ <li>仅仅使用聚集函数中定义的、HAVING 语句和 FROM 语句中的、以及 GROUP BY 中的列
4406
+ </ol>
4407
+ <h3 id="caution_in_parallel_mode">并行模式一些说明</h3>
4408
+ <p>
4409
+ 并行模式下查询分析时需要列名和类型。这样当子查询的 TARGETLIST 中含有一个表达式或函数时,就需要别名和类型(通过映射 through a cast)。
4410
+ 请注意如果没有为表达式或函数提供映射,默认将选择文本类型。对于count(),假定为bigint类型,对于sum()为numeric类型,对于min()/max(),如果参数是date类型,返回类型为date,
4411
+ 其它的假定为numeric类型。avg() 被处理为 sum()/count() (总和除以数量)。</p>
4412
+
4413
+ <h3 id="performance_parallel_mode">关于并行模式的性能</h3>
4414
+ <p>下面是一个相对于执行状态的查询性能的粗略估计:
4415
+ </p>
4416
+
4417
+ <table border>
4418
+ <tr><td>执行状态</td><td>性能</td></tr>
4419
+ <tr><td align = center>L</td><td>一个节点时除去pgpool-II的开销,没有性能损失,因为根本就没有并行查询</td></tr>
4420
+ <tr><td align = center>P</td><td>并行处理是很快的,特别针对顺序扫描。由于扫描大表变成在几个节点上分布式存在的小表上进行并行扫描,故很容易获得加速比。
4421
+ <tr><td align = center>S</td><td>当聚集函数可在并行方式被重写时,它们很快。</td></tr>
4422
+ </td></tr>
4423
+ </table>
4424
+
4425
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
4426
+
4427
+ <!-- ================================================================================ -->
4428
+
4429
+ <h1>入门教程</h1>
4430
+ <p>
4431
+ <a href="tutorial-zh_cn.html">pgpool-II入门教程</a> 已发布。
4432
+ </p>
4433
+
4434
+ <p class="top_link"><a href="#Top">返回顶部</a></p>
4435
+
4436
+ </div>
4437
+
4438
+ <div class="copyright" style="clear: both">
4439
+ <hr>
4440
+ <copyright>
4441
+ Copyright &copy; 2003 &ndash; 2012 pgpool Global Development Group
4442
+ </copyright>
4443
+ </div>
4444
+ </body>
4445
+ </html>